feat(cli-tools): update CLI tools and add new models
- Add Droid and OpenClaw tool cards to CLI tools - Enhance ClaudeToolCard and CodexToolCard to display current base URLs
@@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "9router-app",
|
"name": "9router-app",
|
||||||
"version": "0.2.63",
|
"version": "0.2.66",
|
||||||
"description": "9Router web dashboard",
|
"description": "9Router web dashboard",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev --webpack",
|
"dev": "next dev --webpack",
|
||||||
"build": "next build --webpack",
|
"build": "next build --webpack",
|
||||||
"start": "next start"
|
"start": "next start"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@monaco-editor/react": "^4.7.0",
|
"@monaco-editor/react": "^4.7.0",
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 3.6 KiB After Width: | Height: | Size: 5.0 KiB |
|
Before Width: | Height: | Size: 1.5 KiB After Width: | Height: | Size: 2.7 KiB |
|
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 11 KiB |
|
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 12 KiB |
|
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 15 KiB |
|
Before Width: | Height: | Size: 5.2 KiB After Width: | Height: | Size: 5.7 KiB |
|
Before Width: | Height: | Size: 31 KiB After Width: | Height: | Size: 33 KiB |
|
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 27 KiB |
|
Before Width: | Height: | Size: 7.6 KiB After Width: | Height: | Size: 7.6 KiB |
BIN
public/providers/droid.png
Normal file
|
After Width: | Height: | Size: 6.7 KiB |
|
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 11 KiB |
|
Before Width: | Height: | Size: 8.3 KiB After Width: | Height: | Size: 9.7 KiB |
|
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 27 KiB |
|
Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 2.5 KiB |
|
Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 10 KiB |
|
Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 18 KiB |
|
Before Width: | Height: | Size: 8.7 KiB After Width: | Height: | Size: 8.7 KiB |
|
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 15 KiB |
|
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 15 KiB |
|
Before Width: | Height: | Size: 9.1 KiB After Width: | Height: | Size: 9.9 KiB |
|
Before Width: | Height: | Size: 7.8 KiB After Width: | Height: | Size: 8.8 KiB |
|
Before Width: | Height: | Size: 879 B After Width: | Height: | Size: 1.1 KiB |
BIN
public/providers/openclaw.png
Normal file
|
After Width: | Height: | Size: 16 KiB |
|
Before Width: | Height: | Size: 7.7 KiB After Width: | Height: | Size: 8.6 KiB |
|
Before Width: | Height: | Size: 14 KiB After Width: | Height: | Size: 14 KiB |
|
Before Width: | Height: | Size: 2.6 KiB After Width: | Height: | Size: 3.0 KiB |
@@ -4,7 +4,7 @@ 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, DefaultToolCard } from "./components";
|
import { ClaudeToolCard, CodexToolCard, DroidToolCard, OpenClawToolCard, DefaultToolCard } from "./components";
|
||||||
|
|
||||||
const CLOUD_URL = process.env.NEXT_PUBLIC_CLOUD_URL;
|
const CLOUD_URL = process.env.NEXT_PUBLIC_CLOUD_URL;
|
||||||
|
|
||||||
@@ -156,6 +156,10 @@ export default function CLIToolsPageClient({ machineId }) {
|
|||||||
);
|
);
|
||||||
case "codex":
|
case "codex":
|
||||||
return <CodexToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} cloudEnabled={cloudEnabled} />;
|
return <CodexToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} cloudEnabled={cloudEnabled} />;
|
||||||
|
case "droid":
|
||||||
|
return <DroidToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} hasActiveProviders={hasActiveProviders} cloudEnabled={cloudEnabled} />;
|
||||||
|
case "openclaw":
|
||||||
|
return <OpenClawToolCard key={toolId} {...commonProps} activeProviders={getActiveProviders()} hasActiveProviders={hasActiveProviders} cloudEnabled={cloudEnabled} />;
|
||||||
default:
|
default:
|
||||||
return <DefaultToolCard key={toolId} toolId={toolId} {...commonProps} activeProviders={getActiveProviders()} cloudEnabled={cloudEnabled} />;
|
return <DefaultToolCard key={toolId} toolId={toolId} {...commonProps} activeProviders={getActiveProviders()} cloudEnabled={cloudEnabled} />;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -102,7 +102,15 @@ export default function ClaudeToolCard({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const getEffectiveBaseUrl = () => customBaseUrl || baseUrl;
|
const getEffectiveBaseUrl = () => {
|
||||||
|
const url = customBaseUrl || baseUrl;
|
||||||
|
return url.endsWith("/v1") ? url : `${url}/v1`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getDisplayUrl = () => {
|
||||||
|
const url = customBaseUrl || baseUrl;
|
||||||
|
return url.endsWith("/v1") ? url : `${url}/v1`;
|
||||||
|
};
|
||||||
|
|
||||||
const handleApplySettings = async () => {
|
const handleApplySettings = async () => {
|
||||||
setApplying(true);
|
setApplying(true);
|
||||||
@@ -250,15 +258,26 @@ export default function ClaudeToolCard({
|
|||||||
{!checkingClaude && claudeStatus?.installed && (
|
{!checkingClaude && claudeStatus?.installed && (
|
||||||
<>
|
<>
|
||||||
<div className="flex flex-col gap-2">
|
<div className="flex flex-col gap-2">
|
||||||
|
{/* Current Base URL */}
|
||||||
|
{claudeStatus?.settings?.env?.ANTHROPIC_BASE_URL && (
|
||||||
|
<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">
|
||||||
|
{claudeStatus.settings.env.ANTHROPIC_BASE_URL}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
{/* Base URL */}
|
{/* Base URL */}
|
||||||
<div className="flex items-center gap-2">
|
<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="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>
|
<span className="material-symbols-outlined text-text-muted text-[14px]">arrow_forward</span>
|
||||||
<input
|
<input
|
||||||
type="text"
|
type="text"
|
||||||
value={customBaseUrl || baseUrl}
|
value={getDisplayUrl()}
|
||||||
onChange={(e) => setCustomBaseUrl(e.target.value)}
|
onChange={(e) => setCustomBaseUrl(e.target.value)}
|
||||||
placeholder="https://..."
|
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"
|
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 && (
|
{customBaseUrl && customBaseUrl !== baseUrl && (
|
||||||
|
|||||||
@@ -229,6 +229,21 @@ wire_api = "responses"
|
|||||||
{!checkingCodex && codexStatus?.installed && (
|
{!checkingCodex && codexStatus?.installed && (
|
||||||
<>
|
<>
|
||||||
<div className="flex flex-col gap-2">
|
<div className="flex flex-col gap-2">
|
||||||
|
{/* Current Base URL */}
|
||||||
|
{codexStatus?.config && (() => {
|
||||||
|
const parsed = codexStatus.config.match(/base_url\s*=\s*"([^"]+)"/);
|
||||||
|
const currentBaseUrl = parsed ? parsed[1] : null;
|
||||||
|
return currentBaseUrl ? (
|
||||||
|
<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">
|
||||||
|
{currentBaseUrl}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
) : null;
|
||||||
|
})()}
|
||||||
|
|
||||||
{/* Base URL */}
|
{/* Base URL */}
|
||||||
<div className="flex items-center gap-2">
|
<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="w-32 shrink-0 text-sm font-semibold text-text-main text-right">Base URL</span>
|
||||||
|
|||||||
@@ -0,0 +1,332 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import { useState, useEffect, useRef } from "react";
|
||||||
|
import { Card, Button, ModelSelectModal, ManualConfigModal } from "@/shared/components";
|
||||||
|
import Image from "next/image";
|
||||||
|
|
||||||
|
const CLOUD_URL = process.env.NEXT_PUBLIC_CLOUD_URL;
|
||||||
|
|
||||||
|
export default function DroidToolCard({
|
||||||
|
tool,
|
||||||
|
isExpanded,
|
||||||
|
onToggle,
|
||||||
|
baseUrl,
|
||||||
|
hasActiveProviders,
|
||||||
|
apiKeys,
|
||||||
|
activeProviders,
|
||||||
|
cloudEnabled,
|
||||||
|
}) {
|
||||||
|
const [droidStatus, setDroidStatus] = useState(null);
|
||||||
|
const [checkingDroid, setCheckingDroid] = useState(false);
|
||||||
|
const [applying, setApplying] = useState(false);
|
||||||
|
const [restoring, setRestoring] = useState(false);
|
||||||
|
const [message, setMessage] = useState(null);
|
||||||
|
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("");
|
||||||
|
const hasInitializedModel = useRef(false);
|
||||||
|
|
||||||
|
const getConfigStatus = () => {
|
||||||
|
if (!droidStatus?.installed) return null;
|
||||||
|
const currentConfig = droidStatus.settings?.customModels?.find(m => m.id === "custom:9Router-0");
|
||||||
|
if (!currentConfig) return "not_configured";
|
||||||
|
const localMatch = currentConfig.baseUrl?.includes("localhost") || currentConfig.baseUrl?.includes("127.0.0.1");
|
||||||
|
const cloudMatch = cloudEnabled && CLOUD_URL && currentConfig.baseUrl?.startsWith(CLOUD_URL);
|
||||||
|
if (localMatch || cloudMatch) return "configured";
|
||||||
|
return "other";
|
||||||
|
};
|
||||||
|
|
||||||
|
const configStatus = getConfigStatus();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (apiKeys?.length > 0 && !selectedApiKey) {
|
||||||
|
setSelectedApiKey(apiKeys[0].key);
|
||||||
|
}
|
||||||
|
}, [apiKeys, selectedApiKey]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isExpanded && !droidStatus) {
|
||||||
|
checkDroidStatus();
|
||||||
|
fetchModelAliases();
|
||||||
|
}
|
||||||
|
}, [isExpanded, droidStatus]);
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (droidStatus?.installed && !hasInitializedModel.current) {
|
||||||
|
hasInitializedModel.current = true;
|
||||||
|
const customModel = droidStatus.settings?.customModels?.find(m => m.id === "custom:9Router-0");
|
||||||
|
if (customModel) {
|
||||||
|
if (customModel.model) setSelectedModel(customModel.model);
|
||||||
|
if (customModel.apiKey && apiKeys?.some(k => k.key === customModel.apiKey)) {
|
||||||
|
setSelectedApiKey(customModel.apiKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [droidStatus, apiKeys]);
|
||||||
|
|
||||||
|
const checkDroidStatus = async () => {
|
||||||
|
setCheckingDroid(true);
|
||||||
|
try {
|
||||||
|
const res = await fetch("/api/cli-tools/droid-settings");
|
||||||
|
const data = await res.json();
|
||||||
|
setDroidStatus(data);
|
||||||
|
} catch (error) {
|
||||||
|
setDroidStatus({ installed: false, error: error.message });
|
||||||
|
} finally {
|
||||||
|
setCheckingDroid(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getEffectiveBaseUrl = () => {
|
||||||
|
const url = customBaseUrl || baseUrl;
|
||||||
|
return url.endsWith("/v1") ? url : `${url}/v1`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getDisplayUrl = () => {
|
||||||
|
const url = customBaseUrl || baseUrl;
|
||||||
|
return url.endsWith("/v1") ? url : `${url}/v1`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleApplySettings = async () => {
|
||||||
|
setApplying(true);
|
||||||
|
setMessage(null);
|
||||||
|
try {
|
||||||
|
const keyToUse = selectedApiKey?.trim()
|
||||||
|
|| (apiKeys?.length > 0 ? apiKeys[0].key : null)
|
||||||
|
|| (!cloudEnabled ? "sk_9router" : null);
|
||||||
|
|
||||||
|
const res = await fetch("/api/cli-tools/droid-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!" });
|
||||||
|
checkDroidStatus();
|
||||||
|
} 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/droid-settings", { method: "DELETE" });
|
||||||
|
const data = await res.json();
|
||||||
|
if (res.ok) {
|
||||||
|
setMessage({ type: "success", text: "Settings reset successfully!" });
|
||||||
|
setSelectedModel("");
|
||||||
|
setSelectedApiKey("");
|
||||||
|
checkDroidStatus();
|
||||||
|
} 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);
|
||||||
|
setModalOpen(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const getManualConfigs = () => {
|
||||||
|
const keyToUse = (selectedApiKey && selectedApiKey.trim())
|
||||||
|
? selectedApiKey
|
||||||
|
: (!cloudEnabled ? "sk_9router" : "<API_KEY_FROM_DASHBOARD>");
|
||||||
|
|
||||||
|
const settingsContent = {
|
||||||
|
customModels: [
|
||||||
|
{
|
||||||
|
model: selectedModel || "provider/model-id",
|
||||||
|
id: "custom:9Router-0",
|
||||||
|
index: 0,
|
||||||
|
baseUrl: getEffectiveBaseUrl(),
|
||||||
|
apiKey: keyToUse,
|
||||||
|
displayName: selectedModel || "provider/model-id",
|
||||||
|
maxOutputTokens: 131072,
|
||||||
|
noImageSupport: false,
|
||||||
|
provider: "openai",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
const platform = typeof navigator !== "undefined" && navigator.platform;
|
||||||
|
const isWindows = platform?.toLowerCase().includes("win");
|
||||||
|
const settingsPath = isWindows
|
||||||
|
? "%USERPROFILE%\\.factory\\settings.json"
|
||||||
|
: "~/.factory/settings.json";
|
||||||
|
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
filename: settingsPath,
|
||||||
|
content: JSON.stringify(settingsContent, null, 2),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card padding="sm" 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/droid.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">
|
||||||
|
{checkingDroid && (
|
||||||
|
<div className="flex items-center gap-2 text-text-muted">
|
||||||
|
<span className="material-symbols-outlined animate-spin">progress_activity</span>
|
||||||
|
<span>Checking Factory Droid CLI...</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!checkingDroid && droidStatus && !droidStatus.installed && (
|
||||||
|
<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">Factory Droid CLI not installed</p>
|
||||||
|
<p className="text-sm text-text-muted">Please install Factory Droid CLI to use this feature.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!checkingDroid && droidStatus?.installed && (
|
||||||
|
<>
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
{/* Current Base URL */}
|
||||||
|
{droidStatus?.settings?.customModels?.find(m => m.id === "custom:9Router-0")?.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">
|
||||||
|
{droidStatus.settings.customModels.find(m => m.id === "custom:9Router-0").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 && (
|
||||||
|
<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={!hasActiveProviders} className={`px-2 py-1.5 rounded border text-xs transition-colors shrink-0 whitespace-nowrap ${hasActiveProviders ? "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={handleApplySettings} disabled={!selectedModel} loading={applying}>
|
||||||
|
<span className="material-symbols-outlined text-[14px] mr-1">save</span>Apply
|
||||||
|
</Button>
|
||||||
|
<Button variant="outline" size="sm" onClick={handleResetSettings} disabled={!droidStatus?.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={handleModelSelect}
|
||||||
|
selectedModel={selectedModel}
|
||||||
|
activeProviders={activeProviders}
|
||||||
|
modelAliases={modelAliases}
|
||||||
|
title="Select Model for Factory Droid"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ManualConfigModal
|
||||||
|
isOpen={showManualConfigModal}
|
||||||
|
onClose={() => setShowManualConfigModal(false)}
|
||||||
|
title="Factory Droid - Manual Configuration"
|
||||||
|
configs={getManualConfigs()}
|
||||||
|
/>
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,339 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import { useState, useEffect, useRef } from "react";
|
||||||
|
import { Card, Button, ModelSelectModal, ManualConfigModal } from "@/shared/components";
|
||||||
|
import Image from "next/image";
|
||||||
|
|
||||||
|
const CLOUD_URL = process.env.NEXT_PUBLIC_CLOUD_URL;
|
||||||
|
|
||||||
|
export default function OpenClawToolCard({
|
||||||
|
tool,
|
||||||
|
isExpanded,
|
||||||
|
onToggle,
|
||||||
|
baseUrl,
|
||||||
|
hasActiveProviders,
|
||||||
|
apiKeys,
|
||||||
|
activeProviders,
|
||||||
|
cloudEnabled,
|
||||||
|
}) {
|
||||||
|
const [openclawStatus, setOpenclawStatus] = useState(null);
|
||||||
|
const [checkingOpenclaw, setCheckingOpenclaw] = useState(false);
|
||||||
|
const [applying, setApplying] = useState(false);
|
||||||
|
const [restoring, setRestoring] = useState(false);
|
||||||
|
const [message, setMessage] = useState(null);
|
||||||
|
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("");
|
||||||
|
const hasInitializedModel = useRef(false);
|
||||||
|
|
||||||
|
const getConfigStatus = () => {
|
||||||
|
if (!openclawStatus?.installed) return null;
|
||||||
|
const currentProvider = openclawStatus.settings?.models?.providers?.["9router"];
|
||||||
|
if (!currentProvider) return "not_configured";
|
||||||
|
const localMatch = currentProvider.baseUrl?.includes("localhost") || currentProvider.baseUrl?.includes("127.0.0.1");
|
||||||
|
const cloudMatch = cloudEnabled && CLOUD_URL && currentProvider.baseUrl?.startsWith(CLOUD_URL);
|
||||||
|
if (localMatch || cloudMatch) return "configured";
|
||||||
|
return "other";
|
||||||
|
};
|
||||||
|
|
||||||
|
const configStatus = getConfigStatus();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (apiKeys?.length > 0 && !selectedApiKey) {
|
||||||
|
setSelectedApiKey(apiKeys[0].key);
|
||||||
|
}
|
||||||
|
}, [apiKeys, selectedApiKey]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isExpanded && !openclawStatus) {
|
||||||
|
checkOpenclawStatus();
|
||||||
|
fetchModelAliases();
|
||||||
|
}
|
||||||
|
}, [isExpanded, openclawStatus]);
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (openclawStatus?.installed && !hasInitializedModel.current) {
|
||||||
|
hasInitializedModel.current = true;
|
||||||
|
const provider = openclawStatus.settings?.models?.providers?.["9router"];
|
||||||
|
if (provider) {
|
||||||
|
const primaryModel = openclawStatus.settings?.agents?.defaults?.model?.primary;
|
||||||
|
if (primaryModel) {
|
||||||
|
const modelId = primaryModel.replace("9router/", "");
|
||||||
|
setSelectedModel(modelId);
|
||||||
|
}
|
||||||
|
if (provider.apiKey && apiKeys?.some(k => k.key === provider.apiKey)) {
|
||||||
|
setSelectedApiKey(provider.apiKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [openclawStatus, apiKeys]);
|
||||||
|
|
||||||
|
const checkOpenclawStatus = async () => {
|
||||||
|
setCheckingOpenclaw(true);
|
||||||
|
try {
|
||||||
|
const res = await fetch("/api/cli-tools/openclaw-settings");
|
||||||
|
const data = await res.json();
|
||||||
|
setOpenclawStatus(data);
|
||||||
|
} catch (error) {
|
||||||
|
setOpenclawStatus({ installed: false, error: error.message });
|
||||||
|
} finally {
|
||||||
|
setCheckingOpenclaw(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getEffectiveBaseUrl = () => {
|
||||||
|
const url = customBaseUrl || baseUrl;
|
||||||
|
return url.endsWith("/v1") ? url : `${url}/v1`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getDisplayUrl = () => {
|
||||||
|
const url = customBaseUrl || baseUrl;
|
||||||
|
return url.endsWith("/v1") ? url : `${url}/v1`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleApplySettings = async () => {
|
||||||
|
setApplying(true);
|
||||||
|
setMessage(null);
|
||||||
|
try {
|
||||||
|
const keyToUse = selectedApiKey?.trim()
|
||||||
|
|| (apiKeys?.length > 0 ? apiKeys[0].key : null)
|
||||||
|
|| (!cloudEnabled ? "sk_9router" : null);
|
||||||
|
|
||||||
|
const res = await fetch("/api/cli-tools/openclaw-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!" });
|
||||||
|
checkOpenclawStatus();
|
||||||
|
} 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/openclaw-settings", { method: "DELETE" });
|
||||||
|
const data = await res.json();
|
||||||
|
if (res.ok) {
|
||||||
|
setMessage({ type: "success", text: "Settings reset successfully!" });
|
||||||
|
setSelectedModel("");
|
||||||
|
setSelectedApiKey("");
|
||||||
|
checkOpenclawStatus();
|
||||||
|
} 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);
|
||||||
|
setModalOpen(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const getManualConfigs = () => {
|
||||||
|
const keyToUse = (selectedApiKey && selectedApiKey.trim())
|
||||||
|
? selectedApiKey
|
||||||
|
: (!cloudEnabled ? "sk_9router" : "<API_KEY_FROM_DASHBOARD>");
|
||||||
|
|
||||||
|
const settingsContent = {
|
||||||
|
agents: {
|
||||||
|
defaults: {
|
||||||
|
model: {
|
||||||
|
primary: `9router/${selectedModel || "provider/model-id"}`,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
models: {
|
||||||
|
providers: {
|
||||||
|
"9router": {
|
||||||
|
baseUrl: getEffectiveBaseUrl(),
|
||||||
|
apiKey: keyToUse,
|
||||||
|
api: "openai-completions",
|
||||||
|
models: [
|
||||||
|
{
|
||||||
|
id: selectedModel || "provider/model-id",
|
||||||
|
name: (selectedModel || "provider/model-id").split("/").pop(),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
filename: "~/.openclaw/openclaw.json",
|
||||||
|
content: JSON.stringify(settingsContent, null, 2),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card padding="sm" 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/openclaw.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">
|
||||||
|
{checkingOpenclaw && (
|
||||||
|
<div className="flex items-center gap-2 text-text-muted">
|
||||||
|
<span className="material-symbols-outlined animate-spin">progress_activity</span>
|
||||||
|
<span>Checking Open Claw CLI...</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!checkingOpenclaw && openclawStatus && !openclawStatus.installed && (
|
||||||
|
<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">Open Claw CLI not installed</p>
|
||||||
|
<p className="text-sm text-text-muted">Please install Open Claw CLI to use this feature.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!checkingOpenclaw && openclawStatus?.installed && (
|
||||||
|
<>
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
{/* Current Base URL */}
|
||||||
|
{openclawStatus?.settings?.models?.providers?.["9router"]?.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">
|
||||||
|
{openclawStatus.settings.models.providers["9router"].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 && (
|
||||||
|
<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={!hasActiveProviders} className={`px-2 py-1.5 rounded border text-xs transition-colors shrink-0 whitespace-nowrap ${hasActiveProviders ? "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={handleApplySettings} disabled={!selectedModel} loading={applying}>
|
||||||
|
<span className="material-symbols-outlined text-[14px] mr-1">save</span>Apply
|
||||||
|
</Button>
|
||||||
|
<Button variant="outline" size="sm" onClick={handleResetSettings} disabled={!openclawStatus?.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={handleModelSelect}
|
||||||
|
selectedModel={selectedModel}
|
||||||
|
activeProviders={activeProviders}
|
||||||
|
modelAliases={modelAliases}
|
||||||
|
title="Select Model for Open Claw"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ManualConfigModal
|
||||||
|
isOpen={showManualConfigModal}
|
||||||
|
onClose={() => setShowManualConfigModal(false)}
|
||||||
|
title="Open Claw - Manual Configuration"
|
||||||
|
configs={getManualConfigs()}
|
||||||
|
/>
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,4 +1,6 @@
|
|||||||
export { default as ClaudeToolCard } from "./ClaudeToolCard";
|
export { default as ClaudeToolCard } from "./ClaudeToolCard";
|
||||||
export { default as CodexToolCard } from "./CodexToolCard";
|
export { default as CodexToolCard } from "./CodexToolCard";
|
||||||
|
export { default as DroidToolCard } from "./DroidToolCard";
|
||||||
|
export { default as OpenClawToolCard } from "./OpenClawToolCard";
|
||||||
export { default as DefaultToolCard } from "./DefaultToolCard";
|
export { default as DefaultToolCard } from "./DefaultToolCard";
|
||||||
|
|
||||||
|
|||||||
@@ -102,6 +102,13 @@ export async function POST(request) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Normalize ANTHROPIC_BASE_URL to ensure /v1 suffix
|
||||||
|
if (env.ANTHROPIC_BASE_URL) {
|
||||||
|
env.ANTHROPIC_BASE_URL = env.ANTHROPIC_BASE_URL.endsWith("/v1")
|
||||||
|
? env.ANTHROPIC_BASE_URL
|
||||||
|
: `${env.ANTHROPIC_BASE_URL}/v1`;
|
||||||
|
}
|
||||||
|
|
||||||
// Merge new env with existing settings
|
// Merge new env with existing settings
|
||||||
const newSettings = {
|
const newSettings = {
|
||||||
...currentSettings,
|
...currentSettings,
|
||||||
|
|||||||
@@ -155,9 +155,11 @@ export async function POST(request) {
|
|||||||
parsed._root.model_provider = "9router";
|
parsed._root.model_provider = "9router";
|
||||||
|
|
||||||
// Update or create 9router provider section (no api_key - Codex reads from auth.json)
|
// 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`;
|
||||||
parsed._sections["model_providers.9router"] = {
|
parsed._sections["model_providers.9router"] = {
|
||||||
name: "9Router",
|
name: "9Router",
|
||||||
base_url: `${baseUrl}/v1`,
|
base_url: normalizedBaseUrl,
|
||||||
wire_api: "responses",
|
wire_api: "responses",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
175
src/app/api/cli-tools/droid-settings/route.js
Normal file
@@ -0,0 +1,175 @@
|
|||||||
|
"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 getDroidDir = () => path.join(os.homedir(), ".factory");
|
||||||
|
const getDroidSettingsPath = () => path.join(getDroidDir(), "settings.json");
|
||||||
|
|
||||||
|
// Check if droid CLI is installed
|
||||||
|
const checkDroidInstalled = async () => {
|
||||||
|
try {
|
||||||
|
const isWindows = os.platform() === "win32";
|
||||||
|
const command = isWindows ? "where droid" : "which droid";
|
||||||
|
await execAsync(command);
|
||||||
|
return true;
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Read current settings.json
|
||||||
|
const readSettings = async () => {
|
||||||
|
try {
|
||||||
|
const settingsPath = getDroidSettingsPath();
|
||||||
|
const content = await fs.readFile(settingsPath, "utf-8");
|
||||||
|
return JSON.parse(content);
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code === "ENOENT") return null;
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Check if settings has 9Router customModels
|
||||||
|
const has9RouterConfig = (settings) => {
|
||||||
|
if (!settings || !settings.customModels) return false;
|
||||||
|
return settings.customModels.some(m => m.id === "custom:9Router-0");
|
||||||
|
};
|
||||||
|
|
||||||
|
// GET - Check droid CLI and read current settings
|
||||||
|
export async function GET() {
|
||||||
|
try {
|
||||||
|
const isInstalled = await checkDroidInstalled();
|
||||||
|
|
||||||
|
if (!isInstalled) {
|
||||||
|
return NextResponse.json({
|
||||||
|
installed: false,
|
||||||
|
settings: null,
|
||||||
|
message: "Factory Droid CLI is not installed",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const settings = await readSettings();
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
installed: true,
|
||||||
|
settings,
|
||||||
|
has9Router: has9RouterConfig(settings),
|
||||||
|
settingsPath: getDroidSettingsPath(),
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error checking droid settings:", error);
|
||||||
|
return NextResponse.json({ error: "Failed to check droid settings" }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// POST - Update 9Router customModels (merge with existing settings)
|
||||||
|
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 droidDir = getDroidDir();
|
||||||
|
const settingsPath = getDroidSettingsPath();
|
||||||
|
|
||||||
|
// Ensure directory exists
|
||||||
|
await fs.mkdir(droidDir, { recursive: true });
|
||||||
|
|
||||||
|
// Read existing settings or create new
|
||||||
|
let settings = {};
|
||||||
|
try {
|
||||||
|
const existingSettings = await fs.readFile(settingsPath, "utf-8");
|
||||||
|
settings = JSON.parse(existingSettings);
|
||||||
|
} catch { /* No existing settings */ }
|
||||||
|
|
||||||
|
// Ensure customModels array exists
|
||||||
|
if (!settings.customModels) {
|
||||||
|
settings.customModels = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove existing 9Router config if any
|
||||||
|
settings.customModels = settings.customModels.filter(m => m.id !== "custom:9Router-0");
|
||||||
|
|
||||||
|
// Normalize baseUrl to ensure /v1 suffix
|
||||||
|
const normalizedBaseUrl = baseUrl.endsWith("/v1") ? baseUrl : `${baseUrl}/v1`;
|
||||||
|
|
||||||
|
// Add new 9Router config
|
||||||
|
const customModel = {
|
||||||
|
model: model,
|
||||||
|
id: "custom:9Router-0",
|
||||||
|
index: 0,
|
||||||
|
baseUrl: normalizedBaseUrl,
|
||||||
|
apiKey: apiKey || "your_api_key",
|
||||||
|
displayName: model,
|
||||||
|
maxOutputTokens: 131072,
|
||||||
|
noImageSupport: false,
|
||||||
|
provider: "openai",
|
||||||
|
};
|
||||||
|
|
||||||
|
settings.customModels.unshift(customModel);
|
||||||
|
|
||||||
|
// Write settings
|
||||||
|
await fs.writeFile(settingsPath, JSON.stringify(settings, null, 2));
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: "Factory Droid settings applied successfully!",
|
||||||
|
settingsPath,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error updating droid settings:", error);
|
||||||
|
return NextResponse.json({ error: "Failed to update droid settings" }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DELETE - Remove 9Router customModels only (keep other settings)
|
||||||
|
export async function DELETE() {
|
||||||
|
try {
|
||||||
|
const settingsPath = getDroidSettingsPath();
|
||||||
|
|
||||||
|
// Read existing settings
|
||||||
|
let settings = {};
|
||||||
|
try {
|
||||||
|
const existingSettings = await fs.readFile(settingsPath, "utf-8");
|
||||||
|
settings = JSON.parse(existingSettings);
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code === "ENOENT") {
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: "No settings file to reset",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove 9Router customModels
|
||||||
|
if (settings.customModels) {
|
||||||
|
settings.customModels = settings.customModels.filter(m => m.id !== "custom:9Router-0");
|
||||||
|
|
||||||
|
// Remove customModels array if empty
|
||||||
|
if (settings.customModels.length === 0) {
|
||||||
|
delete settings.customModels;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write updated settings
|
||||||
|
await fs.writeFile(settingsPath, JSON.stringify(settings, null, 2));
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: "9Router settings removed successfully",
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error resetting droid settings:", error);
|
||||||
|
return NextResponse.json({ error: "Failed to reset droid settings" }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
180
src/app/api/cli-tools/openclaw-settings/route.js
Normal file
@@ -0,0 +1,180 @@
|
|||||||
|
"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 getOpenClawDir = () => path.join(os.homedir(), ".openclaw");
|
||||||
|
const getOpenClawSettingsPath = () => path.join(getOpenClawDir(), "openclaw.json");
|
||||||
|
|
||||||
|
// Check if openclaw CLI is installed
|
||||||
|
const checkOpenClawInstalled = async () => {
|
||||||
|
try {
|
||||||
|
const isWindows = os.platform() === "win32";
|
||||||
|
const command = isWindows ? "where openclaw" : "which openclaw";
|
||||||
|
await execAsync(command);
|
||||||
|
return true;
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Read current settings.json
|
||||||
|
const readSettings = async () => {
|
||||||
|
try {
|
||||||
|
const settingsPath = getOpenClawSettingsPath();
|
||||||
|
const content = await fs.readFile(settingsPath, "utf-8");
|
||||||
|
return JSON.parse(content);
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code === "ENOENT") return null;
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Check if settings has 9Router config
|
||||||
|
const has9RouterConfig = (settings) => {
|
||||||
|
if (!settings || !settings.models || !settings.models.providers) return false;
|
||||||
|
return !!settings.models.providers["9router"];
|
||||||
|
};
|
||||||
|
|
||||||
|
// GET - Check openclaw CLI and read current settings
|
||||||
|
export async function GET() {
|
||||||
|
try {
|
||||||
|
const isInstalled = await checkOpenClawInstalled();
|
||||||
|
|
||||||
|
if (!isInstalled) {
|
||||||
|
return NextResponse.json({
|
||||||
|
installed: false,
|
||||||
|
settings: null,
|
||||||
|
message: "Open Claw CLI is not installed",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const settings = await readSettings();
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
installed: true,
|
||||||
|
settings,
|
||||||
|
has9Router: has9RouterConfig(settings),
|
||||||
|
settingsPath: getOpenClawSettingsPath(),
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error checking openclaw settings:", error);
|
||||||
|
return NextResponse.json({ error: "Failed to check openclaw settings" }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// POST - Update 9Router settings (merge with existing settings)
|
||||||
|
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 openclawDir = getOpenClawDir();
|
||||||
|
const settingsPath = getOpenClawSettingsPath();
|
||||||
|
|
||||||
|
// Ensure directory exists
|
||||||
|
await fs.mkdir(openclawDir, { recursive: true });
|
||||||
|
|
||||||
|
// Read existing settings or create new
|
||||||
|
let settings = {};
|
||||||
|
try {
|
||||||
|
const existingSettings = await fs.readFile(settingsPath, "utf-8");
|
||||||
|
settings = JSON.parse(existingSettings);
|
||||||
|
} catch { /* No existing settings */ }
|
||||||
|
|
||||||
|
// Ensure structure exists
|
||||||
|
if (!settings.agents) settings.agents = {};
|
||||||
|
if (!settings.agents.defaults) settings.agents.defaults = {};
|
||||||
|
if (!settings.agents.defaults.model) settings.agents.defaults.model = {};
|
||||||
|
if (!settings.models) settings.models = {};
|
||||||
|
if (!settings.models.providers) settings.models.providers = {};
|
||||||
|
|
||||||
|
// Normalize baseUrl to ensure /v1 suffix
|
||||||
|
const normalizedBaseUrl = baseUrl.endsWith("/v1") ? baseUrl : `${baseUrl}/v1`;
|
||||||
|
|
||||||
|
// Update agents.defaults.model.primary
|
||||||
|
settings.agents.defaults.model.primary = `9router/${model}`;
|
||||||
|
|
||||||
|
// Update models.providers.9router
|
||||||
|
settings.models.providers["9router"] = {
|
||||||
|
baseUrl: normalizedBaseUrl,
|
||||||
|
apiKey: apiKey || "your_api_key",
|
||||||
|
api: "openai-completions",
|
||||||
|
models: [
|
||||||
|
{
|
||||||
|
id: model,
|
||||||
|
name: model.split("/").pop() || model,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
// Write settings
|
||||||
|
await fs.writeFile(settingsPath, JSON.stringify(settings, null, 2));
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: "Open Claw settings applied successfully!",
|
||||||
|
settingsPath,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error updating openclaw settings:", error);
|
||||||
|
return NextResponse.json({ error: "Failed to update openclaw settings" }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DELETE - Remove 9Router settings only (keep other settings)
|
||||||
|
export async function DELETE() {
|
||||||
|
try {
|
||||||
|
const settingsPath = getOpenClawSettingsPath();
|
||||||
|
|
||||||
|
// Read existing settings
|
||||||
|
let settings = {};
|
||||||
|
try {
|
||||||
|
const existingSettings = await fs.readFile(settingsPath, "utf-8");
|
||||||
|
settings = JSON.parse(existingSettings);
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code === "ENOENT") {
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: "No settings file to reset",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove 9Router from models.providers
|
||||||
|
if (settings.models && settings.models.providers) {
|
||||||
|
delete settings.models.providers["9router"];
|
||||||
|
|
||||||
|
// Remove providers object if empty
|
||||||
|
if (Object.keys(settings.models.providers).length === 0) {
|
||||||
|
delete settings.models.providers;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset agents.defaults.model.primary if it uses 9router
|
||||||
|
if (settings.agents?.defaults?.model?.primary?.startsWith("9router/")) {
|
||||||
|
delete settings.agents.defaults.model.primary;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write updated settings
|
||||||
|
await fs.writeFile(settingsPath, JSON.stringify(settings, null, 2));
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: "9Router settings removed successfully",
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error resetting openclaw settings:", error);
|
||||||
|
return NextResponse.json({ error: "Failed to reset openclaw settings" }, { status: 500 });
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import { NextResponse } from "next/server";
|
import { NextResponse } from "next/server";
|
||||||
import { createProviderNode, getProviderNodes } from "@/models";
|
import { createProviderNode, getProviderNodes } from "@/models";
|
||||||
import { OPENAI_COMPATIBLE_PREFIX, ANTHROPIC_COMPATIBLE_PREFIX } from "@/shared/constants/providers";
|
import { OPENAI_COMPATIBLE_PREFIX, ANTHROPIC_COMPATIBLE_PREFIX } from "@/shared/constants/providers";
|
||||||
|
import { generateId } from "@/shared/utils";
|
||||||
|
|
||||||
const OPENAI_COMPATIBLE_DEFAULTS = {
|
const OPENAI_COMPATIBLE_DEFAULTS = {
|
||||||
baseUrl: "https://api.openai.com/v1",
|
baseUrl: "https://api.openai.com/v1",
|
||||||
@@ -44,7 +45,7 @@ export async function POST(request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const node = await createProviderNode({
|
const node = await createProviderNode({
|
||||||
id: `${OPENAI_COMPATIBLE_PREFIX}${apiType}-${crypto.randomUUID()}`,
|
id: `${OPENAI_COMPATIBLE_PREFIX}${apiType}-${generateId()}`,
|
||||||
type: "openai-compatible",
|
type: "openai-compatible",
|
||||||
prefix: prefix.trim(),
|
prefix: prefix.trim(),
|
||||||
apiType,
|
apiType,
|
||||||
@@ -63,7 +64,7 @@ export async function POST(request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const node = await createProviderNode({
|
const node = await createProviderNode({
|
||||||
id: `${ANTHROPIC_COMPATIBLE_PREFIX}${crypto.randomUUID()}`,
|
id: `${ANTHROPIC_COMPATIBLE_PREFIX}${generateId()}`,
|
||||||
type: "anthropic-compatible",
|
type: "anthropic-compatible",
|
||||||
prefix: prefix.trim(),
|
prefix: prefix.trim(),
|
||||||
baseUrl: sanitizedBaseUrl,
|
baseUrl: sanitizedBaseUrl,
|
||||||
|
|||||||
@@ -147,6 +147,12 @@ export async function getProviderNodeById(id) {
|
|||||||
*/
|
*/
|
||||||
export async function createProviderNode(data) {
|
export async function createProviderNode(data) {
|
||||||
const db = await getDb();
|
const db = await getDb();
|
||||||
|
|
||||||
|
// Initialize providerNodes if undefined (backward compatibility)
|
||||||
|
if (!db.data.providerNodes) {
|
||||||
|
db.data.providerNodes = [];
|
||||||
|
}
|
||||||
|
|
||||||
const now = new Date().toISOString();
|
const now = new Date().toISOString();
|
||||||
|
|
||||||
const node = {
|
const node = {
|
||||||
@@ -171,6 +177,10 @@ export async function createProviderNode(data) {
|
|||||||
*/
|
*/
|
||||||
export async function updateProviderNode(id, data) {
|
export async function updateProviderNode(id, data) {
|
||||||
const db = await getDb();
|
const db = await getDb();
|
||||||
|
if (!db.data.providerNodes) {
|
||||||
|
db.data.providerNodes = [];
|
||||||
|
}
|
||||||
|
|
||||||
const index = db.data.providerNodes.findIndex((node) => node.id === id);
|
const index = db.data.providerNodes.findIndex((node) => node.id === id);
|
||||||
|
|
||||||
if (index === -1) return null;
|
if (index === -1) return null;
|
||||||
@@ -191,6 +201,10 @@ export async function updateProviderNode(id, data) {
|
|||||||
*/
|
*/
|
||||||
export async function deleteProviderNode(id) {
|
export async function deleteProviderNode(id) {
|
||||||
const db = await getDb();
|
const db = await getDb();
|
||||||
|
if (!db.data.providerNodes) {
|
||||||
|
db.data.providerNodes = [];
|
||||||
|
}
|
||||||
|
|
||||||
const index = db.data.providerNodes.findIndex((node) => node.id === id);
|
const index = db.data.providerNodes.findIndex((node) => node.id === id);
|
||||||
|
|
||||||
if (index === -1) return null;
|
if (index === -1) return null;
|
||||||
|
|||||||
@@ -30,6 +30,22 @@ export const CLI_TOOLS = {
|
|||||||
description: "OpenAI Codex CLI",
|
description: "OpenAI Codex CLI",
|
||||||
configType: "custom",
|
configType: "custom",
|
||||||
},
|
},
|
||||||
|
droid: {
|
||||||
|
id: "droid",
|
||||||
|
name: "Factory Droid",
|
||||||
|
image: "/providers/droid.png",
|
||||||
|
color: "#00D4FF",
|
||||||
|
description: "Factory Droid AI Assistant",
|
||||||
|
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",
|
||||||
|
|||||||
@@ -2,6 +2,14 @@
|
|||||||
export { cn } from "./cn";
|
export { cn } from "./cn";
|
||||||
export * as api from "./api";
|
export * as api from "./api";
|
||||||
|
|
||||||
|
import { v4 as uuidv4 } from "uuid";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generate unique ID (UUID v4)
|
||||||
|
* @returns {string} UUID v4 string
|
||||||
|
*/
|
||||||
|
export const generateId = uuidv4;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extract error code from error message (401, 429, 503...)
|
* Extract error code from error message (401, 429, 503...)
|
||||||
* @param {string} lastError - Error message
|
* @param {string} lastError - Error message
|
||||||
|
|||||||