feat(auth): sync credentials from Claude CLI

Extend external CLI credential sync to also read from Claude CLI's
~/.claude/.credentials.json file. This allows users who authenticate
with Claude CLI to automatically have those credentials available
in Moltbot without manual configuration.

- Reads OAuth tokens from Claude CLI credential store
- Auto-refreshes when token expires
- Respects same TTL and near-expiry thresholds as Qwen CLI sync
This commit is contained in:
Glucksberg 2026-01-30 00:23:11 +00:00
parent 01e0d3a320
commit e88559a141

View File

@ -1,5 +1,9 @@
import { readQwenCliCredentialsCached } from "../cli-credentials.js";
import {
readClaudeCliCredentialsCached,
readQwenCliCredentialsCached,
} from "../cli-credentials.js";
import {
CLAUDE_CLI_PROFILE_ID,
EXTERNAL_CLI_NEAR_EXPIRY_MS,
EXTERNAL_CLI_SYNC_TTL_MS,
QWEN_CLI_PROFILE_ID,
@ -22,18 +26,20 @@ function shallowEqualOAuthCredentials(a: OAuthCredential | undefined, b: OAuthCr
);
}
function isExternalProfileFresh(cred: AuthProfileCredential | undefined, now: number): boolean {
function isExternalProfileFresh(
cred: AuthProfileCredential | undefined,
now: number,
provider: string,
): boolean {
if (!cred) return false;
if (cred.type !== "oauth" && cred.type !== "token") return false;
if (cred.provider !== "qwen-portal") {
return false;
}
if (cred.provider !== provider) return false;
if (typeof cred.expires !== "number") return true;
return cred.expires > now + EXTERNAL_CLI_NEAR_EXPIRY_MS;
}
/**
* Sync OAuth credentials from external CLI tools (Qwen Code CLI) into the store.
* Sync OAuth credentials from external CLI tools (Claude CLI, Qwen Code CLI) into the store.
*
* Returns true if any credentials were updated.
*/
@ -41,12 +47,50 @@ export function syncExternalCliCredentials(store: AuthProfileStore): boolean {
let mutated = false;
const now = Date.now();
// Sync from Claude CLI (~/.claude/.credentials.json)
const existingClaude = store.profiles[CLAUDE_CLI_PROFILE_ID];
const shouldSyncClaude =
!existingClaude ||
existingClaude.provider !== "anthropic" ||
!isExternalProfileFresh(existingClaude, now, "anthropic");
const claudeCreds = shouldSyncClaude
? readClaudeCliCredentialsCached({ ttlMs: EXTERNAL_CLI_SYNC_TTL_MS })
: null;
if (claudeCreds) {
const existing = store.profiles[CLAUDE_CLI_PROFILE_ID];
const existingTyped =
existing?.type === "oauth" || existing?.type === "token" ? existing : undefined;
const shouldUpdate =
!existingTyped ||
existingTyped.provider !== "anthropic" ||
(typeof existingTyped.expires === "number" && existingTyped.expires <= now) ||
(typeof claudeCreds.expires === "number" &&
typeof existingTyped.expires === "number" &&
claudeCreds.expires > existingTyped.expires);
if (shouldUpdate) {
const isSame =
claudeCreds.type === "oauth" &&
existingTyped?.type === "oauth" &&
shallowEqualOAuthCredentials(existingTyped, claudeCreds);
if (!isSame) {
store.profiles[CLAUDE_CLI_PROFILE_ID] = claudeCreds;
mutated = true;
log.info("synced claude credentials from claude cli", {
profileId: CLAUDE_CLI_PROFILE_ID,
type: claudeCreds.type,
expires: new Date(claudeCreds.expires).toISOString(),
});
}
}
}
// Sync from Qwen Code CLI
const existingQwen = store.profiles[QWEN_CLI_PROFILE_ID];
const shouldSyncQwen =
!existingQwen ||
existingQwen.provider !== "qwen-portal" ||
!isExternalProfileFresh(existingQwen, now);
!isExternalProfileFresh(existingQwen, now, "qwen-portal");
const qwenCreds = shouldSyncQwen
? readQwenCliCredentialsCached({ ttlMs: EXTERNAL_CLI_SYNC_TTL_MS })
: null;