- Added handling for HTTP_STATUS.NOT_ACCEPTABLE in error types and messages. - Enhanced the `prepareClaudeRequest` function to filter built-in tools for non-Anthropic providers and clean up empty tool arrays. - Updated the `openaiToClaudeRequest` function to handle built-in tools more effectively and ensure proper tool conversion. - Improved the `claudeToOpenAIResponse` function to skip processing for built-in server tool blocks. - Refined error message handling in the `parseUpstreamError` function to ensure meaningful output. - Adjusted command checks for tool installations across various settings routes to use `command -v` for better compatibility.
180 lines
5.3 KiB
JavaScript
180 lines
5.3 KiB
JavaScript
"use server";
|
|
|
|
import { NextResponse } from "next/server";
|
|
import { exec } from "child_process";
|
|
import { promisify } from "util";
|
|
import fs from "fs/promises";
|
|
import path from "path";
|
|
import os from "os";
|
|
|
|
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" : "command -v 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 });
|
|
}
|
|
}
|