@aexol/spectral 0.7.1 → 0.7.5

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.
Files changed (219) hide show
  1. package/CHANGELOG.md +5 -0
  2. package/dist/agent/agents.js +1 -1
  3. package/dist/agent/index.js +199 -184
  4. package/dist/commands/serve.js +0 -3
  5. package/dist/designer/data/systems/renault/DESIGN.md +1 -1
  6. package/dist/designer/philosophies.js +668 -0
  7. package/dist/mcp/sampling-handler.js +1 -1
  8. package/dist/memory/commands/status.js +1 -1
  9. package/dist/memory/compaction.js +2 -2
  10. package/dist/memory/config.js +1 -1
  11. package/dist/memory/debug-log.js +1 -1
  12. package/dist/memory/hooks/compaction-hook.js +29 -0
  13. package/dist/memory/index.js +2 -0
  14. package/dist/memory/observer.js +2 -2
  15. package/dist/memory/project-observations-store.js +14 -0
  16. package/dist/memory/tokens.js +1 -1
  17. package/dist/memory/tools/read-project-observations.js +82 -0
  18. package/dist/memory/tools/recall-observation.js +2 -2
  19. package/dist/pi/agent-core/agent-loop.js +501 -0
  20. package/dist/pi/agent-core/agent.js +401 -0
  21. package/dist/pi/agent-core/harness/agent-harness.js +899 -0
  22. package/dist/pi/agent-core/harness/compaction/branch-summarization.js +173 -0
  23. package/dist/pi/agent-core/harness/compaction/compaction.js +532 -0
  24. package/dist/pi/agent-core/harness/compaction/utils.js +130 -0
  25. package/dist/pi/agent-core/harness/env/nodejs.js +485 -0
  26. package/dist/pi/agent-core/harness/messages.js +101 -0
  27. package/dist/pi/agent-core/harness/prompt-templates.js +229 -0
  28. package/dist/pi/agent-core/harness/session/jsonl-repo.js +100 -0
  29. package/dist/pi/agent-core/harness/session/jsonl-storage.js +230 -0
  30. package/dist/pi/agent-core/harness/session/memory-repo.js +41 -0
  31. package/dist/pi/agent-core/harness/session/memory-storage.js +113 -0
  32. package/dist/pi/agent-core/harness/session/repo-utils.js +38 -0
  33. package/dist/pi/agent-core/harness/session/session.js +196 -0
  34. package/dist/pi/agent-core/harness/session/uuid.js +49 -0
  35. package/dist/pi/agent-core/harness/skills.js +310 -0
  36. package/dist/pi/agent-core/harness/system-prompt.js +29 -0
  37. package/dist/pi/agent-core/harness/types.js +93 -0
  38. package/dist/pi/agent-core/harness/utils/shell-output.js +125 -0
  39. package/dist/pi/agent-core/harness/utils/truncate.js +289 -0
  40. package/dist/pi/agent-core/index.js +24 -0
  41. package/dist/pi/agent-core/node.js +2 -0
  42. package/dist/pi/agent-core/proxy.js +277 -0
  43. package/dist/pi/agent-core/types.js +1 -0
  44. package/dist/pi/ai/api-registry.js +43 -0
  45. package/dist/pi/ai/cli.js +120 -0
  46. package/dist/pi/ai/env-api-keys.js +169 -0
  47. package/dist/pi/ai/image-models.generated.js +441 -0
  48. package/dist/pi/ai/image-models.js +22 -0
  49. package/dist/pi/ai/images-api-registry.js +21 -0
  50. package/dist/pi/ai/images.js +13 -0
  51. package/dist/pi/ai/index.js +18 -0
  52. package/dist/pi/ai/models.generated.js +16220 -0
  53. package/dist/pi/ai/models.js +70 -0
  54. package/dist/pi/ai/oauth.js +1 -0
  55. package/dist/pi/ai/providers/anthropic.js +945 -0
  56. package/dist/pi/ai/providers/faux.js +367 -0
  57. package/dist/pi/ai/providers/github-copilot-headers.js +28 -0
  58. package/dist/pi/ai/providers/openai-completions.js +945 -0
  59. package/dist/pi/ai/providers/openai-prompt-cache.js +9 -0
  60. package/dist/pi/ai/providers/register-builtins.js +97 -0
  61. package/dist/pi/ai/providers/simple-options.js +40 -0
  62. package/dist/pi/ai/providers/transform-messages.js +183 -0
  63. package/dist/pi/ai/session-resources.js +21 -0
  64. package/dist/pi/ai/stream.js +26 -0
  65. package/dist/pi/ai/types.js +1 -0
  66. package/dist/pi/ai/utils/diagnostics.js +24 -0
  67. package/dist/pi/ai/utils/event-stream.js +80 -0
  68. package/dist/pi/ai/utils/hash.js +13 -0
  69. package/dist/pi/ai/utils/headers.js +7 -0
  70. package/dist/pi/ai/utils/json-parse.js +112 -0
  71. package/dist/pi/ai/utils/node-http-proxy.js +96 -0
  72. package/dist/pi/ai/utils/oauth/anthropic.js +334 -0
  73. package/dist/pi/ai/utils/oauth/device-code.js +54 -0
  74. package/dist/pi/ai/utils/oauth/github-copilot.js +270 -0
  75. package/dist/pi/ai/utils/oauth/index.js +121 -0
  76. package/dist/pi/ai/utils/oauth/oauth-page.js +104 -0
  77. package/dist/pi/ai/utils/oauth/openai-codex.js +384 -0
  78. package/dist/pi/ai/utils/oauth/pkce.js +30 -0
  79. package/dist/pi/ai/utils/oauth/types.js +1 -0
  80. package/dist/pi/ai/utils/overflow.js +150 -0
  81. package/dist/pi/ai/utils/sanitize-unicode.js +25 -0
  82. package/dist/pi/ai/utils/typebox-helpers.js +20 -0
  83. package/dist/pi/ai/utils/validation.js +280 -0
  84. package/dist/pi/coding-agent/bun/cli.js +7 -0
  85. package/dist/pi/coding-agent/bun/restore-sandbox-env.js +31 -0
  86. package/dist/pi/coding-agent/cli/args.js +340 -0
  87. package/dist/pi/coding-agent/cli/file-processor.js +82 -0
  88. package/dist/pi/coding-agent/cli/initial-message.js +21 -0
  89. package/dist/pi/coding-agent/cli.js +17 -0
  90. package/dist/pi/coding-agent/config.js +414 -0
  91. package/dist/pi/coding-agent/core/agent-session-runtime.js +299 -0
  92. package/dist/pi/coding-agent/core/agent-session-services.js +117 -0
  93. package/dist/pi/coding-agent/core/agent-session.js +2498 -0
  94. package/dist/pi/coding-agent/core/auth-guidance.js +20 -0
  95. package/dist/pi/coding-agent/core/auth-storage.js +441 -0
  96. package/dist/pi/coding-agent/core/bash-executor.js +110 -0
  97. package/dist/pi/coding-agent/core/compaction/branch-summarization.js +242 -0
  98. package/dist/pi/coding-agent/core/compaction/compaction.js +624 -0
  99. package/dist/pi/coding-agent/core/compaction/index.js +6 -0
  100. package/dist/pi/coding-agent/core/compaction/utils.js +152 -0
  101. package/dist/pi/coding-agent/core/defaults.js +1 -0
  102. package/dist/pi/coding-agent/core/diagnostics.js +1 -0
  103. package/dist/pi/coding-agent/core/event-bus.js +24 -0
  104. package/dist/pi/coding-agent/core/exec.js +74 -0
  105. package/dist/pi/coding-agent/core/export-html/ansi-to-html.js +248 -0
  106. package/dist/pi/coding-agent/core/export-html/index.js +225 -0
  107. package/dist/pi/coding-agent/core/export-html/tool-renderer.js +107 -0
  108. package/dist/pi/coding-agent/core/extensions/index.js +8 -0
  109. package/dist/pi/coding-agent/core/extensions/loader.js +485 -0
  110. package/dist/pi/coding-agent/core/extensions/runner.js +824 -0
  111. package/dist/pi/coding-agent/core/extensions/types.js +44 -0
  112. package/dist/pi/coding-agent/core/extensions/wrapper.js +21 -0
  113. package/dist/pi/coding-agent/core/footer-data-provider.js +309 -0
  114. package/dist/pi/coding-agent/core/http-dispatcher.js +47 -0
  115. package/dist/pi/coding-agent/core/index.js +11 -0
  116. package/dist/pi/coding-agent/core/keybindings.js +294 -0
  117. package/dist/pi/coding-agent/core/messages.js +122 -0
  118. package/dist/pi/coding-agent/core/model-registry.js +728 -0
  119. package/dist/pi/coding-agent/core/model-resolver.js +494 -0
  120. package/dist/pi/coding-agent/core/output-guard.js +58 -0
  121. package/dist/pi/coding-agent/core/package-manager.js +2020 -0
  122. package/dist/pi/coding-agent/core/prompt-templates.js +237 -0
  123. package/dist/pi/coding-agent/core/provider-display-names.js +32 -0
  124. package/dist/pi/coding-agent/core/resolve-config-value.js +125 -0
  125. package/dist/pi/coding-agent/core/resource-loader.js +733 -0
  126. package/dist/pi/coding-agent/core/sdk.js +282 -0
  127. package/dist/pi/coding-agent/core/session-cwd.js +37 -0
  128. package/dist/pi/coding-agent/core/session-manager.js +1146 -0
  129. package/dist/pi/coding-agent/core/settings-manager.js +794 -0
  130. package/dist/pi/coding-agent/core/skills.js +386 -0
  131. package/dist/pi/coding-agent/core/slash-commands.js +24 -0
  132. package/dist/pi/coding-agent/core/source-info.js +18 -0
  133. package/dist/pi/coding-agent/core/system-prompt.js +122 -0
  134. package/dist/pi/coding-agent/core/telemetry.js +8 -0
  135. package/dist/pi/coding-agent/core/timings.js +30 -0
  136. package/dist/pi/coding-agent/core/tools/bash.js +341 -0
  137. package/dist/pi/coding-agent/core/tools/edit-diff.js +344 -0
  138. package/dist/pi/coding-agent/core/tools/edit.js +324 -0
  139. package/dist/pi/coding-agent/core/tools/file-mutation-queue.js +36 -0
  140. package/dist/pi/coding-agent/core/tools/find.js +297 -0
  141. package/dist/pi/coding-agent/core/tools/grep.js +303 -0
  142. package/dist/pi/coding-agent/core/tools/index.js +111 -0
  143. package/dist/pi/coding-agent/core/tools/ls.js +168 -0
  144. package/dist/pi/coding-agent/core/tools/output-accumulator.js +183 -0
  145. package/dist/pi/coding-agent/core/tools/path-utils.js +61 -0
  146. package/dist/pi/coding-agent/core/tools/read.js +288 -0
  147. package/dist/pi/coding-agent/core/tools/render-utils.js +48 -0
  148. package/dist/pi/coding-agent/core/tools/tool-definition-wrapper.js +33 -0
  149. package/dist/pi/coding-agent/core/tools/truncate.js +214 -0
  150. package/dist/pi/coding-agent/core/tools/write.js +212 -0
  151. package/dist/pi/coding-agent/index.js +41 -0
  152. package/dist/pi/coding-agent/main.js +5 -0
  153. package/dist/pi/coding-agent/migrations.js +280 -0
  154. package/dist/pi/coding-agent/modes/index.js +7 -0
  155. package/dist/pi/coding-agent/modes/interactive/components/diff.js +132 -0
  156. package/dist/pi/coding-agent/modes/interactive/components/keybinding-hints.js +35 -0
  157. package/dist/pi/coding-agent/modes/interactive/components/visual-truncate.js +32 -0
  158. package/dist/pi/coding-agent/modes/interactive/interactive-mode.js +3 -0
  159. package/dist/pi/coding-agent/modes/interactive/theme/theme.js +1023 -0
  160. package/dist/pi/coding-agent/modes/print-mode.js +130 -0
  161. package/dist/pi/coding-agent/modes/rpc/jsonl.js +48 -0
  162. package/dist/pi/coding-agent/modes/rpc/rpc-client.js +409 -0
  163. package/dist/pi/coding-agent/modes/rpc/rpc-mode.js +600 -0
  164. package/dist/pi/coding-agent/modes/rpc/rpc-types.js +7 -0
  165. package/dist/pi/coding-agent/utils/ansi.js +51 -0
  166. package/dist/pi/coding-agent/utils/changelog.js +86 -0
  167. package/dist/pi/coding-agent/utils/child-process.js +87 -0
  168. package/dist/pi/coding-agent/utils/clipboard-image.js +244 -0
  169. package/dist/pi/coding-agent/utils/clipboard-native.js +13 -0
  170. package/dist/pi/coding-agent/utils/clipboard.js +116 -0
  171. package/dist/pi/coding-agent/utils/exif-orientation.js +157 -0
  172. package/dist/pi/coding-agent/utils/frontmatter.js +25 -0
  173. package/dist/pi/coding-agent/utils/fs-watch.js +24 -0
  174. package/dist/pi/coding-agent/utils/git.js +162 -0
  175. package/dist/pi/coding-agent/utils/html.js +39 -0
  176. package/dist/pi/coding-agent/utils/image-convert.js +38 -0
  177. package/dist/pi/coding-agent/utils/image-resize.js +136 -0
  178. package/dist/pi/coding-agent/utils/mime.js +68 -0
  179. package/dist/pi/coding-agent/utils/paths.js +91 -0
  180. package/dist/pi/coding-agent/utils/photon.js +120 -0
  181. package/dist/pi/coding-agent/utils/pi-user-agent.js +4 -0
  182. package/dist/pi/coding-agent/utils/shell.js +194 -0
  183. package/dist/pi/coding-agent/utils/sleep.js +16 -0
  184. package/dist/pi/coding-agent/utils/syntax-highlight.js +117 -0
  185. package/dist/pi/coding-agent/utils/tools-manager.js +327 -0
  186. package/dist/pi/coding-agent/utils/version-check.js +81 -0
  187. package/dist/pi/coding-agent/utils/windows-self-update.js +76 -0
  188. package/dist/pi/tui/autocomplete.js +631 -0
  189. package/dist/pi/tui/components/box.js +103 -0
  190. package/dist/pi/tui/components/cancellable-loader.js +34 -0
  191. package/dist/pi/tui/components/editor.js +1915 -0
  192. package/dist/pi/tui/components/image.js +88 -0
  193. package/dist/pi/tui/components/input.js +425 -0
  194. package/dist/pi/tui/components/loader.js +68 -0
  195. package/dist/pi/tui/components/markdown.js +633 -0
  196. package/dist/pi/tui/components/select-list.js +158 -0
  197. package/dist/pi/tui/components/settings-list.js +184 -0
  198. package/dist/pi/tui/components/spacer.js +22 -0
  199. package/dist/pi/tui/components/text.js +88 -0
  200. package/dist/pi/tui/components/truncated-text.js +50 -0
  201. package/dist/pi/tui/editor-component.js +1 -0
  202. package/dist/pi/tui/fuzzy.js +109 -0
  203. package/dist/pi/tui/index.js +31 -0
  204. package/dist/pi/tui/keybindings.js +173 -0
  205. package/dist/pi/tui/keys.js +1172 -0
  206. package/dist/pi/tui/kill-ring.js +43 -0
  207. package/dist/pi/tui/stdin-buffer.js +360 -0
  208. package/dist/pi/tui/terminal-image.js +335 -0
  209. package/dist/pi/tui/terminal.js +324 -0
  210. package/dist/pi/tui/tui.js +1076 -0
  211. package/dist/pi/tui/undo-stack.js +24 -0
  212. package/dist/pi/tui/utils.js +1016 -0
  213. package/dist/relay/dispatcher.js +30 -0
  214. package/dist/server/handlers/queue.js +52 -0
  215. package/dist/server/pi-bridge.js +9 -1
  216. package/dist/server/session-stream.js +76 -111
  217. package/dist/server/storage.js +154 -2
  218. package/dist/server/title-generator.js +14 -153
  219. package/package.json +24 -6
