chore: use auth storage
This commit is contained in:
84
index.ts
84
index.ts
@@ -1,7 +1,4 @@
|
|||||||
import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
|
import { AuthStorage, type AuthCredential, type ExtensionAPI } from "@mariozechner/pi-coding-agent";
|
||||||
import { readFileSync } from "node:fs";
|
|
||||||
import { homedir } from "node:os";
|
|
||||||
import { join } from "node:path";
|
|
||||||
import { statusbarConfig } from "./config";
|
import { statusbarConfig } from "./config";
|
||||||
import { contextModule, costModule, directoryModule, modelModule, thinkingModule } from "./modules/basic";
|
import { contextModule, costModule, directoryModule, modelModule, thinkingModule } from "./modules/basic";
|
||||||
import { usageModule } from "./modules/usage";
|
import { usageModule } from "./modules/usage";
|
||||||
@@ -12,7 +9,6 @@ import { zaiUsageProvider } from "./usage/zai";
|
|||||||
import type { ModuleContext, ModuleSpec, RenderedModule, StatusbarState } from "./types";
|
import type { ModuleContext, ModuleSpec, RenderedModule, StatusbarState } from "./types";
|
||||||
import type { Provider, UsageCredential, UsageProvider, UsageReport } from "./usage";
|
import type { Provider, UsageCredential, UsageProvider, UsageReport } from "./usage";
|
||||||
|
|
||||||
const AUTH_PATH = join(homedir(), ".pi", "agent", "auth.json");
|
|
||||||
const REFRESH_MS = 60_000;
|
const REFRESH_MS = 60_000;
|
||||||
const ANTHROPIC_REFRESH_MS = 15 * 60_000;
|
const ANTHROPIC_REFRESH_MS = 15 * 60_000;
|
||||||
|
|
||||||
@@ -32,36 +28,49 @@ function activeProvider(ctx: any): Provider | undefined {
|
|||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
function readPiCredential(provider: Provider): UsageCredential | undefined {
|
function credentialString(raw: unknown, key: string): string | undefined {
|
||||||
try {
|
return isRecord(raw) && typeof raw[key] === "string" ? raw[key] : undefined;
|
||||||
const auth = JSON.parse(readFileSync(AUTH_PATH, "utf8"));
|
}
|
||||||
if (!isRecord(auth) || !isRecord(auth[provider])) return undefined;
|
|
||||||
|
|
||||||
const raw = auth[provider];
|
function buildUsageCredential(raw: AuthCredential | undefined, apiKey: string): UsageCredential {
|
||||||
if (raw.type === "oauth") {
|
if (raw?.type === "oauth") {
|
||||||
return {
|
return {
|
||||||
type: "oauth",
|
type: "oauth",
|
||||||
accessToken: typeof raw.access === "string" ? raw.access : undefined,
|
accessToken: apiKey,
|
||||||
refreshToken: typeof raw.refresh === "string" ? raw.refresh : undefined,
|
refreshToken: raw.refresh,
|
||||||
expiresAt: typeof raw.expires === "number" ? raw.expires : undefined,
|
expiresAt: raw.expires,
|
||||||
accountId: typeof raw.accountId === "string" ? raw.accountId : undefined,
|
accountId: credentialString(raw, "accountId"),
|
||||||
email: typeof raw.email === "string" ? raw.email : undefined,
|
email: credentialString(raw, "email"),
|
||||||
metadata: raw,
|
metadata: raw,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (raw.type === "api_key") {
|
|
||||||
return {
|
return {
|
||||||
type: "api_key",
|
type: "api_key",
|
||||||
apiKey: typeof raw.key === "string" ? raw.key : undefined,
|
apiKey,
|
||||||
accountId: typeof raw.accountId === "string" ? raw.accountId : undefined,
|
accountId: credentialString(raw, "accountId"),
|
||||||
email: typeof raw.email === "string" ? raw.email : undefined,
|
email: credentialString(raw, "email"),
|
||||||
metadata: raw,
|
metadata: raw,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
} catch {}
|
|
||||||
|
|
||||||
return undefined;
|
async function readPiCredential(authStorage: AuthStorage, provider: Provider): Promise<UsageCredential | undefined> {
|
||||||
|
authStorage.reload();
|
||||||
|
const apiKey = await authStorage.getApiKey(provider);
|
||||||
|
if (!apiKey) return undefined;
|
||||||
|
return buildUsageCredential(authStorage.get(provider), apiKey);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function forceRefreshPiCredential(authStorage: AuthStorage, provider: Provider): Promise<UsageCredential> {
|
||||||
|
authStorage.reload();
|
||||||
|
const raw = authStorage.get(provider);
|
||||||
|
const oauthProvider = authStorage.getOAuthProviders().find(candidate => candidate.id === provider);
|
||||||
|
if (raw?.type !== "oauth" || !oauthProvider) throw new Error("login expired");
|
||||||
|
|
||||||
|
// Refresh Provider OAuth Token
|
||||||
|
const refreshed = await oauthProvider.refreshToken(raw);
|
||||||
|
authStorage.set(provider, { type: "oauth", ...refreshed });
|
||||||
|
return buildUsageCredential(authStorage.get(provider), oauthProvider.getApiKey(refreshed));
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderModule(moduleCtx: ModuleContext, spec: ModuleSpec): RenderedModule {
|
function renderModule(moduleCtx: ModuleContext, spec: ModuleSpec): RenderedModule {
|
||||||
@@ -105,6 +114,7 @@ export default function piStatusbarExtension(pi: ExtensionAPI) {
|
|||||||
let latestCtx: any;
|
let latestCtx: any;
|
||||||
let requestRender: (() => void) | undefined;
|
let requestRender: (() => void) | undefined;
|
||||||
const statusbarState: StatusbarState = {};
|
const statusbarState: StatusbarState = {};
|
||||||
|
const authStorage = AuthStorage.create();
|
||||||
|
|
||||||
function updateThinkingLevel() {
|
function updateThinkingLevel() {
|
||||||
try {
|
try {
|
||||||
@@ -154,7 +164,7 @@ export default function piStatusbarExtension(pi: ExtensionAPI) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const credential = readPiCredential(provider);
|
const credential = await readPiCredential(authStorage, provider);
|
||||||
if (!credential) {
|
if (!credential) {
|
||||||
statusbarState.report = { provider, fetchedAt: Date.now(), limits: [] };
|
statusbarState.report = { provider, fetchedAt: Date.now(), limits: [] };
|
||||||
statusbarState.error = "not logged in";
|
statusbarState.error = "not logged in";
|
||||||
@@ -163,25 +173,37 @@ export default function piStatusbarExtension(pi: ExtensionAPI) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
inFlight?.abort();
|
inFlight?.abort();
|
||||||
inFlight = new AbortController();
|
const controller = new AbortController();
|
||||||
|
inFlight = controller;
|
||||||
statusbarState.report = undefined;
|
statusbarState.report = undefined;
|
||||||
statusbarState.error = undefined;
|
statusbarState.error = undefined;
|
||||||
rerender(ctx);
|
rerender(ctx);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const report = await usageProviders[provider].fetchUsage({
|
let activeCredential = credential;
|
||||||
provider,
|
const usageCtx = {
|
||||||
credential,
|
|
||||||
baseUrl: ctx.model?.baseUrl,
|
|
||||||
signal: inFlight.signal,
|
|
||||||
}, {
|
|
||||||
fetch: globalThis.fetch.bind(globalThis),
|
fetch: globalThis.fetch.bind(globalThis),
|
||||||
logger: {
|
logger: {
|
||||||
debug: () => undefined,
|
debug: () => undefined,
|
||||||
warn: () => undefined,
|
warn: () => undefined,
|
||||||
},
|
},
|
||||||
|
};
|
||||||
|
const fetchParams = () => ({
|
||||||
|
provider,
|
||||||
|
credential: activeCredential,
|
||||||
|
baseUrl: ctx.model?.baseUrl,
|
||||||
|
signal: controller.signal,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let report: UsageReport | null;
|
||||||
|
try {
|
||||||
|
report = await usageProviders[provider].fetchUsage(fetchParams(), usageCtx);
|
||||||
|
} catch (error) {
|
||||||
|
if (!(error instanceof Error) || error.message !== "unauthorized") throw error;
|
||||||
|
activeCredential = await forceRefreshPiCredential(authStorage, provider);
|
||||||
|
report = await usageProviders[provider].fetchUsage(fetchParams(), usageCtx);
|
||||||
|
}
|
||||||
|
|
||||||
statusbarState.report = report ?? { provider, fetchedAt: Date.now(), limits: [] };
|
statusbarState.report = report ?? { provider, fetchedAt: Date.now(), limits: [] };
|
||||||
statusbarState.error = report ? undefined : "unavailable";
|
statusbarState.error = report ? undefined : "unavailable";
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
|||||||
Reference in New Issue
Block a user