opencode-codex-auth-rotation 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +37 -0
- package/README.md +170 -0
- package/assets/codex-auth.schema.json +26 -0
- package/assets/opencode-logo-ornate-dark.svg +18 -0
- package/assets/readme-hero.svg +31 -0
- package/config/README.md +103 -0
- package/config/minimal-opencode.json +12 -0
- package/config/opencode-legacy.json +571 -0
- package/config/opencode-modern.json +239 -0
- package/dist/index.d.ts +42 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +299 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/auth/auth.d.ts +43 -0
- package/dist/lib/auth/auth.d.ts.map +1 -0
- package/dist/lib/auth/auth.js +163 -0
- package/dist/lib/auth/auth.js.map +1 -0
- package/dist/lib/auth/browser.d.ts +17 -0
- package/dist/lib/auth/browser.d.ts.map +1 -0
- package/dist/lib/auth/browser.js +76 -0
- package/dist/lib/auth/browser.js.map +1 -0
- package/dist/lib/auth/server.d.ts +10 -0
- package/dist/lib/auth/server.d.ts.map +1 -0
- package/dist/lib/auth/server.js +78 -0
- package/dist/lib/auth/server.js.map +1 -0
- package/dist/lib/config.d.ts +33 -0
- package/dist/lib/config.d.ts.map +1 -0
- package/dist/lib/config.js +84 -0
- package/dist/lib/config.js.map +1 -0
- package/dist/lib/constants.d.ts +67 -0
- package/dist/lib/constants.d.ts.map +1 -0
- package/dist/lib/constants.js +67 -0
- package/dist/lib/constants.js.map +1 -0
- package/dist/lib/logger.d.ts +21 -0
- package/dist/lib/logger.d.ts.map +1 -0
- package/dist/lib/logger.js +77 -0
- package/dist/lib/logger.js.map +1 -0
- package/dist/lib/oauth-success.html +712 -0
- package/dist/lib/prompts/codex-opencode-bridge.d.ts +19 -0
- package/dist/lib/prompts/codex-opencode-bridge.d.ts.map +1 -0
- package/dist/lib/prompts/codex-opencode-bridge.js +152 -0
- package/dist/lib/prompts/codex-opencode-bridge.js.map +1 -0
- package/dist/lib/prompts/codex.d.ts +27 -0
- package/dist/lib/prompts/codex.d.ts.map +1 -0
- package/dist/lib/prompts/codex.js +241 -0
- package/dist/lib/prompts/codex.js.map +1 -0
- package/dist/lib/prompts/opencode-codex.d.ts +21 -0
- package/dist/lib/prompts/opencode-codex.d.ts.map +1 -0
- package/dist/lib/prompts/opencode-codex.js +91 -0
- package/dist/lib/prompts/opencode-codex.js.map +1 -0
- package/dist/lib/request/fetch-helpers.d.ts +73 -0
- package/dist/lib/request/fetch-helpers.d.ts.map +1 -0
- package/dist/lib/request/fetch-helpers.js +221 -0
- package/dist/lib/request/fetch-helpers.js.map +1 -0
- package/dist/lib/request/helpers/input-utils.d.ts +6 -0
- package/dist/lib/request/helpers/input-utils.d.ts.map +1 -0
- package/dist/lib/request/helpers/input-utils.js +174 -0
- package/dist/lib/request/helpers/input-utils.js.map +1 -0
- package/dist/lib/request/helpers/model-map.d.ts +28 -0
- package/dist/lib/request/helpers/model-map.d.ts.map +1 -0
- package/dist/lib/request/helpers/model-map.js +109 -0
- package/dist/lib/request/helpers/model-map.js.map +1 -0
- package/dist/lib/request/request-transformer.d.ts +93 -0
- package/dist/lib/request/request-transformer.d.ts.map +1 -0
- package/dist/lib/request/request-transformer.js +403 -0
- package/dist/lib/request/request-transformer.js.map +1 -0
- package/dist/lib/request/response-handler.d.ts +14 -0
- package/dist/lib/request/response-handler.d.ts.map +1 -0
- package/dist/lib/request/response-handler.js +88 -0
- package/dist/lib/request/response-handler.js.map +1 -0
- package/dist/lib/types.d.ts +184 -0
- package/dist/lib/types.d.ts.map +1 -0
- package/dist/lib/types.js +2 -0
- package/dist/lib/types.js.map +1 -0
- package/dist/src/plugin/accounts.d.ts +21 -0
- package/dist/src/plugin/accounts.d.ts.map +1 -0
- package/dist/src/plugin/accounts.js +53 -0
- package/dist/src/plugin/accounts.js.map +1 -0
- package/dist/src/plugin/config/schema.d.ts +38 -0
- package/dist/src/plugin/config/schema.d.ts.map +1 -0
- package/dist/src/plugin/config/schema.js +32 -0
- package/dist/src/plugin/config/schema.js.map +1 -0
- package/dist/src/plugin/storage.d.ts +15 -0
- package/dist/src/plugin/storage.d.ts.map +1 -0
- package/dist/src/plugin/storage.js +38 -0
- package/dist/src/plugin/storage.js.map +1 -0
- package/dist/src/plugin/usage.d.ts +9 -0
- package/dist/src/plugin/usage.d.ts.map +1 -0
- package/dist/src/plugin/usage.js +76 -0
- package/dist/src/plugin/usage.js.map +1 -0
- package/package.json +74 -0
- package/scripts/check-usage.ts +92 -0
- package/scripts/install-opencode-codex-auth.js +430 -0
- package/scripts/test-all-models.sh +259 -0
- package/scripts/validate-model-map.sh +97 -0
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
import { generatePKCE } from "@openauthjs/openauth/pkce";
|
|
2
|
+
import { randomBytes } from "node:crypto";
|
|
3
|
+
// OAuth constants (from openai/codex)
|
|
4
|
+
export const CLIENT_ID = "app_EMoamEEZ73f0CkXaXp7hrann";
|
|
5
|
+
export const AUTHORIZE_URL = "https://auth.openai.com/oauth/authorize";
|
|
6
|
+
export const TOKEN_URL = "https://auth.openai.com/oauth/token";
|
|
7
|
+
export const REDIRECT_URI = "http://localhost:1455/auth/callback";
|
|
8
|
+
export const SCOPE = "openid profile email offline_access";
|
|
9
|
+
/**
|
|
10
|
+
* Generate a random state value for OAuth flow
|
|
11
|
+
* @returns Random hex string
|
|
12
|
+
*/
|
|
13
|
+
export function createState() {
|
|
14
|
+
return randomBytes(16).toString("hex");
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Parse authorization code and state from user input
|
|
18
|
+
* @param input - User input (URL, code#state, or just code)
|
|
19
|
+
* @returns Parsed authorization data
|
|
20
|
+
*/
|
|
21
|
+
export function parseAuthorizationInput(input) {
|
|
22
|
+
const value = (input || "").trim();
|
|
23
|
+
if (!value)
|
|
24
|
+
return {};
|
|
25
|
+
try {
|
|
26
|
+
const url = new URL(value);
|
|
27
|
+
return {
|
|
28
|
+
code: url.searchParams.get("code") ?? undefined,
|
|
29
|
+
state: url.searchParams.get("state") ?? undefined,
|
|
30
|
+
};
|
|
31
|
+
}
|
|
32
|
+
catch { }
|
|
33
|
+
if (value.includes("#")) {
|
|
34
|
+
const [code, state] = value.split("#", 2);
|
|
35
|
+
return { code, state };
|
|
36
|
+
}
|
|
37
|
+
if (value.includes("code=")) {
|
|
38
|
+
const params = new URLSearchParams(value);
|
|
39
|
+
return {
|
|
40
|
+
code: params.get("code") ?? undefined,
|
|
41
|
+
state: params.get("state") ?? undefined,
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
return { code: value };
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Exchange authorization code for access and refresh tokens
|
|
48
|
+
* @param code - Authorization code from OAuth flow
|
|
49
|
+
* @param verifier - PKCE verifier
|
|
50
|
+
* @param redirectUri - OAuth redirect URI
|
|
51
|
+
* @returns Token result
|
|
52
|
+
*/
|
|
53
|
+
export async function exchangeAuthorizationCode(code, verifier, redirectUri = REDIRECT_URI) {
|
|
54
|
+
const res = await fetch(TOKEN_URL, {
|
|
55
|
+
method: "POST",
|
|
56
|
+
headers: { "Content-Type": "application/x-www-form-urlencoded" },
|
|
57
|
+
body: new URLSearchParams({
|
|
58
|
+
grant_type: "authorization_code",
|
|
59
|
+
client_id: CLIENT_ID,
|
|
60
|
+
code,
|
|
61
|
+
code_verifier: verifier,
|
|
62
|
+
redirect_uri: redirectUri,
|
|
63
|
+
}),
|
|
64
|
+
});
|
|
65
|
+
if (!res.ok) {
|
|
66
|
+
const text = await res.text().catch(() => "");
|
|
67
|
+
console.error("[openai-codex-plugin] code->token failed:", res.status, text);
|
|
68
|
+
return { type: "failed" };
|
|
69
|
+
}
|
|
70
|
+
const json = (await res.json());
|
|
71
|
+
if (!json?.access_token ||
|
|
72
|
+
!json?.refresh_token ||
|
|
73
|
+
typeof json?.expires_in !== "number") {
|
|
74
|
+
console.error("[openai-codex-plugin] token response missing fields:", json);
|
|
75
|
+
return { type: "failed" };
|
|
76
|
+
}
|
|
77
|
+
return {
|
|
78
|
+
type: "success",
|
|
79
|
+
access: json.access_token,
|
|
80
|
+
refresh: json.refresh_token,
|
|
81
|
+
expires: Date.now() + json.expires_in * 1000,
|
|
82
|
+
};
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Decode a JWT token to extract payload
|
|
86
|
+
* @param token - JWT token to decode
|
|
87
|
+
* @returns Decoded payload or null if invalid
|
|
88
|
+
*/
|
|
89
|
+
export function decodeJWT(token) {
|
|
90
|
+
try {
|
|
91
|
+
const parts = token.split(".");
|
|
92
|
+
if (parts.length !== 3)
|
|
93
|
+
return null;
|
|
94
|
+
const payload = parts[1];
|
|
95
|
+
const decoded = Buffer.from(payload, "base64").toString("utf-8");
|
|
96
|
+
return JSON.parse(decoded);
|
|
97
|
+
}
|
|
98
|
+
catch {
|
|
99
|
+
return null;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Refresh access token using refresh token
|
|
104
|
+
* @param refreshToken - Refresh token
|
|
105
|
+
* @returns Token result
|
|
106
|
+
*/
|
|
107
|
+
export async function refreshAccessToken(refreshToken) {
|
|
108
|
+
try {
|
|
109
|
+
const response = await fetch(TOKEN_URL, {
|
|
110
|
+
method: "POST",
|
|
111
|
+
headers: { "Content-Type": "application/x-www-form-urlencoded" },
|
|
112
|
+
body: new URLSearchParams({
|
|
113
|
+
grant_type: "refresh_token",
|
|
114
|
+
refresh_token: refreshToken,
|
|
115
|
+
client_id: CLIENT_ID,
|
|
116
|
+
}),
|
|
117
|
+
});
|
|
118
|
+
if (!response.ok) {
|
|
119
|
+
const text = await response.text().catch(() => "");
|
|
120
|
+
console.error("[openai-codex-plugin] Token refresh failed:", response.status, text);
|
|
121
|
+
return { type: "failed" };
|
|
122
|
+
}
|
|
123
|
+
const json = (await response.json());
|
|
124
|
+
if (!json?.access_token ||
|
|
125
|
+
!json?.refresh_token ||
|
|
126
|
+
typeof json?.expires_in !== "number") {
|
|
127
|
+
console.error("[openai-codex-plugin] Token refresh response missing fields:", json);
|
|
128
|
+
return { type: "failed" };
|
|
129
|
+
}
|
|
130
|
+
return {
|
|
131
|
+
type: "success",
|
|
132
|
+
access: json.access_token,
|
|
133
|
+
refresh: json.refresh_token,
|
|
134
|
+
expires: Date.now() + json.expires_in * 1000,
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
catch (error) {
|
|
138
|
+
const err = error;
|
|
139
|
+
console.error("[openai-codex-plugin] Token refresh error:", err);
|
|
140
|
+
return { type: "failed" };
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Create OAuth authorization flow
|
|
145
|
+
* @returns Authorization flow details
|
|
146
|
+
*/
|
|
147
|
+
export async function createAuthorizationFlow() {
|
|
148
|
+
const pkce = (await generatePKCE());
|
|
149
|
+
const state = createState();
|
|
150
|
+
const url = new URL(AUTHORIZE_URL);
|
|
151
|
+
url.searchParams.set("response_type", "code");
|
|
152
|
+
url.searchParams.set("client_id", CLIENT_ID);
|
|
153
|
+
url.searchParams.set("redirect_uri", REDIRECT_URI);
|
|
154
|
+
url.searchParams.set("scope", SCOPE);
|
|
155
|
+
url.searchParams.set("code_challenge", pkce.challenge);
|
|
156
|
+
url.searchParams.set("code_challenge_method", "S256");
|
|
157
|
+
url.searchParams.set("state", state);
|
|
158
|
+
url.searchParams.set("id_token_add_organizations", "true");
|
|
159
|
+
url.searchParams.set("codex_cli_simplified_flow", "true");
|
|
160
|
+
url.searchParams.set("originator", "codex_cli_rs");
|
|
161
|
+
return { pkce, state, url: url.toString() };
|
|
162
|
+
}
|
|
163
|
+
//# sourceMappingURL=auth.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../lib/auth/auth.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Browser utilities for OAuth flow
|
|
3
|
+
* Handles platform-specific browser opening
|
|
4
|
+
*/
|
|
5
|
+
/**
|
|
6
|
+
* Gets the platform-specific command to open a URL in the default browser
|
|
7
|
+
* @returns Browser opener command for the current platform
|
|
8
|
+
*/
|
|
9
|
+
export declare function getBrowserOpener(): string;
|
|
10
|
+
/**
|
|
11
|
+
* Opens a URL in the default browser
|
|
12
|
+
* Silently fails if browser cannot be opened (user can copy URL manually)
|
|
13
|
+
* @param url - URL to open
|
|
14
|
+
* @returns True if a browser launch was attempted
|
|
15
|
+
*/
|
|
16
|
+
export declare function openBrowserUrl(url: string): boolean;
|
|
17
|
+
//# sourceMappingURL=browser.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"browser.d.ts","sourceRoot":"","sources":["../../../lib/auth/browser.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAOH;;;GAGG;AACH,wBAAgB,gBAAgB,IAAI,MAAM,CAKzC;AAkCD;;;;;GAKG;AACH,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAgBnD"}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Browser utilities for OAuth flow
|
|
3
|
+
* Handles platform-specific browser opening
|
|
4
|
+
*/
|
|
5
|
+
import { spawn } from "node:child_process";
|
|
6
|
+
import fs from "node:fs";
|
|
7
|
+
import path from "node:path";
|
|
8
|
+
import { PLATFORM_OPENERS } from "../constants.js";
|
|
9
|
+
/**
|
|
10
|
+
* Gets the platform-specific command to open a URL in the default browser
|
|
11
|
+
* @returns Browser opener command for the current platform
|
|
12
|
+
*/
|
|
13
|
+
export function getBrowserOpener() {
|
|
14
|
+
const platform = process.platform;
|
|
15
|
+
if (platform === "darwin")
|
|
16
|
+
return PLATFORM_OPENERS.darwin;
|
|
17
|
+
if (platform === "win32")
|
|
18
|
+
return PLATFORM_OPENERS.win32;
|
|
19
|
+
return PLATFORM_OPENERS.linux;
|
|
20
|
+
}
|
|
21
|
+
function commandExists(command) {
|
|
22
|
+
if (!command)
|
|
23
|
+
return false;
|
|
24
|
+
// "start" is a shell builtin on Windows; rely on shell execution
|
|
25
|
+
if (process.platform === "win32" && command.toLowerCase() === "start") {
|
|
26
|
+
return true;
|
|
27
|
+
}
|
|
28
|
+
const pathValue = process.env.PATH || "";
|
|
29
|
+
const entries = pathValue.split(path.delimiter).filter(Boolean);
|
|
30
|
+
if (entries.length === 0)
|
|
31
|
+
return false;
|
|
32
|
+
if (process.platform === "win32") {
|
|
33
|
+
const pathext = (process.env.PATHEXT || ".EXE;.CMD;.BAT;.COM")
|
|
34
|
+
.split(";")
|
|
35
|
+
.filter(Boolean);
|
|
36
|
+
for (const entry of entries) {
|
|
37
|
+
for (const ext of pathext) {
|
|
38
|
+
const candidate = path.join(entry, `${command}${ext}`);
|
|
39
|
+
if (fs.existsSync(candidate))
|
|
40
|
+
return true;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
return false;
|
|
44
|
+
}
|
|
45
|
+
for (const entry of entries) {
|
|
46
|
+
const candidate = path.join(entry, command);
|
|
47
|
+
if (fs.existsSync(candidate))
|
|
48
|
+
return true;
|
|
49
|
+
}
|
|
50
|
+
return false;
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* Opens a URL in the default browser
|
|
54
|
+
* Silently fails if browser cannot be opened (user can copy URL manually)
|
|
55
|
+
* @param url - URL to open
|
|
56
|
+
* @returns True if a browser launch was attempted
|
|
57
|
+
*/
|
|
58
|
+
export function openBrowserUrl(url) {
|
|
59
|
+
try {
|
|
60
|
+
const opener = getBrowserOpener();
|
|
61
|
+
if (!commandExists(opener)) {
|
|
62
|
+
return false;
|
|
63
|
+
}
|
|
64
|
+
const child = spawn(opener, [url], {
|
|
65
|
+
stdio: "ignore",
|
|
66
|
+
shell: process.platform === "win32",
|
|
67
|
+
});
|
|
68
|
+
child.on("error", () => { });
|
|
69
|
+
return true;
|
|
70
|
+
}
|
|
71
|
+
catch (error) {
|
|
72
|
+
// Silently fail - user can manually open the URL from instructions
|
|
73
|
+
return false;
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
//# sourceMappingURL=browser.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"browser.js","sourceRoot":"","sources":["../../../lib/auth/browser.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAC3C,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAEnD;;;GAGG;AACH,MAAM,UAAU,gBAAgB;IAC/B,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,CAAC;IAClC,IAAI,QAAQ,KAAK,QAAQ;QAAE,OAAO,gBAAgB,CAAC,MAAM,CAAC;IAC1D,IAAI,QAAQ,KAAK,OAAO;QAAE,OAAO,gBAAgB,CAAC,KAAK,CAAC;IACxD,OAAO,gBAAgB,CAAC,KAAK,CAAC;AAC/B,CAAC;AAED,SAAS,aAAa,CAAC,OAAe;IACrC,IAAI,CAAC,OAAO;QAAE,OAAO,KAAK,CAAC;IAE3B,iEAAiE;IACjE,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,IAAI,OAAO,CAAC,WAAW,EAAE,KAAK,OAAO,EAAE,CAAC;QACvE,OAAO,IAAI,CAAC;IACb,CAAC;IAED,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,IAAI,EAAE,CAAC;IACzC,MAAM,OAAO,GAAG,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IAChE,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,KAAK,CAAC;IAEvC,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;QAClC,MAAM,OAAO,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,IAAI,qBAAqB,CAAC;aAC5D,KAAK,CAAC,GAAG,CAAC;aACV,MAAM,CAAC,OAAO,CAAC,CAAC;QAClB,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;YAC7B,KAAK,MAAM,GAAG,IAAI,OAAO,EAAE,CAAC;gBAC3B,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,GAAG,OAAO,GAAG,GAAG,EAAE,CAAC,CAAC;gBACvD,IAAI,EAAE,CAAC,UAAU,CAAC,SAAS,CAAC;oBAAE,OAAO,IAAI,CAAC;YAC3C,CAAC;QACF,CAAC;QACD,OAAO,KAAK,CAAC;IACd,CAAC;IAED,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;QAC7B,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;QAC5C,IAAI,EAAE,CAAC,UAAU,CAAC,SAAS,CAAC;YAAE,OAAO,IAAI,CAAC;IAC3C,CAAC;IACD,OAAO,KAAK,CAAC;AACd,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,cAAc,CAAC,GAAW;IACzC,IAAI,CAAC;QACJ,MAAM,MAAM,GAAG,gBAAgB,EAAE,CAAC;QAClC,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,EAAE,CAAC;YAC5B,OAAO,KAAK,CAAC;QACd,CAAC;QACD,MAAM,KAAK,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,EAAE;YAClC,KAAK,EAAE,QAAQ;YACf,KAAK,EAAE,OAAO,CAAC,QAAQ,KAAK,OAAO;SACnC,CAAC,CAAC;QACH,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;QAC5B,OAAO,IAAI,CAAC;IACb,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QAChB,mEAAmE;QACnE,OAAO,KAAK,CAAC;IACd,CAAC;AACF,CAAC"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import type { OAuthServerInfo } from "../types.js";
|
|
2
|
+
/**
|
|
3
|
+
* Start a small local HTTP server that waits for /auth/callback and returns the code
|
|
4
|
+
* @param options - OAuth state for validation
|
|
5
|
+
* @returns Promise that resolves to server info
|
|
6
|
+
*/
|
|
7
|
+
export declare function startLocalOAuthServer({ state }: {
|
|
8
|
+
state: string;
|
|
9
|
+
}): Promise<OAuthServerInfo>;
|
|
10
|
+
//# sourceMappingURL=server.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../../lib/auth/server.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAMnD;;;;GAIG;AACH,wBAAgB,qBAAqB,CAAC,EAAE,KAAK,EAAE,EAAE;IAAE,KAAK,EAAE,MAAM,CAAA;CAAE,GAAG,OAAO,CAAC,eAAe,CAAC,CAkE5F"}
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import http from "node:http";
|
|
2
|
+
import fs from "node:fs";
|
|
3
|
+
import path from "node:path";
|
|
4
|
+
import { fileURLToPath } from "node:url";
|
|
5
|
+
// Resolve path to oauth-success.html (one level up from auth/ subfolder)
|
|
6
|
+
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
7
|
+
const successHtml = fs.readFileSync(path.join(__dirname, "..", "oauth-success.html"), "utf-8");
|
|
8
|
+
/**
|
|
9
|
+
* Start a small local HTTP server that waits for /auth/callback and returns the code
|
|
10
|
+
* @param options - OAuth state for validation
|
|
11
|
+
* @returns Promise that resolves to server info
|
|
12
|
+
*/
|
|
13
|
+
export function startLocalOAuthServer({ state }) {
|
|
14
|
+
const server = http.createServer((req, res) => {
|
|
15
|
+
try {
|
|
16
|
+
const url = new URL(req.url || "", "http://localhost");
|
|
17
|
+
if (url.pathname !== "/auth/callback") {
|
|
18
|
+
res.statusCode = 404;
|
|
19
|
+
res.end("Not found");
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
if (url.searchParams.get("state") !== state) {
|
|
23
|
+
res.statusCode = 400;
|
|
24
|
+
res.end("State mismatch");
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
const code = url.searchParams.get("code");
|
|
28
|
+
if (!code) {
|
|
29
|
+
res.statusCode = 400;
|
|
30
|
+
res.end("Missing authorization code");
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
res.statusCode = 200;
|
|
34
|
+
res.setHeader("Content-Type", "text/html; charset=utf-8");
|
|
35
|
+
res.end(successHtml);
|
|
36
|
+
server._lastCode = code;
|
|
37
|
+
}
|
|
38
|
+
catch {
|
|
39
|
+
res.statusCode = 500;
|
|
40
|
+
res.end("Internal error");
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
return new Promise((resolve) => {
|
|
44
|
+
server
|
|
45
|
+
.listen(1455, "127.0.0.1", () => {
|
|
46
|
+
resolve({
|
|
47
|
+
port: 1455,
|
|
48
|
+
ready: true,
|
|
49
|
+
close: () => server.close(),
|
|
50
|
+
waitForCode: async () => {
|
|
51
|
+
const poll = () => new Promise((r) => setTimeout(r, 100));
|
|
52
|
+
for (let i = 0; i < 600; i++) {
|
|
53
|
+
const lastCode = server._lastCode;
|
|
54
|
+
if (lastCode)
|
|
55
|
+
return { code: lastCode };
|
|
56
|
+
await poll();
|
|
57
|
+
}
|
|
58
|
+
return null;
|
|
59
|
+
},
|
|
60
|
+
});
|
|
61
|
+
})
|
|
62
|
+
.on("error", (err) => {
|
|
63
|
+
console.error("[openai-codex-plugin] Failed to bind http://127.0.0.1:1455 (", err?.code, ") Falling back to manual paste.");
|
|
64
|
+
resolve({
|
|
65
|
+
port: 1455,
|
|
66
|
+
ready: false,
|
|
67
|
+
close: () => {
|
|
68
|
+
try {
|
|
69
|
+
server.close();
|
|
70
|
+
}
|
|
71
|
+
catch { }
|
|
72
|
+
},
|
|
73
|
+
waitForCode: async () => null,
|
|
74
|
+
});
|
|
75
|
+
});
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
//# sourceMappingURL=server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server.js","sourceRoot":"","sources":["../../../lib/auth/server.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import type { PluginConfig, AccountSelectionStrategy } from "./types.js";
|
|
2
|
+
/**
|
|
3
|
+
* Load plugin configuration from ~/.opencode/openai-codex-auth-config.json
|
|
4
|
+
* Falls back to defaults if file doesn't exist or is invalid
|
|
5
|
+
*
|
|
6
|
+
* @returns Plugin configuration
|
|
7
|
+
*/
|
|
8
|
+
export declare function loadPluginConfig(): PluginConfig;
|
|
9
|
+
/**
|
|
10
|
+
* Get the effective CODEX_MODE setting
|
|
11
|
+
* Priority: environment variable > config file > default (true)
|
|
12
|
+
*
|
|
13
|
+
* @param pluginConfig - Plugin configuration from file
|
|
14
|
+
* @returns True if CODEX_MODE should be enabled
|
|
15
|
+
*/
|
|
16
|
+
export declare function getCodexMode(pluginConfig: PluginConfig): boolean;
|
|
17
|
+
/**
|
|
18
|
+
* Get the effective account selection strategy
|
|
19
|
+
* Priority: environment variable > config file > default ("sticky")
|
|
20
|
+
*
|
|
21
|
+
* @param pluginConfig - Plugin configuration from file
|
|
22
|
+
* @returns The account selection strategy to use
|
|
23
|
+
*/
|
|
24
|
+
export declare function getStrategy(pluginConfig: PluginConfig): AccountSelectionStrategy;
|
|
25
|
+
/**
|
|
26
|
+
* Get the effective PID-based offset setting
|
|
27
|
+
* Priority: environment variable > config file > default (false)
|
|
28
|
+
*
|
|
29
|
+
* @param pluginConfig - Plugin configuration from file
|
|
30
|
+
* @returns True if PID-based offset should be enabled
|
|
31
|
+
*/
|
|
32
|
+
export declare function getPidOffset(pluginConfig: PluginConfig): boolean;
|
|
33
|
+
//# sourceMappingURL=config.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../lib/config.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,YAAY,EAAE,wBAAwB,EAAE,MAAM,YAAY,CAAC;AAczE;;;;;GAKG;AACH,wBAAgB,gBAAgB,IAAI,YAAY,CAqB/C;AAED;;;;;;GAMG;AACH,wBAAgB,YAAY,CAAC,YAAY,EAAE,YAAY,GAAG,OAAO,CAQhE;AAED;;;;;;GAMG;AACH,wBAAgB,WAAW,CAAC,YAAY,EAAE,YAAY,GAAG,wBAAwB,CAShF;AAED;;;;;;GAMG;AACH,wBAAgB,YAAY,CAAC,YAAY,EAAE,YAAY,GAAG,OAAO,CAQhE"}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { readFileSync, existsSync } from "node:fs";
|
|
2
|
+
import { join } from "node:path";
|
|
3
|
+
import { homedir } from "node:os";
|
|
4
|
+
const CONFIG_PATH = join(homedir(), ".opencode", "openai-codex-auth-config.json");
|
|
5
|
+
/**
|
|
6
|
+
* Default plugin configuration
|
|
7
|
+
* CODEX_MODE is enabled by default for better Codex CLI parity
|
|
8
|
+
*/
|
|
9
|
+
const DEFAULT_CONFIG = {
|
|
10
|
+
codexMode: true,
|
|
11
|
+
strategy: "sticky",
|
|
12
|
+
pidOffset: false,
|
|
13
|
+
};
|
|
14
|
+
/**
|
|
15
|
+
* Load plugin configuration from ~/.opencode/openai-codex-auth-config.json
|
|
16
|
+
* Falls back to defaults if file doesn't exist or is invalid
|
|
17
|
+
*
|
|
18
|
+
* @returns Plugin configuration
|
|
19
|
+
*/
|
|
20
|
+
export function loadPluginConfig() {
|
|
21
|
+
try {
|
|
22
|
+
if (!existsSync(CONFIG_PATH)) {
|
|
23
|
+
return DEFAULT_CONFIG;
|
|
24
|
+
}
|
|
25
|
+
const fileContent = readFileSync(CONFIG_PATH, "utf-8");
|
|
26
|
+
const userConfig = JSON.parse(fileContent);
|
|
27
|
+
// Merge with defaults
|
|
28
|
+
return {
|
|
29
|
+
...DEFAULT_CONFIG,
|
|
30
|
+
...userConfig,
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
catch (error) {
|
|
34
|
+
console.warn(`[openai-codex-plugin] Failed to load config from ${CONFIG_PATH}:`, error.message);
|
|
35
|
+
return DEFAULT_CONFIG;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* Get the effective CODEX_MODE setting
|
|
40
|
+
* Priority: environment variable > config file > default (true)
|
|
41
|
+
*
|
|
42
|
+
* @param pluginConfig - Plugin configuration from file
|
|
43
|
+
* @returns True if CODEX_MODE should be enabled
|
|
44
|
+
*/
|
|
45
|
+
export function getCodexMode(pluginConfig) {
|
|
46
|
+
// Environment variable takes precedence
|
|
47
|
+
if (process.env.CODEX_MODE !== undefined) {
|
|
48
|
+
return process.env.CODEX_MODE === "1";
|
|
49
|
+
}
|
|
50
|
+
// Use config setting (defaults to true)
|
|
51
|
+
return pluginConfig.codexMode ?? true;
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Get the effective account selection strategy
|
|
55
|
+
* Priority: environment variable > config file > default ("sticky")
|
|
56
|
+
*
|
|
57
|
+
* @param pluginConfig - Plugin configuration from file
|
|
58
|
+
* @returns The account selection strategy to use
|
|
59
|
+
*/
|
|
60
|
+
export function getStrategy(pluginConfig) {
|
|
61
|
+
// Environment variable takes precedence
|
|
62
|
+
const envStrategy = process.env.CODEX_ACCOUNT_STRATEGY;
|
|
63
|
+
if (envStrategy === "sticky" || envStrategy === "round-robin") {
|
|
64
|
+
return envStrategy;
|
|
65
|
+
}
|
|
66
|
+
// Use config setting (defaults to "sticky")
|
|
67
|
+
return pluginConfig.strategy ?? "sticky";
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Get the effective PID-based offset setting
|
|
71
|
+
* Priority: environment variable > config file > default (false)
|
|
72
|
+
*
|
|
73
|
+
* @param pluginConfig - Plugin configuration from file
|
|
74
|
+
* @returns True if PID-based offset should be enabled
|
|
75
|
+
*/
|
|
76
|
+
export function getPidOffset(pluginConfig) {
|
|
77
|
+
// Environment variable takes precedence
|
|
78
|
+
if (process.env.CODEX_PID_OFFSET !== undefined) {
|
|
79
|
+
return process.env.CODEX_PID_OFFSET === "1";
|
|
80
|
+
}
|
|
81
|
+
// Use config setting (defaults to false)
|
|
82
|
+
return pluginConfig.pidOffset ?? false;
|
|
83
|
+
}
|
|
84
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../../lib/config.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACnD,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAGlC,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,WAAW,EAAE,+BAA+B,CAAC,CAAC;AAElF;;;GAGG;AACH,MAAM,cAAc,GAAiB;IACpC,SAAS,EAAE,IAAI;IACf,QAAQ,EAAE,QAAQ;IAClB,SAAS,EAAE,KAAK;CAChB,CAAC;AAEF;;;;;GAKG;AACH,MAAM,UAAU,gBAAgB;IAC/B,IAAI,CAAC;QACJ,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;YAC9B,OAAO,cAAc,CAAC;QACvB,CAAC;QAED,MAAM,WAAW,GAAG,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;QACvD,MAAM,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,CAA0B,CAAC;QAEpE,sBAAsB;QACtB,OAAO;YACN,GAAG,cAAc;YACjB,GAAG,UAAU;SACb,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QAChB,OAAO,CAAC,IAAI,CACX,oDAAoD,WAAW,GAAG,EACjE,KAAe,CAAC,OAAO,CACxB,CAAC;QACF,OAAO,cAAc,CAAC;IACvB,CAAC;AACF,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,YAAY,CAAC,YAA0B;IACtD,wCAAwC;IACxC,IAAI,OAAO,CAAC,GAAG,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;QAC1C,OAAO,OAAO,CAAC,GAAG,CAAC,UAAU,KAAK,GAAG,CAAC;IACvC,CAAC;IAED,wCAAwC;IACxC,OAAO,YAAY,CAAC,SAAS,IAAI,IAAI,CAAC;AACvC,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,WAAW,CAAC,YAA0B;IACrD,wCAAwC;IACxC,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC;IACvD,IAAI,WAAW,KAAK,QAAQ,IAAI,WAAW,KAAK,aAAa,EAAE,CAAC;QAC/D,OAAO,WAAW,CAAC;IACpB,CAAC;IAED,4CAA4C;IAC5C,OAAO,YAAY,CAAC,QAAQ,IAAI,QAAQ,CAAC;AAC1C,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,YAAY,CAAC,YAA0B;IACtD,wCAAwC;IACxC,IAAI,OAAO,CAAC,GAAG,CAAC,gBAAgB,KAAK,SAAS,EAAE,CAAC;QAChD,OAAO,OAAO,CAAC,GAAG,CAAC,gBAAgB,KAAK,GAAG,CAAC;IAC7C,CAAC;IAED,yCAAyC;IACzC,OAAO,YAAY,CAAC,SAAS,IAAI,KAAK,CAAC;AACxC,CAAC"}
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Constants used throughout the plugin
|
|
3
|
+
* Centralized for easy maintenance and configuration
|
|
4
|
+
*/
|
|
5
|
+
/** Plugin identifier for logging and error messages */
|
|
6
|
+
export declare const PLUGIN_NAME = "openai-codex-plugin";
|
|
7
|
+
/** Base URL for ChatGPT backend API */
|
|
8
|
+
export declare const CODEX_BASE_URL = "https://chatgpt.com/backend-api";
|
|
9
|
+
/** Dummy API key used for OpenAI SDK (actual auth via OAuth) */
|
|
10
|
+
export declare const DUMMY_API_KEY = "chatgpt-oauth";
|
|
11
|
+
/** Provider ID for opencode configuration */
|
|
12
|
+
export declare const PROVIDER_ID = "openai";
|
|
13
|
+
/** HTTP Status Codes */
|
|
14
|
+
export declare const HTTP_STATUS: {
|
|
15
|
+
readonly OK: 200;
|
|
16
|
+
readonly UNAUTHORIZED: 401;
|
|
17
|
+
readonly NOT_FOUND: 404;
|
|
18
|
+
readonly TOO_MANY_REQUESTS: 429;
|
|
19
|
+
};
|
|
20
|
+
/** OpenAI-specific headers */
|
|
21
|
+
export declare const OPENAI_HEADERS: {
|
|
22
|
+
readonly BETA: "OpenAI-Beta";
|
|
23
|
+
readonly ACCOUNT_ID: "chatgpt-account-id";
|
|
24
|
+
readonly ORIGINATOR: "originator";
|
|
25
|
+
readonly SESSION_ID: "session_id";
|
|
26
|
+
readonly CONVERSATION_ID: "conversation_id";
|
|
27
|
+
};
|
|
28
|
+
/** OpenAI-specific header values */
|
|
29
|
+
export declare const OPENAI_HEADER_VALUES: {
|
|
30
|
+
readonly BETA_RESPONSES: "responses=experimental";
|
|
31
|
+
readonly ORIGINATOR_CODEX: "codex_cli_rs";
|
|
32
|
+
};
|
|
33
|
+
/** URL path segments */
|
|
34
|
+
export declare const URL_PATHS: {
|
|
35
|
+
readonly RESPONSES: "/responses";
|
|
36
|
+
readonly CODEX_RESPONSES: "/codex/responses";
|
|
37
|
+
};
|
|
38
|
+
/** JWT claim path for ChatGPT account ID */
|
|
39
|
+
export declare const JWT_CLAIM_PATH: "https://api.openai.com/auth";
|
|
40
|
+
/** Error messages */
|
|
41
|
+
export declare const ERROR_MESSAGES: {
|
|
42
|
+
readonly NO_ACCOUNT_ID: "Failed to extract accountId from token";
|
|
43
|
+
readonly TOKEN_REFRESH_FAILED: "Failed to refresh token, authentication required";
|
|
44
|
+
readonly REQUEST_PARSE_ERROR: "Error parsing request";
|
|
45
|
+
};
|
|
46
|
+
/** Log stages for request logging */
|
|
47
|
+
export declare const LOG_STAGES: {
|
|
48
|
+
readonly BEFORE_TRANSFORM: "before-transform";
|
|
49
|
+
readonly AFTER_TRANSFORM: "after-transform";
|
|
50
|
+
readonly RESPONSE: "response";
|
|
51
|
+
readonly ERROR_RESPONSE: "error-response";
|
|
52
|
+
};
|
|
53
|
+
/** Platform-specific browser opener commands */
|
|
54
|
+
export declare const PLATFORM_OPENERS: {
|
|
55
|
+
readonly darwin: "open";
|
|
56
|
+
readonly win32: "start";
|
|
57
|
+
readonly linux: "xdg-open";
|
|
58
|
+
};
|
|
59
|
+
/** OAuth authorization labels */
|
|
60
|
+
export declare const AUTH_LABELS: {
|
|
61
|
+
readonly OAUTH: "ChatGPT Plus/Pro (Codex Subscription)";
|
|
62
|
+
readonly OAUTH_MANUAL: "ChatGPT Plus/Pro (Manual URL Paste)";
|
|
63
|
+
readonly API_KEY: "Manually enter API Key";
|
|
64
|
+
readonly INSTRUCTIONS: "A browser window should open. If it doesn't, copy the URL and open it manually.";
|
|
65
|
+
readonly INSTRUCTIONS_MANUAL: "After logging in, copy the full redirect URL and paste it here.";
|
|
66
|
+
};
|
|
67
|
+
//# sourceMappingURL=constants.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../../lib/constants.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,uDAAuD;AACvD,eAAO,MAAM,WAAW,wBAAwB,CAAC;AAEjD,uCAAuC;AACvC,eAAO,MAAM,cAAc,oCAAoC,CAAC;AAEhE,gEAAgE;AAChE,eAAO,MAAM,aAAa,kBAAkB,CAAC;AAE7C,6CAA6C;AAC7C,eAAO,MAAM,WAAW,WAAW,CAAC;AAEpC,wBAAwB;AACxB,eAAO,MAAM,WAAW;;;;;CAKd,CAAC;AAEX,8BAA8B;AAC9B,eAAO,MAAM,cAAc;;;;;;CAMjB,CAAC;AAEX,oCAAoC;AACpC,eAAO,MAAM,oBAAoB;;;CAGvB,CAAC;AAEX,wBAAwB;AACxB,eAAO,MAAM,SAAS;;;CAGZ,CAAC;AAEX,4CAA4C;AAC5C,eAAO,MAAM,cAAc,EAAG,6BAAsC,CAAC;AAErE,qBAAqB;AACrB,eAAO,MAAM,cAAc;;;;CAIjB,CAAC;AAEX,qCAAqC;AACrC,eAAO,MAAM,UAAU;;;;;CAKb,CAAC;AAEX,gDAAgD;AAChD,eAAO,MAAM,gBAAgB;;;;CAInB,CAAC;AAEX,iCAAiC;AACjC,eAAO,MAAM,WAAW;;;;;;CAQd,CAAC"}
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Constants used throughout the plugin
|
|
3
|
+
* Centralized for easy maintenance and configuration
|
|
4
|
+
*/
|
|
5
|
+
/** Plugin identifier for logging and error messages */
|
|
6
|
+
export const PLUGIN_NAME = "openai-codex-plugin";
|
|
7
|
+
/** Base URL for ChatGPT backend API */
|
|
8
|
+
export const CODEX_BASE_URL = "https://chatgpt.com/backend-api";
|
|
9
|
+
/** Dummy API key used for OpenAI SDK (actual auth via OAuth) */
|
|
10
|
+
export const DUMMY_API_KEY = "chatgpt-oauth";
|
|
11
|
+
/** Provider ID for opencode configuration */
|
|
12
|
+
export const PROVIDER_ID = "openai";
|
|
13
|
+
/** HTTP Status Codes */
|
|
14
|
+
export const HTTP_STATUS = {
|
|
15
|
+
OK: 200,
|
|
16
|
+
UNAUTHORIZED: 401,
|
|
17
|
+
NOT_FOUND: 404,
|
|
18
|
+
TOO_MANY_REQUESTS: 429,
|
|
19
|
+
};
|
|
20
|
+
/** OpenAI-specific headers */
|
|
21
|
+
export const OPENAI_HEADERS = {
|
|
22
|
+
BETA: "OpenAI-Beta",
|
|
23
|
+
ACCOUNT_ID: "chatgpt-account-id",
|
|
24
|
+
ORIGINATOR: "originator",
|
|
25
|
+
SESSION_ID: "session_id",
|
|
26
|
+
CONVERSATION_ID: "conversation_id",
|
|
27
|
+
};
|
|
28
|
+
/** OpenAI-specific header values */
|
|
29
|
+
export const OPENAI_HEADER_VALUES = {
|
|
30
|
+
BETA_RESPONSES: "responses=experimental",
|
|
31
|
+
ORIGINATOR_CODEX: "codex_cli_rs",
|
|
32
|
+
};
|
|
33
|
+
/** URL path segments */
|
|
34
|
+
export const URL_PATHS = {
|
|
35
|
+
RESPONSES: "/responses",
|
|
36
|
+
CODEX_RESPONSES: "/codex/responses",
|
|
37
|
+
};
|
|
38
|
+
/** JWT claim path for ChatGPT account ID */
|
|
39
|
+
export const JWT_CLAIM_PATH = "https://api.openai.com/auth";
|
|
40
|
+
/** Error messages */
|
|
41
|
+
export const ERROR_MESSAGES = {
|
|
42
|
+
NO_ACCOUNT_ID: "Failed to extract accountId from token",
|
|
43
|
+
TOKEN_REFRESH_FAILED: "Failed to refresh token, authentication required",
|
|
44
|
+
REQUEST_PARSE_ERROR: "Error parsing request",
|
|
45
|
+
};
|
|
46
|
+
/** Log stages for request logging */
|
|
47
|
+
export const LOG_STAGES = {
|
|
48
|
+
BEFORE_TRANSFORM: "before-transform",
|
|
49
|
+
AFTER_TRANSFORM: "after-transform",
|
|
50
|
+
RESPONSE: "response",
|
|
51
|
+
ERROR_RESPONSE: "error-response",
|
|
52
|
+
};
|
|
53
|
+
/** Platform-specific browser opener commands */
|
|
54
|
+
export const PLATFORM_OPENERS = {
|
|
55
|
+
darwin: "open",
|
|
56
|
+
win32: "start",
|
|
57
|
+
linux: "xdg-open",
|
|
58
|
+
};
|
|
59
|
+
/** OAuth authorization labels */
|
|
60
|
+
export const AUTH_LABELS = {
|
|
61
|
+
OAUTH: "ChatGPT Plus/Pro (Codex Subscription)",
|
|
62
|
+
OAUTH_MANUAL: "ChatGPT Plus/Pro (Manual URL Paste)",
|
|
63
|
+
API_KEY: "Manually enter API Key",
|
|
64
|
+
INSTRUCTIONS: "A browser window should open. If it doesn't, copy the URL and open it manually.",
|
|
65
|
+
INSTRUCTIONS_MANUAL: "After logging in, copy the full redirect URL and paste it here.",
|
|
66
|
+
};
|
|
67
|
+
//# sourceMappingURL=constants.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"constants.js","sourceRoot":"","sources":["../../lib/constants.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,uDAAuD;AACvD,MAAM,CAAC,MAAM,WAAW,GAAG,qBAAqB,CAAC;AAEjD,uCAAuC;AACvC,MAAM,CAAC,MAAM,cAAc,GAAG,iCAAiC,CAAC;AAEhE,gEAAgE;AAChE,MAAM,CAAC,MAAM,aAAa,GAAG,eAAe,CAAC;AAE7C,6CAA6C;AAC7C,MAAM,CAAC,MAAM,WAAW,GAAG,QAAQ,CAAC;AAEpC,wBAAwB;AACxB,MAAM,CAAC,MAAM,WAAW,GAAG;IAC1B,EAAE,EAAE,GAAG;IACP,YAAY,EAAE,GAAG;IACjB,SAAS,EAAE,GAAG;IACd,iBAAiB,EAAE,GAAG;CACb,CAAC;AAEX,8BAA8B;AAC9B,MAAM,CAAC,MAAM,cAAc,GAAG;IAC7B,IAAI,EAAE,aAAa;IACnB,UAAU,EAAE,oBAAoB;IAChC,UAAU,EAAE,YAAY;IACxB,UAAU,EAAE,YAAY;IACxB,eAAe,EAAE,iBAAiB;CACzB,CAAC;AAEX,oCAAoC;AACpC,MAAM,CAAC,MAAM,oBAAoB,GAAG;IACnC,cAAc,EAAE,wBAAwB;IACxC,gBAAgB,EAAE,cAAc;CACvB,CAAC;AAEX,wBAAwB;AACxB,MAAM,CAAC,MAAM,SAAS,GAAG;IACxB,SAAS,EAAE,YAAY;IACvB,eAAe,EAAE,kBAAkB;CAC1B,CAAC;AAEX,4CAA4C;AAC5C,MAAM,CAAC,MAAM,cAAc,GAAG,6BAAsC,CAAC;AAErE,qBAAqB;AACrB,MAAM,CAAC,MAAM,cAAc,GAAG;IAC7B,aAAa,EAAE,wCAAwC;IACvD,oBAAoB,EAAE,kDAAkD;IACxE,mBAAmB,EAAE,uBAAuB;CACnC,CAAC;AAEX,qCAAqC;AACrC,MAAM,CAAC,MAAM,UAAU,GAAG;IACzB,gBAAgB,EAAE,kBAAkB;IACpC,eAAe,EAAE,iBAAiB;IAClC,QAAQ,EAAE,UAAU;IACpB,cAAc,EAAE,gBAAgB;CACvB,CAAC;AAEX,gDAAgD;AAChD,MAAM,CAAC,MAAM,gBAAgB,GAAG;IAC/B,MAAM,EAAE,MAAM;IACd,KAAK,EAAE,OAAO;IACd,KAAK,EAAE,UAAU;CACR,CAAC;AAEX,iCAAiC;AACjC,MAAM,CAAC,MAAM,WAAW,GAAG;IAC1B,KAAK,EAAE,uCAAuC;IAC9C,YAAY,EAAE,qCAAqC;IACnD,OAAO,EAAE,wBAAwB;IACjC,YAAY,EACX,iFAAiF;IAClF,mBAAmB,EAClB,iEAAiE;CACzD,CAAC"}
|