"use client"; import { useState, useEffect } from "react"; import { Card, Button, ModelSelectModal, ManualConfigModal } from "@/shared/components"; import Image from "next/image"; export default function CodexToolCard({ tool, isExpanded, onToggle, baseUrl, apiKeys, activeProviders, cloudEnabled, initialStatus }) { const [codexStatus, setCodexStatus] = useState(initialStatus || null); const [checkingCodex, setCheckingCodex] = useState(false); const [applying, setApplying] = useState(false); const [restoring, setRestoring] = useState(false); const [message, setMessage] = useState(null); const [showInstallGuide, setShowInstallGuide] = useState(false); const [selectedApiKey, setSelectedApiKey] = useState(""); const [selectedModel, setSelectedModel] = useState(""); const [subagentModel, setSubagentModel] = useState(""); const [modalOpen, setModalOpen] = useState(false); const [subagentModalOpen, setSubagentModalOpen] = useState(false); const [modelAliases, setModelAliases] = useState({}); const [showManualConfigModal, setShowManualConfigModal] = useState(false); const [customBaseUrl, setCustomBaseUrl] = useState(""); useEffect(() => { if (apiKeys?.length > 0 && !selectedApiKey) { setSelectedApiKey(apiKeys[0].key); } }, [apiKeys, selectedApiKey]); useEffect(() => { if (initialStatus) setCodexStatus(initialStatus); }, [initialStatus]); useEffect(() => { if (isExpanded && !codexStatus) { checkCodexStatus(); fetchModelAliases(); } if (isExpanded) fetchModelAliases(); }, [isExpanded]); const fetchModelAliases = async () => { try { const res = await fetch("/api/models/alias"); const data = await res.json(); if (res.ok) setModelAliases(data.aliases || {}); } catch (error) { console.log("Error fetching model aliases:", error); } }; // Parse model and subagent settings from config content useEffect(() => { if (codexStatus?.config) { const modelMatch = codexStatus.config.match(/^model\s*=\s*"([^"]+)"/m); if (modelMatch) setSelectedModel(modelMatch[1]); // Parse subagent settings const subagentModelMatch = codexStatus.config.match(/\[agents\.subagent\]\s*\n\s*model\s*=\s*"([^"]+)"/m); if (subagentModelMatch) setSubagentModel(subagentModelMatch[1]); } }, [codexStatus]); const getConfigStatus = () => { if (!codexStatus?.installed) return null; if (!codexStatus.config) return "not_configured"; const hasBaseUrl = codexStatus.config.includes(baseUrl) || codexStatus.config.includes("localhost") || codexStatus.config.includes("127.0.0.1"); return hasBaseUrl ? "configured" : "other"; }; const configStatus = getConfigStatus(); const getEffectiveBaseUrl = () => { const url = customBaseUrl || `${baseUrl}/v1`; // Ensure URL ends with /v1 return url.endsWith("/v1") ? url : `${url}/v1`; }; const getDisplayUrl = () => customBaseUrl || `${baseUrl}/v1`; const checkCodexStatus = async () => { setCheckingCodex(true); try { const res = await fetch("/api/cli-tools/codex-settings"); const data = await res.json(); setCodexStatus(data); } catch (error) { setCodexStatus({ installed: false, error: error.message }); } finally { setCheckingCodex(false); } }; const handleApplySettings = async () => { setApplying(true); setMessage(null); try { // Use sk_9router for localhost if no key, otherwise use selected key const keyToUse = (selectedApiKey && selectedApiKey.trim()) ? selectedApiKey : (!cloudEnabled ? "sk_9router" : selectedApiKey); const res = await fetch("/api/cli-tools/codex-settings", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ baseUrl: getEffectiveBaseUrl(), apiKey: keyToUse, model: selectedModel, subagentModel: subagentModel || selectedModel }), }); const data = await res.json(); if (res.ok) { setMessage({ type: "success", text: "Settings applied successfully!" }); checkCodexStatus(); } else { setMessage({ type: "error", text: data.error || "Failed to apply settings" }); } } catch (error) { setMessage({ type: "error", text: error.message }); } finally { setApplying(false); } }; const handleResetSettings = async () => { setRestoring(true); setMessage(null); try { const res = await fetch("/api/cli-tools/codex-settings", { method: "DELETE" }); const data = await res.json(); if (res.ok) { setMessage({ type: "success", text: "Settings reset successfully!" }); setSelectedModel(""); setSubagentModel(""); checkCodexStatus(); } else { setMessage({ type: "error", text: data.error || "Failed to reset settings" }); } } catch (error) { setMessage({ type: "error", text: error.message }); } finally { setRestoring(false); } }; const handleModelSelect = (model) => { setSelectedModel(model.value); // Auto-set subagent model if not set if (!subagentModel) { setSubagentModel(model.value); } setModalOpen(false); }; const getManualConfigs = () => { const keyToUse = (selectedApiKey && selectedApiKey.trim()) ? selectedApiKey : (!cloudEnabled ? "sk_9router" : ""); const effectiveSubagentModel = subagentModel || selectedModel; const configContent = `# 9Router Configuration for Codex CLI model = "${selectedModel}" model_provider = "9router" [model_providers.9router] name = "9Router" base_url = "${getEffectiveBaseUrl()}" wire_api = "responses" [agents.subagent] model = "${effectiveSubagentModel}" `; const authContent = JSON.stringify({ auth_mode: "apikey", OPENAI_API_KEY: keyToUse }, null, 2); return [ { filename: "~/.codex/config.toml", content: configContent, }, { filename: "~/.codex/auth.json", content: authContent, }, ]; }; return (
{tool.name} { e.target.style.display = "none"; }} />

