mirror of
https://github.com/decolua/9router.git
synced 2026-05-08 12:01:28 +00:00
914 lines
22 KiB
JavaScript
914 lines
22 KiB
JavaScript
import { Low } from "lowdb";
|
|
import { JSONFile } from "lowdb/node";
|
|
import { v4 as uuidv4 } from "uuid";
|
|
import path from "node:path";
|
|
import os from "node:os";
|
|
import fs from "node:fs";
|
|
|
|
const isCloud = typeof caches !== 'undefined' || typeof caches === 'object';
|
|
|
|
// Get app name - fixed constant to avoid Windows path issues in standalone build
|
|
function getAppName() {
|
|
return "9router";
|
|
}
|
|
|
|
// Get user data directory based on platform
|
|
function getUserDataDir() {
|
|
if (isCloud) return "/tmp"; // Fallback for Workers
|
|
|
|
if (process.env.DATA_DIR) return process.env.DATA_DIR;
|
|
|
|
const platform = process.platform;
|
|
const homeDir = os.homedir();
|
|
const appName = getAppName();
|
|
|
|
if (platform === "win32") {
|
|
return path.join(process.env.APPDATA || path.join(homeDir, "AppData", "Roaming"), appName);
|
|
} else {
|
|
// macOS & Linux: ~/.{appName}
|
|
return path.join(homeDir, `.${appName}`);
|
|
}
|
|
}
|
|
|
|
// Data file path - stored in user home directory
|
|
const DATA_DIR = getUserDataDir();
|
|
const DB_FILE = isCloud ? null : path.join(DATA_DIR, "db.json");
|
|
|
|
// Ensure data directory exists
|
|
if (!isCloud && !fs.existsSync(DATA_DIR)) {
|
|
fs.mkdirSync(DATA_DIR, { recursive: true });
|
|
}
|
|
|
|
// Default data structure
|
|
const defaultData = {
|
|
providerConnections: [],
|
|
providerNodes: [],
|
|
modelAliases: {},
|
|
mitmAlias: {},
|
|
combos: [],
|
|
apiKeys: [],
|
|
settings: {
|
|
cloudEnabled: false,
|
|
stickyRoundRobinLimit: 3,
|
|
requireLogin: true,
|
|
observabilityMaxRecords: 1000,
|
|
observabilityBatchSize: 20,
|
|
observabilityFlushIntervalMs: 5000,
|
|
observabilityMaxJsonSize: 1024
|
|
},
|
|
pricing: {} // NEW: pricing configuration
|
|
};
|
|
|
|
function cloneDefaultData() {
|
|
return {
|
|
providerConnections: [],
|
|
providerNodes: [],
|
|
modelAliases: {},
|
|
mitmAlias: {},
|
|
combos: [],
|
|
apiKeys: [],
|
|
settings: {
|
|
cloudEnabled: false,
|
|
stickyRoundRobinLimit: 3,
|
|
requireLogin: true,
|
|
observabilityMaxRecords: 1000,
|
|
observabilityBatchSize: 20,
|
|
observabilityFlushIntervalMs: 5000,
|
|
observabilityMaxJsonSize: 1024
|
|
},
|
|
pricing: {},
|
|
};
|
|
}
|
|
|
|
function ensureDbShape(data) {
|
|
const defaults = cloneDefaultData();
|
|
const next = data && typeof data === "object" ? data : {};
|
|
let changed = false;
|
|
|
|
for (const [key, defaultValue] of Object.entries(defaults)) {
|
|
if (next[key] === undefined || next[key] === null) {
|
|
next[key] = defaultValue;
|
|
changed = true;
|
|
continue;
|
|
}
|
|
|
|
if (
|
|
key === "settings" &&
|
|
(typeof next.settings !== "object" || Array.isArray(next.settings))
|
|
) {
|
|
next.settings = { ...defaultValue };
|
|
changed = true;
|
|
continue;
|
|
}
|
|
|
|
if (
|
|
key === "settings" &&
|
|
typeof next.settings === "object" &&
|
|
!Array.isArray(next.settings)
|
|
) {
|
|
for (const [settingKey, settingDefault] of Object.entries(defaultValue)) {
|
|
if (next.settings[settingKey] === undefined) {
|
|
next.settings[settingKey] = settingDefault;
|
|
changed = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return { data: next, changed };
|
|
}
|
|
|
|
// Singleton instance
|
|
let dbInstance = null;
|
|
|
|
/**
|
|
* Get database instance (singleton)
|
|
*/
|
|
export async function getDb() {
|
|
if (isCloud) {
|
|
// Return in-memory DB for Workers
|
|
if (!dbInstance) {
|
|
const data = cloneDefaultData();
|
|
dbInstance = new Low({ read: async () => {}, write: async () => {} }, data);
|
|
dbInstance.data = data;
|
|
}
|
|
return dbInstance;
|
|
}
|
|
|
|
if (!dbInstance) {
|
|
const adapter = new JSONFile(DB_FILE);
|
|
dbInstance = new Low(adapter, cloneDefaultData());
|
|
}
|
|
|
|
// Always read latest disk state to avoid stale singleton data across route workers.
|
|
try {
|
|
await dbInstance.read();
|
|
} catch (error) {
|
|
if (error instanceof SyntaxError) {
|
|
console.warn('[DB] Corrupt JSON detected, resetting to defaults...');
|
|
dbInstance.data = cloneDefaultData();
|
|
await dbInstance.write();
|
|
} else {
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
// Initialize/migrate missing keys for older DB schema versions.
|
|
if (!dbInstance.data) {
|
|
dbInstance.data = cloneDefaultData();
|
|
await dbInstance.write();
|
|
} else {
|
|
const { data, changed } = ensureDbShape(dbInstance.data);
|
|
dbInstance.data = data;
|
|
if (changed) {
|
|
await dbInstance.write();
|
|
}
|
|
}
|
|
|
|
return dbInstance;
|
|
}
|
|
|
|
// ============ Provider Connections ============
|
|
|
|
/**
|
|
* Get all provider connections
|
|
*/
|
|
export async function getProviderConnections(filter = {}) {
|
|
const db = await getDb();
|
|
let connections = db.data.providerConnections || [];
|
|
|
|
if (filter.provider) {
|
|
connections = connections.filter(c => c.provider === filter.provider);
|
|
}
|
|
if (filter.isActive !== undefined) {
|
|
connections = connections.filter(c => c.isActive === filter.isActive);
|
|
}
|
|
|
|
// Sort by priority (lower = higher priority)
|
|
connections.sort((a, b) => (a.priority || 999) - (b.priority || 999));
|
|
|
|
return connections;
|
|
}
|
|
|
|
// ============ Provider Nodes ============
|
|
|
|
/**
|
|
* Get provider nodes
|
|
*/
|
|
export async function getProviderNodes(filter = {}) {
|
|
const db = await getDb();
|
|
let nodes = db.data.providerNodes || [];
|
|
|
|
if (filter.type) {
|
|
nodes = nodes.filter((node) => node.type === filter.type);
|
|
}
|
|
|
|
return nodes;
|
|
}
|
|
|
|
/**
|
|
* Get provider node by ID
|
|
*/
|
|
export async function getProviderNodeById(id) {
|
|
const db = await getDb();
|
|
return db.data.providerNodes.find((node) => node.id === id) || null;
|
|
}
|
|
|
|
/**
|
|
* Create provider node
|
|
*/
|
|
export async function createProviderNode(data) {
|
|
const db = await getDb();
|
|
|
|
// Initialize providerNodes if undefined (backward compatibility)
|
|
if (!db.data.providerNodes) {
|
|
db.data.providerNodes = [];
|
|
}
|
|
|
|
const now = new Date().toISOString();
|
|
|
|
const node = {
|
|
id: data.id || uuidv4(),
|
|
type: data.type,
|
|
name: data.name,
|
|
prefix: data.prefix,
|
|
apiType: data.apiType,
|
|
baseUrl: data.baseUrl,
|
|
createdAt: now,
|
|
updatedAt: now,
|
|
};
|
|
|
|
db.data.providerNodes.push(node);
|
|
await db.write();
|
|
|
|
return node;
|
|
}
|
|
|
|
/**
|
|
* Update provider node
|
|
*/
|
|
export async function updateProviderNode(id, data) {
|
|
const db = await getDb();
|
|
if (!db.data.providerNodes) {
|
|
db.data.providerNodes = [];
|
|
}
|
|
|
|
const index = db.data.providerNodes.findIndex((node) => node.id === id);
|
|
|
|
if (index === -1) return null;
|
|
|
|
db.data.providerNodes[index] = {
|
|
...db.data.providerNodes[index],
|
|
...data,
|
|
updatedAt: new Date().toISOString(),
|
|
};
|
|
|
|
await db.write();
|
|
|
|
return db.data.providerNodes[index];
|
|
}
|
|
|
|
/**
|
|
* Delete provider node
|
|
*/
|
|
export async function deleteProviderNode(id) {
|
|
const db = await getDb();
|
|
if (!db.data.providerNodes) {
|
|
db.data.providerNodes = [];
|
|
}
|
|
|
|
const index = db.data.providerNodes.findIndex((node) => node.id === id);
|
|
|
|
if (index === -1) return null;
|
|
|
|
const [removed] = db.data.providerNodes.splice(index, 1);
|
|
await db.write();
|
|
|
|
return removed;
|
|
}
|
|
|
|
/**
|
|
* Delete all provider connections by provider ID
|
|
*/
|
|
export async function deleteProviderConnectionsByProvider(providerId) {
|
|
const db = await getDb();
|
|
const beforeCount = db.data.providerConnections.length;
|
|
db.data.providerConnections = db.data.providerConnections.filter(
|
|
(connection) => connection.provider !== providerId
|
|
);
|
|
const deletedCount = beforeCount - db.data.providerConnections.length;
|
|
await db.write();
|
|
return deletedCount;
|
|
}
|
|
|
|
/**
|
|
* Get provider connection by ID
|
|
*/
|
|
export async function getProviderConnectionById(id) {
|
|
const db = await getDb();
|
|
return db.data.providerConnections.find(c => c.id === id) || null;
|
|
}
|
|
|
|
/**
|
|
* Create or update provider connection (upsert by provider + email/name)
|
|
*/
|
|
export async function createProviderConnection(data) {
|
|
const db = await getDb();
|
|
const now = new Date().toISOString();
|
|
|
|
// Check for existing connection with same provider and email (for OAuth)
|
|
// or same provider and name (for API key)
|
|
let existingIndex = -1;
|
|
if (data.authType === "oauth" && data.email) {
|
|
existingIndex = db.data.providerConnections.findIndex(
|
|
c => c.provider === data.provider && c.authType === "oauth" && c.email === data.email
|
|
);
|
|
} else if (data.authType === "apikey" && data.name) {
|
|
existingIndex = db.data.providerConnections.findIndex(
|
|
c => c.provider === data.provider && c.authType === "apikey" && c.name === data.name
|
|
);
|
|
}
|
|
|
|
// If exists, update instead of create
|
|
if (existingIndex !== -1) {
|
|
db.data.providerConnections[existingIndex] = {
|
|
...db.data.providerConnections[existingIndex],
|
|
...data,
|
|
updatedAt: now,
|
|
};
|
|
await db.write();
|
|
return db.data.providerConnections[existingIndex];
|
|
}
|
|
|
|
// Generate name for OAuth if not provided
|
|
let connectionName = data.name || null;
|
|
if (!connectionName && data.authType === "oauth") {
|
|
if (data.email) {
|
|
connectionName = data.email;
|
|
} else {
|
|
// Count existing connections for this provider to generate index
|
|
const existingCount = db.data.providerConnections.filter(
|
|
c => c.provider === data.provider
|
|
).length;
|
|
connectionName = `Account ${existingCount + 1}`;
|
|
}
|
|
}
|
|
|
|
// Auto-increment priority if not provided
|
|
let connectionPriority = data.priority;
|
|
if (!connectionPriority) {
|
|
const providerConnections = db.data.providerConnections.filter(
|
|
c => c.provider === data.provider
|
|
);
|
|
const maxPriority = providerConnections.reduce((max, c) => Math.max(max, c.priority || 0), 0);
|
|
connectionPriority = maxPriority + 1;
|
|
}
|
|
|
|
// Create new connection - only save fields with actual values
|
|
const connection = {
|
|
id: uuidv4(),
|
|
provider: data.provider,
|
|
authType: data.authType || "oauth",
|
|
name: connectionName,
|
|
priority: connectionPriority,
|
|
isActive: data.isActive !== undefined ? data.isActive : true,
|
|
createdAt: now,
|
|
updatedAt: now,
|
|
};
|
|
|
|
// Only add optional fields if they have values
|
|
const optionalFields = [
|
|
"displayName", "email", "globalPriority", "defaultModel",
|
|
"accessToken", "refreshToken", "expiresAt", "tokenType",
|
|
"scope", "idToken", "projectId", "apiKey", "testStatus",
|
|
"lastTested", "lastError", "lastErrorAt", "rateLimitedUntil", "expiresIn", "errorCode",
|
|
"consecutiveUseCount"
|
|
];
|
|
|
|
for (const field of optionalFields) {
|
|
if (data[field] !== undefined && data[field] !== null) {
|
|
connection[field] = data[field];
|
|
}
|
|
}
|
|
|
|
// Only add providerSpecificData if it has content
|
|
if (data.providerSpecificData && Object.keys(data.providerSpecificData).length > 0) {
|
|
connection.providerSpecificData = data.providerSpecificData;
|
|
}
|
|
|
|
db.data.providerConnections.push(connection);
|
|
await db.write();
|
|
|
|
// Reorder to ensure consistency
|
|
await reorderProviderConnections(data.provider);
|
|
|
|
return connection;
|
|
}
|
|
|
|
/**
|
|
* Update provider connection
|
|
*/
|
|
export async function updateProviderConnection(id, data) {
|
|
const db = await getDb();
|
|
const index = db.data.providerConnections.findIndex(c => c.id === id);
|
|
|
|
if (index === -1) return null;
|
|
|
|
const providerId = db.data.providerConnections[index].provider;
|
|
|
|
db.data.providerConnections[index] = {
|
|
...db.data.providerConnections[index],
|
|
...data,
|
|
updatedAt: new Date().toISOString(),
|
|
};
|
|
|
|
await db.write();
|
|
|
|
// Reorder if priority was changed
|
|
if (data.priority !== undefined) {
|
|
await reorderProviderConnections(providerId);
|
|
}
|
|
|
|
return db.data.providerConnections[index];
|
|
}
|
|
|
|
/**
|
|
* Delete provider connection
|
|
*/
|
|
export async function deleteProviderConnection(id) {
|
|
const db = await getDb();
|
|
const index = db.data.providerConnections.findIndex(c => c.id === id);
|
|
|
|
if (index === -1) return false;
|
|
|
|
const providerId = db.data.providerConnections[index].provider;
|
|
|
|
db.data.providerConnections.splice(index, 1);
|
|
await db.write();
|
|
|
|
// Reorder to fill gaps
|
|
await reorderProviderConnections(providerId);
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Reorder provider connections to ensure unique, sequential priorities
|
|
*/
|
|
export async function reorderProviderConnections(providerId) {
|
|
const db = await getDb();
|
|
if (!db.data.providerConnections) return;
|
|
|
|
const providerConnections = db.data.providerConnections
|
|
.filter(c => c.provider === providerId)
|
|
.sort((a, b) => {
|
|
// Sort by priority first
|
|
const pDiff = (a.priority || 0) - (b.priority || 0);
|
|
if (pDiff !== 0) return pDiff;
|
|
// Use updatedAt as tie-breaker (newer first)
|
|
return new Date(b.updatedAt || 0) - new Date(a.updatedAt || 0);
|
|
});
|
|
|
|
// Re-assign sequential priorities
|
|
providerConnections.forEach((conn, index) => {
|
|
conn.priority = index + 1;
|
|
});
|
|
|
|
await db.write();
|
|
}
|
|
|
|
// ============ Model Aliases ============
|
|
|
|
/**
|
|
* Get all model aliases
|
|
*/
|
|
export async function getModelAliases() {
|
|
const db = await getDb();
|
|
return db.data.modelAliases || {};
|
|
}
|
|
|
|
/**
|
|
* Set model alias
|
|
*/
|
|
export async function setModelAlias(alias, model) {
|
|
const db = await getDb();
|
|
db.data.modelAliases[alias] = model;
|
|
await db.write();
|
|
}
|
|
|
|
/**
|
|
* Delete model alias
|
|
*/
|
|
export async function deleteModelAlias(alias) {
|
|
const db = await getDb();
|
|
delete db.data.modelAliases[alias];
|
|
await db.write();
|
|
}
|
|
|
|
// ============ MITM Alias ============
|
|
|
|
export async function getMitmAlias(toolName) {
|
|
const db = await getDb();
|
|
const all = db.data.mitmAlias || {};
|
|
if (toolName) return all[toolName] || {};
|
|
return all;
|
|
}
|
|
|
|
export async function setMitmAliasAll(toolName, mappings) {
|
|
const db = await getDb();
|
|
if (!db.data.mitmAlias) db.data.mitmAlias = {};
|
|
db.data.mitmAlias[toolName] = mappings || {};
|
|
await db.write();
|
|
}
|
|
|
|
// ============ Combos ============
|
|
|
|
/**
|
|
* Get all combos
|
|
*/
|
|
export async function getCombos() {
|
|
const db = await getDb();
|
|
return db.data.combos || [];
|
|
}
|
|
|
|
/**
|
|
* Get combo by ID
|
|
*/
|
|
export async function getComboById(id) {
|
|
const db = await getDb();
|
|
return (db.data.combos || []).find(c => c.id === id) || null;
|
|
}
|
|
|
|
/**
|
|
* Get combo by name
|
|
*/
|
|
export async function getComboByName(name) {
|
|
const db = await getDb();
|
|
return (db.data.combos || []).find(c => c.name === name) || null;
|
|
}
|
|
|
|
/**
|
|
* Create combo
|
|
*/
|
|
export async function createCombo(data) {
|
|
const db = await getDb();
|
|
if (!db.data.combos) db.data.combos = [];
|
|
|
|
const now = new Date().toISOString();
|
|
const combo = {
|
|
id: uuidv4(),
|
|
name: data.name,
|
|
models: data.models || [],
|
|
createdAt: now,
|
|
updatedAt: now,
|
|
};
|
|
|
|
db.data.combos.push(combo);
|
|
await db.write();
|
|
return combo;
|
|
}
|
|
|
|
/**
|
|
* Update combo
|
|
*/
|
|
export async function updateCombo(id, data) {
|
|
const db = await getDb();
|
|
if (!db.data.combos) db.data.combos = [];
|
|
|
|
const index = db.data.combos.findIndex(c => c.id === id);
|
|
if (index === -1) return null;
|
|
|
|
db.data.combos[index] = {
|
|
...db.data.combos[index],
|
|
...data,
|
|
updatedAt: new Date().toISOString(),
|
|
};
|
|
|
|
await db.write();
|
|
return db.data.combos[index];
|
|
}
|
|
|
|
/**
|
|
* Delete combo
|
|
*/
|
|
export async function deleteCombo(id) {
|
|
const db = await getDb();
|
|
if (!db.data.combos) return false;
|
|
|
|
const index = db.data.combos.findIndex(c => c.id === id);
|
|
if (index === -1) return false;
|
|
|
|
db.data.combos.splice(index, 1);
|
|
await db.write();
|
|
return true;
|
|
}
|
|
|
|
// ============ API Keys ============
|
|
|
|
/**
|
|
* Get all API keys
|
|
*/
|
|
export async function getApiKeys() {
|
|
const db = await getDb();
|
|
return db.data.apiKeys || [];
|
|
}
|
|
|
|
/**
|
|
* Generate short random key (8 chars)
|
|
*/
|
|
function generateShortKey() {
|
|
const chars = "abcdefghijklmnopqrstuvwxyz0123456789";
|
|
let result = "";
|
|
for (let i = 0; i < 8; i++) {
|
|
result += chars.charAt(Math.floor(Math.random() * chars.length));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Create API key
|
|
* @param {string} name - Key name
|
|
* @param {string} machineId - MachineId (required)
|
|
*/
|
|
export async function createApiKey(name, machineId) {
|
|
if (!machineId) {
|
|
throw new Error("machineId is required");
|
|
}
|
|
|
|
const db = await getDb();
|
|
const now = new Date().toISOString();
|
|
|
|
// Always use new format: sk-{machineId}-{keyId}-{crc8}
|
|
const { generateApiKeyWithMachine } = await import("@/shared/utils/apiKey");
|
|
const result = generateApiKeyWithMachine(machineId);
|
|
|
|
const apiKey = {
|
|
id: uuidv4(),
|
|
name: name,
|
|
key: result.key,
|
|
machineId: machineId,
|
|
createdAt: now,
|
|
};
|
|
|
|
db.data.apiKeys.push(apiKey);
|
|
await db.write();
|
|
|
|
return apiKey;
|
|
}
|
|
|
|
/**
|
|
* Delete API key
|
|
*/
|
|
export async function deleteApiKey(id) {
|
|
const db = await getDb();
|
|
const index = db.data.apiKeys.findIndex(k => k.id === id);
|
|
|
|
if (index === -1) return false;
|
|
|
|
db.data.apiKeys.splice(index, 1);
|
|
await db.write();
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Validate API key
|
|
*/
|
|
export async function validateApiKey(key) {
|
|
const db = await getDb();
|
|
return db.data.apiKeys.some(k => k.key === key);
|
|
}
|
|
|
|
// ============ Data Cleanup ============
|
|
|
|
/**
|
|
* Remove null/empty fields from all provider connections to reduce db size
|
|
*/
|
|
export async function cleanupProviderConnections() {
|
|
const db = await getDb();
|
|
const fieldsToCheck = [
|
|
"displayName", "email", "globalPriority", "defaultModel",
|
|
"accessToken", "refreshToken", "expiresAt", "tokenType",
|
|
"scope", "idToken", "projectId", "apiKey", "testStatus",
|
|
"lastTested", "lastError", "lastErrorAt", "rateLimitedUntil", "expiresIn",
|
|
"consecutiveUseCount"
|
|
];
|
|
|
|
let cleaned = 0;
|
|
for (const connection of db.data.providerConnections) {
|
|
for (const field of fieldsToCheck) {
|
|
if (connection[field] === null || connection[field] === undefined) {
|
|
delete connection[field];
|
|
cleaned++;
|
|
}
|
|
}
|
|
// Remove empty providerSpecificData
|
|
if (connection.providerSpecificData && Object.keys(connection.providerSpecificData).length === 0) {
|
|
delete connection.providerSpecificData;
|
|
cleaned++;
|
|
}
|
|
}
|
|
|
|
if (cleaned > 0) {
|
|
await db.write();
|
|
}
|
|
return cleaned;
|
|
}
|
|
|
|
// ============ Settings ============
|
|
|
|
/**
|
|
* Get settings
|
|
*/
|
|
export async function getSettings() {
|
|
const db = await getDb();
|
|
return db.data.settings || { cloudEnabled: false };
|
|
}
|
|
|
|
/**
|
|
* Update settings
|
|
*/
|
|
export async function updateSettings(updates) {
|
|
const db = await getDb();
|
|
db.data.settings = {
|
|
...db.data.settings,
|
|
...updates
|
|
};
|
|
await db.write();
|
|
return db.data.settings;
|
|
}
|
|
|
|
/**
|
|
* Check if cloud is enabled
|
|
*/
|
|
export async function isCloudEnabled() {
|
|
const settings = await getSettings();
|
|
return settings.cloudEnabled === true;
|
|
}
|
|
|
|
/**
|
|
* Get cloud URL (UI config > env > default)
|
|
*/
|
|
export async function getCloudUrl() {
|
|
const settings = await getSettings();
|
|
return settings.cloudUrl
|
|
|| process.env.CLOUD_URL
|
|
|| process.env.NEXT_PUBLIC_CLOUD_URL
|
|
|| "";
|
|
}
|
|
|
|
// ============ Pricing ============
|
|
|
|
/**
|
|
* Get pricing configuration
|
|
* Returns merged user pricing with defaults
|
|
*/
|
|
export async function getPricing() {
|
|
const db = await getDb();
|
|
const userPricing = db.data.pricing || {};
|
|
|
|
// Import default pricing
|
|
const { getDefaultPricing } = await import("@/shared/constants/pricing.js");
|
|
const defaultPricing = getDefaultPricing();
|
|
|
|
// Merge user pricing with defaults
|
|
// User pricing overrides defaults for specific provider/model combinations
|
|
const mergedPricing = {};
|
|
|
|
for (const [provider, models] of Object.entries(defaultPricing)) {
|
|
mergedPricing[provider] = { ...models };
|
|
|
|
// Apply user overrides if they exist
|
|
if (userPricing[provider]) {
|
|
for (const [model, pricing] of Object.entries(userPricing[provider])) {
|
|
if (mergedPricing[provider][model]) {
|
|
mergedPricing[provider][model] = { ...mergedPricing[provider][model], ...pricing };
|
|
} else {
|
|
mergedPricing[provider][model] = pricing;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add any user-only pricing entries
|
|
for (const [provider, models] of Object.entries(userPricing)) {
|
|
if (!mergedPricing[provider]) {
|
|
mergedPricing[provider] = { ...models };
|
|
} else {
|
|
for (const [model, pricing] of Object.entries(models)) {
|
|
if (!mergedPricing[provider][model]) {
|
|
mergedPricing[provider][model] = pricing;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return mergedPricing;
|
|
}
|
|
|
|
/**
|
|
* Get pricing for a specific provider and model
|
|
*/
|
|
export async function getPricingForModel(provider, model) {
|
|
const pricing = await getPricing();
|
|
|
|
// Try direct lookup
|
|
if (pricing[provider]?.[model]) {
|
|
return pricing[provider][model];
|
|
}
|
|
|
|
// Try mapping provider ID to alias
|
|
// We need to duplicate the mapping here or import it
|
|
// Since we can't easily import from open-sse, we'll implement the mapping locally
|
|
const PROVIDER_ID_TO_ALIAS = {
|
|
claude: "cc",
|
|
codex: "cx",
|
|
"gemini-cli": "gc",
|
|
qwen: "qw",
|
|
iflow: "if",
|
|
antigravity: "ag",
|
|
github: "gh",
|
|
openai: "openai",
|
|
anthropic: "anthropic",
|
|
gemini: "gemini",
|
|
openrouter: "openrouter",
|
|
glm: "glm",
|
|
kimi: "kimi",
|
|
minimax: "minimax",
|
|
};
|
|
|
|
const alias = PROVIDER_ID_TO_ALIAS[provider];
|
|
if (alias && pricing[alias]) {
|
|
return pricing[alias][model] || null;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Update pricing configuration
|
|
* @param {object} pricingData - New pricing data to merge
|
|
*/
|
|
export async function updatePricing(pricingData) {
|
|
const db = await getDb();
|
|
|
|
// Ensure pricing object exists
|
|
if (!db.data.pricing) {
|
|
db.data.pricing = {};
|
|
}
|
|
|
|
// Merge new pricing data
|
|
for (const [provider, models] of Object.entries(pricingData)) {
|
|
if (!db.data.pricing[provider]) {
|
|
db.data.pricing[provider] = {};
|
|
}
|
|
|
|
for (const [model, pricing] of Object.entries(models)) {
|
|
db.data.pricing[provider][model] = pricing;
|
|
}
|
|
}
|
|
|
|
await db.write();
|
|
return db.data.pricing;
|
|
}
|
|
|
|
/**
|
|
* Reset pricing to defaults for specific provider/model
|
|
* @param {string} provider - Provider ID
|
|
* @param {string} model - Model ID (optional, if not provided resets entire provider)
|
|
*/
|
|
export async function resetPricing(provider, model) {
|
|
const db = await getDb();
|
|
|
|
if (!db.data.pricing) {
|
|
db.data.pricing = {};
|
|
}
|
|
|
|
if (model) {
|
|
// Reset specific model
|
|
if (db.data.pricing[provider]) {
|
|
delete db.data.pricing[provider][model];
|
|
// Clean up empty provider objects
|
|
if (Object.keys(db.data.pricing[provider]).length === 0) {
|
|
delete db.data.pricing[provider];
|
|
}
|
|
}
|
|
} else {
|
|
// Reset entire provider
|
|
delete db.data.pricing[provider];
|
|
}
|
|
|
|
await db.write();
|
|
return db.data.pricing;
|
|
}
|
|
|
|
/**
|
|
* Reset all pricing to defaults
|
|
*/
|
|
export async function resetAllPricing() {
|
|
const db = await getDb();
|
|
db.data.pricing = {};
|
|
await db.write();
|
|
return db.data.pricing;
|
|
}
|