mirror of
https://github.com/anomalyco/opencode.git
synced 2026-04-23 22:34:53 +00:00
Use Node's createServer for MCP OAuth callback and Codex plugin OAuth servers instead of Bun.serve, making them work under Node.js.
196 lines
6.1 KiB
TypeScript
196 lines
6.1 KiB
TypeScript
import { createConnection } from "net"
|
|
import { createServer } from "http"
|
|
import { Log } from "../util/log"
|
|
import { OAUTH_CALLBACK_PORT, OAUTH_CALLBACK_PATH } from "./oauth-provider"
|
|
|
|
const log = Log.create({ service: "mcp.oauth-callback" })
|
|
|
|
const HTML_SUCCESS = `<!DOCTYPE html>
|
|
<html>
|
|
<head>
|
|
<title>OpenCode - Authorization Successful</title>
|
|
<style>
|
|
body { font-family: system-ui, -apple-system, sans-serif; display: flex; justify-content: center; align-items: center; height: 100vh; margin: 0; background: #1a1a2e; color: #eee; }
|
|
.container { text-align: center; padding: 2rem; }
|
|
h1 { color: #4ade80; margin-bottom: 1rem; }
|
|
p { color: #aaa; }
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<div class="container">
|
|
<h1>Authorization Successful</h1>
|
|
<p>You can close this window and return to OpenCode.</p>
|
|
</div>
|
|
<script>setTimeout(() => window.close(), 2000);</script>
|
|
</body>
|
|
</html>`
|
|
|
|
const HTML_ERROR = (error: string) => `<!DOCTYPE html>
|
|
<html>
|
|
<head>
|
|
<title>OpenCode - Authorization Failed</title>
|
|
<style>
|
|
body { font-family: system-ui, -apple-system, sans-serif; display: flex; justify-content: center; align-items: center; height: 100vh; margin: 0; background: #1a1a2e; color: #eee; }
|
|
.container { text-align: center; padding: 2rem; }
|
|
h1 { color: #f87171; margin-bottom: 1rem; }
|
|
p { color: #aaa; }
|
|
.error { color: #fca5a5; font-family: monospace; margin-top: 1rem; padding: 1rem; background: rgba(248,113,113,0.1); border-radius: 0.5rem; }
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<div class="container">
|
|
<h1>Authorization Failed</h1>
|
|
<p>An error occurred during authorization.</p>
|
|
<div class="error">${error}</div>
|
|
</div>
|
|
</body>
|
|
</html>`
|
|
|
|
interface PendingAuth {
|
|
resolve: (code: string) => void
|
|
reject: (error: Error) => void
|
|
timeout: ReturnType<typeof setTimeout>
|
|
}
|
|
|
|
export namespace McpOAuthCallback {
|
|
let server: ReturnType<typeof createServer> | undefined
|
|
const pendingAuths = new Map<string, PendingAuth>()
|
|
|
|
const CALLBACK_TIMEOUT_MS = 5 * 60 * 1000 // 5 minutes
|
|
|
|
function handleRequest(req: import("http").IncomingMessage, res: import("http").ServerResponse) {
|
|
const url = new URL(req.url || "/", `http://localhost:${OAUTH_CALLBACK_PORT}`)
|
|
|
|
if (url.pathname !== OAUTH_CALLBACK_PATH) {
|
|
res.writeHead(404)
|
|
res.end("Not found")
|
|
return
|
|
}
|
|
|
|
const code = url.searchParams.get("code")
|
|
const state = url.searchParams.get("state")
|
|
const error = url.searchParams.get("error")
|
|
const errorDescription = url.searchParams.get("error_description")
|
|
|
|
log.info("received oauth callback", { hasCode: !!code, state, error })
|
|
|
|
// Enforce state parameter presence
|
|
if (!state) {
|
|
const errorMsg = "Missing required state parameter - potential CSRF attack"
|
|
log.error("oauth callback missing state parameter", { url: url.toString() })
|
|
res.writeHead(400, { "Content-Type": "text/html" })
|
|
res.end(HTML_ERROR(errorMsg))
|
|
return
|
|
}
|
|
|
|
if (error) {
|
|
const errorMsg = errorDescription || error
|
|
if (pendingAuths.has(state)) {
|
|
const pending = pendingAuths.get(state)!
|
|
clearTimeout(pending.timeout)
|
|
pendingAuths.delete(state)
|
|
pending.reject(new Error(errorMsg))
|
|
}
|
|
res.writeHead(200, { "Content-Type": "text/html" })
|
|
res.end(HTML_ERROR(errorMsg))
|
|
return
|
|
}
|
|
|
|
if (!code) {
|
|
res.writeHead(400, { "Content-Type": "text/html" })
|
|
res.end(HTML_ERROR("No authorization code provided"))
|
|
return
|
|
}
|
|
|
|
// Validate state parameter
|
|
if (!pendingAuths.has(state)) {
|
|
const errorMsg = "Invalid or expired state parameter - potential CSRF attack"
|
|
log.error("oauth callback with invalid state", { state, pendingStates: Array.from(pendingAuths.keys()) })
|
|
res.writeHead(400, { "Content-Type": "text/html" })
|
|
res.end(HTML_ERROR(errorMsg))
|
|
return
|
|
}
|
|
|
|
const pending = pendingAuths.get(state)!
|
|
|
|
clearTimeout(pending.timeout)
|
|
pendingAuths.delete(state)
|
|
pending.resolve(code)
|
|
|
|
res.writeHead(200, { "Content-Type": "text/html" })
|
|
res.end(HTML_SUCCESS)
|
|
}
|
|
|
|
export async function ensureRunning(): Promise<void> {
|
|
if (server) return
|
|
|
|
const running = await isPortInUse()
|
|
if (running) {
|
|
log.info("oauth callback server already running on another instance", { port: OAUTH_CALLBACK_PORT })
|
|
return
|
|
}
|
|
|
|
server = createServer(handleRequest)
|
|
await new Promise<void>((resolve, reject) => {
|
|
server!.listen(OAUTH_CALLBACK_PORT, () => {
|
|
log.info("oauth callback server started", { port: OAUTH_CALLBACK_PORT })
|
|
resolve()
|
|
})
|
|
server!.on("error", reject)
|
|
})
|
|
}
|
|
|
|
export function waitForCallback(oauthState: string): Promise<string> {
|
|
return new Promise((resolve, reject) => {
|
|
const timeout = setTimeout(() => {
|
|
if (pendingAuths.has(oauthState)) {
|
|
pendingAuths.delete(oauthState)
|
|
reject(new Error("OAuth callback timeout - authorization took too long"))
|
|
}
|
|
}, CALLBACK_TIMEOUT_MS)
|
|
|
|
pendingAuths.set(oauthState, { resolve, reject, timeout })
|
|
})
|
|
}
|
|
|
|
export function cancelPending(mcpName: string): void {
|
|
const pending = pendingAuths.get(mcpName)
|
|
if (pending) {
|
|
clearTimeout(pending.timeout)
|
|
pendingAuths.delete(mcpName)
|
|
pending.reject(new Error("Authorization cancelled"))
|
|
}
|
|
}
|
|
|
|
export async function isPortInUse(): Promise<boolean> {
|
|
return new Promise((resolve) => {
|
|
const socket = createConnection(OAUTH_CALLBACK_PORT, "127.0.0.1")
|
|
socket.on("connect", () => {
|
|
socket.destroy()
|
|
resolve(true)
|
|
})
|
|
socket.on("error", () => {
|
|
resolve(false)
|
|
})
|
|
})
|
|
}
|
|
|
|
export async function stop(): Promise<void> {
|
|
if (server) {
|
|
await new Promise<void>((resolve) => server!.close(() => resolve()))
|
|
server = undefined
|
|
log.info("oauth callback server stopped")
|
|
}
|
|
|
|
for (const [name, pending] of pendingAuths) {
|
|
clearTimeout(pending.timeout)
|
|
pending.reject(new Error("OAuth callback server stopped"))
|
|
}
|
|
pendingAuths.clear()
|
|
}
|
|
|
|
export function isRunning(): boolean {
|
|
return server !== undefined
|
|
}
|
|
}
|