{tool.name}

{configStatus === "configured" && Connected} {configStatus === "not_configured" && Not configured} {configStatus === "other" && Other}

{tool.description}

expand_more
{isExpanded && (
{checkingCodex && (
progress_activity Checking Codex CLI...
)} {!checkingCodex && codexStatus && !codexStatus.installed && (
warning

Codex CLI not detected locally

Manual configuration is still available if 9router is deployed on a remote server.

{showInstallGuide && (

Installation Guide

macOS / Linux / Windows:

npm install -g @openai/codex

After installation, run codex to verify.

Codex uses ~/.codex/auth.json with OPENAI_API_KEY. Click "Apply" to auto-configure.

)}
)} {!checkingCodex && codexStatus?.installed && ( <>
{/* Current Base URL */} {codexStatus?.config && (() => { const parsed = codexStatus.config.match(/base_url\s*=\s*"([^"]+)"/); const currentBaseUrl = parsed ? parsed[1] : null; return currentBaseUrl ? (
Current arrow_forward {currentBaseUrl}
) : null; })()} {/* Base URL */}
Base URL arrow_forward setCustomBaseUrl(e.target.value)} placeholder="https://.../v1" className="min-w-0 px-2 py-2 bg-surface rounded border border-border text-xs focus:outline-none focus:ring-1 focus:ring-primary/50 sm:py-1.5" /> {customBaseUrl && customBaseUrl !== `${baseUrl}/v1` && ( )}
{/* API Key */}
API Key arrow_forward {apiKeys.length > 0 ? ( ) : ( {cloudEnabled ? "No API keys - Create one in Keys page" : "sk_9router (default)"} )}
{/* Model */}
Model arrow_forward setSelectedModel(e.target.value)} placeholder="provider/model-id" className="min-w-0 px-2 py-2 bg-surface rounded border border-border text-xs focus:outline-none focus:ring-1 focus:ring-primary/50 sm:py-1.5" /> {selectedModel && }
{/* Subagent Model */}
Subagent Model arrow_forward setSubagentModel(e.target.value)} placeholder={selectedModel || "provider/model-id (defaults to main model)"} className="min-w-0 px-2 py-2 bg-surface rounded border border-border text-xs focus:outline-none focus:ring-1 focus:ring-primary/50 sm:py-1.5" /> {subagentModel && ( )}
{message && (
{message.type === "success" ? "check_circle" : "error"} {message.text}
)}
)}
)} setModalOpen(false)} onSelect={handleModelSelect} selectedModel={selectedModel} activeProviders={activeProviders} modelAliases={modelAliases} title="Select Model for Codex" /> setSubagentModalOpen(false)} onSelect={(model) => { setSubagentModel(model.value); setSubagentModalOpen(false); }} selectedModel={subagentModel} activeProviders={activeProviders} modelAliases={modelAliases} title="Select Subagent Model for Codex" /> setShowManualConfigModal(false)} title="Codex CLI - Manual Configuration" configs={getManualConfigs()} />
); }