mirror of
https://github.com/decolua/9router.git
synced 2026-05-08 12:01:28 +00:00
Add OpenCode CLI
This commit is contained in:
BIN
public/providers/opencode.png
Normal file
BIN
public/providers/opencode.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 16 KiB |
@@ -4,13 +4,14 @@ import { useState, useEffect, useCallback } from "react";
|
|||||||
import { Card, CardSkeleton } from "@/shared/components";
|
import { Card, CardSkeleton } from "@/shared/components";
|
||||||
import { CLI_TOOLS } from "@/shared/constants/cliTools";
|
import { CLI_TOOLS } from "@/shared/constants/cliTools";
|
||||||
import { PROVIDER_MODELS, getModelsByProviderId, PROVIDER_ID_TO_ALIAS } from "@/shared/constants/models";
|
import { PROVIDER_MODELS, getModelsByProviderId, PROVIDER_ID_TO_ALIAS } from "@/shared/constants/models";
|
||||||
import { ClaudeToolCard, CodexToolCard, DroidToolCard, OpenClawToolCard, DefaultToolCard, AntigravityToolCard } from "./components";
|
import { ClaudeToolCard, CodexToolCard, DroidToolCard, OpenClawToolCard, DefaultToolCard, AntigravityToolCard, OpenCodeToolCard } from "./components";
|
||||||
|
|
||||||
const CLOUD_URL = process.env.NEXT_PUBLIC_CLOUD_URL;
|
const CLOUD_URL = process.env.NEXT_PUBLIC_CLOUD_URL;
|
||||||
|
|
||||||
const STATUS_ENDPOINTS = {
|
const STATUS_ENDPOINTS = {
|
||||||
claude: "/api/cli-tools/claude-settings",
|
claude: "/api/cli-tools/claude-settings",
|
||||||
codex: "/api/cli-tools/codex-settings",
|
codex: "/api/cli-tools/codex-settings",
|
||||||
|
opencode: "/api/cli-tools/opencode-settings",
|
||||||
droid: "/api/cli-tools/droid-settings",
|
droid: "/api/cli-tools/droid-settings",
|
||||||
openclaw: "/api/cli-tools/openclaw-settings",
|
openclaw: "/api/cli-tools/openclaw-settings",
|
||||||
antigravity: "/api/cli-tools/antigravity-mitm",
|
antigravity: "/api/cli-tools/antigravity-mitm",
|
||||||
@@ -199,6 +200,8 @@ export default function CLIToolsPageClient({ machineId }) {
|
|||||||
);
|
);
|
||||||
case "codex":
|
case "codex":
|
||||||
return <CodexToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} cloudEnabled={cloudEnabled} initialStatus={toolStatuses.codex} />;
|
return <CodexToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} cloudEnabled={cloudEnabled} initialStatus={toolStatuses.codex} />;
|
||||||
|
case "opencode":
|
||||||
|
return <OpenCodeToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} cloudEnabled={cloudEnabled} initialStatus={toolStatuses.opencode} />;
|
||||||
case "droid":
|
case "droid":
|
||||||
return <DroidToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} hasActiveProviders={hasActiveProviders} cloudEnabled={cloudEnabled} initialStatus={toolStatuses.droid} />;
|
return <DroidToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} hasActiveProviders={hasActiveProviders} cloudEnabled={cloudEnabled} initialStatus={toolStatuses.droid} />;
|
||||||
case "openclaw":
|
case "openclaw":
|
||||||
@@ -206,7 +209,7 @@ export default function CLIToolsPageClient({ machineId }) {
|
|||||||
case "antigravity":
|
case "antigravity":
|
||||||
return <AntigravityToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} hasActiveProviders={hasActiveProviders} cloudEnabled={cloudEnabled} initialStatus={toolStatuses.antigravity} />;
|
return <AntigravityToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} hasActiveProviders={hasActiveProviders} cloudEnabled={cloudEnabled} initialStatus={toolStatuses.antigravity} />;
|
||||||
default:
|
default:
|
||||||
return <DefaultToolCard key={toolId} toolId={toolId} {...commonProps} activeProviders={getActiveProviders()} cloudEnabled={cloudEnabled} />;
|
return <DefaultToolCard key={toolId} toolId={toolId} {...commonProps} activeProviders={getActiveProviders()} cloudEnabled={cloudEnabled} tunnelEnabled={tunnelEnabled} />;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -209,7 +209,7 @@ export default function AntigravityToolCard({
|
|||||||
const isRunning = status?.running;
|
const isRunning = status?.running;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card padding="sm" className="overflow-hidden">
|
<Card padding="xs" className="overflow-hidden">
|
||||||
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
||||||
<div className="flex items-center gap-3">
|
<div className="flex items-center gap-3">
|
||||||
<div className="size-8 flex items-center justify-center shrink-0">
|
<div className="size-8 flex items-center justify-center shrink-0">
|
||||||
|
|||||||
@@ -206,7 +206,7 @@ export default function ClaudeToolCard({
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card padding="sm" className="overflow-hidden">
|
<Card padding="xs" className="overflow-hidden">
|
||||||
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
||||||
<div className="flex items-center gap-3">
|
<div className="flex items-center gap-3">
|
||||||
<div className="size-8 flex items-center justify-center shrink-0">
|
<div className="size-8 flex items-center justify-center shrink-0">
|
||||||
|
|||||||
@@ -169,7 +169,7 @@ wire_api = "responses"
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card padding="sm" className="overflow-hidden">
|
<Card padding="xs" className="overflow-hidden">
|
||||||
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
||||||
<div className="flex items-center gap-3">
|
<div className="flex items-center gap-3">
|
||||||
<div className="size-8 flex items-center justify-center shrink-0">
|
<div className="size-8 flex items-center justify-center shrink-0">
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import { useState } from "react";
|
|||||||
import { Card, ModelSelectModal } from "@/shared/components";
|
import { Card, ModelSelectModal } from "@/shared/components";
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
|
|
||||||
export default function DefaultToolCard({ toolId, tool, isExpanded, onToggle, baseUrl, apiKeys, activeProviders = [], cloudEnabled = false }) {
|
export default function DefaultToolCard({ toolId, tool, isExpanded, onToggle, baseUrl, apiKeys, activeProviders = [], cloudEnabled = false, tunnelEnabled = false }) {
|
||||||
const [copiedField, setCopiedField] = useState(null);
|
const [copiedField, setCopiedField] = useState(null);
|
||||||
const [showModelModal, setShowModelModal] = useState(false);
|
const [showModelModal, setShowModelModal] = useState(false);
|
||||||
const [modelValue, setModelValue] = useState("");
|
const [modelValue, setModelValue] = useState("");
|
||||||
@@ -125,11 +125,11 @@ export default function DefaultToolCard({ toolId, tool, isExpanded, onToggle, ba
|
|||||||
return (
|
return (
|
||||||
<div className="flex flex-col gap-2 mb-4">
|
<div className="flex flex-col gap-2 mb-4">
|
||||||
{tool.notes.map((note, index) => {
|
{tool.notes.map((note, index) => {
|
||||||
// Skip cloudCheck note if cloud is enabled
|
// Skip cloudCheck note if tunnel or cloud is enabled
|
||||||
if (note.type === "cloudCheck" && cloudEnabled) return null;
|
if (note.type === "cloudCheck" && (cloudEnabled || tunnelEnabled)) return null;
|
||||||
|
|
||||||
const isWarning = note.type === "warning";
|
const isWarning = note.type === "warning";
|
||||||
const isError = note.type === "cloudCheck" && !cloudEnabled;
|
const isError = note.type === "cloudCheck" && !cloudEnabled && !tunnelEnabled;
|
||||||
|
|
||||||
let bgClass = "bg-blue-500/10 border-blue-500/30";
|
let bgClass = "bg-blue-500/10 border-blue-500/30";
|
||||||
let textClass = "text-blue-600 dark:text-blue-400";
|
let textClass = "text-blue-600 dark:text-blue-400";
|
||||||
@@ -160,6 +160,7 @@ export default function DefaultToolCard({ toolId, tool, isExpanded, onToggle, ba
|
|||||||
};
|
};
|
||||||
|
|
||||||
const canShowGuide = () => {
|
const canShowGuide = () => {
|
||||||
|
if (tool.requiresExternalUrl && !cloudEnabled && !tunnelEnabled) return false;
|
||||||
if (tool.requiresCloud && !cloudEnabled) return false;
|
if (tool.requiresCloud && !cloudEnabled) return false;
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
@@ -258,7 +259,7 @@ export default function DefaultToolCard({ toolId, tool, isExpanded, onToggle, ba
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card padding="sm" className="overflow-hidden">
|
<Card padding="xs" className="overflow-hidden">
|
||||||
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
||||||
<div className="flex items-center gap-3">
|
<div className="flex items-center gap-3">
|
||||||
<div className="size-8 rounded-lg flex items-center justify-center shrink-0">
|
<div className="size-8 rounded-lg flex items-center justify-center shrink-0">
|
||||||
|
|||||||
@@ -200,7 +200,7 @@ export default function DroidToolCard({
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card padding="sm" className="overflow-hidden">
|
<Card padding="xs" className="overflow-hidden">
|
||||||
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
||||||
<div className="flex items-center gap-3">
|
<div className="flex items-center gap-3">
|
||||||
<div className="size-8 flex items-center justify-center shrink-0">
|
<div className="size-8 flex items-center justify-center shrink-0">
|
||||||
|
|||||||
@@ -213,7 +213,7 @@ export default function OpenClawToolCard({
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card padding="sm" className="overflow-hidden">
|
<Card padding="xs" className="overflow-hidden">
|
||||||
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
||||||
<div className="flex items-center gap-3">
|
<div className="flex items-center gap-3">
|
||||||
<div className="size-8 flex items-center justify-center shrink-0">
|
<div className="size-8 flex items-center justify-center shrink-0">
|
||||||
|
|||||||
@@ -0,0 +1,308 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import { useState, useEffect } from "react";
|
||||||
|
import { Card, Button, ModelSelectModal, ManualConfigModal } from "@/shared/components";
|
||||||
|
import Image from "next/image";
|
||||||
|
|
||||||
|
export default function OpenCodeToolCard({ tool, isExpanded, onToggle, baseUrl, apiKeys, activeProviders, cloudEnabled, initialStatus }) {
|
||||||
|
const [status, setStatus] = useState(initialStatus || null);
|
||||||
|
const [checking, setChecking] = 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 [modalOpen, setModalOpen] = 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) setStatus(initialStatus);
|
||||||
|
}, [initialStatus]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isExpanded && !status) {
|
||||||
|
checkStatus();
|
||||||
|
fetchModelAliases();
|
||||||
|
}
|
||||||
|
if (isExpanded) fetchModelAliases();
|
||||||
|
}, [isExpanded]);
|
||||||
|
|
||||||
|
// Sync model from existing config
|
||||||
|
useEffect(() => {
|
||||||
|
if (status?.config?.model?.startsWith("9router/")) {
|
||||||
|
setSelectedModel(status.config.model.replace("9router/", ""));
|
||||||
|
}
|
||||||
|
}, [status]);
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getConfigStatus = () => {
|
||||||
|
if (!status?.installed) return null;
|
||||||
|
if (!status.config) return "not_configured";
|
||||||
|
const url = status.config?.provider?.["9router"]?.options?.baseURL || "";
|
||||||
|
const isLocal = url.includes("localhost") || url.includes("127.0.0.1");
|
||||||
|
return status.has9Router && (isLocal || url.includes(baseUrl)) ? "configured" : status.has9Router ? "other" : "not_configured";
|
||||||
|
};
|
||||||
|
|
||||||
|
const configStatus = getConfigStatus();
|
||||||
|
|
||||||
|
const getEffectiveBaseUrl = () => {
|
||||||
|
const url = customBaseUrl || baseUrl;
|
||||||
|
return url.endsWith("/v1") ? url : `${url}/v1`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getDisplayUrl = () => customBaseUrl || `${baseUrl}/v1`;
|
||||||
|
|
||||||
|
const checkStatus = async () => {
|
||||||
|
setChecking(true);
|
||||||
|
try {
|
||||||
|
const res = await fetch("/api/cli-tools/opencode-settings");
|
||||||
|
const data = await res.json();
|
||||||
|
setStatus(data);
|
||||||
|
} catch (error) {
|
||||||
|
setStatus({ installed: false, error: error.message });
|
||||||
|
} finally {
|
||||||
|
setChecking(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleApply = async () => {
|
||||||
|
setApplying(true);
|
||||||
|
setMessage(null);
|
||||||
|
try {
|
||||||
|
const keyToUse = (selectedApiKey && selectedApiKey.trim())
|
||||||
|
? selectedApiKey
|
||||||
|
: (!cloudEnabled ? "sk_9router" : selectedApiKey);
|
||||||
|
|
||||||
|
const res = await fetch("/api/cli-tools/opencode-settings", {
|
||||||
|
method: "POST",
|
||||||
|
headers: { "Content-Type": "application/json" },
|
||||||
|
body: JSON.stringify({ baseUrl: getEffectiveBaseUrl(), apiKey: keyToUse, model: selectedModel }),
|
||||||
|
});
|
||||||
|
const data = await res.json();
|
||||||
|
if (res.ok) {
|
||||||
|
setMessage({ type: "success", text: "Settings applied successfully!" });
|
||||||
|
checkStatus();
|
||||||
|
} else {
|
||||||
|
setMessage({ type: "error", text: data.error || "Failed to apply settings" });
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
setMessage({ type: "error", text: error.message });
|
||||||
|
} finally {
|
||||||
|
setApplying(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleReset = async () => {
|
||||||
|
setRestoring(true);
|
||||||
|
setMessage(null);
|
||||||
|
try {
|
||||||
|
const res = await fetch("/api/cli-tools/opencode-settings", { method: "DELETE" });
|
||||||
|
const data = await res.json();
|
||||||
|
if (res.ok) {
|
||||||
|
setMessage({ type: "success", text: "Settings reset successfully!" });
|
||||||
|
setSelectedModel("");
|
||||||
|
checkStatus();
|
||||||
|
} else {
|
||||||
|
setMessage({ type: "error", text: data.error || "Failed to reset settings" });
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
setMessage({ type: "error", text: error.message });
|
||||||
|
} finally {
|
||||||
|
setRestoring(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getManualConfigs = () => {
|
||||||
|
const keyToUse = (selectedApiKey && selectedApiKey.trim())
|
||||||
|
? selectedApiKey
|
||||||
|
: (!cloudEnabled ? "sk_9router" : "<API_KEY_FROM_DASHBOARD>");
|
||||||
|
|
||||||
|
return [{
|
||||||
|
filename: "~/.config/opencode/opencode.json",
|
||||||
|
content: JSON.stringify({
|
||||||
|
provider: {
|
||||||
|
"9router": {
|
||||||
|
npm: "@ai-sdk/openai-compatible",
|
||||||
|
options: { baseURL: getEffectiveBaseUrl(), apiKey: keyToUse },
|
||||||
|
models: { [selectedModel || "provider/model-id"]: { name: selectedModel || "provider/model-id" } },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
model: `9router/${selectedModel || "provider/model-id"}`,
|
||||||
|
}, null, 2),
|
||||||
|
}];
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card padding="xs" className="overflow-hidden">
|
||||||
|
<div className="flex items-center justify-between hover:cursor-pointer" onClick={onToggle}>
|
||||||
|
<div className="flex items-center gap-3">
|
||||||
|
<div className="size-8 flex items-center justify-center shrink-0">
|
||||||
|
<Image src="/providers/opencode.png" alt={tool.name} width={32} height={32} className="size-8 object-contain rounded-lg" sizes="32px" onError={(e) => { e.target.style.display = "none"; }} />
|
||||||
|
</div>
|
||||||
|
<div className="min-w-0">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<h3 className="font-medium text-sm">{tool.name}</h3>
|
||||||
|
{configStatus === "configured" && <span className="px-1.5 py-0.5 text-[10px] font-medium bg-green-500/10 text-green-600 dark:text-green-400 rounded-full">Connected</span>}
|
||||||
|
{configStatus === "not_configured" && <span className="px-1.5 py-0.5 text-[10px] font-medium bg-yellow-500/10 text-yellow-600 dark:text-yellow-400 rounded-full">Not configured</span>}
|
||||||
|
{configStatus === "other" && <span className="px-1.5 py-0.5 text-[10px] font-medium bg-blue-500/10 text-blue-600 dark:text-blue-400 rounded-full">Other</span>}
|
||||||
|
</div>
|
||||||
|
<p className="text-xs text-text-muted truncate">{tool.description}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<span className={`material-symbols-outlined text-text-muted text-[20px] transition-transform ${isExpanded ? "rotate-180" : ""}`}>expand_more</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{isExpanded && (
|
||||||
|
<div className="mt-4 pt-4 border-t border-border flex flex-col gap-4">
|
||||||
|
{checking && (
|
||||||
|
<div className="flex items-center gap-2 text-text-muted">
|
||||||
|
<span className="material-symbols-outlined animate-spin">progress_activity</span>
|
||||||
|
<span>Checking OpenCode CLI...</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!checking && status && !status.installed && (
|
||||||
|
<div className="flex flex-col gap-4">
|
||||||
|
<div className="flex items-center gap-3 p-4 bg-yellow-500/10 border border-yellow-500/30 rounded-lg">
|
||||||
|
<span className="material-symbols-outlined text-yellow-500">warning</span>
|
||||||
|
<div className="flex-1">
|
||||||
|
<p className="font-medium text-yellow-600 dark:text-yellow-400">OpenCode CLI not installed</p>
|
||||||
|
<p className="text-sm text-text-muted">Please install OpenCode CLI to use auto-apply feature.</p>
|
||||||
|
</div>
|
||||||
|
<Button variant="outline" size="sm" onClick={() => setShowInstallGuide(!showInstallGuide)}>
|
||||||
|
<span className="material-symbols-outlined text-[18px] mr-1">{showInstallGuide ? "expand_less" : "help"}</span>
|
||||||
|
{showInstallGuide ? "Hide" : "How to Install"}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
{showInstallGuide && (
|
||||||
|
<div className="p-4 bg-surface border border-border rounded-lg">
|
||||||
|
<h4 className="font-medium mb-3">Installation Guide</h4>
|
||||||
|
<div className="space-y-3 text-sm">
|
||||||
|
<div>
|
||||||
|
<p className="text-text-muted mb-1">macOS / Linux:</p>
|
||||||
|
<code className="block px-3 py-2 bg-black/5 dark:bg-white/5 rounded font-mono text-xs">npm install -g opencode-ai</code>
|
||||||
|
</div>
|
||||||
|
<p className="text-text-muted">After installation, run <code className="px-1 bg-black/5 dark:bg-white/5 rounded">opencode</code> to verify.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!checking && status?.installed && (
|
||||||
|
<>
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
{/* Current base URL */}
|
||||||
|
{status?.config?.provider?.["9router"]?.options?.baseURL && (
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="w-32 shrink-0 text-sm font-semibold text-text-main text-right">Current</span>
|
||||||
|
<span className="material-symbols-outlined text-text-muted text-[14px]">arrow_forward</span>
|
||||||
|
<span className="flex-1 px-2 py-1.5 text-xs text-text-muted truncate">
|
||||||
|
{status.config.provider["9router"].options.baseURL}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Base URL */}
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="w-32 shrink-0 text-sm font-semibold text-text-main text-right">Base URL</span>
|
||||||
|
<span className="material-symbols-outlined text-text-muted text-[14px]">arrow_forward</span>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={getDisplayUrl()}
|
||||||
|
onChange={(e) => setCustomBaseUrl(e.target.value)}
|
||||||
|
placeholder="https://.../v1"
|
||||||
|
className="flex-1 px-2 py-1.5 bg-surface rounded border border-border text-xs focus:outline-none focus:ring-1 focus:ring-primary/50"
|
||||||
|
/>
|
||||||
|
{customBaseUrl && customBaseUrl !== `${baseUrl}/v1` && (
|
||||||
|
<button onClick={() => setCustomBaseUrl("")} className="p-1 text-text-muted hover:text-primary rounded transition-colors" title="Reset to default">
|
||||||
|
<span className="material-symbols-outlined text-[14px]">restart_alt</span>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* API Key */}
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="w-32 shrink-0 text-sm font-semibold text-text-main text-right">API Key</span>
|
||||||
|
<span className="material-symbols-outlined text-text-muted text-[14px]">arrow_forward</span>
|
||||||
|
{apiKeys.length > 0 ? (
|
||||||
|
<select value={selectedApiKey} onChange={(e) => setSelectedApiKey(e.target.value)} className="flex-1 px-2 py-1.5 bg-surface rounded text-xs border border-border focus:outline-none focus:ring-1 focus:ring-primary/50">
|
||||||
|
{apiKeys.map((key) => <option key={key.id} value={key.key}>{key.key}</option>)}
|
||||||
|
</select>
|
||||||
|
) : (
|
||||||
|
<span className="flex-1 text-xs text-text-muted px-2 py-1.5">
|
||||||
|
{cloudEnabled ? "No API keys - Create one in Keys page" : "sk_9router (default)"}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Model */}
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="w-32 shrink-0 text-sm font-semibold text-text-main text-right">Model</span>
|
||||||
|
<span className="material-symbols-outlined text-text-muted text-[14px]">arrow_forward</span>
|
||||||
|
<input type="text" value={selectedModel} onChange={(e) => setSelectedModel(e.target.value)} placeholder="provider/model-id" className="flex-1 px-2 py-1.5 bg-surface rounded border border-border text-xs focus:outline-none focus:ring-1 focus:ring-primary/50" />
|
||||||
|
<button onClick={() => setModalOpen(true)} disabled={!activeProviders?.length} className={`px-2 py-1.5 rounded border text-xs transition-colors shrink-0 whitespace-nowrap ${activeProviders?.length ? "bg-surface border-border text-text-main hover:border-primary cursor-pointer" : "opacity-50 cursor-not-allowed border-border"}`}>Select Model</button>
|
||||||
|
{selectedModel && <button onClick={() => setSelectedModel("")} className="p-1 text-text-muted hover:text-red-500 rounded transition-colors" title="Clear"><span className="material-symbols-outlined text-[14px]">close</span></button>}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{message && (
|
||||||
|
<div className={`flex items-center gap-2 px-2 py-1.5 rounded text-xs ${message.type === "success" ? "bg-green-500/10 text-green-600" : "bg-red-500/10 text-red-600"}`}>
|
||||||
|
<span className="material-symbols-outlined text-[14px]">{message.type === "success" ? "check_circle" : "error"}</span>
|
||||||
|
<span>{message.text}</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Button variant="primary" size="sm" onClick={handleApply} disabled={!selectedModel} loading={applying}>
|
||||||
|
<span className="material-symbols-outlined text-[14px] mr-1">save</span>Apply
|
||||||
|
</Button>
|
||||||
|
<Button variant="outline" size="sm" onClick={handleReset} disabled={!status.has9Router} loading={restoring}>
|
||||||
|
<span className="material-symbols-outlined text-[14px] mr-1">restore</span>Reset
|
||||||
|
</Button>
|
||||||
|
<Button variant="ghost" size="sm" onClick={() => setShowManualConfigModal(true)}>
|
||||||
|
<span className="material-symbols-outlined text-[14px] mr-1">content_copy</span>Manual Config
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<ModelSelectModal
|
||||||
|
isOpen={modalOpen}
|
||||||
|
onClose={() => setModalOpen(false)}
|
||||||
|
onSelect={(model) => { setSelectedModel(model.value); setModalOpen(false); }}
|
||||||
|
selectedModel={selectedModel}
|
||||||
|
activeProviders={activeProviders}
|
||||||
|
modelAliases={modelAliases}
|
||||||
|
title="Select Model for OpenCode"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ManualConfigModal
|
||||||
|
isOpen={showManualConfigModal}
|
||||||
|
onClose={() => setShowManualConfigModal(false)}
|
||||||
|
title="OpenCode - Manual Configuration"
|
||||||
|
configs={getManualConfigs()}
|
||||||
|
/>
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -4,4 +4,5 @@ export { default as DroidToolCard } from "./DroidToolCard";
|
|||||||
export { default as OpenClawToolCard } from "./OpenClawToolCard";
|
export { default as OpenClawToolCard } from "./OpenClawToolCard";
|
||||||
export { default as DefaultToolCard } from "./DefaultToolCard";
|
export { default as DefaultToolCard } from "./DefaultToolCard";
|
||||||
export { default as AntigravityToolCard } from "./AntigravityToolCard";
|
export { default as AntigravityToolCard } from "./AntigravityToolCard";
|
||||||
|
export { default as OpenCodeToolCard } from "./OpenCodeToolCard";
|
||||||
|
|
||||||
|
|||||||
153
src/app/api/cli-tools/opencode-settings/route.js
Normal file
153
src/app/api/cli-tools/opencode-settings/route.js
Normal file
@@ -0,0 +1,153 @@
|
|||||||
|
"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";
|
||||||
|
|
||||||
|
const execAsync = promisify(exec);
|
||||||
|
|
||||||
|
const getConfigDir = () => path.join(os.homedir(), ".config", "opencode");
|
||||||
|
const getConfigPath = () => path.join(getConfigDir(), "opencode.json");
|
||||||
|
|
||||||
|
const checkOpenCodeInstalled = async () => {
|
||||||
|
try {
|
||||||
|
const isWindows = os.platform() === "win32";
|
||||||
|
const command = isWindows ? "where opencode" : "command -v opencode";
|
||||||
|
await execAsync(command, { windowsHide: true });
|
||||||
|
return true;
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const readConfig = async () => {
|
||||||
|
try {
|
||||||
|
const content = await fs.readFile(getConfigPath(), "utf-8");
|
||||||
|
return JSON.parse(content);
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code === "ENOENT") return null;
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const has9RouterConfig = (config) => {
|
||||||
|
if (!config?.provider) return false;
|
||||||
|
return !!config.provider["9router"];
|
||||||
|
};
|
||||||
|
|
||||||
|
// GET - Check opencode CLI and read current settings
|
||||||
|
export async function GET() {
|
||||||
|
try {
|
||||||
|
const isInstalled = await checkOpenCodeInstalled();
|
||||||
|
|
||||||
|
if (!isInstalled) {
|
||||||
|
return NextResponse.json({
|
||||||
|
installed: false,
|
||||||
|
config: null,
|
||||||
|
message: "OpenCode CLI is not installed",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const config = await readConfig();
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
installed: true,
|
||||||
|
config,
|
||||||
|
has9Router: has9RouterConfig(config),
|
||||||
|
configPath: getConfigPath(),
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error checking opencode settings:", error);
|
||||||
|
return NextResponse.json({ error: "Failed to check opencode settings" }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// POST - Apply 9Router as openai-compatible provider
|
||||||
|
export async function POST(request) {
|
||||||
|
try {
|
||||||
|
const { baseUrl, apiKey, model } = await request.json();
|
||||||
|
|
||||||
|
if (!baseUrl || !model) {
|
||||||
|
return NextResponse.json({ error: "baseUrl and model are required" }, { status: 400 });
|
||||||
|
}
|
||||||
|
|
||||||
|
const configDir = getConfigDir();
|
||||||
|
const configPath = getConfigPath();
|
||||||
|
|
||||||
|
await fs.mkdir(configDir, { recursive: true });
|
||||||
|
|
||||||
|
// Read existing config or start fresh
|
||||||
|
let config = {};
|
||||||
|
try {
|
||||||
|
const existing = await fs.readFile(configPath, "utf-8");
|
||||||
|
config = JSON.parse(existing);
|
||||||
|
} catch { /* No existing config */ }
|
||||||
|
|
||||||
|
const normalizedBaseUrl = baseUrl.endsWith("/v1") ? baseUrl : `${baseUrl}/v1`;
|
||||||
|
const keyToUse = apiKey || "sk_9router";
|
||||||
|
|
||||||
|
// Merge 9router provider
|
||||||
|
if (!config.provider) config.provider = {};
|
||||||
|
config.provider["9router"] = {
|
||||||
|
npm: "@ai-sdk/openai-compatible",
|
||||||
|
options: {
|
||||||
|
baseURL: normalizedBaseUrl,
|
||||||
|
apiKey: keyToUse,
|
||||||
|
},
|
||||||
|
models: {
|
||||||
|
[model]: { name: model },
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
// Set as active model
|
||||||
|
config.model = `9router/${model}`;
|
||||||
|
|
||||||
|
await fs.writeFile(configPath, JSON.stringify(config, null, 2));
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: "OpenCode settings applied successfully!",
|
||||||
|
configPath,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error updating opencode settings:", error);
|
||||||
|
return NextResponse.json({ error: "Failed to update opencode settings" }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DELETE - Remove 9Router provider from config
|
||||||
|
export async function DELETE() {
|
||||||
|
try {
|
||||||
|
const configPath = getConfigPath();
|
||||||
|
|
||||||
|
let config = {};
|
||||||
|
try {
|
||||||
|
const existing = await fs.readFile(configPath, "utf-8");
|
||||||
|
config = JSON.parse(existing);
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code === "ENOENT") {
|
||||||
|
return NextResponse.json({ success: true, message: "No config file to reset" });
|
||||||
|
}
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove 9router provider
|
||||||
|
if (config.provider) delete config.provider["9router"];
|
||||||
|
|
||||||
|
// Reset model if it was pointing to 9router
|
||||||
|
if (config.model?.startsWith("9router/")) delete config.model;
|
||||||
|
|
||||||
|
await fs.writeFile(configPath, JSON.stringify(config, null, 2));
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: "9Router settings removed from OpenCode",
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error resetting opencode settings:", error);
|
||||||
|
return NextResponse.json({ error: "Failed to reset opencode settings" }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -22,6 +22,14 @@ export const CLI_TOOLS = {
|
|||||||
{ id: "haiku", name: "Claude Haiku", alias: "haiku", envKey: "ANTHROPIC_DEFAULT_HAIKU_MODEL", defaultValue: "cc/claude-haiku-4-5-20251001" },
|
{ id: "haiku", name: "Claude Haiku", alias: "haiku", envKey: "ANTHROPIC_DEFAULT_HAIKU_MODEL", defaultValue: "cc/claude-haiku-4-5-20251001" },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
openclaw: {
|
||||||
|
id: "openclaw",
|
||||||
|
name: "Open Claw",
|
||||||
|
image: "/providers/openclaw.png",
|
||||||
|
color: "#FF6B35",
|
||||||
|
description: "Open Claw AI Assistant",
|
||||||
|
configType: "custom",
|
||||||
|
},
|
||||||
codex: {
|
codex: {
|
||||||
id: "codex",
|
id: "codex",
|
||||||
name: "OpenAI Codex CLI",
|
name: "OpenAI Codex CLI",
|
||||||
@@ -30,6 +38,14 @@ export const CLI_TOOLS = {
|
|||||||
description: "OpenAI Codex CLI",
|
description: "OpenAI Codex CLI",
|
||||||
configType: "custom",
|
configType: "custom",
|
||||||
},
|
},
|
||||||
|
opencode: {
|
||||||
|
id: "opencode",
|
||||||
|
name: "OpenCode",
|
||||||
|
image: "/providers/opencode.png",
|
||||||
|
color: "#E87040",
|
||||||
|
description: "OpenCode AI Terminal Assistant",
|
||||||
|
configType: "custom",
|
||||||
|
},
|
||||||
antigravity: {
|
antigravity: {
|
||||||
id: "antigravity",
|
id: "antigravity",
|
||||||
name: "Antigravity",
|
name: "Antigravity",
|
||||||
@@ -55,14 +71,6 @@ export const CLI_TOOLS = {
|
|||||||
description: "Factory Droid AI Assistant",
|
description: "Factory Droid AI Assistant",
|
||||||
configType: "custom",
|
configType: "custom",
|
||||||
},
|
},
|
||||||
openclaw: {
|
|
||||||
id: "openclaw",
|
|
||||||
name: "Open Claw",
|
|
||||||
image: "/providers/openclaw.png",
|
|
||||||
color: "#FF6B35",
|
|
||||||
description: "Open Claw AI Assistant",
|
|
||||||
configType: "custom",
|
|
||||||
},
|
|
||||||
cursor: {
|
cursor: {
|
||||||
id: "cursor",
|
id: "cursor",
|
||||||
name: "Cursor",
|
name: "Cursor",
|
||||||
@@ -70,10 +78,10 @@ export const CLI_TOOLS = {
|
|||||||
color: "#000000",
|
color: "#000000",
|
||||||
description: "Cursor AI Code Editor",
|
description: "Cursor AI Code Editor",
|
||||||
configType: "guide",
|
configType: "guide",
|
||||||
requiresCloud: true,
|
requiresExternalUrl: true,
|
||||||
notes: [
|
notes: [
|
||||||
{ type: "warning", text: "Requires Cursor Pro account to use this feature." },
|
{ type: "warning", text: "Requires Cursor Pro account to use this feature." },
|
||||||
{ type: "cloudCheck", text: "Cursor routes requests through its own server, so local endpoint is not supported. Please enable Cloud Endpoint in Settings." },
|
{ type: "cloudCheck", text: "Cursor routes requests through its own server, so local endpoint is not supported. Please enable Tunnel or Cloud Endpoint in Settings." },
|
||||||
],
|
],
|
||||||
guideSteps: [
|
guideSteps: [
|
||||||
{ step: 1, title: "Open Settings", desc: "Go to Settings → Models" },
|
{ step: 1, title: "Open Settings", desc: "Go to Settings → Models" },
|
||||||
|
|||||||
Reference in New Issue
Block a user