@pixelbyte-software/pixcode 1.51.2 → 1.51.3
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/CODE_OF_CONDUCT.md +41 -41
- package/CONTRIBUTING.md +155 -155
- package/LICENSE +718 -718
- package/README.de.md +169 -169
- package/README.ja.md +167 -167
- package/README.ko.md +167 -167
- package/README.md +419 -419
- package/README.ru.md +169 -169
- package/README.tr.md +298 -298
- package/README.zh-CN.md +167 -167
- package/SECURITY.md +46 -46
- package/dist/api-automation.html +110 -110
- package/dist/api-docs.html +548 -548
- package/dist/assets/{index-EN9ngyxf.js → index-17CwxHSZ.js} +185 -185
- package/dist/assets/index-B9N-gfOQ.css +32 -0
- package/dist/clear-cache.html +85 -85
- package/dist/convert-icons.md +52 -52
- package/dist/docs.html +308 -308
- package/dist/favicon.svg +8 -8
- package/dist/features.html +133 -133
- package/dist/generate-icons.js +48 -48
- package/dist/humans.txt +15 -15
- package/dist/icons/codex-white.svg +3 -3
- package/dist/icons/codex.svg +3 -3
- package/dist/icons/cursor-white.svg +11 -11
- package/dist/icons/icon-128x128.svg +9 -9
- package/dist/icons/icon-144x144.svg +9 -9
- package/dist/icons/icon-152x152.svg +9 -9
- package/dist/icons/icon-192x192.svg +9 -9
- package/dist/icons/icon-384x384.svg +9 -9
- package/dist/icons/icon-512x512.svg +9 -9
- package/dist/icons/icon-72x72.svg +9 -9
- package/dist/icons/icon-96x96.svg +9 -9
- package/dist/icons/icon-template.svg +9 -9
- package/dist/icons/qwen-logo.svg +14 -14
- package/dist/index.html +59 -59
- package/dist/landing.html +268 -268
- package/dist/llms-full.txt +119 -119
- package/dist/llms.txt +53 -53
- package/dist/logo.svg +12 -12
- package/dist/manifest.json +60 -60
- package/dist/openapi.yaml +1696 -1696
- package/dist/orchestration.html +125 -125
- package/dist/robots.txt +4 -4
- package/dist/site.css +692 -692
- package/dist/sitemap.xml +51 -51
- package/dist/sw.js +132 -132
- package/dist-server/server/cli.js +96 -96
- package/dist-server/server/daemon/manager.js +33 -33
- package/dist-server/server/daemon-manager.js +64 -64
- package/dist-server/server/index.js +122 -3
- package/dist-server/server/index.js.map +1 -1
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js +84 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js.map +1 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js +43 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js.map +1 -0
- package/dist-server/server/modules/orchestration/hermes/hermes.routes.js +55 -1
- package/dist-server/server/modules/orchestration/hermes/hermes.routes.js.map +1 -1
- package/dist-server/server/modules/orchestration/index.js +1 -0
- package/dist-server/server/modules/orchestration/index.js.map +1 -1
- package/dist-server/server/routes/commands.js +25 -25
- package/dist-server/server/routes/git.js +17 -17
- package/dist-server/server/routes/live-view.js +46 -46
- package/dist-server/server/services/hermes-gateway.js +310 -0
- package/dist-server/server/services/hermes-gateway.js.map +1 -1
- package/dist-server/server/services/public-api-manifest.js +59 -51
- package/dist-server/server/services/public-api-manifest.js.map +1 -1
- package/package.json +222 -222
- package/scripts/fix-node-pty.js +67 -67
- package/scripts/github/create-v1.38-issues.mjs +351 -351
- package/scripts/github/create-vscode-workbench-issues.mjs +121 -121
- package/scripts/hermes/configure-pixcode-mcp.mjs +165 -163
- package/scripts/hermes/pixcode-mcp-server.mjs +1009 -958
- package/scripts/smoke/changes-panel-layout.mjs +48 -48
- package/scripts/smoke/chat-composer-fixed-layout.mjs +55 -55
- package/scripts/smoke/chat-message-timeline-order.mjs +41 -41
- package/scripts/smoke/chat-realtime-hydration.mjs +44 -44
- package/scripts/smoke/chat-session-provider-pools.mjs +35 -35
- package/scripts/smoke/chat-session-state.mjs +19 -19
- package/scripts/smoke/code-editor-theme.mjs +55 -55
- package/scripts/smoke/code-editor-vscode-engine.mjs +91 -91
- package/scripts/smoke/command-center-agent-writes.mjs +79 -79
- package/scripts/smoke/command-center-non-git.mjs +46 -46
- package/scripts/smoke/context-packet.mjs +43 -43
- package/scripts/smoke/control-room-ux-redesign.mjs +91 -91
- package/scripts/smoke/daemon-entrypoint.mjs +20 -20
- package/scripts/smoke/default-landing-routing.mjs +33 -33
- package/scripts/smoke/desktop-native-notifications.mjs +30 -30
- package/scripts/smoke/desktop-tray-icon.mjs +33 -33
- package/scripts/smoke/discord-release-workflow.mjs +24 -24
- package/scripts/smoke/git-install-update.mjs +255 -255
- package/scripts/smoke/handoff-artifact-protocol.mjs +50 -50
- package/scripts/smoke/hermes-api-install.mjs +56 -56
- package/scripts/smoke/hermes-gateway-persistence.mjs +104 -104
- package/scripts/smoke/hermes-mcp-pixcode-roundtrip.mjs +426 -367
- package/scripts/smoke/hermes-rest-chat-api.mjs +162 -162
- package/scripts/smoke/hermes-rest-chat-live.mjs +45 -45
- package/scripts/smoke/hermes-rest-codex-launch.mjs +209 -209
- package/scripts/smoke/hermes-rest-gateway.mjs +79 -70
- package/scripts/smoke/hermes-rest-live.mjs +42 -42
- package/scripts/smoke/hermes-roundtrip.mjs +167 -167
- package/scripts/smoke/hermes-settings-commands.mjs +349 -346
- package/scripts/smoke/hermes-smoke-launcher-guard.mjs +34 -34
- package/scripts/smoke/live-view-diagnostics.mjs +53 -53
- package/scripts/smoke/live-view-environment.mjs +92 -92
- package/scripts/smoke/live-view-integration.mjs +450 -450
- package/scripts/smoke/mac-desktop-runtime.mjs +37 -37
- package/scripts/smoke/mobile-tunnel-guidance.mjs +29 -29
- package/scripts/smoke/model-registry.mjs +36 -36
- package/scripts/smoke/multi-project-ui.mjs +45 -45
- package/scripts/smoke/multi-worker-slots.mjs +42 -42
- package/scripts/smoke/notification-center.mjs +87 -87
- package/scripts/smoke/notification-inapp-preference.mjs +23 -23
- package/scripts/smoke/notification-taxonomy.mjs +58 -58
- package/scripts/smoke/orchestration-api.mjs +172 -172
- package/scripts/smoke/orchestration-execution-dashboard.mjs +33 -33
- package/scripts/smoke/orchestration-live-run.mjs +176 -176
- package/scripts/smoke/orchestration-mobile-scroll.mjs +29 -29
- package/scripts/smoke/orchestration-model-sync.mjs +30 -30
- package/scripts/smoke/orchestration-permission-fallback.mjs +34 -34
- package/scripts/smoke/orchestration-runtime-guards.mjs +48 -48
- package/scripts/smoke/orchestration-user-facing-output.mjs +25 -25
- package/scripts/smoke/permission-policy.mjs +50 -50
- package/scripts/smoke/pixcode-workbench-1-48.mjs +167 -167
- package/scripts/smoke/provider-models-opencode-live.mjs +66 -66
- package/scripts/smoke/provider-rest-api.mjs +124 -124
- package/scripts/smoke/provider-selection-status.mjs +52 -52
- package/scripts/smoke/run-state-refresh.mjs +52 -52
- package/scripts/smoke/runtime-manager.mjs +99 -99
- package/scripts/smoke/shell-manual-disconnect.mjs +30 -30
- package/scripts/smoke/side-panel-editor-layout.mjs +34 -34
- package/scripts/smoke/static-root-routing.mjs +21 -21
- package/scripts/smoke/strict-handoff-compact.mjs +60 -60
- package/scripts/smoke/taskmaster-config.mjs +24 -24
- package/scripts/smoke/taskmaster-execution-telegram.mjs +3 -3
- package/scripts/smoke/taskmaster-onboarding.mjs +3 -3
- package/scripts/smoke/taskmaster-run-graph.mjs +3 -3
- package/scripts/smoke/telegram-control.mjs +242 -242
- package/scripts/smoke/tunnel-persistence.mjs +56 -56
- package/scripts/smoke/update-issue-progress.mjs +69 -69
- package/scripts/smoke/update-ux.mjs +55 -55
- package/scripts/smoke/v138-completion.mjs +132 -132
- package/scripts/smoke/v138-desktop-release-hardening.mjs +69 -69
- package/scripts/smoke/v138-diagnostics.mjs +63 -63
- package/scripts/smoke/v138-issue-planner.mjs +33 -33
- package/scripts/smoke/v143-remote-control.mjs +76 -76
- package/scripts/smoke/v144-production-loop.mjs +47 -47
- package/scripts/smoke/v145-platformization.mjs +46 -46
- package/scripts/smoke/v146-control-room-ui.mjs +150 -150
- package/scripts/smoke/version-modal-autoshow.mjs +29 -29
- package/scripts/smoke/vscode-workbench-layout.mjs +63 -63
- package/scripts/smoke/vscode-workbench-polish.mjs +461 -436
- package/scripts/smoke/workflow-fallback-replay.mjs +56 -56
- package/scripts/smoke/workflow-templates.mjs +43 -43
- package/scripts/smoke/workflow-trace-timeline.mjs +46 -46
- package/scripts/update-git-install.mjs +293 -293
- package/server/claude-sdk.js +920 -920
- package/server/cli.js +1039 -1039
- package/server/constants/config.js +4 -4
- package/server/cursor-cli.js +344 -344
- package/server/daemon/manager.js +563 -563
- package/server/daemon-manager.js +964 -964
- package/server/database/db.js +921 -921
- package/server/database/json-store.js +197 -197
- package/server/gemini-cli.js +550 -550
- package/server/gemini-response-handler.js +79 -79
- package/server/index.js +128 -2
- package/server/load-env.js +35 -35
- package/server/middleware/auth.js +175 -175
- package/server/modules/orchestration/a2a/adapter-registry.ts +108 -108
- package/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.ts +63 -63
- package/server/modules/orchestration/a2a/adapters/claude-code.adapter.ts +286 -286
- package/server/modules/orchestration/a2a/adapters/codex.adapter.ts +244 -244
- package/server/modules/orchestration/a2a/adapters/cursor.adapter.ts +249 -249
- package/server/modules/orchestration/a2a/adapters/gemini.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/adapters/json-event.adapter.test.ts +60 -0
- package/server/modules/orchestration/a2a/adapters/json-event.adapter.ts +101 -0
- package/server/modules/orchestration/a2a/adapters/opencode.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/adapters/qwen.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/agent-card.ts +55 -55
- package/server/modules/orchestration/a2a/routes.ts +590 -590
- package/server/modules/orchestration/a2a/task-store.ts +178 -178
- package/server/modules/orchestration/a2a/types.ts +126 -126
- package/server/modules/orchestration/a2a/validator.ts +113 -113
- package/server/modules/orchestration/hermes/hermes.routes.ts +642 -583
- package/server/modules/orchestration/index.ts +101 -100
- package/server/modules/orchestration/preview/port-watcher.ts +112 -112
- package/server/modules/orchestration/preview/preview-proxy.ts +60 -60
- package/server/modules/orchestration/preview/types.ts +19 -19
- package/server/modules/orchestration/security/permission-policy.ts +401 -401
- package/server/modules/orchestration/tasks/orchestration-task-store.ts +41 -41
- package/server/modules/orchestration/tasks/orchestration-task.routes.ts +64 -64
- package/server/modules/orchestration/tasks/orchestration-task.service.ts +209 -209
- package/server/modules/orchestration/tasks/orchestration-task.types.ts +40 -40
- package/server/modules/orchestration/tasks/task-run-graph.ts +155 -155
- package/server/modules/orchestration/workflows/approval-queue.ts +106 -106
- package/server/modules/orchestration/workflows/built-in-workflows.ts +127 -127
- package/server/modules/orchestration/workflows/context-packet.ts +186 -186
- package/server/modules/orchestration/workflows/handoff-artifact.ts +175 -175
- package/server/modules/orchestration/workflows/workflow-fallback-policy.ts +161 -161
- package/server/modules/orchestration/workflows/workflow-replay.ts +254 -254
- package/server/modules/orchestration/workflows/workflow-runner.ts +2070 -2070
- package/server/modules/orchestration/workflows/workflow-store.ts +97 -97
- package/server/modules/orchestration/workflows/workflow-templates.ts +272 -272
- package/server/modules/orchestration/workflows/workflow-trace.ts +424 -424
- package/server/modules/orchestration/workflows/workflow.routes.ts +586 -586
- package/server/modules/orchestration/workflows/workflow.types.ts +111 -111
- package/server/modules/orchestration/workflows/workspace-target.ts +122 -122
- package/server/modules/orchestration/workspace/docker-workspace.ts +136 -136
- package/server/modules/orchestration/workspace/path-safety.ts +55 -55
- package/server/modules/orchestration/workspace/types.ts +52 -52
- package/server/modules/orchestration/workspace/workspace-manager.ts +102 -102
- package/server/modules/orchestration/workspace/worktree-workspace.ts +126 -126
- package/server/modules/providers/index.ts +2 -2
- package/server/modules/providers/list/claude/claude-auth.provider.ts +146 -146
- package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
- package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
- package/server/modules/providers/list/claude/claude.provider.ts +15 -15
- package/server/modules/providers/list/codex/codex-auth.provider.ts +117 -117
- package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
- package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
- package/server/modules/providers/list/codex/codex.provider.ts +15 -15
- package/server/modules/providers/list/cursor/cursor-auth.provider.ts +147 -147
- package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
- package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
- package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
- package/server/modules/providers/list/gemini/gemini-auth.provider.ts +173 -173
- package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
- package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
- package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
- package/server/modules/providers/list/opencode/opencode-auth.provider.ts +131 -131
- package/server/modules/providers/list/opencode/opencode-mcp.provider.ts +126 -126
- package/server/modules/providers/list/opencode/opencode-sessions.provider.ts +286 -286
- package/server/modules/providers/list/opencode/opencode.provider.ts +29 -29
- package/server/modules/providers/list/qwen/qwen-auth.provider.ts +146 -146
- package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -114
- package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +265 -265
- package/server/modules/providers/list/qwen/qwen.provider.ts +21 -21
- package/server/modules/providers/provider.registry.ts +40 -40
- package/server/modules/providers/provider.routes.ts +944 -944
- package/server/modules/providers/services/mcp.service.ts +86 -86
- package/server/modules/providers/services/provider-auth.service.ts +26 -26
- package/server/modules/providers/services/sessions.service.ts +45 -45
- package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
- package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
- package/server/modules/providers/shared/provider-configs.ts +142 -142
- package/server/modules/providers/tests/mcp.test.ts +293 -293
- package/server/openai-codex.js +462 -462
- package/server/opencode-cli.js +491 -491
- package/server/opencode-response-handler.js +111 -111
- package/server/projects.js +3008 -3008
- package/server/qwen-code-cli.js +410 -410
- package/server/qwen-response-handler.js +73 -73
- package/server/routes/agent.js +1435 -1435
- package/server/routes/auth.js +159 -159
- package/server/routes/codex.js +20 -20
- package/server/routes/commands.js +570 -570
- package/server/routes/cursor.js +61 -61
- package/server/routes/diagnostics.js +41 -41
- package/server/routes/gemini.js +25 -25
- package/server/routes/git.js +1650 -1650
- package/server/routes/live-view.js +411 -411
- package/server/routes/mcp-utils.js +13 -13
- package/server/routes/messages.js +62 -62
- package/server/routes/network.js +125 -125
- package/server/routes/platformization.js +212 -212
- package/server/routes/plugins.js +320 -320
- package/server/routes/production-agent-loop.js +90 -90
- package/server/routes/projects.js +917 -917
- package/server/routes/public-api.js +34 -34
- package/server/routes/qwen.js +27 -27
- package/server/routes/remote.js +55 -55
- package/server/routes/settings.js +321 -321
- package/server/routes/telegram.js +140 -140
- package/server/routes/user.js +125 -125
- package/server/routes/webhooks.js +63 -63
- package/server/services/control-room.js +102 -102
- package/server/services/diagnostics.js +165 -165
- package/server/services/external-access.js +375 -375
- package/server/services/hermes-gateway.js +1562 -1247
- package/server/services/hermes-install-jobs.js +729 -729
- package/server/services/install-jobs.js +715 -715
- package/server/services/live-view.js +956 -956
- package/server/services/managed-runtimes.js +493 -493
- package/server/services/model-registry.js +144 -144
- package/server/services/notification-orchestrator.js +365 -365
- package/server/services/notification-taxonomy.js +204 -204
- package/server/services/platformization.js +815 -815
- package/server/services/production-agent-loop.js +248 -248
- package/server/services/provider-cli-versions.js +149 -149
- package/server/services/provider-credentials.js +189 -189
- package/server/services/provider-models.js +396 -396
- package/server/services/public-api-manifest.js +190 -182
- package/server/services/remote-connection.js +127 -127
- package/server/services/runtime-manager.js +323 -323
- package/server/services/startup-update.js +234 -234
- package/server/services/telegram/bot.js +331 -331
- package/server/services/telegram/control-center.js +979 -979
- package/server/services/telegram/telegram-http-client.js +151 -151
- package/server/services/telegram/translations.js +340 -340
- package/server/services/vapid-keys.js +36 -36
- package/server/services/webhooks.js +216 -216
- package/server/sessionManager.js +225 -225
- package/server/shared/interfaces.ts +54 -54
- package/server/shared/types.ts +172 -172
- package/server/shared/utils.ts +193 -193
- package/server/tsconfig.json +36 -36
- package/server/utils/colors.js +21 -21
- package/server/utils/commandParser.js +305 -305
- package/server/utils/frontmatter.js +18 -18
- package/server/utils/gitConfig.js +34 -34
- package/server/utils/plugin-loader.js +457 -457
- package/server/utils/plugin-process-manager.js +185 -185
- package/server/utils/port-access.js +209 -209
- package/server/utils/runtime-paths.js +37 -37
- package/server/utils/url-detection.js +71 -71
- package/server/vite-daemon.js +79 -79
- package/shared/modelConstants.js +161 -161
- package/shared/networkHosts.js +22 -22
- package/dist/assets/index-DMz0zv6T.css +0 -32
|
@@ -1,242 +1,242 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
|
|
3
|
-
import assert from 'node:assert/strict';
|
|
4
|
-
import { mkdirSync, readFileSync, rmSync } from 'node:fs';
|
|
5
|
-
import path from 'node:path';
|
|
6
|
-
|
|
7
|
-
const runtimeDir = path.resolve('.pixcode-dev', 'smoke-telegram-control');
|
|
8
|
-
mkdirSync(runtimeDir, { recursive: true });
|
|
9
|
-
process.env.DATABASE_PATH = path.join(runtimeDir, 'auth.db');
|
|
10
|
-
|
|
11
|
-
const checks = [
|
|
12
|
-
{
|
|
13
|
-
name: 'telegram bot wires the remote control center and callback queries',
|
|
14
|
-
file: 'server/services/telegram/bot.js',
|
|
15
|
-
test: (source) => (
|
|
16
|
-
source.includes('handleTelegramControlMessage')
|
|
17
|
-
&& source.includes('handleTelegramControlCallback')
|
|
18
|
-
&& source.includes("bot.on('callback_query'")
|
|
19
|
-
),
|
|
20
|
-
},
|
|
21
|
-
{
|
|
22
|
-
name: 'telegram HTTP client polls callback queries and can answer them',
|
|
23
|
-
file: 'server/services/telegram/telegram-http-client.js',
|
|
24
|
-
test: (source) => (
|
|
25
|
-
source.includes("allowed_updates: ['message', 'callback_query']")
|
|
26
|
-
&& source.includes('answerCallbackQuery')
|
|
27
|
-
&& source.includes('editMessageText')
|
|
28
|
-
&& source.includes("this.emit('callback_query'")
|
|
29
|
-
),
|
|
30
|
-
},
|
|
31
|
-
{
|
|
32
|
-
name: 'telegram control center exposes provider, model, workflow, install, and settings actions',
|
|
33
|
-
file: 'server/services/telegram/control-center.js',
|
|
34
|
-
test: (source) => (
|
|
35
|
-
source.includes('showMainMenu')
|
|
36
|
-
&& source.includes('showProviderMenu')
|
|
37
|
-
&& source.includes('showModelMenu')
|
|
38
|
-
&& source.includes('showWorkflowMenu')
|
|
39
|
-
&& source.includes('runWorkflow')
|
|
40
|
-
&& source.includes('startCliInstall')
|
|
41
|
-
&& source.includes('updateTelegramControlState')
|
|
42
|
-
&& source.includes('/api/agent')
|
|
43
|
-
&& source.includes('/api/orchestration/workflows')
|
|
44
|
-
),
|
|
45
|
-
},
|
|
46
|
-
{
|
|
47
|
-
name: 'telegram link state persists remote-control preferences',
|
|
48
|
-
file: 'server/database/db.js',
|
|
49
|
-
test: (source) => (
|
|
50
|
-
source.includes('telegram_control')
|
|
51
|
-
&& source.includes('getControlState')
|
|
52
|
-
&& source.includes('updateControlState')
|
|
53
|
-
&& source.includes('remoteControlEnabled')
|
|
54
|
-
),
|
|
55
|
-
},
|
|
56
|
-
{
|
|
57
|
-
name: 'telegram settings UI exposes remote-control toggles',
|
|
58
|
-
file: 'src/components/settings/view/tabs/telegram-settings/TelegramSettingsTab.tsx',
|
|
59
|
-
test: (source) => (
|
|
60
|
-
source.includes('controlEnabled')
|
|
61
|
-
&& source.includes('progressMode')
|
|
62
|
-
&& source.includes('telegram.control.title')
|
|
63
|
-
&& source.includes('telegram.control.progressMode')
|
|
64
|
-
),
|
|
65
|
-
},
|
|
66
|
-
];
|
|
67
|
-
|
|
68
|
-
const failures = [];
|
|
69
|
-
|
|
70
|
-
for (const check of checks) {
|
|
71
|
-
let source = '';
|
|
72
|
-
try {
|
|
73
|
-
source = readFileSync(check.file, 'utf8');
|
|
74
|
-
} catch {
|
|
75
|
-
failures.push(`${check.name} (${check.file} missing)`);
|
|
76
|
-
continue;
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
if (!check.test(source)) failures.push(check.name);
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
try {
|
|
83
|
-
const {
|
|
84
|
-
handleTelegramControlCallback,
|
|
85
|
-
handleTelegramControlMessage,
|
|
86
|
-
} = await import('../../server/services/telegram/control-center.js');
|
|
87
|
-
const {
|
|
88
|
-
handleIncomingTelegramMessage,
|
|
89
|
-
setTelegramBotForTesting,
|
|
90
|
-
} = await import('../../server/services/telegram/bot.js');
|
|
91
|
-
const { telegramLinksDb } = await import('../../server/database/db.js');
|
|
92
|
-
|
|
93
|
-
const userId = 4242;
|
|
94
|
-
telegramLinksDb.unlink(userId);
|
|
95
|
-
telegramLinksDb.setPairingCode(userId, '123456', new Date(Date.now() + 600_000).toISOString(), 'tr');
|
|
96
|
-
telegramLinksDb.verify(userId, 'chat-4242', 'ali');
|
|
97
|
-
|
|
98
|
-
const sent = [];
|
|
99
|
-
const bot = {
|
|
100
|
-
sendMessage: async (chatId, text, extra = {}) => {
|
|
101
|
-
sent.push({ chatId, text, extra });
|
|
102
|
-
return { ok: true };
|
|
103
|
-
},
|
|
104
|
-
};
|
|
105
|
-
|
|
106
|
-
const link = telegramLinksDb.getByUserId(userId);
|
|
107
|
-
const expectReply = async (input, expectedFragment) => {
|
|
108
|
-
sent.length = 0;
|
|
109
|
-
const handled = await handleTelegramControlMessage({
|
|
110
|
-
bot,
|
|
111
|
-
msg: { chat: { id: 4242 }, text: input },
|
|
112
|
-
link,
|
|
113
|
-
});
|
|
114
|
-
assert.equal(handled, true, `${input} should be handled`);
|
|
115
|
-
assert.ok(sent.length > 0, `${input} should send at least one reply`);
|
|
116
|
-
assert.ok(
|
|
117
|
-
sent[0].text.includes(expectedFragment),
|
|
118
|
-
`${input} should include "${expectedFragment}" but got "${sent[0].text}"`,
|
|
119
|
-
);
|
|
120
|
-
};
|
|
121
|
-
|
|
122
|
-
await expectReply('/start', 'Pixcode Telegram kontrol merkezi');
|
|
123
|
-
await expectReply('/help', 'Komutlar:');
|
|
124
|
-
await expectReply('/start@Otobot', 'Pixcode Telegram kontrol merkezi');
|
|
125
|
-
await expectReply('/help@Otobot', 'Komutlar:');
|
|
126
|
-
await expectReply('/', 'Komutlar:');
|
|
127
|
-
|
|
128
|
-
const menuEvents = [];
|
|
129
|
-
const menuBot = {
|
|
130
|
-
sendMessage: async (chatId, text, extra = {}) => {
|
|
131
|
-
menuEvents.push({ type: 'send', chatId, text, extra });
|
|
132
|
-
return { ok: true, message_id: 77 };
|
|
133
|
-
},
|
|
134
|
-
editMessageText: async (text, extra = {}) => {
|
|
135
|
-
menuEvents.push({ type: 'edit', text, extra });
|
|
136
|
-
return { ok: true, message_id: extra.message_id };
|
|
137
|
-
},
|
|
138
|
-
answerCallbackQuery: async () => ({ ok: true }),
|
|
139
|
-
};
|
|
140
|
-
const findButton = (markup, predicate) => {
|
|
141
|
-
for (const row of markup?.inline_keyboard || []) {
|
|
142
|
-
for (const candidate of row) {
|
|
143
|
-
if (predicate(candidate)) return candidate;
|
|
144
|
-
}
|
|
145
|
-
}
|
|
146
|
-
return null;
|
|
147
|
-
};
|
|
148
|
-
|
|
149
|
-
menuEvents.length = 0;
|
|
150
|
-
await handleTelegramControlMessage({
|
|
151
|
-
bot: menuBot,
|
|
152
|
-
msg: { chat: { id: 4242 }, text: '/settings' },
|
|
153
|
-
link,
|
|
154
|
-
});
|
|
155
|
-
const settingsMenu = menuEvents.at(-1);
|
|
156
|
-
const languageButton = findButton(
|
|
157
|
-
settingsMenu.extra.reply_markup,
|
|
158
|
-
(candidate) => /language|dil/i.test(candidate.text),
|
|
159
|
-
);
|
|
160
|
-
assert.ok(languageButton, 'settings menu should expose a language button');
|
|
161
|
-
|
|
162
|
-
menuEvents.length = 0;
|
|
163
|
-
await handleTelegramControlCallback({
|
|
164
|
-
bot: menuBot,
|
|
165
|
-
query: {
|
|
166
|
-
id: 'query-language-menu',
|
|
167
|
-
data: languageButton.callback_data,
|
|
168
|
-
message: { chat: { id: 4242 }, message_id: 77 },
|
|
169
|
-
},
|
|
170
|
-
link,
|
|
171
|
-
});
|
|
172
|
-
assert.equal(menuEvents.length, 1, 'callback menus should replace the existing menu message');
|
|
173
|
-
assert.equal(menuEvents[0].type, 'edit', 'callback menus should use editMessageText');
|
|
174
|
-
const trButton = findButton(menuEvents[0].extra.reply_markup, (candidate) => candidate.text === 'tr');
|
|
175
|
-
assert.ok(trButton, 'language menu should include Turkish');
|
|
176
|
-
|
|
177
|
-
menuEvents.length = 0;
|
|
178
|
-
await handleTelegramControlCallback({
|
|
179
|
-
bot: menuBot,
|
|
180
|
-
query: {
|
|
181
|
-
id: 'query-language-tr',
|
|
182
|
-
data: trButton.callback_data,
|
|
183
|
-
message: { chat: { id: 4242 }, message_id: 77 },
|
|
184
|
-
},
|
|
185
|
-
link,
|
|
186
|
-
});
|
|
187
|
-
assert.equal(telegramLinksDb.getByUserId(userId).language, 'tr', 'language selection should persist');
|
|
188
|
-
assert.equal(menuEvents.length, 1, 'language selection should not send a confirmation plus a second menu');
|
|
189
|
-
assert.equal(menuEvents[0].type, 'edit', 'language selection should replace the menu in place');
|
|
190
|
-
assert.ok(
|
|
191
|
-
menuEvents[0].text.includes('Pixcode Telegram kontrol merkezi'),
|
|
192
|
-
`Turkish menu should render after selection, got "${menuEvents[0].text}"`,
|
|
193
|
-
);
|
|
194
|
-
assert.ok(
|
|
195
|
-
!menuEvents[0].text.includes('Project:'),
|
|
196
|
-
'Turkish menu should not keep English summary labels after language selection',
|
|
197
|
-
);
|
|
198
|
-
|
|
199
|
-
const botLevelMessages = [];
|
|
200
|
-
setTelegramBotForTesting({
|
|
201
|
-
sendMessage: async (chatId, text, extra = {}) => {
|
|
202
|
-
botLevelMessages.push({ chatId, text, extra });
|
|
203
|
-
return { ok: true };
|
|
204
|
-
},
|
|
205
|
-
answerCallbackQuery: async () => ({ ok: true }),
|
|
206
|
-
});
|
|
207
|
-
|
|
208
|
-
const expectBotReply = async (input, expectedFragment) => {
|
|
209
|
-
botLevelMessages.length = 0;
|
|
210
|
-
await handleIncomingTelegramMessage({
|
|
211
|
-
chat: { id: 'chat-4242' },
|
|
212
|
-
text: input,
|
|
213
|
-
message_id: 1,
|
|
214
|
-
from: { username: 'ali' },
|
|
215
|
-
});
|
|
216
|
-
assert.ok(botLevelMessages.length > 0, `${input} should reply from bot.js`);
|
|
217
|
-
assert.ok(
|
|
218
|
-
botLevelMessages[0].text.includes(expectedFragment),
|
|
219
|
-
`${input} should include "${expectedFragment}" but got "${botLevelMessages[0].text}"`,
|
|
220
|
-
);
|
|
221
|
-
assert.ok(
|
|
222
|
-
!botLevelMessages.some((entry) => entry.text.includes('Mesaj son oturumuna iletildi')),
|
|
223
|
-
`${input} should never hit the bridge queue reply`,
|
|
224
|
-
);
|
|
225
|
-
};
|
|
226
|
-
|
|
227
|
-
await expectBotReply('/start', 'Nasıl başlarsın');
|
|
228
|
-
await expectBotReply('/help', 'Komutlar:');
|
|
229
|
-
|
|
230
|
-
telegramLinksDb.unlink(userId);
|
|
231
|
-
} catch (error) {
|
|
232
|
-
failures.push(error?.message || String(error));
|
|
233
|
-
} finally {
|
|
234
|
-
rmSync(runtimeDir, { recursive: true, force: true });
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
if (failures.length > 0) {
|
|
238
|
-
console.error(`Telegram control smoke failed:\n- ${failures.join('\n- ')}`);
|
|
239
|
-
process.exit(1);
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
console.log('telegram control smoke passed');
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
import { mkdirSync, readFileSync, rmSync } from 'node:fs';
|
|
5
|
+
import path from 'node:path';
|
|
6
|
+
|
|
7
|
+
const runtimeDir = path.resolve('.pixcode-dev', 'smoke-telegram-control');
|
|
8
|
+
mkdirSync(runtimeDir, { recursive: true });
|
|
9
|
+
process.env.DATABASE_PATH = path.join(runtimeDir, 'auth.db');
|
|
10
|
+
|
|
11
|
+
const checks = [
|
|
12
|
+
{
|
|
13
|
+
name: 'telegram bot wires the remote control center and callback queries',
|
|
14
|
+
file: 'server/services/telegram/bot.js',
|
|
15
|
+
test: (source) => (
|
|
16
|
+
source.includes('handleTelegramControlMessage')
|
|
17
|
+
&& source.includes('handleTelegramControlCallback')
|
|
18
|
+
&& source.includes("bot.on('callback_query'")
|
|
19
|
+
),
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
name: 'telegram HTTP client polls callback queries and can answer them',
|
|
23
|
+
file: 'server/services/telegram/telegram-http-client.js',
|
|
24
|
+
test: (source) => (
|
|
25
|
+
source.includes("allowed_updates: ['message', 'callback_query']")
|
|
26
|
+
&& source.includes('answerCallbackQuery')
|
|
27
|
+
&& source.includes('editMessageText')
|
|
28
|
+
&& source.includes("this.emit('callback_query'")
|
|
29
|
+
),
|
|
30
|
+
},
|
|
31
|
+
{
|
|
32
|
+
name: 'telegram control center exposes provider, model, workflow, install, and settings actions',
|
|
33
|
+
file: 'server/services/telegram/control-center.js',
|
|
34
|
+
test: (source) => (
|
|
35
|
+
source.includes('showMainMenu')
|
|
36
|
+
&& source.includes('showProviderMenu')
|
|
37
|
+
&& source.includes('showModelMenu')
|
|
38
|
+
&& source.includes('showWorkflowMenu')
|
|
39
|
+
&& source.includes('runWorkflow')
|
|
40
|
+
&& source.includes('startCliInstall')
|
|
41
|
+
&& source.includes('updateTelegramControlState')
|
|
42
|
+
&& source.includes('/api/agent')
|
|
43
|
+
&& source.includes('/api/orchestration/workflows')
|
|
44
|
+
),
|
|
45
|
+
},
|
|
46
|
+
{
|
|
47
|
+
name: 'telegram link state persists remote-control preferences',
|
|
48
|
+
file: 'server/database/db.js',
|
|
49
|
+
test: (source) => (
|
|
50
|
+
source.includes('telegram_control')
|
|
51
|
+
&& source.includes('getControlState')
|
|
52
|
+
&& source.includes('updateControlState')
|
|
53
|
+
&& source.includes('remoteControlEnabled')
|
|
54
|
+
),
|
|
55
|
+
},
|
|
56
|
+
{
|
|
57
|
+
name: 'telegram settings UI exposes remote-control toggles',
|
|
58
|
+
file: 'src/components/settings/view/tabs/telegram-settings/TelegramSettingsTab.tsx',
|
|
59
|
+
test: (source) => (
|
|
60
|
+
source.includes('controlEnabled')
|
|
61
|
+
&& source.includes('progressMode')
|
|
62
|
+
&& source.includes('telegram.control.title')
|
|
63
|
+
&& source.includes('telegram.control.progressMode')
|
|
64
|
+
),
|
|
65
|
+
},
|
|
66
|
+
];
|
|
67
|
+
|
|
68
|
+
const failures = [];
|
|
69
|
+
|
|
70
|
+
for (const check of checks) {
|
|
71
|
+
let source = '';
|
|
72
|
+
try {
|
|
73
|
+
source = readFileSync(check.file, 'utf8');
|
|
74
|
+
} catch {
|
|
75
|
+
failures.push(`${check.name} (${check.file} missing)`);
|
|
76
|
+
continue;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
if (!check.test(source)) failures.push(check.name);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
try {
|
|
83
|
+
const {
|
|
84
|
+
handleTelegramControlCallback,
|
|
85
|
+
handleTelegramControlMessage,
|
|
86
|
+
} = await import('../../server/services/telegram/control-center.js');
|
|
87
|
+
const {
|
|
88
|
+
handleIncomingTelegramMessage,
|
|
89
|
+
setTelegramBotForTesting,
|
|
90
|
+
} = await import('../../server/services/telegram/bot.js');
|
|
91
|
+
const { telegramLinksDb } = await import('../../server/database/db.js');
|
|
92
|
+
|
|
93
|
+
const userId = 4242;
|
|
94
|
+
telegramLinksDb.unlink(userId);
|
|
95
|
+
telegramLinksDb.setPairingCode(userId, '123456', new Date(Date.now() + 600_000).toISOString(), 'tr');
|
|
96
|
+
telegramLinksDb.verify(userId, 'chat-4242', 'ali');
|
|
97
|
+
|
|
98
|
+
const sent = [];
|
|
99
|
+
const bot = {
|
|
100
|
+
sendMessage: async (chatId, text, extra = {}) => {
|
|
101
|
+
sent.push({ chatId, text, extra });
|
|
102
|
+
return { ok: true };
|
|
103
|
+
},
|
|
104
|
+
};
|
|
105
|
+
|
|
106
|
+
const link = telegramLinksDb.getByUserId(userId);
|
|
107
|
+
const expectReply = async (input, expectedFragment) => {
|
|
108
|
+
sent.length = 0;
|
|
109
|
+
const handled = await handleTelegramControlMessage({
|
|
110
|
+
bot,
|
|
111
|
+
msg: { chat: { id: 4242 }, text: input },
|
|
112
|
+
link,
|
|
113
|
+
});
|
|
114
|
+
assert.equal(handled, true, `${input} should be handled`);
|
|
115
|
+
assert.ok(sent.length > 0, `${input} should send at least one reply`);
|
|
116
|
+
assert.ok(
|
|
117
|
+
sent[0].text.includes(expectedFragment),
|
|
118
|
+
`${input} should include "${expectedFragment}" but got "${sent[0].text}"`,
|
|
119
|
+
);
|
|
120
|
+
};
|
|
121
|
+
|
|
122
|
+
await expectReply('/start', 'Pixcode Telegram kontrol merkezi');
|
|
123
|
+
await expectReply('/help', 'Komutlar:');
|
|
124
|
+
await expectReply('/start@Otobot', 'Pixcode Telegram kontrol merkezi');
|
|
125
|
+
await expectReply('/help@Otobot', 'Komutlar:');
|
|
126
|
+
await expectReply('/', 'Komutlar:');
|
|
127
|
+
|
|
128
|
+
const menuEvents = [];
|
|
129
|
+
const menuBot = {
|
|
130
|
+
sendMessage: async (chatId, text, extra = {}) => {
|
|
131
|
+
menuEvents.push({ type: 'send', chatId, text, extra });
|
|
132
|
+
return { ok: true, message_id: 77 };
|
|
133
|
+
},
|
|
134
|
+
editMessageText: async (text, extra = {}) => {
|
|
135
|
+
menuEvents.push({ type: 'edit', text, extra });
|
|
136
|
+
return { ok: true, message_id: extra.message_id };
|
|
137
|
+
},
|
|
138
|
+
answerCallbackQuery: async () => ({ ok: true }),
|
|
139
|
+
};
|
|
140
|
+
const findButton = (markup, predicate) => {
|
|
141
|
+
for (const row of markup?.inline_keyboard || []) {
|
|
142
|
+
for (const candidate of row) {
|
|
143
|
+
if (predicate(candidate)) return candidate;
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
return null;
|
|
147
|
+
};
|
|
148
|
+
|
|
149
|
+
menuEvents.length = 0;
|
|
150
|
+
await handleTelegramControlMessage({
|
|
151
|
+
bot: menuBot,
|
|
152
|
+
msg: { chat: { id: 4242 }, text: '/settings' },
|
|
153
|
+
link,
|
|
154
|
+
});
|
|
155
|
+
const settingsMenu = menuEvents.at(-1);
|
|
156
|
+
const languageButton = findButton(
|
|
157
|
+
settingsMenu.extra.reply_markup,
|
|
158
|
+
(candidate) => /language|dil/i.test(candidate.text),
|
|
159
|
+
);
|
|
160
|
+
assert.ok(languageButton, 'settings menu should expose a language button');
|
|
161
|
+
|
|
162
|
+
menuEvents.length = 0;
|
|
163
|
+
await handleTelegramControlCallback({
|
|
164
|
+
bot: menuBot,
|
|
165
|
+
query: {
|
|
166
|
+
id: 'query-language-menu',
|
|
167
|
+
data: languageButton.callback_data,
|
|
168
|
+
message: { chat: { id: 4242 }, message_id: 77 },
|
|
169
|
+
},
|
|
170
|
+
link,
|
|
171
|
+
});
|
|
172
|
+
assert.equal(menuEvents.length, 1, 'callback menus should replace the existing menu message');
|
|
173
|
+
assert.equal(menuEvents[0].type, 'edit', 'callback menus should use editMessageText');
|
|
174
|
+
const trButton = findButton(menuEvents[0].extra.reply_markup, (candidate) => candidate.text === 'tr');
|
|
175
|
+
assert.ok(trButton, 'language menu should include Turkish');
|
|
176
|
+
|
|
177
|
+
menuEvents.length = 0;
|
|
178
|
+
await handleTelegramControlCallback({
|
|
179
|
+
bot: menuBot,
|
|
180
|
+
query: {
|
|
181
|
+
id: 'query-language-tr',
|
|
182
|
+
data: trButton.callback_data,
|
|
183
|
+
message: { chat: { id: 4242 }, message_id: 77 },
|
|
184
|
+
},
|
|
185
|
+
link,
|
|
186
|
+
});
|
|
187
|
+
assert.equal(telegramLinksDb.getByUserId(userId).language, 'tr', 'language selection should persist');
|
|
188
|
+
assert.equal(menuEvents.length, 1, 'language selection should not send a confirmation plus a second menu');
|
|
189
|
+
assert.equal(menuEvents[0].type, 'edit', 'language selection should replace the menu in place');
|
|
190
|
+
assert.ok(
|
|
191
|
+
menuEvents[0].text.includes('Pixcode Telegram kontrol merkezi'),
|
|
192
|
+
`Turkish menu should render after selection, got "${menuEvents[0].text}"`,
|
|
193
|
+
);
|
|
194
|
+
assert.ok(
|
|
195
|
+
!menuEvents[0].text.includes('Project:'),
|
|
196
|
+
'Turkish menu should not keep English summary labels after language selection',
|
|
197
|
+
);
|
|
198
|
+
|
|
199
|
+
const botLevelMessages = [];
|
|
200
|
+
setTelegramBotForTesting({
|
|
201
|
+
sendMessage: async (chatId, text, extra = {}) => {
|
|
202
|
+
botLevelMessages.push({ chatId, text, extra });
|
|
203
|
+
return { ok: true };
|
|
204
|
+
},
|
|
205
|
+
answerCallbackQuery: async () => ({ ok: true }),
|
|
206
|
+
});
|
|
207
|
+
|
|
208
|
+
const expectBotReply = async (input, expectedFragment) => {
|
|
209
|
+
botLevelMessages.length = 0;
|
|
210
|
+
await handleIncomingTelegramMessage({
|
|
211
|
+
chat: { id: 'chat-4242' },
|
|
212
|
+
text: input,
|
|
213
|
+
message_id: 1,
|
|
214
|
+
from: { username: 'ali' },
|
|
215
|
+
});
|
|
216
|
+
assert.ok(botLevelMessages.length > 0, `${input} should reply from bot.js`);
|
|
217
|
+
assert.ok(
|
|
218
|
+
botLevelMessages[0].text.includes(expectedFragment),
|
|
219
|
+
`${input} should include "${expectedFragment}" but got "${botLevelMessages[0].text}"`,
|
|
220
|
+
);
|
|
221
|
+
assert.ok(
|
|
222
|
+
!botLevelMessages.some((entry) => entry.text.includes('Mesaj son oturumuna iletildi')),
|
|
223
|
+
`${input} should never hit the bridge queue reply`,
|
|
224
|
+
);
|
|
225
|
+
};
|
|
226
|
+
|
|
227
|
+
await expectBotReply('/start', 'Nasıl başlarsın');
|
|
228
|
+
await expectBotReply('/help', 'Komutlar:');
|
|
229
|
+
|
|
230
|
+
telegramLinksDb.unlink(userId);
|
|
231
|
+
} catch (error) {
|
|
232
|
+
failures.push(error?.message || String(error));
|
|
233
|
+
} finally {
|
|
234
|
+
rmSync(runtimeDir, { recursive: true, force: true });
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
if (failures.length > 0) {
|
|
238
|
+
console.error(`Telegram control smoke failed:\n- ${failures.join('\n- ')}`);
|
|
239
|
+
process.exit(1);
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
console.log('telegram control smoke passed');
|
|
@@ -1,56 +1,56 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
|
|
3
|
-
import assert from 'node:assert/strict';
|
|
4
|
-
import { readFileSync } from 'node:fs';
|
|
5
|
-
|
|
6
|
-
const externalAccess = readFileSync('server/services/external-access.js', 'utf8');
|
|
7
|
-
const networkRoutes = readFileSync('server/routes/network.js', 'utf8');
|
|
8
|
-
const serverIndex = readFileSync('server/index.js', 'utf8');
|
|
9
|
-
|
|
10
|
-
assert.match(
|
|
11
|
-
externalAccess,
|
|
12
|
-
/TUNNEL_PERSISTENCE_PATH/,
|
|
13
|
-
'Tunnel service should persist the user-requested tunnel state outside process memory.',
|
|
14
|
-
);
|
|
15
|
-
assert.match(
|
|
16
|
-
externalAccess,
|
|
17
|
-
/persistTunnelPreference/,
|
|
18
|
-
'Tunnel service should write tunnel start/stop intent to disk.',
|
|
19
|
-
);
|
|
20
|
-
assert.match(
|
|
21
|
-
externalAccess,
|
|
22
|
-
/desired:\s*true/,
|
|
23
|
-
'Starting a tunnel should mark tunnel intent as desired until the user stops it.',
|
|
24
|
-
);
|
|
25
|
-
assert.match(
|
|
26
|
-
externalAccess,
|
|
27
|
-
/desired:\s*false/,
|
|
28
|
-
'Stopping a tunnel should clear persisted tunnel intent.',
|
|
29
|
-
);
|
|
30
|
-
assert.match(
|
|
31
|
-
externalAccess,
|
|
32
|
-
/restoreRequestedTunnel/,
|
|
33
|
-
'Tunnel service should expose a startup restore hook.',
|
|
34
|
-
);
|
|
35
|
-
assert.match(
|
|
36
|
-
externalAccess,
|
|
37
|
-
/restoring/,
|
|
38
|
-
'Tunnel restore should distinguish automatic restart attempts from direct user starts.',
|
|
39
|
-
);
|
|
40
|
-
assert.match(
|
|
41
|
-
networkRoutes,
|
|
42
|
-
/persistPreference:\s*true/,
|
|
43
|
-
'Manual tunnel starts should persist the user preference through the network route.',
|
|
44
|
-
);
|
|
45
|
-
assert.match(
|
|
46
|
-
serverIndex,
|
|
47
|
-
/restoreRequestedTunnel/,
|
|
48
|
-
'Server startup should restore a requested tunnel after updates/restarts.',
|
|
49
|
-
);
|
|
50
|
-
assert.match(
|
|
51
|
-
serverIndex,
|
|
52
|
-
/restoreRequestedTunnel\(\{ port: Number\(SERVER_PORT\) \}\)/,
|
|
53
|
-
'Server startup should restore the tunnel against the current backend port.',
|
|
54
|
-
);
|
|
55
|
-
|
|
56
|
-
console.log('tunnel persistence smoke passed');
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
import { readFileSync } from 'node:fs';
|
|
5
|
+
|
|
6
|
+
const externalAccess = readFileSync('server/services/external-access.js', 'utf8');
|
|
7
|
+
const networkRoutes = readFileSync('server/routes/network.js', 'utf8');
|
|
8
|
+
const serverIndex = readFileSync('server/index.js', 'utf8');
|
|
9
|
+
|
|
10
|
+
assert.match(
|
|
11
|
+
externalAccess,
|
|
12
|
+
/TUNNEL_PERSISTENCE_PATH/,
|
|
13
|
+
'Tunnel service should persist the user-requested tunnel state outside process memory.',
|
|
14
|
+
);
|
|
15
|
+
assert.match(
|
|
16
|
+
externalAccess,
|
|
17
|
+
/persistTunnelPreference/,
|
|
18
|
+
'Tunnel service should write tunnel start/stop intent to disk.',
|
|
19
|
+
);
|
|
20
|
+
assert.match(
|
|
21
|
+
externalAccess,
|
|
22
|
+
/desired:\s*true/,
|
|
23
|
+
'Starting a tunnel should mark tunnel intent as desired until the user stops it.',
|
|
24
|
+
);
|
|
25
|
+
assert.match(
|
|
26
|
+
externalAccess,
|
|
27
|
+
/desired:\s*false/,
|
|
28
|
+
'Stopping a tunnel should clear persisted tunnel intent.',
|
|
29
|
+
);
|
|
30
|
+
assert.match(
|
|
31
|
+
externalAccess,
|
|
32
|
+
/restoreRequestedTunnel/,
|
|
33
|
+
'Tunnel service should expose a startup restore hook.',
|
|
34
|
+
);
|
|
35
|
+
assert.match(
|
|
36
|
+
externalAccess,
|
|
37
|
+
/restoring/,
|
|
38
|
+
'Tunnel restore should distinguish automatic restart attempts from direct user starts.',
|
|
39
|
+
);
|
|
40
|
+
assert.match(
|
|
41
|
+
networkRoutes,
|
|
42
|
+
/persistPreference:\s*true/,
|
|
43
|
+
'Manual tunnel starts should persist the user preference through the network route.',
|
|
44
|
+
);
|
|
45
|
+
assert.match(
|
|
46
|
+
serverIndex,
|
|
47
|
+
/restoreRequestedTunnel/,
|
|
48
|
+
'Server startup should restore a requested tunnel after updates/restarts.',
|
|
49
|
+
);
|
|
50
|
+
assert.match(
|
|
51
|
+
serverIndex,
|
|
52
|
+
/restoreRequestedTunnel\(\{ port: Number\(SERVER_PORT\) \}\)/,
|
|
53
|
+
'Server startup should restore the tunnel against the current backend port.',
|
|
54
|
+
);
|
|
55
|
+
|
|
56
|
+
console.log('tunnel persistence smoke passed');
|