226 lines
6.8 KiB
JavaScript
226 lines
6.8 KiB
JavaScript
"use server";
|
|
|
|
import { NextResponse } from "next/server";
|
|
import { exec } from "child_process";
|
|
import { promisify } from "util";
|
|
import fs from "fs/promises";
|
|
import path from "path";
|
|
import os from "os";
|
|
import { parseTOML, stringifyTOML } from "confbox";
|
|
|
|
const execAsync = promisify(exec);
|
|
|
|
const getCodexDir = () => path.join(os.homedir(), ".codex");
|
|
const getCodexConfigPath = () => path.join(getCodexDir(), "config.toml");
|
|
const getCodexAuthPath = () => path.join(getCodexDir(), "auth.json");
|
|
|
|
// Flatten confbox-parsed TOML into a writable object, preserving nested tables
|
|
const parsedToWritable = (obj) => obj ?? {};
|
|
|
|
// Set a nested key from a flat dotted path, creating intermediate objects as needed
|
|
const setNestedSection = (obj, dottedKey, value) => {
|
|
const keys = dottedKey.split(".");
|
|
let cur = obj;
|
|
for (let i = 0; i < keys.length - 1; i++) {
|
|
if (cur[keys[i]] == null || typeof cur[keys[i]] !== "object") {
|
|
cur[keys[i]] = {};
|
|
}
|
|
cur = cur[keys[i]];
|
|
}
|
|
cur[keys[keys.length - 1]] = value;
|
|
};
|
|
|
|
// Delete a nested key from a flat dotted path
|
|
const deleteNestedSection = (obj, dottedKey) => {
|
|
const keys = dottedKey.split(".");
|
|
let cur = obj;
|
|
for (let i = 0; i < keys.length - 1; i++) {
|
|
cur = cur?.[keys[i]];
|
|
if (cur == null) return;
|
|
}
|
|
delete cur[keys[keys.length - 1]];
|
|
};
|
|
|
|
// Check if codex CLI is installed (via which/where or config file exists)
|
|
const checkCodexInstalled = async () => {
|
|
try {
|
|
const isWindows = os.platform() === "win32";
|
|
const command = isWindows ? "where codex" : "which codex";
|
|
const env = isWindows
|
|
? { ...process.env, PATH: `${process.env.APPDATA}\\npm;${process.env.PATH}` }
|
|
: process.env;
|
|
await execAsync(command, { windowsHide: true, env });
|
|
return true;
|
|
} catch {
|
|
try {
|
|
await fs.access(getCodexConfigPath());
|
|
return true;
|
|
} catch {
|
|
return false;
|
|
}
|
|
}
|
|
};
|
|
|
|
// Read current config.toml
|
|
const readConfig = async () => {
|
|
try {
|
|
const configPath = getCodexConfigPath();
|
|
const content = await fs.readFile(configPath, "utf-8");
|
|
return content;
|
|
} catch (error) {
|
|
if (error.code === "ENOENT") return null;
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
// Check if config has 9Router settings
|
|
const has9RouterConfig = (config) => {
|
|
if (!config) return false;
|
|
return config.includes("model_provider = \"9router\"") || config.includes("[model_providers.9router]");
|
|
};
|
|
|
|
// GET - Check codex CLI and read current settings
|
|
export async function GET() {
|
|
try {
|
|
const isInstalled = await checkCodexInstalled();
|
|
|
|
if (!isInstalled) {
|
|
return NextResponse.json({
|
|
installed: false,
|
|
config: null,
|
|
message: "Codex CLI is not installed",
|
|
});
|
|
}
|
|
|
|
const config = await readConfig();
|
|
|
|
return NextResponse.json({
|
|
installed: true,
|
|
config,
|
|
has9Router: has9RouterConfig(config),
|
|
configPath: getCodexConfigPath(),
|
|
});
|
|
} catch (error) {
|
|
console.log("Error checking codex settings:", error);
|
|
return NextResponse.json({ error: "Failed to check codex settings" }, { status: 500 });
|
|
}
|
|
}
|
|
|
|
// POST - Update 9Router settings (merge with existing config)
|
|
export async function POST(request) {
|
|
try {
|
|
const { baseUrl, apiKey, model } = await request.json();
|
|
|
|
if (!baseUrl || !apiKey || !model) {
|
|
return NextResponse.json({ error: "baseUrl, apiKey and model are required" }, { status: 400 });
|
|
}
|
|
|
|
const codexDir = getCodexDir();
|
|
const configPath = getCodexConfigPath();
|
|
|
|
// Ensure directory exists
|
|
await fs.mkdir(codexDir, { recursive: true });
|
|
|
|
// Read and parse existing config
|
|
let parsed = {};
|
|
try {
|
|
const existingConfig = await fs.readFile(configPath, "utf-8");
|
|
parsed = parsedToWritable(parseTOML(existingConfig));
|
|
} catch { /* No existing config */ }
|
|
|
|
// Update only 9Router related fields (api_key goes to auth.json, not config.toml)
|
|
parsed.model = model;
|
|
parsed.model_provider = "9router";
|
|
|
|
// Update or create 9router provider section (no api_key - Codex reads from auth.json)
|
|
// Ensure /v1 suffix is added only once
|
|
const normalizedBaseUrl = baseUrl.endsWith("/v1") ? baseUrl : `${baseUrl}/v1`;
|
|
setNestedSection(parsed, "model_providers.9router", {
|
|
name: "9Router",
|
|
base_url: normalizedBaseUrl,
|
|
wire_api: "responses",
|
|
});
|
|
|
|
// Write merged config
|
|
const configContent = stringifyTOML(parsed);
|
|
await fs.writeFile(configPath, configContent);
|
|
|
|
// Update auth.json with OPENAI_API_KEY (Codex reads this first)
|
|
const authPath = getCodexAuthPath();
|
|
let authData = {};
|
|
try {
|
|
const existingAuth = await fs.readFile(authPath, "utf-8");
|
|
authData = JSON.parse(existingAuth);
|
|
} catch { /* No existing auth */ }
|
|
|
|
authData.OPENAI_API_KEY = apiKey;
|
|
await fs.writeFile(authPath, JSON.stringify(authData, null, 2));
|
|
|
|
return NextResponse.json({
|
|
success: true,
|
|
message: "Codex settings applied successfully!",
|
|
configPath,
|
|
});
|
|
} catch (error) {
|
|
console.log("Error updating codex settings:", error);
|
|
return NextResponse.json({ error: "Failed to update codex settings" }, { status: 500 });
|
|
}
|
|
}
|
|
|
|
// DELETE - Remove 9Router settings only (keep other settings)
|
|
export async function DELETE() {
|
|
try {
|
|
const configPath = getCodexConfigPath();
|
|
|
|
// Read and parse existing config
|
|
let parsed = {};
|
|
try {
|
|
const existingConfig = await fs.readFile(configPath, "utf-8");
|
|
parsed = parsedToWritable(parseTOML(existingConfig));
|
|
} catch (error) {
|
|
if (error.code === "ENOENT") {
|
|
return NextResponse.json({
|
|
success: true,
|
|
message: "No config file to reset",
|
|
});
|
|
}
|
|
throw error;
|
|
}
|
|
|
|
// Remove 9Router related root fields only if they point to 9router
|
|
if (parsed.model_provider === "9router") {
|
|
delete parsed.model;
|
|
delete parsed.model_provider;
|
|
}
|
|
|
|
// Remove 9router provider section
|
|
deleteNestedSection(parsed, "model_providers.9router");
|
|
|
|
// Write updated config
|
|
const configContent = stringifyTOML(parsed);
|
|
await fs.writeFile(configPath, configContent);
|
|
|
|
// Remove OPENAI_API_KEY from auth.json
|
|
const authPath = getCodexAuthPath();
|
|
try {
|
|
const existingAuth = await fs.readFile(authPath, "utf-8");
|
|
const authData = JSON.parse(existingAuth);
|
|
delete authData.OPENAI_API_KEY;
|
|
|
|
// Write back or delete if empty
|
|
if (Object.keys(authData).length === 0) {
|
|
await fs.unlink(authPath);
|
|
} else {
|
|
await fs.writeFile(authPath, JSON.stringify(authData, null, 2));
|
|
}
|
|
} catch { /* No auth file */ }
|
|
|
|
return NextResponse.json({
|
|
success: true,
|
|
message: "9Router settings removed successfully",
|
|
});
|
|
} catch (error) {
|
|
console.log("Error resetting codex settings:", error);
|
|
return NextResponse.json({ error: "Failed to reset codex settings" }, { status: 500 });
|
|
}
|
|
}
|