This does not implement the full Login with ChatGPT experience, but it
should unblock people.
**What works**
* The `codex` multitool now has a `login` subcommand, so you can run
`codex login`, which should write `CODEX_HOME/auth.json` if you complete
the flow successfully. The TUI will now read the `OPENAI_API_KEY` from
`auth.json`.
* The TUI should refresh the token if it has expired and the necessary
information is in `auth.json`.
* There is a `LoginScreen` in the TUI that tells you to run `codex
login` if both (1) your model provider expects to use `OPENAI_API_KEY`
as its env var, and (2) `OPENAI_API_KEY` is not set.
**What does not work**
* The `LoginScreen` does not support the login flow from within the TUI.
Instead, it tells you to quit, run `codex login`, and then run `codex`
again.
* `codex exec` does read from `auth.json` yet, nor does it direct the
user to go through the login flow if `OPENAI_API_KEY` is not be found.
* The `maybeRedeemCredits()` function from `get-api-key.tsx` has not
been ported from TypeScript to `login_with_chatgpt.py` yet:
a67a67f325/codex-cli/src/utils/get-api-key.tsx (L84-L89)
**Implementation**
Currently, the OAuth flow requires running a local webserver on
`127.0.0.1:1455`. It seemed wasteful to incur the additional binary cost
of a webserver dependency in the Rust CLI just to support login, so
instead we implement this logic in Python, as Python has a `http.server`
module as part of its standard library. Specifically, we bundle the
contents of a single Python file as a string in the Rust CLI and then
use it to spawn a subprocess as `python3 -c
{{SOURCE_FOR_PYTHON_SERVER}}`.
As such, the most significant files in this PR are:
```
codex-rs/login/src/login_with_chatgpt.py
codex-rs/login/src/lib.rs
```
Now that the CLI may load `OPENAI_API_KEY` from the environment _or_
`CODEX_HOME/auth.json`, we need a new abstraction for reading/writing
this variable, so we introduce:
```
codex-rs/core/src/openai_api_key.rs
```
Note that `std::env::set_var()` is [rightfully] `unsafe` in Rust 2024,
so we use a LazyLock<RwLock<Option<String>>> to store `OPENAI_API_KEY`
so it is read in a thread-safe manner.
Ultimately, it should be possible to go through the entire login flow
from the TUI. This PR introduces a placeholder `LoginScreen` UI for that
right now, though the new `codex login` subcommand introduced in this PR
should be a viable workaround until the UI is ready.
**Testing**
Because the login flow is currently implemented in a standalone Python
file, you can test it without building any Rust code as follows:
```
rm -rf /tmp/codex_home && mkdir /tmp/codex_home
CODEX_HOME=/tmp/codex_home python3 codex-rs/login/src/login_with_chatgpt.py
```
For reference:
* the original TypeScript implementation was introduced in
https://github.com/openai/codex/pull/963
* support for redeeming credits was later added in
https://github.com/openai/codex/pull/974
767 lines
25 KiB
TypeScript
767 lines
25 KiB
TypeScript
import type { Choice } from "./get-api-key-components";
|
|
import type { Request, Response } from "express";
|
|
|
|
import { ApiKeyPrompt, WaitingForAuth } from "./get-api-key-components";
|
|
import chalk from "chalk";
|
|
import express from "express";
|
|
import fs from "fs/promises";
|
|
import { render } from "ink";
|
|
import crypto from "node:crypto";
|
|
import { URL } from "node:url";
|
|
import open from "open";
|
|
import os from "os";
|
|
import path from "path";
|
|
import React from "react";
|
|
|
|
function promptUserForChoice(): Promise<Choice> {
|
|
return new Promise<Choice>((resolve) => {
|
|
const instance = render(
|
|
<ApiKeyPrompt
|
|
onDone={(choice: Choice) => {
|
|
resolve(choice);
|
|
instance.unmount();
|
|
}}
|
|
/>,
|
|
);
|
|
});
|
|
}
|
|
|
|
interface OidcConfiguration {
|
|
issuer: string;
|
|
authorization_endpoint: string;
|
|
token_endpoint: string;
|
|
}
|
|
|
|
async function getOidcConfiguration(
|
|
issuer: string,
|
|
): Promise<OidcConfiguration> {
|
|
const discoveryUrl = new URL(issuer);
|
|
discoveryUrl.pathname = "/.well-known/openid-configuration";
|
|
|
|
if (issuer === "https://auth.openai.com") {
|
|
// Account for legacy quirk in production tenant
|
|
discoveryUrl.pathname = "/v2.0" + discoveryUrl.pathname;
|
|
}
|
|
|
|
const res = await fetch(discoveryUrl.toString());
|
|
if (!res.ok) {
|
|
throw new Error("Failed to fetch OIDC configuration");
|
|
}
|
|
return (await res.json()) as OidcConfiguration;
|
|
}
|
|
|
|
interface IDTokenClaims {
|
|
"exp": number;
|
|
"https://api.openai.com/auth": {
|
|
organization_id: string;
|
|
project_id: string;
|
|
completed_platform_onboarding: boolean;
|
|
is_org_owner: boolean;
|
|
chatgpt_subscription_active_start: string;
|
|
chatgpt_subscription_active_until: string;
|
|
chatgpt_plan_type: string;
|
|
};
|
|
}
|
|
|
|
interface AccessTokenClaims {
|
|
"https://api.openai.com/auth": {
|
|
chatgpt_plan_type: string;
|
|
};
|
|
}
|
|
|
|
function generatePKCECodes(): {
|
|
code_verifier: string;
|
|
code_challenge: string;
|
|
} {
|
|
const code_verifier = crypto.randomBytes(64).toString("hex");
|
|
const code_challenge = crypto
|
|
.createHash("sha256")
|
|
.update(code_verifier)
|
|
.digest("base64url");
|
|
return { code_verifier, code_challenge };
|
|
}
|
|
|
|
async function maybeRedeemCredits(
|
|
issuer: string,
|
|
clientId: string,
|
|
refreshToken: string,
|
|
idToken?: string,
|
|
): Promise<void> {
|
|
try {
|
|
let currentIdToken = idToken;
|
|
let idClaims: IDTokenClaims | undefined;
|
|
|
|
if (
|
|
currentIdToken &&
|
|
typeof currentIdToken === "string" &&
|
|
currentIdToken.split(".")[1]
|
|
) {
|
|
idClaims = JSON.parse(
|
|
Buffer.from(currentIdToken.split(".")[1]!, "base64url").toString(
|
|
"utf8",
|
|
),
|
|
) as IDTokenClaims;
|
|
} else {
|
|
currentIdToken = "";
|
|
}
|
|
|
|
// Validate idToken expiration
|
|
// if expired, attempt token-exchange for a fresh idToken
|
|
if (!idClaims || !idClaims.exp || Date.now() >= idClaims.exp * 1000) {
|
|
// eslint-disable-next-line no-console
|
|
console.log(chalk.dim("Refreshing credentials..."));
|
|
try {
|
|
const refreshRes = await fetch("https://auth.openai.com/oauth/token", {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({
|
|
client_id: clientId,
|
|
grant_type: "refresh_token",
|
|
refresh_token: refreshToken,
|
|
scope: "openid profile email",
|
|
}),
|
|
});
|
|
if (!refreshRes.ok) {
|
|
// eslint-disable-next-line no-console
|
|
console.warn(
|
|
`Failed to refresh credentials: ${refreshRes.status} ${refreshRes.statusText}\n${chalk.dim(await refreshRes.text())}`,
|
|
);
|
|
// eslint-disable-next-line no-console
|
|
console.warn(
|
|
`Please sign in again to redeem credits: ${chalk.bold("codex --login")}`,
|
|
);
|
|
return;
|
|
}
|
|
const refreshData = (await refreshRes.json()) as {
|
|
id_token: string;
|
|
refresh_token?: string;
|
|
};
|
|
currentIdToken = refreshData.id_token;
|
|
idClaims = JSON.parse(
|
|
Buffer.from(currentIdToken.split(".")[1]!, "base64url").toString(
|
|
"utf8",
|
|
),
|
|
) as IDTokenClaims;
|
|
if (refreshData.refresh_token) {
|
|
try {
|
|
const home = os.homedir();
|
|
const authDir = path.join(home, ".codex");
|
|
const authFile = path.join(authDir, "auth.json");
|
|
const existingJson = JSON.parse(
|
|
await fs.readFile(authFile, "utf-8"),
|
|
);
|
|
existingJson.tokens.id_token = currentIdToken;
|
|
existingJson.tokens.refresh_token = refreshData.refresh_token;
|
|
existingJson.last_refresh = new Date().toISOString();
|
|
await fs.writeFile(
|
|
authFile,
|
|
JSON.stringify(existingJson, null, 2),
|
|
{ mode: 0o600 },
|
|
);
|
|
} catch (err) {
|
|
// eslint-disable-next-line no-console
|
|
console.warn("Unable to update refresh token in auth file:", err);
|
|
}
|
|
}
|
|
} catch (err) {
|
|
// eslint-disable-next-line no-console
|
|
console.warn("Unable to refresh ID token via token-exchange:", err);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Confirm the subscription is active for more than 7 days
|
|
const subStart =
|
|
idClaims["https://api.openai.com/auth"]
|
|
?.chatgpt_subscription_active_start;
|
|
if (
|
|
typeof subStart === "string" &&
|
|
Date.now() - new Date(subStart).getTime() < 7 * 24 * 60 * 60 * 1000
|
|
) {
|
|
// eslint-disable-next-line no-console
|
|
console.warn(
|
|
"Sorry, your subscription must be active for more than 7 days to redeem credits.\nMore info: " +
|
|
chalk.dim("https://help.openai.com/en/articles/11381614") +
|
|
chalk.bold(
|
|
"\nPlease try again on " +
|
|
new Date(
|
|
new Date(subStart).getTime() + 7 * 24 * 60 * 60 * 1000,
|
|
).toLocaleDateString() +
|
|
" " +
|
|
new Date(
|
|
new Date(subStart).getTime() + 7 * 24 * 60 * 60 * 1000,
|
|
).toLocaleTimeString(),
|
|
),
|
|
);
|
|
return;
|
|
}
|
|
|
|
const completed = Boolean(
|
|
idClaims["https://api.openai.com/auth"]?.completed_platform_onboarding,
|
|
);
|
|
const isOwner = Boolean(
|
|
idClaims["https://api.openai.com/auth"]?.is_org_owner,
|
|
);
|
|
const needsSetup = !completed && isOwner;
|
|
|
|
const planType = idClaims["https://api.openai.com/auth"]
|
|
?.chatgpt_plan_type as string | undefined;
|
|
|
|
if (needsSetup || !(planType === "plus" || planType === "pro")) {
|
|
// eslint-disable-next-line no-console
|
|
console.warn(
|
|
"Users with Plus or Pro subscriptions can redeem free API credits.\nMore info: " +
|
|
chalk.dim("https://help.openai.com/en/articles/11381614"),
|
|
);
|
|
return;
|
|
}
|
|
|
|
const apiHost =
|
|
issuer === "https://auth.openai.com"
|
|
? "https://api.openai.com"
|
|
: "https://api.openai.org";
|
|
|
|
const redeemRes = await fetch(`${apiHost}/v1/billing/redeem_credits`, {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({ id_token: currentIdToken }),
|
|
});
|
|
|
|
if (!redeemRes.ok) {
|
|
// eslint-disable-next-line no-console
|
|
console.warn(
|
|
`Credit redemption request failed: ${redeemRes.status} ${redeemRes.statusText}`,
|
|
);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
const redeemData = (await redeemRes.json()) as {
|
|
granted_chatgpt_subscriber_api_credits?: number;
|
|
};
|
|
const granted = redeemData?.granted_chatgpt_subscriber_api_credits ?? 0;
|
|
if (granted > 0) {
|
|
// eslint-disable-next-line no-console
|
|
console.log(
|
|
chalk.green(
|
|
`${chalk.bold(
|
|
`Thanks for being a ChatGPT ${
|
|
planType === "plus" ? "Plus" : "Pro"
|
|
} subscriber!`,
|
|
)}\nIf you haven't already redeemed, you should receive ${
|
|
planType === "plus" ? "$5" : "$50"
|
|
} in API credits\nCredits: ${chalk.dim(chalk.underline("https://platform.openai.com/settings/organization/billing/credit-grants"))}\nMore info: ${chalk.dim(chalk.underline("https://help.openai.com/en/articles/11381614"))}`,
|
|
),
|
|
);
|
|
} else {
|
|
// eslint-disable-next-line no-console
|
|
console.log(
|
|
chalk.green(
|
|
`It looks like no credits were granted:\n${JSON.stringify(
|
|
redeemData,
|
|
null,
|
|
2,
|
|
)}\nCredits: ${chalk.dim(
|
|
chalk.underline(
|
|
"https://platform.openai.com/settings/organization/billing/credit-grants",
|
|
),
|
|
)}\nMore info: ${chalk.dim(
|
|
chalk.underline("https://help.openai.com/en/articles/11381614"),
|
|
)}`,
|
|
),
|
|
);
|
|
}
|
|
} catch (parseErr) {
|
|
// eslint-disable-next-line no-console
|
|
console.warn("Unable to parse credit redemption response:", parseErr);
|
|
}
|
|
} catch (err) {
|
|
// eslint-disable-next-line no-console
|
|
console.warn("Unable to redeem ChatGPT subscriber API credits:", err);
|
|
}
|
|
}
|
|
|
|
async function handleCallback(
|
|
req: Request,
|
|
issuer: string,
|
|
oidcConfig: OidcConfiguration,
|
|
codeVerifier: string,
|
|
clientId: string,
|
|
redirectUri: string,
|
|
expectedState: string,
|
|
): Promise<{ access_token: string; success_url: string }> {
|
|
const state = (req.query as Record<string, string>)["state"] as
|
|
| string
|
|
| undefined;
|
|
if (!state || state !== expectedState) {
|
|
throw new Error("Invalid state parameter");
|
|
}
|
|
|
|
const code = (req.query as Record<string, string>)["code"] as
|
|
| string
|
|
| undefined;
|
|
if (!code) {
|
|
throw new Error("Missing authorization code");
|
|
}
|
|
|
|
const params = new URLSearchParams();
|
|
params.append("grant_type", "authorization_code");
|
|
params.append("code", code);
|
|
params.append("redirect_uri", redirectUri);
|
|
params.append("client_id", clientId);
|
|
params.append("code_verifier", codeVerifier);
|
|
|
|
oidcConfig.token_endpoint = `${issuer}/oauth/token`;
|
|
const tokenRes = await fetch(oidcConfig.token_endpoint, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/x-www-form-urlencoded",
|
|
},
|
|
body: params.toString(),
|
|
});
|
|
|
|
if (!tokenRes.ok) {
|
|
throw new Error("Failed to exchange authorization code for tokens");
|
|
}
|
|
|
|
const tokenData = (await tokenRes.json()) as {
|
|
id_token: string;
|
|
access_token: string;
|
|
refresh_token: string;
|
|
};
|
|
|
|
const idTokenParts = tokenData.id_token.split(".");
|
|
if (idTokenParts.length !== 3) {
|
|
throw new Error("Invalid ID token");
|
|
}
|
|
const accessTokenParts = tokenData.access_token.split(".");
|
|
if (accessTokenParts.length !== 3) {
|
|
throw new Error("Invalid access token");
|
|
}
|
|
|
|
const idTokenClaims = JSON.parse(
|
|
Buffer.from(idTokenParts[1]!, "base64url").toString("utf8"),
|
|
) as IDTokenClaims;
|
|
|
|
const accessTokenClaims = JSON.parse(
|
|
Buffer.from(accessTokenParts[1]!, "base64url").toString("utf8"),
|
|
) as AccessTokenClaims;
|
|
|
|
const org_id = idTokenClaims["https://api.openai.com/auth"]?.organization_id;
|
|
|
|
if (!org_id) {
|
|
throw new Error("Missing organization in id_token claims");
|
|
}
|
|
const project_id = idTokenClaims["https://api.openai.com/auth"]?.project_id;
|
|
|
|
if (!project_id) {
|
|
throw new Error("Missing project in id_token claims");
|
|
}
|
|
|
|
const randomId = crypto.randomBytes(6).toString("hex");
|
|
const exchangeParams = new URLSearchParams({
|
|
grant_type: "urn:ietf:params:oauth:grant-type:token-exchange",
|
|
client_id: clientId,
|
|
requested_token: "openai-api-key",
|
|
subject_token: tokenData.id_token,
|
|
subject_token_type: "urn:ietf:params:oauth:token-type:id_token",
|
|
name: `Codex CLI [auto-generated] (${new Date().toISOString().slice(0, 10)}) [${
|
|
randomId
|
|
}]`,
|
|
});
|
|
const exchangeRes = await fetch(oidcConfig.token_endpoint, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/x-www-form-urlencoded",
|
|
},
|
|
body: exchangeParams.toString(),
|
|
});
|
|
if (!exchangeRes.ok) {
|
|
throw new Error(`Failed to create API key: ${await exchangeRes.text()}`);
|
|
}
|
|
|
|
const exchanged = (await exchangeRes.json()) as {
|
|
access_token: string;
|
|
// NOTE(mbolin): I did not see the "key" property set in practice. Note
|
|
// this property is not read by the code.
|
|
key: string;
|
|
};
|
|
|
|
// Determine whether the organization still requires additional
|
|
// setup (e.g., adding a payment method) based on the ID-token
|
|
// claim provided by the auth service.
|
|
const completedOnboarding = Boolean(
|
|
idTokenClaims["https://api.openai.com/auth"]?.completed_platform_onboarding,
|
|
);
|
|
const chatgptPlanType =
|
|
accessTokenClaims["https://api.openai.com/auth"]?.chatgpt_plan_type;
|
|
const isOrgOwner = Boolean(
|
|
idTokenClaims["https://api.openai.com/auth"]?.is_org_owner,
|
|
);
|
|
const needsSetup = !completedOnboarding && isOrgOwner;
|
|
|
|
// Build the success URL on the same host/port as the callback and
|
|
// include the required query parameters for the front-end page.
|
|
// console.log("Redirecting to success page");
|
|
const successUrl = new URL("/success", redirectUri);
|
|
if (issuer === "https://auth.openai.com") {
|
|
successUrl.searchParams.set("platform_url", "https://platform.openai.com");
|
|
} else {
|
|
successUrl.searchParams.set(
|
|
"platform_url",
|
|
"https://platform.api.openai.org",
|
|
);
|
|
}
|
|
successUrl.searchParams.set("id_token", tokenData.id_token);
|
|
successUrl.searchParams.set("needs_setup", needsSetup ? "true" : "false");
|
|
successUrl.searchParams.set("org_id", org_id);
|
|
successUrl.searchParams.set("project_id", project_id);
|
|
successUrl.searchParams.set("plan_type", chatgptPlanType);
|
|
|
|
try {
|
|
const home = os.homedir();
|
|
const authDir = path.join(home, ".codex");
|
|
await fs.mkdir(authDir, { recursive: true });
|
|
const authFile = path.join(authDir, "auth.json");
|
|
const authData = {
|
|
tokens: tokenData,
|
|
last_refresh: new Date().toISOString(),
|
|
OPENAI_API_KEY: exchanged.access_token,
|
|
};
|
|
await fs.writeFile(authFile, JSON.stringify(authData, null, 2), {
|
|
mode: 0o600,
|
|
});
|
|
} catch (err) {
|
|
// eslint-disable-next-line no-console
|
|
console.warn("Unable to save auth file:", err);
|
|
}
|
|
|
|
await maybeRedeemCredits(
|
|
issuer,
|
|
clientId,
|
|
tokenData.refresh_token,
|
|
tokenData.id_token,
|
|
);
|
|
|
|
return {
|
|
access_token: exchanged.access_token,
|
|
success_url: successUrl.toString(),
|
|
};
|
|
}
|
|
|
|
const LOGIN_SUCCESS_HTML = String.raw`
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="utf-8" />
|
|
<title>Sign into Codex CLI</title>
|
|
<link rel="icon" href='data:image/svg+xml,%3Csvg xmlns="http://www.w3.org/2000/svg" width="32" height="32" fill="none" viewBox="0 0 32 32"%3E%3Cpath stroke="%23000" stroke-linecap="round" stroke-width="2.484" d="M22.356 19.797H17.17M9.662 12.29l1.979 3.576a.511.511 0 0 1-.005.504l-1.974 3.409M30.758 16c0 8.15-6.607 14.758-14.758 14.758-8.15 0-14.758-6.607-14.758-14.758C1.242 7.85 7.85 1.242 16 1.242c8.15 0 14.758 6.608 14.758 14.758Z"/%3E%3C/svg%3E' type="image/svg+xml">
|
|
<style>
|
|
.container {
|
|
margin: auto;
|
|
height: 100%;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
position: relative;
|
|
background: white;
|
|
font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;
|
|
}
|
|
.inner-container {
|
|
width: 400px;
|
|
flex-direction: column;
|
|
justify-content: flex-start;
|
|
align-items: center;
|
|
gap: 20px;
|
|
display: inline-flex;
|
|
}
|
|
.content {
|
|
align-self: stretch;
|
|
flex-direction: column;
|
|
justify-content: flex-start;
|
|
align-items: center;
|
|
gap: 20px;
|
|
display: flex;
|
|
}
|
|
.svg-wrapper {
|
|
position: relative;
|
|
}
|
|
.title {
|
|
text-align: center;
|
|
color: var(--text-primary, #0D0D0D);
|
|
font-size: 28px;
|
|
font-weight: 400;
|
|
line-height: 36.40px;
|
|
word-wrap: break-word;
|
|
}
|
|
.setup-box {
|
|
width: 600px;
|
|
padding: 16px 20px;
|
|
background: var(--bg-primary, white);
|
|
box-shadow: 0px 4px 16px rgba(0, 0, 0, 0.05);
|
|
border-radius: 16px;
|
|
outline: 1px var(--border-default, rgba(13, 13, 13, 0.10)) solid;
|
|
outline-offset: -1px;
|
|
justify-content: flex-start;
|
|
align-items: center;
|
|
gap: 16px;
|
|
display: inline-flex;
|
|
}
|
|
.setup-content {
|
|
flex: 1 1 0;
|
|
justify-content: flex-start;
|
|
align-items: center;
|
|
gap: 24px;
|
|
display: flex;
|
|
}
|
|
.setup-text {
|
|
flex: 1 1 0;
|
|
flex-direction: column;
|
|
justify-content: flex-start;
|
|
align-items: flex-start;
|
|
gap: 4px;
|
|
display: inline-flex;
|
|
}
|
|
.setup-title {
|
|
align-self: stretch;
|
|
color: var(--text-primary, #0D0D0D);
|
|
font-size: 14px;
|
|
font-weight: 510;
|
|
line-height: 20px;
|
|
word-wrap: break-word;
|
|
}
|
|
.setup-description {
|
|
align-self: stretch;
|
|
color: var(--text-secondary, #5D5D5D);
|
|
font-size: 14px;
|
|
font-weight: 400;
|
|
line-height: 20px;
|
|
word-wrap: break-word;
|
|
}
|
|
.redirect-box {
|
|
justify-content: flex-start;
|
|
align-items: center;
|
|
gap: 8px;
|
|
display: flex;
|
|
}
|
|
.close-button,
|
|
.redirect-button {
|
|
height: 28px;
|
|
padding: 8px 16px;
|
|
background: var(--interactive-bg-primary-default, #0D0D0D);
|
|
border-radius: 999px;
|
|
justify-content: center;
|
|
align-items: center;
|
|
gap: 4px;
|
|
display: flex;
|
|
}
|
|
.close-button,
|
|
.redirect-text {
|
|
color: var(--interactive-label-primary-default, white);
|
|
font-size: 14px;
|
|
font-weight: 510;
|
|
line-height: 20px;
|
|
word-wrap: break-word;
|
|
text-decoration: none;
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<div class="container">
|
|
<div class="inner-container">
|
|
<div class="content">
|
|
<div data-svg-wrapper class="svg-wrapper">
|
|
<svg width="56" height="56" viewBox="0 0 56 56" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
<path d="M4.6665 28.0003C4.6665 15.1137 15.1132 4.66699 27.9998 4.66699C40.8865 4.66699 51.3332 15.1137 51.3332 28.0003C51.3332 40.887 40.8865 51.3337 27.9998 51.3337C15.1132 51.3337 4.6665 40.887 4.6665 28.0003ZM37.5093 18.5088C36.4554 17.7672 34.9999 18.0203 34.2583 19.0742L24.8508 32.4427L20.9764 28.1808C20.1095 27.2272 18.6338 27.1569 17.6803 28.0238C16.7267 28.8906 16.6565 30.3664 17.5233 31.3199L23.3566 37.7366C23.833 38.2606 24.5216 38.5399 25.2284 38.4958C25.9353 38.4517 26.5838 38.089 26.9914 37.5098L38.0747 21.7598C38.8163 20.7059 38.5632 19.2504 37.5093 18.5088Z" fill="var(--green-400, #04B84C)"/>
|
|
</svg>
|
|
</div>
|
|
<div class="title">Signed in to Codex CLI</div>
|
|
</div>
|
|
<div class="close-box" style="display: none;">
|
|
<div class="setup-description">You may now close this page</div>
|
|
</div>
|
|
<div class="setup-box" style="display: none;">
|
|
<div class="setup-content">
|
|
<div class="setup-text">
|
|
<div class="setup-title">Finish setting up your API organization</div>
|
|
<div class="setup-description">Add a payment method to use your organization.</div>
|
|
</div>
|
|
<div class="redirect-box">
|
|
<div data-hasendicon="false" data-hasstarticon="false" data-ishovered="false" data-isinactive="false" data-ispressed="false" data-size="large" data-type="primary" class="redirect-button">
|
|
<div class="redirect-text">Redirecting in 3s...</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<script>
|
|
(function () {
|
|
const params = new URLSearchParams(window.location.search);
|
|
const needsSetup = params.get('needs_setup') === 'true';
|
|
const platformUrl = params.get('platform_url') || 'https://platform.openai.com';
|
|
const orgId = params.get('org_id');
|
|
const projectId = params.get('project_id');
|
|
const planType = params.get('plan_type');
|
|
const idToken = params.get('id_token');
|
|
// Show different message and optional redirect when setup is required
|
|
if (needsSetup) {
|
|
const setupBox = document.querySelector('.setup-box');
|
|
setupBox.style.display = 'flex';
|
|
const redirectUrlObj = new URL('/org-setup', platformUrl);
|
|
redirectUrlObj.searchParams.set('p', planType);
|
|
redirectUrlObj.searchParams.set('t', idToken);
|
|
redirectUrlObj.searchParams.set('with_org', orgId);
|
|
redirectUrlObj.searchParams.set('project_id', projectId);
|
|
const redirectUrl = redirectUrlObj.toString();
|
|
const message = document.querySelector('.redirect-text');
|
|
let countdown = 3;
|
|
function tick() {
|
|
message.textContent =
|
|
'Redirecting in ' + countdown + 's…';
|
|
if (countdown === 0) {
|
|
window.location.replace(redirectUrl);
|
|
} else {
|
|
countdown -= 1;
|
|
setTimeout(tick, 1000);
|
|
}
|
|
}
|
|
tick();
|
|
} else {
|
|
const closeBox = document.querySelector('.close-box');
|
|
closeBox.style.display = 'flex';
|
|
}
|
|
})();
|
|
</script>
|
|
</body>
|
|
</html>`;
|
|
|
|
async function signInFlow(issuer: string, clientId: string): Promise<string> {
|
|
const app = express();
|
|
|
|
let codeVerifier = "";
|
|
let redirectUri = "";
|
|
let server: ReturnType<typeof app.listen>;
|
|
const state = crypto.randomBytes(32).toString("hex");
|
|
|
|
const apiKeyPromise = new Promise<string>((resolve, reject) => {
|
|
let _apiKey: string | undefined;
|
|
|
|
app.get("/success", (_req: Request, res: Response) => {
|
|
res.type("text/html").send(LOGIN_SUCCESS_HTML);
|
|
if (_apiKey) {
|
|
resolve(_apiKey);
|
|
} else {
|
|
// eslint-disable-next-line no-console
|
|
console.error(
|
|
"Sorry, it seems like the authentication flow failed. Please try again, or submit an issue on our GitHub if it continues.",
|
|
);
|
|
process.exit(1);
|
|
}
|
|
});
|
|
|
|
// Callback route -------------------------------------------------------
|
|
app.get("/auth/callback", async (req: Request, res: Response) => {
|
|
try {
|
|
const oidcConfig = await getOidcConfiguration(issuer);
|
|
oidcConfig.token_endpoint = `${issuer}/oauth/token`;
|
|
oidcConfig.authorization_endpoint = `${issuer}/oauth/authorize`;
|
|
const { access_token, success_url } = await handleCallback(
|
|
req,
|
|
issuer,
|
|
oidcConfig,
|
|
codeVerifier,
|
|
clientId,
|
|
redirectUri,
|
|
state,
|
|
);
|
|
_apiKey = access_token;
|
|
res.redirect(success_url);
|
|
} catch (err) {
|
|
reject(err);
|
|
}
|
|
});
|
|
|
|
server = app.listen(1455, "127.0.0.1", async () => {
|
|
const address = server.address();
|
|
if (typeof address === "string" || !address) {
|
|
// eslint-disable-next-line no-console
|
|
console.log(
|
|
"It seems like you might already be trying to sign in (port :1455 already in use)",
|
|
);
|
|
process.exit(1);
|
|
return;
|
|
}
|
|
const port = address.port;
|
|
redirectUri = `http://localhost:${port}/auth/callback`;
|
|
|
|
try {
|
|
const oidcConfig = await getOidcConfiguration(issuer);
|
|
oidcConfig.token_endpoint = `${issuer}/oauth/token`;
|
|
oidcConfig.authorization_endpoint = `${issuer}/oauth/authorize`;
|
|
const pkce = generatePKCECodes();
|
|
codeVerifier = pkce.code_verifier;
|
|
|
|
const authUrl = new URL(oidcConfig.authorization_endpoint);
|
|
authUrl.searchParams.append("response_type", "code");
|
|
authUrl.searchParams.append("client_id", clientId);
|
|
authUrl.searchParams.append("redirect_uri", redirectUri);
|
|
authUrl.searchParams.append(
|
|
"scope",
|
|
"openid profile email offline_access",
|
|
);
|
|
authUrl.searchParams.append("code_challenge", pkce.code_challenge);
|
|
authUrl.searchParams.append("code_challenge_method", "S256");
|
|
authUrl.searchParams.append("id_token_add_organizations", "true");
|
|
authUrl.searchParams.append("state", state);
|
|
|
|
// Open the browser immediately.
|
|
open(authUrl.toString());
|
|
setTimeout(() => {
|
|
// eslint-disable-next-line no-console
|
|
console.log(
|
|
`\nOpening login page in your browser: ${authUrl.toString()}\n`,
|
|
);
|
|
}, 500);
|
|
} catch (err) {
|
|
reject(err);
|
|
}
|
|
});
|
|
});
|
|
|
|
// Ensure the server is closed afterwards.
|
|
return apiKeyPromise.finally(() => {
|
|
if (server) {
|
|
server.close();
|
|
}
|
|
});
|
|
}
|
|
|
|
export async function getApiKey(
|
|
issuer: string,
|
|
clientId: string,
|
|
forceLogin: boolean = false,
|
|
): Promise<string> {
|
|
if (!forceLogin && process.env["OPENAI_API_KEY"]) {
|
|
return process.env["OPENAI_API_KEY"]!;
|
|
}
|
|
const choice = await promptUserForChoice();
|
|
if (choice.type === "apikey") {
|
|
process.env["OPENAI_API_KEY"] = choice.key;
|
|
return choice.key;
|
|
}
|
|
const spinner = render(<WaitingForAuth />);
|
|
try {
|
|
const key = await signInFlow(issuer, clientId);
|
|
spinner.clear();
|
|
spinner.unmount();
|
|
process.env["OPENAI_API_KEY"] = key;
|
|
return key;
|
|
} catch (err) {
|
|
spinner.clear();
|
|
spinner.unmount();
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
export { maybeRedeemCredits };
|