@@ -0,0 +1,96 @@
1
+ import { HttpProxyAgent } from "http-proxy-agent";
2
+ import { HttpsProxyAgent } from "https-proxy-agent";
3
+ const DEFAULT_PROXY_PORTS = {
4
+ ftp: 21,
5
+ gopher: 70,
6
+ http: 80,
7
+ https: 443,
8
+ ws: 80,
9
+ wss: 443,
10
+ };
11
+ export const UNSUPPORTED_PROXY_PROTOCOL_MESSAGE = "Unsupported proxy protocol. SOCKS and PAC proxy URLs are not supported; use an HTTP or HTTPS proxy URL.";
12
+ function getProxyEnv(key) {
13
+ return process.env[key.toLowerCase()] || process.env[key.toUpperCase()] || "";
14
+ }
15
+ function parseProxyTargetUrl(targetUrl) {
16
+ if (targetUrl instanceof URL) {
17
+ return targetUrl;
18
+ }
19
+ try {
20
+ return new URL(targetUrl);
21
+ }
22
+ catch {
23
+ return undefined;
24
+ }
25
+ }
26
+ function shouldProxyHostname(hostname, port) {
27
+ const noProxy = getProxyEnv("no_proxy").toLowerCase();
28
+ if (!noProxy) {
29
+ return true;
30
+ }
31
+ if (noProxy === "*") {
32
+ return false;
33
+ }
34
+ return noProxy.split(/[,\s]/).every((proxy) => {
35
+ if (!proxy) {
36
+ return true;
37
+ }
38
+ const parsedProxy = proxy.match(/^(.+):(\d+)$/);
39
+ let proxyHostname = parsedProxy ? parsedProxy[1] : proxy;
40
+ const proxyPort = parsedProxy ? Number.parseInt(parsedProxy[2], 10) : 0;
41
+ if (proxyPort && proxyPort !== port) {
42
+ return true;
43
+ }
44
+ if (!/^[.*]/.test(proxyHostname)) {
45
+ return hostname !== proxyHostname;
46
+ }
47
+ if (proxyHostname.startsWith("*")) {
48
+ proxyHostname = proxyHostname.slice(1);
49
+ }
50
+ return !hostname.endsWith(proxyHostname);
51
+ });
52
+ }
53
+ function getProxyForUrl(targetUrl) {
54
+ const parsedUrl = parseProxyTargetUrl(targetUrl);
55
+ if (!parsedUrl?.protocol || !parsedUrl.host) {
56
+ return "";
57
+ }
58
+ const protocol = parsedUrl.protocol.split(":", 1)[0];
59
+ const hostname = parsedUrl.host.replace(/:\d*$/, "");
60
+ const port = Number.parseInt(parsedUrl.port, 10) || DEFAULT_PROXY_PORTS[protocol] || 0;
61
+ if (!shouldProxyHostname(hostname, port)) {
62
+ return "";
63
+ }
64
+ let proxy = getProxyEnv(`${protocol}_proxy`) || getProxyEnv("all_proxy");
65
+ if (proxy && !proxy.includes("://")) {
66
+ proxy = `${protocol}://${proxy}`;
67
+ }
68
+ return proxy;
69
+ }
70
+ export function resolveHttpProxyUrlForTarget(targetUrl) {
71
+ const proxy = getProxyForUrl(targetUrl);
72
+ if (!proxy) {
73
+ return undefined;
74
+ }
75
+ let proxyUrl;
76
+ try {
77
+ proxyUrl = new URL(proxy);
78
+ }
79
+ catch (error) {
80
+ throw new Error(`Invalid proxy URL ${JSON.stringify(proxy)}: ${error instanceof Error ? error.message : String(error)}`);
81
+ }
82
+ if (proxyUrl.protocol !== "http:" && proxyUrl.protocol !== "https:") {
83
+ throw new Error(`${UNSUPPORTED_PROXY_PROTOCOL_MESSAGE} Got ${proxyUrl.protocol}`);
84
+ }
85
+ return proxyUrl;
86
+ }
87
+ export function createHttpProxyAgentsForTarget(targetUrl) {
88
+ const proxyUrl = resolveHttpProxyUrlForTarget(targetUrl);
89
+ if (!proxyUrl) {
90
+ return undefined;
91
+ }
92
+ return {
93
+ httpAgent: new HttpProxyAgent(proxyUrl),
94
+ httpsAgent: new HttpsProxyAgent(proxyUrl),
95
+ };
96
+ }
@@ -0,0 +1,334 @@
1
+ /**
2
+ * Anthropic OAuth flow (Claude Pro/Max)
3
+ *
4
+ * NOTE: This module uses Node.js http.createServer for the OAuth callback server.
5
+ * It is only intended for CLI use, not browser environments.
6
+ */
7
+ import { oauthErrorHtml, oauthSuccessHtml } from "./oauth-page.js";
8
+ import { generatePKCE } from "./pkce.js";
9
+ let nodeApis = null;
10
+ let nodeApisPromise = null;
11
+ const decode = (s) => atob(s);
12
+ const CLIENT_ID = decode("OWQxYzI1MGEtZTYxYi00NGQ5LTg4ZWQtNTk0NGQxOTYyZjVl");
13
+ const AUTHORIZE_URL = "https://claude.ai/oauth/authorize";
14
+ const TOKEN_URL = "https://platform.claude.com/v1/oauth/token";
15
+ const CALLBACK_HOST = process.env.PI_OAUTH_CALLBACK_HOST || "127.0.0.1";
16
+ const CALLBACK_PORT = 53692;
17
+ const CALLBACK_PATH = "/callback";
18
+ const REDIRECT_URI = `http://localhost:${CALLBACK_PORT}${CALLBACK_PATH}`;
19
+ const SCOPES = "org:create_api_key user:profile user:inference user:sessions:claude_code user:mcp_servers user:file_upload";
20
+ async function getNodeApis() {
21
+ if (nodeApis)
22
+ return nodeApis;
23
+ if (!nodeApisPromise) {
24
+ if (typeof process === "undefined" || (!process.versions?.node && !process.versions?.bun)) {
25
+ throw new Error("Anthropic OAuth is only available in Node.js environments");
26
+ }
27
+ nodeApisPromise = import("node:http").then((httpModule) => ({
28
+ createServer: httpModule.createServer,
29
+ }));
30
+ }
31
+ nodeApis = await nodeApisPromise;
32
+ return nodeApis;
33
+ }
34
+ function parseAuthorizationInput(input) {
35
+ const value = input.trim();
36
+ if (!value)
37
+ return {};
38
+ try {
39
+ const url = new URL(value);
40
+ return {
41
+ code: url.searchParams.get("code") ?? undefined,
42
+ state: url.searchParams.get("state") ?? undefined,
43
+ };
44
+ }
45
+ catch {
46
+ // not a URL
47
+ }
48
+ if (value.includes("#")) {
49
+ const [code, state] = value.split("#", 2);
50
+ return { code, state };
51
+ }
52
+ if (value.includes("code=")) {
53
+ const params = new URLSearchParams(value);
54
+ return {
55
+ code: params.get("code") ?? undefined,
56
+ state: params.get("state") ?? undefined,
57
+ };
58
+ }
59
+ return { code: value };
60
+ }
61
+ function formatErrorDetails(error) {
62
+ if (error instanceof Error) {
63
+ const details = [`${error.name}: ${error.message}`];
64
+ const errorWithCode = error;
65
+ if (errorWithCode.code)
66
+ details.push(`code=${errorWithCode.code}`);
67
+ if (typeof errorWithCode.errno !== "undefined")
68
+ details.push(`errno=${String(errorWithCode.errno)}`);
69
+ if (typeof error.cause !== "undefined") {
70
+ details.push(`cause=${formatErrorDetails(error.cause)}`);
71
+ }
72
+ if (error.stack) {
73
+ details.push(`stack=${error.stack}`);
74
+ }
75
+ return details.join("; ");
76
+ }
77
+ return String(error);
78
+ }
79
+ async function startCallbackServer(expectedState) {
80
+ const { createServer } = await getNodeApis();
81
+ return new Promise((resolve, reject) => {
82
+ let settleWait;
83
+ const waitForCodePromise = new Promise((resolveWait) => {
84
+ let settled = false;
85
+ settleWait = (value) => {
86
+ if (settled)
87
+ return;
88
+ settled = true;
89
+ resolveWait(value);
90
+ };
91
+ });
92
+ const server = createServer((req, res) => {
93
+ try {
94
+ const url = new URL(req.url || "", "http://localhost");
95
+ if (url.pathname !== CALLBACK_PATH) {
96
+ res.writeHead(404, { "Content-Type": "text/html; charset=utf-8" });
97
+ res.end(oauthErrorHtml("Callback route not found."));
98
+ return;
99
+ }
100
+ const code = url.searchParams.get("code");
101
+ const state = url.searchParams.get("state");
102
+ const error = url.searchParams.get("error");
103
+ if (error) {
104
+ res.writeHead(400, { "Content-Type": "text/html; charset=utf-8" });
105
+ res.end(oauthErrorHtml("Anthropic authentication did not complete.", `Error: ${error}`));
106
+ return;
107
+ }
108
+ if (!code || !state) {
109
+ res.writeHead(400, { "Content-Type": "text/html; charset=utf-8" });
110
+ res.end(oauthErrorHtml("Missing code or state parameter."));
111
+ return;
112
+ }
113
+ if (state !== expectedState) {
114
+ res.writeHead(400, { "Content-Type": "text/html; charset=utf-8" });
115
+ res.end(oauthErrorHtml("State mismatch."));
116
+ return;
117
+ }
118
+ res.writeHead(200, { "Content-Type": "text/html; charset=utf-8" });
119
+ res.end(oauthSuccessHtml("Anthropic authentication completed. You can close this window."));
120
+ settleWait?.({ code, state });
121
+ }
122
+ catch {
123
+ res.writeHead(500, { "Content-Type": "text/plain; charset=utf-8" });
124
+ res.end("Internal error");
125
+ }
126
+ });
127
+ server.on("error", (err) => {
128
+ reject(err);
129
+ });
130
+ server.listen(CALLBACK_PORT, CALLBACK_HOST, () => {
131
+ resolve({
132
+ server,
133
+ redirectUri: REDIRECT_URI,
134
+ cancelWait: () => {
135
+ settleWait?.(null);
136
+ },
137
+ waitForCode: () => waitForCodePromise,
138
+ });
139
+ });
140
+ });
141
+ }
142
+ async function postJson(url, body) {
143
+ const response = await fetch(url, {
144
+ method: "POST",
145
+ headers: {
146
+ "Content-Type": "application/json",
147
+ Accept: "application/json",
148
+ },
149
+ body: JSON.stringify(body),
150
+ signal: AbortSignal.timeout(30_000),
151
+ });
152
+ const responseBody = await response.text();
153
+ if (!response.ok) {
154
+ throw new Error(`HTTP request failed. status=${response.status}; url=${url}; body=${responseBody}`);
155
+ }
156
+ return responseBody;
157
+ }
158
+ async function exchangeAuthorizationCode(code, state, verifier, redirectUri) {
159
+ let responseBody;
160
+ try {
161
+ responseBody = await postJson(TOKEN_URL, {
162
+ grant_type: "authorization_code",
163
+ client_id: CLIENT_ID,
164
+ code,
165
+ state,
166
+ redirect_uri: redirectUri,
167
+ code_verifier: verifier,
168
+ });
169
+ }
170
+ catch (error) {
171
+ throw new Error(`Token exchange request failed. url=${TOKEN_URL}; redirect_uri=${redirectUri}; response_type=authorization_code; details=${formatErrorDetails(error)}`);
172
+ }
173
+ let tokenData;
174
+ try {
175
+ tokenData = JSON.parse(responseBody);
176
+ }
177
+ catch (error) {
178
+ throw new Error(`Token exchange returned invalid JSON. url=${TOKEN_URL}; body=${responseBody}; details=${formatErrorDetails(error)}`);
179
+ }
180
+ return {
181
+ refresh: tokenData.refresh_token,
182
+ access: tokenData.access_token,
183
+ expires: Date.now() + tokenData.expires_in * 1000 - 5 * 60 * 1000,
184
+ };
185
+ }
186
+ /**
187
+ * Login with Anthropic OAuth (authorization code + PKCE)
188
+ */
189
+ export async function loginAnthropic(options) {
190
+ const { verifier, challenge } = await generatePKCE();
191
+ const server = await startCallbackServer(verifier);
192
+ let code;
193
+ let state;
194
+ let redirectUriForExchange = REDIRECT_URI;
195
+ try {
196
+ const authParams = new URLSearchParams({
197
+ code: "true",
198
+ client_id: CLIENT_ID,
199
+ response_type: "code",
200
+ redirect_uri: REDIRECT_URI,
201
+ scope: SCOPES,
202
+ code_challenge: challenge,
203
+ code_challenge_method: "S256",
204
+ state: verifier,
205
+ });
206
+ options.onAuth({
207
+ url: `${AUTHORIZE_URL}?${authParams.toString()}`,
208
+ instructions: "Complete login in your browser. If the browser is on another machine, paste the final redirect URL here.",
209
+ });
210
+ if (options.onManualCodeInput) {
211
+ let manualInput;
212
+ let manualError;
213
+ const manualPromise = options
214
+ .onManualCodeInput()
215
+ .then((input) => {
216
+ manualInput = input;
217
+ server.cancelWait();
218
+ })
219
+ .catch((err) => {
220
+ manualError = err instanceof Error ? err : new Error(String(err));
221
+ server.cancelWait();
222
+ });
223
+ const result = await server.waitForCode();
224
+ if (manualError) {
225
+ throw manualError;
226
+ }
227
+ if (result?.code) {
228
+ code = result.code;
229
+ state = result.state;
230
+ redirectUriForExchange = REDIRECT_URI;
231
+ }
232
+ else if (manualInput) {
233
+ const parsed = parseAuthorizationInput(manualInput);
234
+ if (parsed.state && parsed.state !== verifier) {
235
+ throw new Error("OAuth state mismatch");
236
+ }
237
+ code = parsed.code;
238
+ state = parsed.state ?? verifier;
239
+ }
240
+ if (!code) {
241
+ await manualPromise;
242
+ if (manualError) {
243
+ throw manualError;
244
+ }
245
+ if (manualInput) {
246
+ const parsed = parseAuthorizationInput(manualInput);
247
+ if (parsed.state && parsed.state !== verifier) {
248
+ throw new Error("OAuth state mismatch");
249
+ }
250
+ code = parsed.code;
251
+ state = parsed.state ?? verifier;
252
+ }
253
+ }
254
+ }
255
+ else {
256
+ const result = await server.waitForCode();
257
+ if (result?.code) {
258
+ code = result.code;
259
+ state = result.state;
260
+ redirectUriForExchange = REDIRECT_URI;
261
+ }
262
+ }
263
+ if (!code) {
264
+ const input = await options.onPrompt({
265
+ message: "Paste the authorization code or full redirect URL:",
266
+ placeholder: REDIRECT_URI,
267
+ });
268
+ const parsed = parseAuthorizationInput(input);
269
+ if (parsed.state && parsed.state !== verifier) {
270
+ throw new Error("OAuth state mismatch");
271
+ }
272
+ code = parsed.code;
273
+ state = parsed.state ?? verifier;
274
+ }
275
+ if (!code) {
276
+ throw new Error("Missing authorization code");
277
+ }
278
+ if (!state) {
279
+ throw new Error("Missing OAuth state");
280
+ }
281
+ options.onProgress?.("Exchanging authorization code for tokens...");
282
+ return exchangeAuthorizationCode(code, state, verifier, redirectUriForExchange);
283
+ }
284
+ finally {
285
+ server.server.close();
286
+ }
287
+ }
288
+ /**
289
+ * Refresh Anthropic OAuth token
290
+ */
291
+ export async function refreshAnthropicToken(refreshToken) {
292
+ let responseBody;
293
+ try {
294
+ responseBody = await postJson(TOKEN_URL, {
295
+ grant_type: "refresh_token",
296
+ client_id: CLIENT_ID,
297
+ refresh_token: refreshToken,
298
+ });
299
+ }
300
+ catch (error) {
301
+ throw new Error(`Anthropic token refresh request failed. url=${TOKEN_URL}; details=${formatErrorDetails(error)}`);
302
+ }
303
+ let data;
304
+ try {
305
+ data = JSON.parse(responseBody);
306
+ }
307
+ catch (error) {
308
+ throw new Error(`Anthropic token refresh returned invalid JSON. url=${TOKEN_URL}; body=${responseBody}; details=${formatErrorDetails(error)}`);
309
+ }
310
+ return {
311
+ refresh: data.refresh_token,
312
+ access: data.access_token,
313
+ expires: Date.now() + data.expires_in * 1000 - 5 * 60 * 1000,
314
+ };
315
+ }
316
+ export const anthropicOAuthProvider = {
317
+ id: "anthropic",
318
+ name: "Anthropic (Claude Pro/Max)",
319
+ usesCallbackServer: true,
320
+ async login(callbacks) {
321
+ return loginAnthropic({
322
+ onAuth: callbacks.onAuth,
323
+ onPrompt: callbacks.onPrompt,
324
+ onProgress: callbacks.onProgress,
325
+ onManualCodeInput: callbacks.onManualCodeInput,
326
+ });
327
+ },
328
+ async refreshToken(credentials) {
329
+ return refreshAnthropicToken(credentials.refresh);
330
+ },
331
+ getApiKey(credentials) {
332
+ return credentials.access;
333
+ },
334
+ };
@@ -0,0 +1,54 @@
1
+ const CANCEL_MESSAGE = "Login cancelled";
2
+ const TIMEOUT_MESSAGE = "Device flow timed out";
3
+ const SLOW_DOWN_TIMEOUT_MESSAGE = "Device flow timed out after one or more slow_down responses. This is often caused by clock drift in WSL or VM environments. Please sync or restart the VM clock and try again.";
4
+ const MINIMUM_INTERVAL_MS = 1000;
5
+ // RFC 8628 section 3.2: if the authorization server omits `interval`, the client must use 5 seconds.
6
+ const DEFAULT_POLL_INTERVAL_SECONDS = 5;
7
+ // RFC 8628 section 3.5: `slow_down` means the polling interval must increase by 5 seconds.
8
+ const SLOW_DOWN_INTERVAL_INCREMENT_MS = 5000;
9
+ function abortableSleep(ms, signal, cancelMessage) {
10
+ return new Promise((resolve, reject) => {
11
+ if (signal?.aborted) {
12
+ reject(new Error(cancelMessage));
13
+ return;
14
+ }
15
+ const onAbort = () => {
16
+ clearTimeout(timeout);
17
+ reject(new Error(cancelMessage));
18
+ };
19
+ const timeout = setTimeout(() => {
20
+ signal?.removeEventListener("abort", onAbort);
21
+ resolve();
22
+ }, ms);
23
+ signal?.addEventListener("abort", onAbort, { once: true });
24
+ });
25
+ }
26
+ export async function pollOAuthDeviceCodeFlow(options) {
27
+ const deadline = typeof options.expiresInSeconds === "number"
28
+ ? Date.now() + options.expiresInSeconds * 1000
29
+ : Number.POSITIVE_INFINITY;
30
+ let intervalMs = Math.max(MINIMUM_INTERVAL_MS, Math.floor((options.intervalSeconds ?? DEFAULT_POLL_INTERVAL_SECONDS) * 1000));
31
+ let slowDownResponses = 0;
32
+ while (Date.now() < deadline) {
33
+ if (options.signal?.aborted) {
34
+ throw new Error(CANCEL_MESSAGE);
35
+ }
36
+ const remainingMs = deadline - Date.now();
37
+ await abortableSleep(Math.min(intervalMs, remainingMs), options.signal, CANCEL_MESSAGE);
38
+ const result = await options.poll();
39
+ if (result.status === "complete") {
40
+ return result.accessToken;
41
+ }
42
+ if (result.status === "pending") {
43
+ continue;
44
+ }
45
+ if (result.status === "slow_down") {
46
+ slowDownResponses += 1;
47
+ // RFC 8628 section 3.5: apply this increase to this and all subsequent requests.
48
+ intervalMs = Math.max(MINIMUM_INTERVAL_MS, intervalMs + SLOW_DOWN_INTERVAL_INCREMENT_MS);
49
+ continue;
50
+ }
51
+ throw new Error(result.message);
52
+ }
53
+ throw new Error(slowDownResponses > 0 ? SLOW_DOWN_TIMEOUT_MESSAGE : TIMEOUT_MESSAGE);
54
+ }