@jsonstudio/rcc 0.89.1803 → 0.89.1959
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/configsamples/config.json +19 -0
- package/configsamples/provider/deepseek/config.v1.json +59 -0
- package/dist/build-info.js +2 -2
- package/dist/cli/commands/claude.d.ts +4 -0
- package/dist/cli/commands/claude.js +56 -0
- package/dist/cli/commands/claude.js.map +1 -0
- package/dist/cli/commands/clock-admin.d.ts +20 -0
- package/dist/cli/commands/clock-admin.js +234 -0
- package/dist/cli/commands/clock-admin.js.map +1 -0
- package/dist/cli/commands/code.d.ts +0 -42
- package/dist/cli/commands/code.js +4 -414
- package/dist/cli/commands/code.js.map +1 -1
- package/dist/cli/commands/codex.d.ts +4 -0
- package/dist/cli/commands/codex.js +43 -0
- package/dist/cli/commands/codex.js.map +1 -0
- package/dist/cli/commands/examples.js +13 -16
- package/dist/cli/commands/examples.js.map +1 -1
- package/dist/cli/commands/init/basic.d.ts +40 -0
- package/dist/cli/commands/init/basic.js +482 -0
- package/dist/cli/commands/init/basic.js.map +1 -0
- package/dist/cli/commands/init/camoufox.d.ts +7 -0
- package/dist/cli/commands/init/camoufox.js +59 -0
- package/dist/cli/commands/init/camoufox.js.map +1 -0
- package/dist/cli/commands/init/interactive.d.ts +18 -0
- package/dist/cli/commands/init/interactive.js +223 -0
- package/dist/cli/commands/init/interactive.js.map +1 -0
- package/dist/cli/commands/init/shared.d.ts +66 -0
- package/dist/cli/commands/init/shared.js +9 -0
- package/dist/cli/commands/init/shared.js.map +1 -0
- package/dist/cli/commands/init/workflows.d.ts +29 -0
- package/dist/cli/commands/init/workflows.js +341 -0
- package/dist/cli/commands/init/workflows.js.map +1 -0
- package/dist/cli/commands/init.d.ts +2 -26
- package/dist/cli/commands/init.js +220 -53
- package/dist/cli/commands/init.js.map +1 -1
- package/dist/cli/commands/launcher-kernel.d.ts +78 -0
- package/dist/cli/commands/launcher-kernel.js +1194 -0
- package/dist/cli/commands/launcher-kernel.js.map +1 -0
- package/dist/cli/commands/start.js +27 -1
- package/dist/cli/commands/start.js.map +1 -1
- package/dist/cli/commands/status.d.ts +2 -0
- package/dist/cli/commands/status.js +24 -1
- package/dist/cli/commands/status.js.map +1 -1
- package/dist/cli/commands/stop.d.ts +1 -0
- package/dist/cli/commands/stop.js +201 -4
- package/dist/cli/commands/stop.js.map +1 -1
- package/dist/cli/commands/tmux-inject.d.ts +20 -0
- package/dist/cli/commands/tmux-inject.js +212 -0
- package/dist/cli/commands/tmux-inject.js.map +1 -0
- package/dist/cli/config/init-provider-catalog.js +34 -0
- package/dist/cli/config/init-provider-catalog.js.map +1 -1
- package/dist/cli/register/claude-command.d.ts +3 -0
- package/dist/cli/register/claude-command.js +5 -0
- package/dist/cli/register/claude-command.js.map +1 -0
- package/dist/cli/register/clock-admin-command.d.ts +3 -0
- package/dist/cli/register/clock-admin-command.js +5 -0
- package/dist/cli/register/clock-admin-command.js.map +1 -0
- package/dist/cli/register/codex-command.d.ts +3 -0
- package/dist/cli/register/codex-command.js +5 -0
- package/dist/cli/register/codex-command.js.map +1 -0
- package/dist/cli/register/status-config-commands.d.ts +2 -0
- package/dist/cli/register/status-config-commands.js.map +1 -1
- package/dist/cli/register/tmux-inject-command.d.ts +3 -0
- package/dist/cli/register/tmux-inject-command.js +5 -0
- package/dist/cli/register/tmux-inject-command.js.map +1 -0
- package/dist/cli/server/port-utils.d.ts +3 -2
- package/dist/cli/server/port-utils.js +171 -32
- package/dist/cli/server/port-utils.js.map +1 -1
- package/dist/cli.js +45 -6
- package/dist/cli.js.map +1 -1
- package/dist/client/gemini/gemini-protocol-client.js +56 -5
- package/dist/client/gemini/gemini-protocol-client.js.map +1 -1
- package/dist/commands/token-daemon.js +59 -7
- package/dist/commands/token-daemon.js.map +1 -1
- package/dist/commands/validate.js +87 -15
- package/dist/commands/validate.js.map +1 -1
- package/dist/config/routecodex-config-loader.js +31 -2
- package/dist/config/routecodex-config-loader.js.map +1 -1
- package/dist/docs/daemon-admin-ui.html +948 -74
- package/dist/index.d.ts +1 -0
- package/dist/index.js +325 -37
- package/dist/index.js.map +1 -1
- package/dist/manager/quota/provider-quota-center.js +8 -14
- package/dist/manager/quota/provider-quota-center.js.map +1 -1
- package/dist/modules/llmswitch/bridge.d.ts +39 -0
- package/dist/modules/llmswitch/bridge.js +169 -0
- package/dist/modules/llmswitch/bridge.js.map +1 -1
- package/dist/modules/pipeline/utils/colored-logger.js +1 -1
- package/dist/modules/pipeline/utils/colored-logger.js.map +1 -1
- package/dist/providers/auth/deepseek-account-auth.d.ts +39 -0
- package/dist/providers/auth/deepseek-account-auth.js +329 -0
- package/dist/providers/auth/deepseek-account-auth.js.map +1 -0
- package/dist/providers/auth/deepseek-account-token-acquirer.d.ts +15 -0
- package/dist/providers/auth/deepseek-account-token-acquirer.js +644 -0
- package/dist/providers/auth/deepseek-account-token-acquirer.js.map +1 -0
- package/dist/providers/auth/oauth-lifecycle.js +26 -4
- package/dist/providers/auth/oauth-lifecycle.js.map +1 -1
- package/dist/providers/auth/oauth-repair-cooldown.d.ts +5 -0
- package/dist/providers/auth/oauth-repair-cooldown.js +39 -0
- package/dist/providers/auth/oauth-repair-cooldown.js.map +1 -1
- package/dist/providers/auth/token-scanner/index.d.ts +6 -0
- package/dist/providers/auth/token-scanner/index.js +53 -0
- package/dist/providers/auth/token-scanner/index.js.map +1 -1
- package/dist/providers/core/api/provider-config.d.ts +17 -2
- package/dist/providers/core/api/provider-types.d.ts +6 -0
- package/dist/providers/core/api/provider-types.js.map +1 -1
- package/dist/providers/core/config/camoufox-launcher.d.ts +7 -0
- package/dist/providers/core/config/camoufox-launcher.js +68 -21
- package/dist/providers/core/config/camoufox-launcher.js.map +1 -1
- package/dist/providers/core/config/service-profiles.js +19 -0
- package/dist/providers/core/config/service-profiles.js.map +1 -1
- package/dist/providers/core/contracts/deepseek-provider-contract.d.ts +34 -0
- package/dist/providers/core/contracts/deepseek-provider-contract.js +100 -0
- package/dist/providers/core/contracts/deepseek-provider-contract.js.map +1 -0
- package/dist/providers/core/runtime/anthropic-http-provider.d.ts +0 -5
- package/dist/providers/core/runtime/anthropic-http-provider.js +0 -26
- package/dist/providers/core/runtime/anthropic-http-provider.js.map +1 -1
- package/dist/providers/core/runtime/deepseek-http-provider.d.ts +35 -0
- package/dist/providers/core/runtime/deepseek-http-provider.js +373 -0
- package/dist/providers/core/runtime/deepseek-http-provider.js.map +1 -0
- package/dist/providers/core/runtime/deepseek-session-pow.d.ts +55 -0
- package/dist/providers/core/runtime/deepseek-session-pow.js +422 -0
- package/dist/providers/core/runtime/deepseek-session-pow.js.map +1 -0
- package/dist/providers/core/runtime/gemini-cli-http-provider.d.ts +0 -3
- package/dist/providers/core/runtime/gemini-cli-http-provider.js +0 -72
- package/dist/providers/core/runtime/gemini-cli-http-provider.js.map +1 -1
- package/dist/providers/core/runtime/gemini-http-provider.d.ts +1 -7
- package/dist/providers/core/runtime/gemini-http-provider.js +3 -110
- package/dist/providers/core/runtime/gemini-http-provider.js.map +1 -1
- package/dist/providers/core/runtime/http-request-executor.d.ts +1 -0
- package/dist/providers/core/runtime/http-request-executor.js +4 -0
- package/dist/providers/core/runtime/http-request-executor.js.map +1 -1
- package/dist/providers/core/runtime/http-transport-provider.d.ts +10 -4
- package/dist/providers/core/runtime/http-transport-provider.js +308 -82
- package/dist/providers/core/runtime/http-transport-provider.js.map +1 -1
- package/dist/providers/core/runtime/iflow-http-provider.d.ts +0 -4
- package/dist/providers/core/runtime/iflow-http-provider.js +0 -28
- package/dist/providers/core/runtime/iflow-http-provider.js.map +1 -1
- package/dist/providers/core/runtime/provider-factory.d.ts +5 -0
- package/dist/providers/core/runtime/provider-factory.js +59 -6
- package/dist/providers/core/runtime/provider-factory.js.map +1 -1
- package/dist/providers/core/runtime/responses-provider.d.ts +0 -2
- package/dist/providers/core/runtime/responses-provider.js +0 -11
- package/dist/providers/core/runtime/responses-provider.js.map +1 -1
- package/dist/providers/core/strategies/oauth-device-flow.js +16 -1
- package/dist/providers/core/strategies/oauth-device-flow.js.map +1 -1
- package/dist/providers/core/utils/provider-type-utils.js +2 -1
- package/dist/providers/core/utils/provider-type-utils.js.map +1 -1
- package/dist/providers/profile/families/anthropic-profile.d.ts +2 -0
- package/dist/providers/profile/families/anthropic-profile.js +32 -0
- package/dist/providers/profile/families/anthropic-profile.js.map +1 -0
- package/dist/providers/profile/families/antigravity-profile.d.ts +2 -0
- package/dist/providers/profile/families/antigravity-profile.js +109 -0
- package/dist/providers/profile/families/antigravity-profile.js.map +1 -0
- package/dist/providers/profile/families/glm-profile.d.ts +2 -0
- package/dist/providers/profile/families/glm-profile.js +48 -0
- package/dist/providers/profile/families/glm-profile.js.map +1 -0
- package/dist/providers/profile/families/iflow-profile.d.ts +2 -0
- package/dist/providers/profile/families/iflow-profile.js +232 -0
- package/dist/providers/profile/families/iflow-profile.js.map +1 -0
- package/dist/providers/profile/families/qwen-profile.d.ts +2 -0
- package/dist/providers/profile/families/qwen-profile.js +14 -0
- package/dist/providers/profile/families/qwen-profile.js.map +1 -0
- package/dist/providers/profile/families/responses-profile.d.ts +2 -0
- package/dist/providers/profile/families/responses-profile.js +28 -0
- package/dist/providers/profile/families/responses-profile.js.map +1 -0
- package/dist/providers/profile/profile-contracts.d.ts +74 -0
- package/dist/providers/profile/profile-contracts.js +2 -0
- package/dist/providers/profile/profile-contracts.js.map +1 -0
- package/dist/providers/profile/profile-registry.d.ts +3 -0
- package/dist/providers/profile/profile-registry.js +40 -0
- package/dist/providers/profile/profile-registry.js.map +1 -0
- package/dist/providers/profile/provider-directory.d.ts +2 -0
- package/dist/providers/profile/provider-directory.js +55 -0
- package/dist/providers/profile/provider-directory.js.map +1 -0
- package/dist/providers/profile/provider-profile-loader.js +43 -3
- package/dist/providers/profile/provider-profile-loader.js.map +1 -1
- package/dist/providers/profile/provider-profile.d.ts +8 -0
- package/dist/scripts/deepseek/pow-solver.mjs +146 -0
- package/dist/scripts/deepseek/sha3_wasm_bg.7b9ca65ddd.wasm +0 -0
- package/dist/server/handlers/config-admin-handler.js +27 -0
- package/dist/server/handlers/config-admin-handler.js.map +1 -1
- package/dist/server/runtime/http-server/clock-client-registry.d.ts +113 -0
- package/dist/server/runtime/http-server/clock-client-registry.js +592 -0
- package/dist/server/runtime/http-server/clock-client-registry.js.map +1 -0
- package/dist/server/runtime/http-server/clock-client-routes.d.ts +2 -0
- package/dist/server/runtime/http-server/clock-client-routes.js +481 -0
- package/dist/server/runtime/http-server/clock-client-routes.js.map +1 -0
- package/dist/server/runtime/http-server/clock-daemon-inject-config.d.ts +1 -0
- package/dist/server/runtime/http-server/clock-daemon-inject-config.js +11 -0
- package/dist/server/runtime/http-server/clock-daemon-inject-config.js.map +1 -0
- package/dist/server/runtime/http-server/daemon-admin/auth-handler.js +3 -3
- package/dist/server/runtime/http-server/daemon-admin/auth-handler.js.map +1 -1
- package/dist/server/runtime/http-server/daemon-admin/auth-session.d.ts +1 -0
- package/dist/server/runtime/http-server/daemon-admin/auth-session.js +18 -2
- package/dist/server/runtime/http-server/daemon-admin/auth-session.js.map +1 -1
- package/dist/server/runtime/http-server/daemon-admin/control-handler.js +2 -15
- package/dist/server/runtime/http-server/daemon-admin/control-handler.js.map +1 -1
- package/dist/server/runtime/http-server/daemon-admin/credentials-handler.js +65 -7
- package/dist/server/runtime/http-server/daemon-admin/credentials-handler.js.map +1 -1
- package/dist/server/runtime/http-server/executor-metadata.js +37 -1
- package/dist/server/runtime/http-server/executor-metadata.js.map +1 -1
- package/dist/server/runtime/http-server/executor-provider.js +55 -0
- package/dist/server/runtime/http-server/executor-provider.js.map +1 -1
- package/dist/server/runtime/http-server/executor-response.js +49 -1
- package/dist/server/runtime/http-server/executor-response.js.map +1 -1
- package/dist/server/runtime/http-server/index.d.ts +10 -0
- package/dist/server/runtime/http-server/index.js +534 -9
- package/dist/server/runtime/http-server/index.js.map +1 -1
- package/dist/server/runtime/http-server/managed-process-probe.d.ts +6 -0
- package/dist/server/runtime/http-server/managed-process-probe.js +294 -0
- package/dist/server/runtime/http-server/managed-process-probe.js.map +1 -0
- package/dist/server/runtime/http-server/middleware.js +16 -1
- package/dist/server/runtime/http-server/middleware.js.map +1 -1
- package/dist/server/runtime/http-server/provider-utils.js +6 -2
- package/dist/server/runtime/http-server/provider-utils.js.map +1 -1
- package/dist/server/runtime/http-server/request-executor.d.ts +1 -0
- package/dist/server/runtime/http-server/request-executor.js +360 -35
- package/dist/server/runtime/http-server/request-executor.js.map +1 -1
- package/dist/server/runtime/http-server/routes.js +95 -3
- package/dist/server/runtime/http-server/routes.js.map +1 -1
- package/dist/server/runtime/http-server/stats-manager.d.ts +10 -0
- package/dist/server/runtime/http-server/stats-manager.js +119 -16
- package/dist/server/runtime/http-server/stats-manager.js.map +1 -1
- package/dist/server/runtime/http-server/tmux-session-probe.d.ts +3 -0
- package/dist/server/runtime/http-server/tmux-session-probe.js +101 -0
- package/dist/server/runtime/http-server/tmux-session-probe.js.map +1 -0
- package/dist/server/utils/stage-logger.js +21 -5
- package/dist/server/utils/stage-logger.js.map +1 -1
- package/dist/token-daemon/index.js +59 -10
- package/dist/token-daemon/index.js.map +1 -1
- package/dist/token-daemon/server-utils.d.ts +1 -0
- package/dist/token-daemon/server-utils.js +4 -1
- package/dist/token-daemon/server-utils.js.map +1 -1
- package/dist/token-daemon/token-daemon.js +38 -4
- package/dist/token-daemon/token-daemon.js.map +1 -1
- package/dist/token-daemon/token-types.d.ts +1 -1
- package/dist/token-daemon/token-types.js +2 -1
- package/dist/token-daemon/token-types.js.map +1 -1
- package/dist/token-daemon/token-utils.js +5 -2
- package/dist/token-daemon/token-utils.js.map +1 -1
- package/dist/utils/clock-client-token.d.ts +3 -0
- package/dist/utils/clock-client-token.js +54 -0
- package/dist/utils/clock-client-token.js.map +1 -0
- package/dist/utils/managed-server-pids.d.ts +25 -0
- package/dist/utils/managed-server-pids.js +176 -0
- package/dist/utils/managed-server-pids.js.map +1 -0
- package/dist/utils/process-lifecycle-logger.d.ts +8 -0
- package/dist/utils/process-lifecycle-logger.js +151 -0
- package/dist/utils/process-lifecycle-logger.js.map +1 -0
- package/dist/utils/runtime-exit-forensics.d.ts +30 -0
- package/dist/utils/runtime-exit-forensics.js +101 -0
- package/dist/utils/runtime-exit-forensics.js.map +1 -0
- package/dist/utils/shutdown-caller-context.d.ts +22 -0
- package/dist/utils/shutdown-caller-context.js +25 -0
- package/dist/utils/shutdown-caller-context.js.map +1 -0
- package/docs/PROVIDERS_BUILTIN.md +8 -0
- package/docs/PROVIDER_TYPES.md +3 -1
- package/docs/SERVERTOOL_PRE_COMMAND_HOOKS.md +85 -0
- package/docs/clock-client-daemon-design.md +343 -0
- package/docs/daemon-admin-ui.html +948 -74
- package/docs/providers/deepseek-web-provider-design.md +192 -0
- package/docs/routing-instructions.md +4 -1
- package/docs/stop-message-auto.md +4 -3
- package/docs/v2-architecture/PROVIDER-V2-CHANGESET-RELEASE-CHECKLIST.md +80 -0
- package/docs/v2-architecture/PROVIDER-V2-LAYERING-ADR-DRAFT.md +225 -0
- package/docs/v2-architecture/PROVIDER-V2-MIGRATION-MATRIX-DRAFT.md +88 -0
- package/docs/v2-architecture/PROVIDER-V2-PHASED-MIGRATION-ROLLBACK-DRAFT.md +164 -0
- package/docs/v2-architecture/PROVIDER-V2-PROFILE-API-REGISTRY-DRAFT.md +201 -0
- package/docs/v2-architecture/PROVIDER-V2-PROFILE-GEMINI-DRAFT.md +56 -0
- package/docs/v2-architecture/PROVIDER-V2-REFACTOR-OVERVIEW-DRAFT.md +102 -0
- package/docs/v2-architecture/PROVIDER-V2-VERIFICATION-MATRIX-DRAFT.md +163 -0
- package/package.json +10 -9
- package/scripts/copy-compat-assets.mjs +18 -0
- package/scripts/copy-modules-config.mjs +1 -0
- package/scripts/deepseek/pow-solver.mjs +146 -0
- package/scripts/deepseek/sha3_wasm_bg.7b9ca65ddd.wasm +0 -0
- package/scripts/ensure-cli-executable.mjs +64 -0
- package/scripts/install-global.sh +5 -2
- package/scripts/install.sh +1 -1
- package/scripts/monitor/daemon-kill-watch.mjs +184 -0
- package/scripts/monitor/port-kill-watch.sh +74 -0
- package/scripts/quick-install.sh +1 -1
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
import fs from 'fs/promises';
|
|
2
|
+
import fsSync from 'fs';
|
|
3
|
+
import os from 'os';
|
|
4
|
+
import path from 'path';
|
|
5
|
+
const DEFAULT_LOG_DIR = path.join(os.homedir(), '.routecodex', 'logs');
|
|
6
|
+
const DEFAULT_LOG_PATH = path.join(DEFAULT_LOG_DIR, 'process-lifecycle.jsonl');
|
|
7
|
+
let writeQueue = Promise.resolve();
|
|
8
|
+
function parseBool(value, defaultValue) {
|
|
9
|
+
if (typeof value !== 'string') {
|
|
10
|
+
return defaultValue;
|
|
11
|
+
}
|
|
12
|
+
const normalized = value.trim().toLowerCase();
|
|
13
|
+
if (!normalized) {
|
|
14
|
+
return defaultValue;
|
|
15
|
+
}
|
|
16
|
+
if (normalized === '1' || normalized === 'true' || normalized === 'yes' || normalized === 'on') {
|
|
17
|
+
return true;
|
|
18
|
+
}
|
|
19
|
+
if (normalized === '0' || normalized === 'false' || normalized === 'no' || normalized === 'off') {
|
|
20
|
+
return false;
|
|
21
|
+
}
|
|
22
|
+
return defaultValue;
|
|
23
|
+
}
|
|
24
|
+
function resolveLogPath() {
|
|
25
|
+
const envPath = process.env.ROUTECODEX_PROCESS_LIFECYCLE_LOG || process.env.RCC_PROCESS_LIFECYCLE_LOG;
|
|
26
|
+
if (typeof envPath === 'string' && envPath.trim()) {
|
|
27
|
+
return envPath.trim();
|
|
28
|
+
}
|
|
29
|
+
return DEFAULT_LOG_PATH;
|
|
30
|
+
}
|
|
31
|
+
function shouldConsoleLog() {
|
|
32
|
+
return parseBool(process.env.ROUTECODEX_PROCESS_LIFECYCLE_CONSOLE || process.env.RCC_PROCESS_LIFECYCLE_CONSOLE, true);
|
|
33
|
+
}
|
|
34
|
+
function serializeUnknown(value) {
|
|
35
|
+
if (value === null || value === undefined) {
|
|
36
|
+
return null;
|
|
37
|
+
}
|
|
38
|
+
const primitiveType = typeof value;
|
|
39
|
+
if (primitiveType === 'string' || primitiveType === 'number' || primitiveType === 'boolean') {
|
|
40
|
+
return value;
|
|
41
|
+
}
|
|
42
|
+
if (Array.isArray(value)) {
|
|
43
|
+
return value.map((item) => serializeUnknown(item));
|
|
44
|
+
}
|
|
45
|
+
if (value instanceof Error) {
|
|
46
|
+
const out = {
|
|
47
|
+
name: value.name,
|
|
48
|
+
message: value.message
|
|
49
|
+
};
|
|
50
|
+
if (value.stack) {
|
|
51
|
+
out.stack = value.stack;
|
|
52
|
+
}
|
|
53
|
+
return out;
|
|
54
|
+
}
|
|
55
|
+
try {
|
|
56
|
+
return JSON.parse(JSON.stringify(value));
|
|
57
|
+
}
|
|
58
|
+
catch {
|
|
59
|
+
return String(value);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
function buildRecord(event) {
|
|
63
|
+
return {
|
|
64
|
+
ts: new Date().toISOString(),
|
|
65
|
+
pid: process.pid,
|
|
66
|
+
ppid: process.ppid,
|
|
67
|
+
event: event.event,
|
|
68
|
+
source: event.source,
|
|
69
|
+
...(event.details ? { details: serializeUnknown(event.details) } : {})
|
|
70
|
+
};
|
|
71
|
+
}
|
|
72
|
+
function formatConsoleLine(record) {
|
|
73
|
+
const details = record.details && typeof record.details === 'object' && !Array.isArray(record.details)
|
|
74
|
+
? record.details
|
|
75
|
+
: null;
|
|
76
|
+
const parts = [];
|
|
77
|
+
if (details) {
|
|
78
|
+
for (const key of ['signal', 'targetPid', 'port', 'result', 'reason']) {
|
|
79
|
+
const value = details[key];
|
|
80
|
+
if (value !== undefined && value !== null && value !== '') {
|
|
81
|
+
parts.push(`${key}=${String(value)}`);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
const suffix = parts.length ? ` ${parts.join(' ')}` : '';
|
|
86
|
+
return `[routecodex.lifecycle][${record.ts}] ${record.event} source=${record.source}${suffix}`;
|
|
87
|
+
}
|
|
88
|
+
function printBlue(line) {
|
|
89
|
+
const canColor = Boolean(process.stdout?.isTTY);
|
|
90
|
+
if (canColor) {
|
|
91
|
+
console.log(`\x1b[94m${line}\x1b[0m`);
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
console.log(line);
|
|
95
|
+
}
|
|
96
|
+
async function appendRecord(record) {
|
|
97
|
+
const logPath = resolveLogPath();
|
|
98
|
+
const dir = path.dirname(logPath);
|
|
99
|
+
if (!fsSync.existsSync(dir)) {
|
|
100
|
+
await fs.mkdir(dir, { recursive: true });
|
|
101
|
+
}
|
|
102
|
+
await fs.appendFile(logPath, `${JSON.stringify(record)}\n`, 'utf8');
|
|
103
|
+
}
|
|
104
|
+
function appendRecordSync(record) {
|
|
105
|
+
try {
|
|
106
|
+
const logPath = resolveLogPath();
|
|
107
|
+
const dir = path.dirname(logPath);
|
|
108
|
+
if (!fsSync.existsSync(dir)) {
|
|
109
|
+
fsSync.mkdirSync(dir, { recursive: true });
|
|
110
|
+
}
|
|
111
|
+
fsSync.appendFileSync(logPath, `${JSON.stringify(record)}\n`, 'utf8');
|
|
112
|
+
}
|
|
113
|
+
catch {
|
|
114
|
+
// Never throw from lifecycle logging.
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
function enqueueWrite(task) {
|
|
118
|
+
writeQueue = writeQueue.then(async () => {
|
|
119
|
+
try {
|
|
120
|
+
await task();
|
|
121
|
+
}
|
|
122
|
+
catch {
|
|
123
|
+
// Never throw from lifecycle logging.
|
|
124
|
+
}
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
export function logProcessLifecycle(event) {
|
|
128
|
+
const record = buildRecord(event);
|
|
129
|
+
if (shouldConsoleLog()) {
|
|
130
|
+
printBlue(formatConsoleLine(record));
|
|
131
|
+
}
|
|
132
|
+
enqueueWrite(async () => {
|
|
133
|
+
await appendRecord(record);
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
export function logProcessLifecycleSync(event) {
|
|
137
|
+
const record = buildRecord(event);
|
|
138
|
+
if (shouldConsoleLog()) {
|
|
139
|
+
printBlue(formatConsoleLine(record));
|
|
140
|
+
}
|
|
141
|
+
appendRecordSync(record);
|
|
142
|
+
}
|
|
143
|
+
export async function flushProcessLifecycleLogQueue() {
|
|
144
|
+
try {
|
|
145
|
+
await writeQueue;
|
|
146
|
+
}
|
|
147
|
+
catch {
|
|
148
|
+
// Never throw from lifecycle logging.
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
//# sourceMappingURL=process-lifecycle-logger.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"process-lifecycle-logger.js","sourceRoot":"","sources":["../../src/utils/process-lifecycle-logger.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
export type RuntimeExitMarker = {
|
|
2
|
+
kind: string;
|
|
3
|
+
code: number | null;
|
|
4
|
+
signal?: string;
|
|
5
|
+
message?: string;
|
|
6
|
+
recordedAt: string;
|
|
7
|
+
};
|
|
8
|
+
export type RuntimeLifecycleState = {
|
|
9
|
+
runId: string;
|
|
10
|
+
pid: number;
|
|
11
|
+
port: number;
|
|
12
|
+
startedAt: string;
|
|
13
|
+
buildVersion?: string;
|
|
14
|
+
buildMode?: string;
|
|
15
|
+
exit?: RuntimeExitMarker;
|
|
16
|
+
};
|
|
17
|
+
export type UngracefulInference = {
|
|
18
|
+
shouldReport: boolean;
|
|
19
|
+
reason: string;
|
|
20
|
+
};
|
|
21
|
+
export declare function resolveRuntimeLifecyclePath(port: number, routeCodexHomeDir?: string): string;
|
|
22
|
+
export declare function safeReadRuntimeLifecycle(filePath: string): RuntimeLifecycleState | null;
|
|
23
|
+
export declare function safeWriteRuntimeLifecycle(filePath: string, state: RuntimeLifecycleState): boolean;
|
|
24
|
+
export declare function safeMarkRuntimeExit(filePath: string, marker: RuntimeExitMarker): boolean;
|
|
25
|
+
export declare function isPidAliveForForensics(pid: number): boolean;
|
|
26
|
+
export declare function inferUngracefulPreviousExit(args: {
|
|
27
|
+
previous: RuntimeLifecycleState | null;
|
|
28
|
+
currentPid: number;
|
|
29
|
+
isPidAlive?: (pid: number) => boolean;
|
|
30
|
+
}): UngracefulInference;
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
import fsSync from 'node:fs';
|
|
2
|
+
import path from 'node:path';
|
|
3
|
+
import { homedir } from 'node:os';
|
|
4
|
+
function normalizePort(port) {
|
|
5
|
+
return Number.isFinite(port) && port > 0 ? Math.floor(port) : 0;
|
|
6
|
+
}
|
|
7
|
+
export function resolveRuntimeLifecyclePath(port, routeCodexHomeDir) {
|
|
8
|
+
const home = routeCodexHomeDir || path.join(homedir(), '.routecodex');
|
|
9
|
+
return path.join(home, 'state', 'runtime-lifecycle', `server-${normalizePort(port)}.json`);
|
|
10
|
+
}
|
|
11
|
+
function ensureParentDir(filePath) {
|
|
12
|
+
const dir = path.dirname(filePath);
|
|
13
|
+
if (!fsSync.existsSync(dir)) {
|
|
14
|
+
fsSync.mkdirSync(dir, { recursive: true });
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
export function safeReadRuntimeLifecycle(filePath) {
|
|
18
|
+
try {
|
|
19
|
+
if (!fsSync.existsSync(filePath)) {
|
|
20
|
+
return null;
|
|
21
|
+
}
|
|
22
|
+
const raw = fsSync.readFileSync(filePath, 'utf8');
|
|
23
|
+
if (!raw.trim()) {
|
|
24
|
+
return null;
|
|
25
|
+
}
|
|
26
|
+
const parsed = JSON.parse(raw);
|
|
27
|
+
if (!parsed || typeof parsed !== 'object') {
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
const pid = Number(parsed.pid);
|
|
31
|
+
const port = Number(parsed.port);
|
|
32
|
+
const runId = String(parsed.runId || '').trim();
|
|
33
|
+
if (!Number.isFinite(pid) || pid <= 0 || !Number.isFinite(port) || port <= 0 || !runId) {
|
|
34
|
+
return null;
|
|
35
|
+
}
|
|
36
|
+
return parsed;
|
|
37
|
+
}
|
|
38
|
+
catch {
|
|
39
|
+
return null;
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
export function safeWriteRuntimeLifecycle(filePath, state) {
|
|
43
|
+
try {
|
|
44
|
+
ensureParentDir(filePath);
|
|
45
|
+
fsSync.writeFileSync(filePath, JSON.stringify(state, null, 2), 'utf8');
|
|
46
|
+
return true;
|
|
47
|
+
}
|
|
48
|
+
catch {
|
|
49
|
+
return false;
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
export function safeMarkRuntimeExit(filePath, marker) {
|
|
53
|
+
try {
|
|
54
|
+
const existing = safeReadRuntimeLifecycle(filePath);
|
|
55
|
+
if (!existing) {
|
|
56
|
+
return false;
|
|
57
|
+
}
|
|
58
|
+
const next = {
|
|
59
|
+
...existing,
|
|
60
|
+
exit: marker
|
|
61
|
+
};
|
|
62
|
+
return safeWriteRuntimeLifecycle(filePath, next);
|
|
63
|
+
}
|
|
64
|
+
catch {
|
|
65
|
+
return false;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
export function isPidAliveForForensics(pid) {
|
|
69
|
+
if (!Number.isFinite(pid) || pid <= 0) {
|
|
70
|
+
return false;
|
|
71
|
+
}
|
|
72
|
+
try {
|
|
73
|
+
process.kill(pid, 0);
|
|
74
|
+
return true;
|
|
75
|
+
}
|
|
76
|
+
catch {
|
|
77
|
+
return false;
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
export function inferUngracefulPreviousExit(args) {
|
|
81
|
+
const previous = args.previous;
|
|
82
|
+
if (!previous) {
|
|
83
|
+
return { shouldReport: false, reason: 'no_previous_state' };
|
|
84
|
+
}
|
|
85
|
+
if (previous.exit) {
|
|
86
|
+
return { shouldReport: false, reason: 'previous_exit_recorded' };
|
|
87
|
+
}
|
|
88
|
+
const previousPid = Number(previous.pid);
|
|
89
|
+
if (!Number.isFinite(previousPid) || previousPid <= 0) {
|
|
90
|
+
return { shouldReport: false, reason: 'invalid_previous_pid' };
|
|
91
|
+
}
|
|
92
|
+
if (Number(previousPid) === Number(args.currentPid)) {
|
|
93
|
+
return { shouldReport: false, reason: 'same_pid_reentry' };
|
|
94
|
+
}
|
|
95
|
+
const isPidAlive = args.isPidAlive ?? isPidAliveForForensics;
|
|
96
|
+
if (isPidAlive(previousPid)) {
|
|
97
|
+
return { shouldReport: false, reason: 'previous_pid_alive' };
|
|
98
|
+
}
|
|
99
|
+
return { shouldReport: true, reason: 'previous_missing_exit_marker_pid_dead' };
|
|
100
|
+
}
|
|
101
|
+
//# sourceMappingURL=runtime-exit-forensics.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"runtime-exit-forensics.js","sourceRoot":"","sources":["../../src/utils/runtime-exit-forensics.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
interface ShutdownCallerContext {
|
|
2
|
+
source: string;
|
|
3
|
+
requestTs: string;
|
|
4
|
+
remoteIp?: string;
|
|
5
|
+
method?: string;
|
|
6
|
+
path?: string;
|
|
7
|
+
userAgent?: string;
|
|
8
|
+
forwardedFor?: string;
|
|
9
|
+
origin?: string;
|
|
10
|
+
referer?: string;
|
|
11
|
+
authPresent?: boolean;
|
|
12
|
+
callerPid?: string;
|
|
13
|
+
callerTs?: string;
|
|
14
|
+
callerCwd?: string;
|
|
15
|
+
callerCmd?: string;
|
|
16
|
+
}
|
|
17
|
+
export declare function setShutdownCallerContext(value: ShutdownCallerContext): void;
|
|
18
|
+
export declare function clearShutdownCallerContext(): void;
|
|
19
|
+
export declare function getShutdownCallerContext(opts?: {
|
|
20
|
+
maxAgeMs?: number;
|
|
21
|
+
}): ShutdownCallerContext | null;
|
|
22
|
+
export type { ShutdownCallerContext };
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
const DEFAULT_MAX_AGE_MS = 120_000;
|
|
2
|
+
let cached = null;
|
|
3
|
+
export function setShutdownCallerContext(value) {
|
|
4
|
+
cached = {
|
|
5
|
+
value: { ...value },
|
|
6
|
+
setAtMs: Date.now()
|
|
7
|
+
};
|
|
8
|
+
}
|
|
9
|
+
export function clearShutdownCallerContext() {
|
|
10
|
+
cached = null;
|
|
11
|
+
}
|
|
12
|
+
export function getShutdownCallerContext(opts) {
|
|
13
|
+
if (!cached) {
|
|
14
|
+
return null;
|
|
15
|
+
}
|
|
16
|
+
const maxAgeMs = Number.isFinite(opts?.maxAgeMs) && opts?.maxAgeMs > 0
|
|
17
|
+
? Number(opts?.maxAgeMs)
|
|
18
|
+
: DEFAULT_MAX_AGE_MS;
|
|
19
|
+
if (Date.now() - cached.setAtMs > maxAgeMs) {
|
|
20
|
+
cached = null;
|
|
21
|
+
return null;
|
|
22
|
+
}
|
|
23
|
+
return { ...cached.value };
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=shutdown-caller-context.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"shutdown-caller-context.js","sourceRoot":"","sources":["../../src/utils/shutdown-caller-context.ts"],"names":[],"mappings":"AAiBA,MAAM,kBAAkB,GAAG,OAAO,CAAC;AAEnC,IAAI,MAAM,GAA6D,IAAI,CAAC;AAE5E,MAAM,UAAU,wBAAwB,CAAC,KAA4B;IACnE,MAAM,GAAG;QACP,KAAK,EAAE,EAAE,GAAG,KAAK,EAAE;QACnB,OAAO,EAAE,IAAI,CAAC,GAAG,EAAE;KACpB,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,0BAA0B;IACxC,MAAM,GAAG,IAAI,CAAC;AAChB,CAAC;AAED,MAAM,UAAU,wBAAwB,CAAC,IAA4B;IACnE,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO,IAAI,CAAC;IACd,CAAC;IACD,MAAM,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,QAAQ,CAAC,IAAK,IAAI,EAAE,QAAmB,GAAG,CAAC;QAChF,CAAC,CAAC,MAAM,CAAC,IAAI,EAAE,QAAQ,CAAC;QACxB,CAAC,CAAC,kBAAkB,CAAC;IACvB,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,MAAM,CAAC,OAAO,GAAG,QAAQ,EAAE,CAAC;QAC3C,MAAM,GAAG,IAAI,CAAC;QACd,OAAO,IAAI,CAAC;IACd,CAAC;IACD,OAAO,EAAE,GAAG,MAAM,CAAC,KAAK,EAAE,CAAC;AAC7B,CAAC"}
|
|
@@ -87,6 +87,14 @@
|
|
|
87
87
|
- 关键点:`auth.type: "iflow-oauth"` + `auth.tokenFile`;`compatibilityProfile: "chat:iflow"`
|
|
88
88
|
- 你需要:先完成一次 OAuth 登录生成 tokenFile
|
|
89
89
|
|
|
90
|
+
### DeepSeek Web(账号)
|
|
91
|
+
|
|
92
|
+
- 样本:`configsamples/provider/deepseek/config.v1.json`
|
|
93
|
+
- 建议配置:`type: "openai"` + `compatibilityProfile: "chat:deepseek-web"`
|
|
94
|
+
- 认证字段:`auth.type: "deepseek-account"`,并通过 `auth.entries[*].tokenFile` 提供账号 token 文件(支持多账号轮转)
|
|
95
|
+
- 可选参数:`deepseek.strictToolRequired`、`deepseek.textToolFallback`、`deepseek.powTimeoutMs`、`deepseek.powMaxAttempts`、`deepseek.sessionReuseTtlMs`
|
|
96
|
+
- 说明:Provider 层负责账号登录/会话/PoW/传输;工具调用标准化由 llmswitch-core compat 完成
|
|
97
|
+
|
|
90
98
|
### Kimi(API Key)
|
|
91
99
|
|
|
92
100
|
- 样本:`configsamples/provider/kimi/config.v1.json`
|
package/docs/PROVIDER_TYPES.md
CHANGED
|
@@ -31,10 +31,12 @@
|
|
|
31
31
|
}
|
|
32
32
|
```
|
|
33
33
|
|
|
34
|
-
### OpenAI-compatible(例如 GLM/Qwen/Kimi 等)
|
|
34
|
+
### OpenAI-compatible(例如 GLM/Qwen/Kimi/DeepSeek Web 等)
|
|
35
35
|
|
|
36
36
|
仍然建议 `type: "openai"`,并通过 `compatibilityProfile` 做最小字段适配:
|
|
37
37
|
|
|
38
|
+
- DeepSeek Web 账号模式建议:`type: "openai"` + `compatibilityProfile: "chat:deepseek-web"` + `auth.type = "deepseek-account"`。
|
|
39
|
+
|
|
38
40
|
```jsonc
|
|
39
41
|
{
|
|
40
42
|
"id": "qwen",
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
# ServerTool Pre-Command Hooks
|
|
2
|
+
|
|
3
|
+
## Goal
|
|
4
|
+
|
|
5
|
+
在 `tool_call` 下发到客户端执行前,做一层可注册的 `pre_command` 拦截(不消费工具调用、不替代执行),用于:
|
|
6
|
+
|
|
7
|
+
- 参数形状清理(例如 `exec_command` 参数规范化)
|
|
8
|
+
- 预执行审计(shell side-effect 记录)
|
|
9
|
+
- 与 Claude Code `PreToolUse` 类似的“执行前 hook”能力
|
|
10
|
+
|
|
11
|
+
## Activation (runtime)
|
|
12
|
+
|
|
13
|
+
优先级最高的激活方式是路由指令(与 `stopMessage` 同路径):
|
|
14
|
+
|
|
15
|
+
- `<**precommand:<file://precommand/your-script.sh>**>`:激活预执行脚本
|
|
16
|
+
- `<**precommand:clear**>`:清除当前 precommand 状态
|
|
17
|
+
- `<**precommand**>` / `<**precommand:on**>`:加载默认脚本(默认 `default.sh`,可用 `ROUTECODEX_PRECOMMAND_DEFAULT_SCRIPT` 覆盖)
|
|
18
|
+
|
|
19
|
+
脚本必须位于 `~/.routecodex/precommand` 下(支持 `ROUTECODEX_USER_DIR` 覆盖根目录)。
|
|
20
|
+
|
|
21
|
+
运行时会把 hook event JSON 通过 stdin 和环境变量 `ROUTECODEX_PRE_COMMAND_HOOK_EVENT` 传给脚本:
|
|
22
|
+
|
|
23
|
+
- 脚本输出空内容:不改参数
|
|
24
|
+
- 脚本输出 JSON:用于改写工具参数(支持 `toolArguments`/`arguments` 或直接输出参数对象)
|
|
25
|
+
|
|
26
|
+
## Config file (fallback)
|
|
27
|
+
|
|
28
|
+
当未激活 runtime precommand 时,回退读取:`~/.routecodex/hooks/pre-command-hooks.json`
|
|
29
|
+
|
|
30
|
+
可覆盖:
|
|
31
|
+
|
|
32
|
+
- `ROUTECODEX_PRE_COMMAND_HOOKS_FILE`
|
|
33
|
+
- `RCC_PRE_COMMAND_HOOKS_FILE`
|
|
34
|
+
- `LLMSWITCH_PRE_COMMAND_HOOKS_FILE`
|
|
35
|
+
|
|
36
|
+
示例:
|
|
37
|
+
|
|
38
|
+
```json
|
|
39
|
+
{
|
|
40
|
+
"enabled": true,
|
|
41
|
+
"hooks": [
|
|
42
|
+
{
|
|
43
|
+
"id": "normalize-npm-cmd",
|
|
44
|
+
"tool": "exec_command",
|
|
45
|
+
"priority": 10,
|
|
46
|
+
"cmdRegex": "^npm\\s+",
|
|
47
|
+
"jq": ".cmd = (\"set -euo pipefail; \" + .cmd)"
|
|
48
|
+
},
|
|
49
|
+
{
|
|
50
|
+
"id": "audit-pre-command",
|
|
51
|
+
"tool": ["exec_command", "shell"],
|
|
52
|
+
"priority": 20,
|
|
53
|
+
"shell": "cat >> ~/.routecodex/logs/pre-command-audit.jsonl"
|
|
54
|
+
}
|
|
55
|
+
]
|
|
56
|
+
}
|
|
57
|
+
```
|
|
58
|
+
|
|
59
|
+
## `jq` 注册(shell 操作)
|
|
60
|
+
|
|
61
|
+
可直接用 `jq` 向 hooks 文件追加注册项:
|
|
62
|
+
|
|
63
|
+
```bash
|
|
64
|
+
HOOK_FILE="${ROUTECODEX_PRE_COMMAND_HOOKS_FILE:-$HOME/.routecodex/hooks/pre-command-hooks.json}"
|
|
65
|
+
mkdir -p "$(dirname "$HOOK_FILE")"
|
|
66
|
+
[ -f "$HOOK_FILE" ] || printf '{"enabled":true,"hooks":[]}' > "$HOOK_FILE"
|
|
67
|
+
|
|
68
|
+
jq '.hooks += [
|
|
69
|
+
{
|
|
70
|
+
"id": "block-applypatch-via-exec",
|
|
71
|
+
"tool": "exec_command",
|
|
72
|
+
"priority": 5,
|
|
73
|
+
"cmdRegex": "apply_patch",
|
|
74
|
+
"jq": ".cmd = \"echo \\\"Warning: use apply_patch tool\\\"; \" + .cmd"
|
|
75
|
+
}
|
|
76
|
+
]' "$HOOK_FILE" > "$HOOK_FILE.tmp" && mv "$HOOK_FILE.tmp" "$HOOK_FILE"
|
|
77
|
+
```
|
|
78
|
+
|
|
79
|
+
## Runtime behavior
|
|
80
|
+
|
|
81
|
+
- 执行顺序:`priority` 升序,`priority` 相同按注册顺序。
|
|
82
|
+
- `jq` action:把当前 arguments JSON 作为输入,输出 JSON 对象作为新的 tool arguments。
|
|
83
|
+
- `shell` action:以 hook event JSON 作为 stdin 执行 shell 命令(用于审计/旁路记录)。
|
|
84
|
+
- 失败策略:hook 报错记为 trace(`error`),不阻断原工具调用下发。
|
|
85
|
+
- trace:通过 servertool hook trace 通道记录 `match/miss/error`。
|