mirror of
https://github.com/diegosouzapw/OmniRoute.git
synced 2026-05-23 04:28:06 +00:00
Migra os 8 comandos restantes (doctor, setup, providers, config, status,
logs, update, provider) de parseArgs manual para a API do Commander.
Cada arquivo exporta register<Nome>(program) e run*Command(opts = {}).
Deleta bin/cli/index.mjs (substituído por commands/registry.mjs).
Remove dependência de bin/cli/args.mjs em todos os comandos migrados.
Corrige CWE-310 em doctor.mjs: authTagLength explícito no createDecipheriv.
167 lines
5.4 KiB
TypeScript
167 lines
5.4 KiB
TypeScript
import test from "node:test";
|
|
import assert from "node:assert/strict";
|
|
import fs from "node:fs";
|
|
import os from "node:os";
|
|
import path from "node:path";
|
|
import bcrypt from "bcryptjs";
|
|
import Database from "better-sqlite3";
|
|
|
|
const ORIGINAL_DATA_DIR = process.env.DATA_DIR;
|
|
const ORIGINAL_STORAGE_ENCRYPTION_KEY = process.env.STORAGE_ENCRYPTION_KEY;
|
|
const ORIGINAL_FETCH = globalThis.fetch;
|
|
|
|
function createTempDataDir() {
|
|
return fs.mkdtempSync(path.join(os.tmpdir(), "omniroute-cli-setup-"));
|
|
}
|
|
|
|
async function withTempEnv(fn: (dataDir: string) => Promise<void>) {
|
|
const dataDir = createTempDataDir();
|
|
process.env.DATA_DIR = dataDir;
|
|
delete process.env.STORAGE_ENCRYPTION_KEY;
|
|
|
|
try {
|
|
await fn(dataDir);
|
|
} finally {
|
|
fs.rmSync(dataDir, { recursive: true, force: true });
|
|
if (ORIGINAL_DATA_DIR === undefined) {
|
|
delete process.env.DATA_DIR;
|
|
} else {
|
|
process.env.DATA_DIR = ORIGINAL_DATA_DIR;
|
|
}
|
|
if (ORIGINAL_STORAGE_ENCRYPTION_KEY === undefined) {
|
|
delete process.env.STORAGE_ENCRYPTION_KEY;
|
|
} else {
|
|
process.env.STORAGE_ENCRYPTION_KEY = ORIGINAL_STORAGE_ENCRYPTION_KEY;
|
|
}
|
|
globalThis.fetch = ORIGINAL_FETCH;
|
|
}
|
|
}
|
|
|
|
test("setup command writes password, setup state, and provider in non-interactive mode", async () => {
|
|
await withTempEnv(async (dataDir) => {
|
|
const { runSetupCommand } = await import("../../bin/cli/commands/setup.mjs");
|
|
|
|
const exitCode = await runSetupCommand({
|
|
nonInteractive: true,
|
|
password: "super-secret",
|
|
addProvider: true,
|
|
provider: "openai",
|
|
providerName: "OpenAI CLI",
|
|
apiKey: "sk-test",
|
|
defaultModel: "gpt-4o-mini",
|
|
});
|
|
|
|
assert.equal(exitCode, 0);
|
|
|
|
const db = new Database(path.join(dataDir, "storage.sqlite"));
|
|
const rows = db
|
|
.prepare("SELECT key, value FROM key_value WHERE namespace = 'settings'")
|
|
.all() as Array<{ key: string; value: string }>;
|
|
const settings = Object.fromEntries(rows.map((row) => [row.key, JSON.parse(row.value)]));
|
|
|
|
assert.equal(settings.setupComplete, true);
|
|
assert.equal(settings.requireLogin, true);
|
|
assert.equal(await bcrypt.compare("super-secret", settings.password as string), true);
|
|
|
|
const provider = db.prepare("SELECT * FROM provider_connections").get() as {
|
|
provider: string;
|
|
auth_type: string;
|
|
name: string;
|
|
api_key: string;
|
|
default_model: string;
|
|
is_active: number;
|
|
};
|
|
db.close();
|
|
|
|
assert.equal(provider.provider, "openai");
|
|
assert.equal(provider.auth_type, "apikey");
|
|
assert.equal(provider.name, "OpenAI CLI");
|
|
assert.equal(provider.api_key, "sk-test");
|
|
assert.equal(provider.default_model, "gpt-4o-mini");
|
|
assert.equal(provider.is_active, 1);
|
|
});
|
|
});
|
|
|
|
test("setup command can mark onboarding complete without provider in non-interactive mode", async () => {
|
|
await withTempEnv(async (dataDir) => {
|
|
const { runSetupCommand } = await import("../../bin/cli/commands/setup.mjs");
|
|
|
|
const exitCode = await runSetupCommand({ nonInteractive: true, password: "super-secret" });
|
|
|
|
assert.equal(exitCode, 0);
|
|
|
|
const db = new Database(path.join(dataDir, "storage.sqlite"));
|
|
const setupComplete = db
|
|
.prepare("SELECT value FROM key_value WHERE namespace = 'settings' AND key = 'setupComplete'")
|
|
.get() as { value: string };
|
|
const providerCount = db
|
|
.prepare("SELECT COUNT(*) as count FROM provider_connections")
|
|
.get() as { count: number };
|
|
db.close();
|
|
|
|
assert.equal(JSON.parse(setupComplete.value), true);
|
|
assert.equal(providerCount.count, 0);
|
|
});
|
|
});
|
|
|
|
test("setup command disables login when no password is configured", async () => {
|
|
await withTempEnv(async (dataDir) => {
|
|
const { runSetupCommand } = await import("../../bin/cli/commands/setup.mjs");
|
|
|
|
const exitCode = await runSetupCommand({
|
|
nonInteractive: true,
|
|
addProvider: true,
|
|
provider: "openai",
|
|
apiKey: "sk-test",
|
|
});
|
|
|
|
assert.equal(exitCode, 0);
|
|
|
|
const db = new Database(path.join(dataDir, "storage.sqlite"));
|
|
const requireLogin = db
|
|
.prepare("SELECT value FROM key_value WHERE namespace = 'settings' AND key = 'requireLogin'")
|
|
.get() as { value: string };
|
|
db.close();
|
|
|
|
assert.equal(JSON.parse(requireLogin.value), false);
|
|
});
|
|
});
|
|
|
|
test("setup command can test provider and persist active status", async () => {
|
|
await withTempEnv(async (dataDir) => {
|
|
const calls: string[] = [];
|
|
globalThis.fetch = (async (url: URL | RequestInfo) => {
|
|
calls.push(String(url));
|
|
return new Response(JSON.stringify({ data: [] }), {
|
|
status: 200,
|
|
headers: { "Content-Type": "application/json" },
|
|
});
|
|
}) as typeof fetch;
|
|
|
|
const { runSetupCommand } = await import("../../bin/cli/commands/setup.mjs");
|
|
|
|
const exitCode = await runSetupCommand({
|
|
nonInteractive: true,
|
|
addProvider: true,
|
|
provider: "openai",
|
|
apiKey: "sk-test",
|
|
testProvider: true,
|
|
});
|
|
|
|
assert.equal(exitCode, 0);
|
|
assert.equal(calls.length, 1);
|
|
assert.equal(calls[0], "https://api.openai.com/v1/models");
|
|
|
|
const db = new Database(path.join(dataDir, "storage.sqlite"));
|
|
const provider = db.prepare("SELECT * FROM provider_connections").get() as {
|
|
test_status: string;
|
|
last_tested: string;
|
|
last_error: string | null;
|
|
};
|
|
db.close();
|
|
|
|
assert.equal(provider.test_status, "active");
|
|
assert.ok(provider.last_tested);
|
|
assert.equal(provider.last_error, null);
|
|
});
|
|
});
|