@bloxystudios/bloxycode 1.0.2 → 1.0.4
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/bin/bloxycode +11 -8
- package/package.json +16 -143
- package/postinstall.mjs +129 -0
- package/LICENSE +0 -21
- package/README.md +0 -286
- package/src/acp/README.md +0 -164
- package/src/acp/agent.ts +0 -1437
- package/src/acp/session.ts +0 -105
- package/src/acp/types.ts +0 -22
- package/src/agent/agent.ts +0 -356
- package/src/agent/generate.txt +0 -75
- package/src/agent/prompt/bloxy.txt +0 -46
- package/src/agent/prompt/compaction.txt +0 -12
- package/src/agent/prompt/explore.txt +0 -18
- package/src/agent/prompt/summary.txt +0 -11
- package/src/agent/prompt/title.txt +0 -44
- package/src/auth/index.ts +0 -73
- package/src/bloxy/event.ts +0 -41
- package/src/bloxy/index.ts +0 -5
- package/src/bloxy/parser.ts +0 -263
- package/src/bloxy/prompt.ts +0 -121
- package/src/bloxy/runner.ts +0 -193
- package/src/bloxy/state.ts +0 -246
- package/src/bun/index.ts +0 -134
- package/src/bus/bus-event.ts +0 -43
- package/src/bus/global.ts +0 -10
- package/src/bus/index.ts +0 -105
- package/src/cli/bootstrap.ts +0 -17
- package/src/cli/cmd/acp.ts +0 -69
- package/src/cli/cmd/agent.ts +0 -257
- package/src/cli/cmd/auth.ts +0 -400
- package/src/cli/cmd/cmd.ts +0 -7
- package/src/cli/cmd/debug/agent.ts +0 -167
- package/src/cli/cmd/debug/config.ts +0 -16
- package/src/cli/cmd/debug/file.ts +0 -97
- package/src/cli/cmd/debug/index.ts +0 -48
- package/src/cli/cmd/debug/lsp.ts +0 -52
- package/src/cli/cmd/debug/ripgrep.ts +0 -87
- package/src/cli/cmd/debug/scrap.ts +0 -16
- package/src/cli/cmd/debug/skill.ts +0 -16
- package/src/cli/cmd/debug/snapshot.ts +0 -52
- package/src/cli/cmd/export.ts +0 -88
- package/src/cli/cmd/generate.ts +0 -38
- package/src/cli/cmd/github.ts +0 -1548
- package/src/cli/cmd/import.ts +0 -98
- package/src/cli/cmd/mcp.ts +0 -755
- package/src/cli/cmd/models.ts +0 -77
- package/src/cli/cmd/pr.ts +0 -112
- package/src/cli/cmd/run.ts +0 -395
- package/src/cli/cmd/serve.ts +0 -20
- package/src/cli/cmd/session.ts +0 -135
- package/src/cli/cmd/stats.ts +0 -402
- package/src/cli/cmd/tui/app.tsx +0 -771
- package/src/cli/cmd/tui/attach.ts +0 -39
- package/src/cli/cmd/tui/component/border.tsx +0 -21
- package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
- package/src/cli/cmd/tui/component/dialog-command.tsx +0 -148
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
- package/src/cli/cmd/tui/component/dialog-model.tsx +0 -234
- package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -256
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -114
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
- package/src/cli/cmd/tui/component/dialog-stash.tsx +0 -87
- package/src/cli/cmd/tui/component/dialog-status.tsx +0 -164
- package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
- package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
- package/src/cli/cmd/tui/component/logo.tsx +0 -102
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -653
- package/src/cli/cmd/tui/component/prompt/frecency.tsx +0 -89
- package/src/cli/cmd/tui/component/prompt/history.tsx +0 -108
- package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1138
- package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
- package/src/cli/cmd/tui/component/tips.tsx +0 -153
- package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
- package/src/cli/cmd/tui/context/args.tsx +0 -14
- package/src/cli/cmd/tui/context/directory.ts +0 -13
- package/src/cli/cmd/tui/context/exit.tsx +0 -23
- package/src/cli/cmd/tui/context/helper.tsx +0 -25
- package/src/cli/cmd/tui/context/keybind.tsx +0 -101
- package/src/cli/cmd/tui/context/kv.tsx +0 -52
- package/src/cli/cmd/tui/context/local.tsx +0 -402
- package/src/cli/cmd/tui/context/prompt.tsx +0 -18
- package/src/cli/cmd/tui/context/route.tsx +0 -46
- package/src/cli/cmd/tui/context/sdk.tsx +0 -94
- package/src/cli/cmd/tui/context/sync.tsx +0 -470
- package/src/cli/cmd/tui/context/theme/aura.json +0 -69
- package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
- package/src/cli/cmd/tui/context/theme/bloxycode.json +0 -245
- package/src/cli/cmd/tui/context/theme/carbonfox.json +0 -248
- package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -233
- package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -233
- package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
- package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -228
- package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
- package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
- package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
- package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
- package/src/cli/cmd/tui/context/theme/github.json +0 -233
- package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -242
- package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
- package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -237
- package/src/cli/cmd/tui/context/theme/material.json +0 -235
- package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
- package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
- package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
- package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
- package/src/cli/cmd/tui/context/theme/nord.json +0 -223
- package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
- package/src/cli/cmd/tui/context/theme/orng.json +0 -249
- package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
- package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
- package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
- package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
- package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
- package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
- package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
- package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
- package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
- package/src/cli/cmd/tui/context/theme.tsx +0 -1152
- package/src/cli/cmd/tui/event.ts +0 -48
- package/src/cli/cmd/tui/routes/home.tsx +0 -140
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -64
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -109
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -26
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
- package/src/cli/cmd/tui/routes/session/footer.tsx +0 -91
- package/src/cli/cmd/tui/routes/session/header.tsx +0 -142
- package/src/cli/cmd/tui/routes/session/index.tsx +0 -2048
- package/src/cli/cmd/tui/routes/session/permission.tsx +0 -508
- package/src/cli/cmd/tui/routes/session/question.tsx +0 -453
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -313
- package/src/cli/cmd/tui/thread.ts +0 -165
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -57
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -83
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -204
- package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -38
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -77
- package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -385
- package/src/cli/cmd/tui/ui/dialog.tsx +0 -167
- package/src/cli/cmd/tui/ui/link.tsx +0 -28
- package/src/cli/cmd/tui/ui/spinner.ts +0 -368
- package/src/cli/cmd/tui/ui/toast.tsx +0 -100
- package/src/cli/cmd/tui/util/clipboard.ts +0 -160
- package/src/cli/cmd/tui/util/editor.ts +0 -32
- package/src/cli/cmd/tui/util/signal.ts +0 -7
- package/src/cli/cmd/tui/util/terminal.ts +0 -114
- package/src/cli/cmd/tui/util/transcript.ts +0 -98
- package/src/cli/cmd/tui/worker.ts +0 -152
- package/src/cli/cmd/uninstall.ts +0 -357
- package/src/cli/cmd/upgrade.ts +0 -73
- package/src/cli/cmd/web.ts +0 -81
- package/src/cli/error.ts +0 -57
- package/src/cli/network.ts +0 -53
- package/src/cli/ui.ts +0 -86
- package/src/cli/upgrade.ts +0 -25
- package/src/command/index.ts +0 -173
- package/src/command/template/bloxy-resume.txt +0 -15
- package/src/command/template/bloxy-status.txt +0 -25
- package/src/command/template/bloxy-validate.txt +0 -22
- package/src/command/template/bloxy.txt +0 -14
- package/src/command/template/initialize.txt +0 -10
- package/src/command/template/review.txt +0 -99
- package/src/config/config.ts +0 -1367
- package/src/config/markdown.ts +0 -93
- package/src/env/index.ts +0 -26
- package/src/file/ignore.ts +0 -83
- package/src/file/index.ts +0 -415
- package/src/file/ripgrep.ts +0 -407
- package/src/file/time.ts +0 -69
- package/src/file/watcher.ts +0 -127
- package/src/flag/flag.ts +0 -79
- package/src/format/formatter.ts +0 -357
- package/src/format/index.ts +0 -137
- package/src/global/index.ts +0 -55
- package/src/id/id.ts +0 -83
- package/src/ide/index.ts +0 -76
- package/src/index.ts +0 -159
- package/src/installation/index.ts +0 -246
- package/src/lsp/client.ts +0 -252
- package/src/lsp/index.ts +0 -485
- package/src/lsp/language.ts +0 -119
- package/src/lsp/server.ts +0 -2046
- package/src/mcp/auth.ts +0 -135
- package/src/mcp/index.ts +0 -934
- package/src/mcp/oauth-callback.ts +0 -200
- package/src/mcp/oauth-provider.ts +0 -154
- package/src/patch/index.ts +0 -680
- package/src/permission/arity.ts +0 -163
- package/src/permission/index.ts +0 -210
- package/src/permission/next.ts +0 -280
- package/src/plugin/antigravity.ts +0 -378
- package/src/plugin/codex.ts +0 -506
- package/src/plugin/copilot.ts +0 -298
- package/src/plugin/index.ts +0 -136
- package/src/project/bootstrap.ts +0 -35
- package/src/project/instance.ts +0 -91
- package/src/project/project.ts +0 -371
- package/src/project/state.ts +0 -66
- package/src/project/vcs.ts +0 -76
- package/src/provider/auth.ts +0 -147
- package/src/provider/models-snapshot.ts +0 -2
- package/src/provider/models.ts +0 -133
- package/src/provider/provider.ts +0 -1241
- package/src/provider/sdk/openai-compatible/src/README.md +0 -5
- package/src/provider/sdk/openai-compatible/src/index.ts +0 -2
- package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +0 -100
- package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +0 -303
- package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +0 -22
- package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +0 -18
- package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +0 -22
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +0 -207
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +0 -1732
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +0 -177
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +0 -1
- package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +0 -88
- package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +0 -128
- package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +0 -115
- package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +0 -65
- package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +0 -104
- package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +0 -103
- package/src/provider/transform.ts +0 -741
- package/src/pty/index.ts +0 -241
- package/src/question/index.ts +0 -171
- package/src/scheduler/index.ts +0 -61
- package/src/server/error.ts +0 -36
- package/src/server/event.ts +0 -7
- package/src/server/mdns.ts +0 -59
- package/src/server/routes/config.ts +0 -92
- package/src/server/routes/experimental.ts +0 -208
- package/src/server/routes/file.ts +0 -197
- package/src/server/routes/global.ts +0 -135
- package/src/server/routes/mcp.ts +0 -225
- package/src/server/routes/permission.ts +0 -68
- package/src/server/routes/project.ts +0 -82
- package/src/server/routes/provider.ts +0 -165
- package/src/server/routes/pty.ts +0 -169
- package/src/server/routes/question.ts +0 -98
- package/src/server/routes/session.ts +0 -939
- package/src/server/routes/tui.ts +0 -379
- package/src/server/server.ts +0 -604
- package/src/session/compaction.ts +0 -225
- package/src/session/fallback.ts +0 -246
- package/src/session/index.ts +0 -498
- package/src/session/instruction.ts +0 -164
- package/src/session/llm.ts +0 -298
- package/src/session/message-v2.ts +0 -747
- package/src/session/message.ts +0 -189
- package/src/session/processor.ts +0 -450
- package/src/session/prompt/anthropic-20250930.txt +0 -166
- package/src/session/prompt/anthropic.txt +0 -105
- package/src/session/prompt/beast.txt +0 -147
- package/src/session/prompt/build-switch.txt +0 -5
- package/src/session/prompt/codex_header.txt +0 -79
- package/src/session/prompt/copilot-gpt-5.txt +0 -143
- package/src/session/prompt/gemini.txt +0 -155
- package/src/session/prompt/max-steps.txt +0 -16
- package/src/session/prompt/plan-reminder-anthropic.txt +0 -67
- package/src/session/prompt/plan.txt +0 -26
- package/src/session/prompt/qwen.txt +0 -109
- package/src/session/prompt.ts +0 -1822
- package/src/session/retry.ts +0 -99
- package/src/session/revert.ts +0 -121
- package/src/session/status.ts +0 -100
- package/src/session/summary.ts +0 -217
- package/src/session/system.ts +0 -52
- package/src/session/todo.ts +0 -37
- package/src/share/share-next.ts +0 -200
- package/src/share/share.ts +0 -92
- package/src/shell/shell.ts +0 -67
- package/src/skill/index.ts +0 -1
- package/src/skill/skill.ts +0 -135
- package/src/snapshot/index.ts +0 -236
- package/src/storage/storage.ts +0 -227
- package/src/tool/apply_patch.ts +0 -281
- package/src/tool/apply_patch.txt +0 -33
- package/src/tool/bash.ts +0 -258
- package/src/tool/bash.txt +0 -115
- package/src/tool/batch.ts +0 -175
- package/src/tool/batch.txt +0 -24
- package/src/tool/bloxy-control.ts +0 -123
- package/src/tool/bloxy-control.txt +0 -13
- package/src/tool/codesearch.ts +0 -132
- package/src/tool/codesearch.txt +0 -12
- package/src/tool/edit.ts +0 -655
- package/src/tool/edit.txt +0 -10
- package/src/tool/external-directory.ts +0 -32
- package/src/tool/glob.ts +0 -77
- package/src/tool/glob.txt +0 -6
- package/src/tool/grep.ts +0 -154
- package/src/tool/grep.txt +0 -8
- package/src/tool/invalid.ts +0 -17
- package/src/tool/ls.ts +0 -121
- package/src/tool/ls.txt +0 -1
- package/src/tool/lsp.ts +0 -96
- package/src/tool/lsp.txt +0 -19
- package/src/tool/multiedit.ts +0 -46
- package/src/tool/multiedit.txt +0 -41
- package/src/tool/plan-enter.txt +0 -14
- package/src/tool/plan-exit.txt +0 -13
- package/src/tool/plan.ts +0 -130
- package/src/tool/question.ts +0 -33
- package/src/tool/question.txt +0 -10
- package/src/tool/read.ts +0 -211
- package/src/tool/read.txt +0 -12
- package/src/tool/registry.ts +0 -161
- package/src/tool/skill.ts +0 -82
- package/src/tool/task.ts +0 -191
- package/src/tool/task.txt +0 -60
- package/src/tool/todo.ts +0 -53
- package/src/tool/todoread.txt +0 -14
- package/src/tool/todowrite.txt +0 -167
- package/src/tool/tool.ts +0 -89
- package/src/tool/truncation.ts +0 -106
- package/src/tool/webfetch.ts +0 -188
- package/src/tool/webfetch.txt +0 -13
- package/src/tool/websearch.ts +0 -150
- package/src/tool/websearch.txt +0 -14
- package/src/tool/write.ts +0 -85
- package/src/tool/write.txt +0 -8
- package/src/util/archive.ts +0 -16
- package/src/util/binary.ts +0 -41
- package/src/util/color.ts +0 -19
- package/src/util/context.ts +0 -25
- package/src/util/defer.ts +0 -12
- package/src/util/error.ts +0 -54
- package/src/util/eventloop.ts +0 -20
- package/src/util/filesystem.ts +0 -93
- package/src/util/fn.ts +0 -11
- package/src/util/format.ts +0 -20
- package/src/util/iife.ts +0 -3
- package/src/util/keybind.ts +0 -103
- package/src/util/lazy.ts +0 -23
- package/src/util/locale.ts +0 -81
- package/src/util/lock.ts +0 -98
- package/src/util/log.ts +0 -180
- package/src/util/queue.ts +0 -32
- package/src/util/rpc.ts +0 -66
- package/src/util/scrap.ts +0 -10
- package/src/util/signal.ts +0 -12
- package/src/util/slug.ts +0 -74
- package/src/util/timeout.ts +0 -14
- package/src/util/token.ts +0 -7
- package/src/util/wildcard.ts +0 -56
- package/src/worktree/index.ts +0 -549
package/src/plugin/codex.ts
DELETED
|
@@ -1,506 +0,0 @@
|
|
|
1
|
-
import type { Hooks, PluginInput } from "@opencode-ai/plugin"
|
|
2
|
-
import { Log } from "../util/log"
|
|
3
|
-
import { Installation } from "../installation"
|
|
4
|
-
import { Auth, OAUTH_DUMMY_KEY } from "../auth"
|
|
5
|
-
import os from "os"
|
|
6
|
-
|
|
7
|
-
const log = Log.create({ service: "plugin.codex" })
|
|
8
|
-
|
|
9
|
-
const CLIENT_ID = "app_EMoamEEZ73f0CkXaXp7hrann"
|
|
10
|
-
const ISSUER = "https://auth.openai.com"
|
|
11
|
-
const CODEX_API_ENDPOINT = "https://chatgpt.com/backend-api/codex/responses"
|
|
12
|
-
const OAUTH_PORT = 1455
|
|
13
|
-
|
|
14
|
-
interface PkceCodes {
|
|
15
|
-
verifier: string
|
|
16
|
-
challenge: string
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
async function generatePKCE(): Promise<PkceCodes> {
|
|
20
|
-
const verifier = generateRandomString(43)
|
|
21
|
-
const encoder = new TextEncoder()
|
|
22
|
-
const data = encoder.encode(verifier)
|
|
23
|
-
const hash = await crypto.subtle.digest("SHA-256", data)
|
|
24
|
-
const challenge = base64UrlEncode(hash)
|
|
25
|
-
return { verifier, challenge }
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
function generateRandomString(length: number): string {
|
|
29
|
-
const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-._~"
|
|
30
|
-
const bytes = crypto.getRandomValues(new Uint8Array(length))
|
|
31
|
-
return Array.from(bytes)
|
|
32
|
-
.map((b) => chars[b % chars.length])
|
|
33
|
-
.join("")
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
function base64UrlEncode(buffer: ArrayBuffer): string {
|
|
37
|
-
const bytes = new Uint8Array(buffer)
|
|
38
|
-
const binary = String.fromCharCode(...bytes)
|
|
39
|
-
return btoa(binary).replace(/\+/g, "-").replace(/\//g, "_").replace(/=+$/, "")
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
function generateState(): string {
|
|
43
|
-
return base64UrlEncode(crypto.getRandomValues(new Uint8Array(32)).buffer)
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
export interface IdTokenClaims {
|
|
47
|
-
chatgpt_account_id?: string
|
|
48
|
-
organizations?: Array<{ id: string }>
|
|
49
|
-
email?: string
|
|
50
|
-
"https://api.openai.com/auth"?: {
|
|
51
|
-
chatgpt_account_id?: string
|
|
52
|
-
}
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
export function parseJwtClaims(token: string): IdTokenClaims | undefined {
|
|
56
|
-
const parts = token.split(".")
|
|
57
|
-
if (parts.length !== 3) return undefined
|
|
58
|
-
try {
|
|
59
|
-
return JSON.parse(Buffer.from(parts[1], "base64url").toString())
|
|
60
|
-
} catch {
|
|
61
|
-
return undefined
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
export function extractAccountIdFromClaims(claims: IdTokenClaims): string | undefined {
|
|
66
|
-
return (
|
|
67
|
-
claims.chatgpt_account_id ||
|
|
68
|
-
claims["https://api.openai.com/auth"]?.chatgpt_account_id ||
|
|
69
|
-
claims.organizations?.[0]?.id
|
|
70
|
-
)
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
export function extractAccountId(tokens: TokenResponse): string | undefined {
|
|
74
|
-
if (tokens.id_token) {
|
|
75
|
-
const claims = parseJwtClaims(tokens.id_token)
|
|
76
|
-
const accountId = claims && extractAccountIdFromClaims(claims)
|
|
77
|
-
if (accountId) return accountId
|
|
78
|
-
}
|
|
79
|
-
if (tokens.access_token) {
|
|
80
|
-
const claims = parseJwtClaims(tokens.access_token)
|
|
81
|
-
return claims ? extractAccountIdFromClaims(claims) : undefined
|
|
82
|
-
}
|
|
83
|
-
return undefined
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
function buildAuthorizeUrl(redirectUri: string, pkce: PkceCodes, state: string): string {
|
|
87
|
-
const params = new URLSearchParams({
|
|
88
|
-
response_type: "code",
|
|
89
|
-
client_id: CLIENT_ID,
|
|
90
|
-
redirect_uri: redirectUri,
|
|
91
|
-
scope: "openid profile email offline_access",
|
|
92
|
-
code_challenge: pkce.challenge,
|
|
93
|
-
code_challenge_method: "S256",
|
|
94
|
-
id_token_add_organizations: "true",
|
|
95
|
-
codex_cli_simplified_flow: "true",
|
|
96
|
-
state,
|
|
97
|
-
originator: "opencode",
|
|
98
|
-
})
|
|
99
|
-
return `${ISSUER}/oauth/authorize?${params.toString()}`
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
interface TokenResponse {
|
|
103
|
-
id_token: string
|
|
104
|
-
access_token: string
|
|
105
|
-
refresh_token: string
|
|
106
|
-
expires_in?: number
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
async function exchangeCodeForTokens(code: string, redirectUri: string, pkce: PkceCodes): Promise<TokenResponse> {
|
|
110
|
-
const response = await fetch(`${ISSUER}/oauth/token`, {
|
|
111
|
-
method: "POST",
|
|
112
|
-
headers: { "Content-Type": "application/x-www-form-urlencoded" },
|
|
113
|
-
body: new URLSearchParams({
|
|
114
|
-
grant_type: "authorization_code",
|
|
115
|
-
code,
|
|
116
|
-
redirect_uri: redirectUri,
|
|
117
|
-
client_id: CLIENT_ID,
|
|
118
|
-
code_verifier: pkce.verifier,
|
|
119
|
-
}).toString(),
|
|
120
|
-
})
|
|
121
|
-
if (!response.ok) {
|
|
122
|
-
throw new Error(`Token exchange failed: ${response.status}`)
|
|
123
|
-
}
|
|
124
|
-
return response.json()
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
async function refreshAccessToken(refreshToken: string): Promise<TokenResponse> {
|
|
128
|
-
const response = await fetch(`${ISSUER}/oauth/token`, {
|
|
129
|
-
method: "POST",
|
|
130
|
-
headers: { "Content-Type": "application/x-www-form-urlencoded" },
|
|
131
|
-
body: new URLSearchParams({
|
|
132
|
-
grant_type: "refresh_token",
|
|
133
|
-
refresh_token: refreshToken,
|
|
134
|
-
client_id: CLIENT_ID,
|
|
135
|
-
}).toString(),
|
|
136
|
-
})
|
|
137
|
-
if (!response.ok) {
|
|
138
|
-
throw new Error(`Token refresh failed: ${response.status}`)
|
|
139
|
-
}
|
|
140
|
-
return response.json()
|
|
141
|
-
}
|
|
142
|
-
|
|
143
|
-
const HTML_SUCCESS = `<!doctype html>
|
|
144
|
-
<html>
|
|
145
|
-
<head>
|
|
146
|
-
<title>BloxyCode - Codex Authorization Successful</title>
|
|
147
|
-
<style>
|
|
148
|
-
body {
|
|
149
|
-
font-family:
|
|
150
|
-
system-ui,
|
|
151
|
-
-apple-system,
|
|
152
|
-
sans-serif;
|
|
153
|
-
display: flex;
|
|
154
|
-
justify-content: center;
|
|
155
|
-
align-items: center;
|
|
156
|
-
height: 100vh;
|
|
157
|
-
margin: 0;
|
|
158
|
-
background: #131010;
|
|
159
|
-
color: #f1ecec;
|
|
160
|
-
}
|
|
161
|
-
.container {
|
|
162
|
-
text-align: center;
|
|
163
|
-
padding: 2rem;
|
|
164
|
-
}
|
|
165
|
-
h1 {
|
|
166
|
-
color: #f1ecec;
|
|
167
|
-
margin-bottom: 1rem;
|
|
168
|
-
}
|
|
169
|
-
p {
|
|
170
|
-
color: #b7b1b1;
|
|
171
|
-
}
|
|
172
|
-
</style>
|
|
173
|
-
</head>
|
|
174
|
-
<body>
|
|
175
|
-
<div class="container">
|
|
176
|
-
<h1>Authorization Successful</h1>
|
|
177
|
-
<p>You can close this window and return to BloxyCode.</p>
|
|
178
|
-
</div>
|
|
179
|
-
<script>
|
|
180
|
-
setTimeout(() => window.close(), 2000)
|
|
181
|
-
</script>
|
|
182
|
-
</body>
|
|
183
|
-
</html>`
|
|
184
|
-
|
|
185
|
-
const HTML_ERROR = (error: string) => `<!doctype html>
|
|
186
|
-
<html>
|
|
187
|
-
<head>
|
|
188
|
-
<title>BloxyCode - Codex Authorization Failed</title>
|
|
189
|
-
<style>
|
|
190
|
-
body {
|
|
191
|
-
font-family:
|
|
192
|
-
system-ui,
|
|
193
|
-
-apple-system,
|
|
194
|
-
sans-serif;
|
|
195
|
-
display: flex;
|
|
196
|
-
justify-content: center;
|
|
197
|
-
align-items: center;
|
|
198
|
-
height: 100vh;
|
|
199
|
-
margin: 0;
|
|
200
|
-
background: #131010;
|
|
201
|
-
color: #f1ecec;
|
|
202
|
-
}
|
|
203
|
-
.container {
|
|
204
|
-
text-align: center;
|
|
205
|
-
padding: 2rem;
|
|
206
|
-
}
|
|
207
|
-
h1 {
|
|
208
|
-
color: #fc533a;
|
|
209
|
-
margin-bottom: 1rem;
|
|
210
|
-
}
|
|
211
|
-
p {
|
|
212
|
-
color: #b7b1b1;
|
|
213
|
-
}
|
|
214
|
-
.error {
|
|
215
|
-
color: #ff917b;
|
|
216
|
-
font-family: monospace;
|
|
217
|
-
margin-top: 1rem;
|
|
218
|
-
padding: 1rem;
|
|
219
|
-
background: #3c140d;
|
|
220
|
-
border-radius: 0.5rem;
|
|
221
|
-
}
|
|
222
|
-
</style>
|
|
223
|
-
</head>
|
|
224
|
-
<body>
|
|
225
|
-
<div class="container">
|
|
226
|
-
<h1>Authorization Failed</h1>
|
|
227
|
-
<p>An error occurred during authorization.</p>
|
|
228
|
-
<div class="error">${error}</div>
|
|
229
|
-
</div>
|
|
230
|
-
</body>
|
|
231
|
-
</html>`
|
|
232
|
-
|
|
233
|
-
interface PendingOAuth {
|
|
234
|
-
pkce: PkceCodes
|
|
235
|
-
state: string
|
|
236
|
-
resolve: (tokens: TokenResponse) => void
|
|
237
|
-
reject: (error: Error) => void
|
|
238
|
-
}
|
|
239
|
-
|
|
240
|
-
let oauthServer: ReturnType<typeof Bun.serve> | undefined
|
|
241
|
-
let pendingOAuth: PendingOAuth | undefined
|
|
242
|
-
|
|
243
|
-
async function startOAuthServer(): Promise<{ port: number; redirectUri: string }> {
|
|
244
|
-
if (oauthServer) {
|
|
245
|
-
return { port: OAUTH_PORT, redirectUri: `http://localhost:${OAUTH_PORT}/auth/callback` }
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
oauthServer = Bun.serve({
|
|
249
|
-
port: OAUTH_PORT,
|
|
250
|
-
fetch(req) {
|
|
251
|
-
const url = new URL(req.url)
|
|
252
|
-
|
|
253
|
-
if (url.pathname === "/auth/callback") {
|
|
254
|
-
const code = url.searchParams.get("code")
|
|
255
|
-
const state = url.searchParams.get("state")
|
|
256
|
-
const error = url.searchParams.get("error")
|
|
257
|
-
const errorDescription = url.searchParams.get("error_description")
|
|
258
|
-
|
|
259
|
-
if (error) {
|
|
260
|
-
const errorMsg = errorDescription || error
|
|
261
|
-
pendingOAuth?.reject(new Error(errorMsg))
|
|
262
|
-
pendingOAuth = undefined
|
|
263
|
-
return new Response(HTML_ERROR(errorMsg), {
|
|
264
|
-
headers: { "Content-Type": "text/html" },
|
|
265
|
-
})
|
|
266
|
-
}
|
|
267
|
-
|
|
268
|
-
if (!code) {
|
|
269
|
-
const errorMsg = "Missing authorization code"
|
|
270
|
-
pendingOAuth?.reject(new Error(errorMsg))
|
|
271
|
-
pendingOAuth = undefined
|
|
272
|
-
return new Response(HTML_ERROR(errorMsg), {
|
|
273
|
-
status: 400,
|
|
274
|
-
headers: { "Content-Type": "text/html" },
|
|
275
|
-
})
|
|
276
|
-
}
|
|
277
|
-
|
|
278
|
-
if (!pendingOAuth || state !== pendingOAuth.state) {
|
|
279
|
-
const errorMsg = "Invalid state - potential CSRF attack"
|
|
280
|
-
pendingOAuth?.reject(new Error(errorMsg))
|
|
281
|
-
pendingOAuth = undefined
|
|
282
|
-
return new Response(HTML_ERROR(errorMsg), {
|
|
283
|
-
status: 400,
|
|
284
|
-
headers: { "Content-Type": "text/html" },
|
|
285
|
-
})
|
|
286
|
-
}
|
|
287
|
-
|
|
288
|
-
const current = pendingOAuth
|
|
289
|
-
pendingOAuth = undefined
|
|
290
|
-
|
|
291
|
-
exchangeCodeForTokens(code, `http://localhost:${OAUTH_PORT}/auth/callback`, current.pkce)
|
|
292
|
-
.then((tokens) => current.resolve(tokens))
|
|
293
|
-
.catch((err) => current.reject(err))
|
|
294
|
-
|
|
295
|
-
return new Response(HTML_SUCCESS, {
|
|
296
|
-
headers: { "Content-Type": "text/html" },
|
|
297
|
-
})
|
|
298
|
-
}
|
|
299
|
-
|
|
300
|
-
if (url.pathname === "/cancel") {
|
|
301
|
-
pendingOAuth?.reject(new Error("Login cancelled"))
|
|
302
|
-
pendingOAuth = undefined
|
|
303
|
-
return new Response("Login cancelled", { status: 200 })
|
|
304
|
-
}
|
|
305
|
-
|
|
306
|
-
return new Response("Not found", { status: 404 })
|
|
307
|
-
},
|
|
308
|
-
})
|
|
309
|
-
|
|
310
|
-
log.info("codex oauth server started", { port: OAUTH_PORT })
|
|
311
|
-
return { port: OAUTH_PORT, redirectUri: `http://localhost:${OAUTH_PORT}/auth/callback` }
|
|
312
|
-
}
|
|
313
|
-
|
|
314
|
-
function stopOAuthServer() {
|
|
315
|
-
if (oauthServer) {
|
|
316
|
-
oauthServer.stop()
|
|
317
|
-
oauthServer = undefined
|
|
318
|
-
log.info("codex oauth server stopped")
|
|
319
|
-
}
|
|
320
|
-
}
|
|
321
|
-
|
|
322
|
-
function waitForOAuthCallback(pkce: PkceCodes, state: string): Promise<TokenResponse> {
|
|
323
|
-
return new Promise((resolve, reject) => {
|
|
324
|
-
const timeout = setTimeout(
|
|
325
|
-
() => {
|
|
326
|
-
if (pendingOAuth) {
|
|
327
|
-
pendingOAuth = undefined
|
|
328
|
-
reject(new Error("OAuth callback timeout - authorization took too long"))
|
|
329
|
-
}
|
|
330
|
-
},
|
|
331
|
-
5 * 60 * 1000,
|
|
332
|
-
) // 5 minute timeout
|
|
333
|
-
|
|
334
|
-
pendingOAuth = {
|
|
335
|
-
pkce,
|
|
336
|
-
state,
|
|
337
|
-
resolve: (tokens) => {
|
|
338
|
-
clearTimeout(timeout)
|
|
339
|
-
resolve(tokens)
|
|
340
|
-
},
|
|
341
|
-
reject: (error) => {
|
|
342
|
-
clearTimeout(timeout)
|
|
343
|
-
reject(error)
|
|
344
|
-
},
|
|
345
|
-
}
|
|
346
|
-
})
|
|
347
|
-
}
|
|
348
|
-
|
|
349
|
-
export async function CodexAuthPlugin(input: PluginInput): Promise<Hooks> {
|
|
350
|
-
return {
|
|
351
|
-
auth: {
|
|
352
|
-
provider: "openai",
|
|
353
|
-
async loader(getAuth, provider) {
|
|
354
|
-
const auth = await getAuth()
|
|
355
|
-
if (auth.type !== "oauth") return {}
|
|
356
|
-
|
|
357
|
-
// Filter models to only allowed Codex models for OAuth
|
|
358
|
-
const allowedModels = new Set([
|
|
359
|
-
"gpt-5.1-codex-max",
|
|
360
|
-
"gpt-5.1-codex-mini",
|
|
361
|
-
"gpt-5.2",
|
|
362
|
-
"gpt-5.2-codex",
|
|
363
|
-
"gpt-5.1-codex",
|
|
364
|
-
])
|
|
365
|
-
for (const modelId of Object.keys(provider.models)) {
|
|
366
|
-
if (!allowedModels.has(modelId)) {
|
|
367
|
-
delete provider.models[modelId]
|
|
368
|
-
}
|
|
369
|
-
}
|
|
370
|
-
|
|
371
|
-
// Zero out costs for Codex (included with ChatGPT subscription)
|
|
372
|
-
for (const model of Object.values(provider.models)) {
|
|
373
|
-
model.cost = {
|
|
374
|
-
input: 0,
|
|
375
|
-
output: 0,
|
|
376
|
-
cache: { read: 0, write: 0 },
|
|
377
|
-
}
|
|
378
|
-
}
|
|
379
|
-
|
|
380
|
-
return {
|
|
381
|
-
apiKey: OAUTH_DUMMY_KEY,
|
|
382
|
-
async fetch(requestInput: RequestInfo | URL, init?: RequestInit) {
|
|
383
|
-
// Remove dummy API key authorization header
|
|
384
|
-
if (init?.headers) {
|
|
385
|
-
if (init.headers instanceof Headers) {
|
|
386
|
-
init.headers.delete("authorization")
|
|
387
|
-
init.headers.delete("Authorization")
|
|
388
|
-
} else if (Array.isArray(init.headers)) {
|
|
389
|
-
init.headers = init.headers.filter(([key]) => key.toLowerCase() !== "authorization")
|
|
390
|
-
} else {
|
|
391
|
-
delete init.headers["authorization"]
|
|
392
|
-
delete init.headers["Authorization"]
|
|
393
|
-
}
|
|
394
|
-
}
|
|
395
|
-
|
|
396
|
-
const currentAuth = await getAuth()
|
|
397
|
-
if (currentAuth.type !== "oauth") return fetch(requestInput, init)
|
|
398
|
-
|
|
399
|
-
// Cast to include accountId field
|
|
400
|
-
const authWithAccount = currentAuth as typeof currentAuth & { accountId?: string }
|
|
401
|
-
|
|
402
|
-
// Check if token needs refresh
|
|
403
|
-
if (!currentAuth.access || currentAuth.expires < Date.now()) {
|
|
404
|
-
log.info("refreshing codex access token")
|
|
405
|
-
const tokens = await refreshAccessToken(currentAuth.refresh)
|
|
406
|
-
const newAccountId = extractAccountId(tokens) || authWithAccount.accountId
|
|
407
|
-
await input.client.auth.set({
|
|
408
|
-
path: { id: "openai" },
|
|
409
|
-
body: {
|
|
410
|
-
type: "oauth",
|
|
411
|
-
refresh: tokens.refresh_token,
|
|
412
|
-
access: tokens.access_token,
|
|
413
|
-
expires: Date.now() + (tokens.expires_in ?? 3600) * 1000,
|
|
414
|
-
...(newAccountId && { accountId: newAccountId }),
|
|
415
|
-
},
|
|
416
|
-
})
|
|
417
|
-
currentAuth.access = tokens.access_token
|
|
418
|
-
authWithAccount.accountId = newAccountId
|
|
419
|
-
}
|
|
420
|
-
|
|
421
|
-
// Build headers
|
|
422
|
-
const headers = new Headers()
|
|
423
|
-
if (init?.headers) {
|
|
424
|
-
if (init.headers instanceof Headers) {
|
|
425
|
-
init.headers.forEach((value, key) => headers.set(key, value))
|
|
426
|
-
} else if (Array.isArray(init.headers)) {
|
|
427
|
-
for (const [key, value] of init.headers) {
|
|
428
|
-
if (value !== undefined) headers.set(key, String(value))
|
|
429
|
-
}
|
|
430
|
-
} else {
|
|
431
|
-
for (const [key, value] of Object.entries(init.headers)) {
|
|
432
|
-
if (value !== undefined) headers.set(key, String(value))
|
|
433
|
-
}
|
|
434
|
-
}
|
|
435
|
-
}
|
|
436
|
-
|
|
437
|
-
// Set authorization header with access token
|
|
438
|
-
headers.set("authorization", `Bearer ${currentAuth.access}`)
|
|
439
|
-
|
|
440
|
-
// Set ChatGPT-Account-Id header for organization subscriptions
|
|
441
|
-
if (authWithAccount.accountId) {
|
|
442
|
-
headers.set("ChatGPT-Account-Id", authWithAccount.accountId)
|
|
443
|
-
}
|
|
444
|
-
|
|
445
|
-
// Rewrite URL to Codex endpoint
|
|
446
|
-
const parsed =
|
|
447
|
-
requestInput instanceof URL
|
|
448
|
-
? requestInput
|
|
449
|
-
: new URL(typeof requestInput === "string" ? requestInput : requestInput.url)
|
|
450
|
-
const url =
|
|
451
|
-
parsed.pathname.includes("/v1/responses") || parsed.pathname.includes("/chat/completions")
|
|
452
|
-
? new URL(CODEX_API_ENDPOINT)
|
|
453
|
-
: parsed
|
|
454
|
-
|
|
455
|
-
return fetch(url, {
|
|
456
|
-
...init,
|
|
457
|
-
headers,
|
|
458
|
-
})
|
|
459
|
-
},
|
|
460
|
-
}
|
|
461
|
-
},
|
|
462
|
-
methods: [
|
|
463
|
-
{
|
|
464
|
-
label: "ChatGPT Pro/Plus",
|
|
465
|
-
type: "oauth",
|
|
466
|
-
authorize: async () => {
|
|
467
|
-
const { redirectUri } = await startOAuthServer()
|
|
468
|
-
const pkce = await generatePKCE()
|
|
469
|
-
const state = generateState()
|
|
470
|
-
const authUrl = buildAuthorizeUrl(redirectUri, pkce, state)
|
|
471
|
-
|
|
472
|
-
const callbackPromise = waitForOAuthCallback(pkce, state)
|
|
473
|
-
|
|
474
|
-
return {
|
|
475
|
-
url: authUrl,
|
|
476
|
-
instructions: "Complete authorization in your browser. This window will close automatically.",
|
|
477
|
-
method: "auto" as const,
|
|
478
|
-
callback: async () => {
|
|
479
|
-
const tokens = await callbackPromise
|
|
480
|
-
stopOAuthServer()
|
|
481
|
-
const accountId = extractAccountId(tokens)
|
|
482
|
-
return {
|
|
483
|
-
type: "success" as const,
|
|
484
|
-
refresh: tokens.refresh_token,
|
|
485
|
-
access: tokens.access_token,
|
|
486
|
-
expires: Date.now() + (tokens.expires_in ?? 3600) * 1000,
|
|
487
|
-
accountId,
|
|
488
|
-
}
|
|
489
|
-
},
|
|
490
|
-
}
|
|
491
|
-
},
|
|
492
|
-
},
|
|
493
|
-
{
|
|
494
|
-
label: "Manually enter API Key",
|
|
495
|
-
type: "api",
|
|
496
|
-
},
|
|
497
|
-
],
|
|
498
|
-
},
|
|
499
|
-
"chat.headers": async (input, output) => {
|
|
500
|
-
if (input.model.providerID !== "openai") return
|
|
501
|
-
output.headers.originator = "opencode"
|
|
502
|
-
output.headers["User-Agent"] = `opencode/${Installation.VERSION} (${os.platform()} ${os.release()}; ${os.arch()})`
|
|
503
|
-
output.headers.session_id = input.sessionID
|
|
504
|
-
},
|
|
505
|
-
}
|
|
506
|
-
}
|