@pixelbyte-software/pixcode 1.51.1 → 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-DARIZgoD.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 +125 -4
- 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 -164
- 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 +131 -3
- 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,91 +1,91 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
|
|
3
|
-
import assert from 'node:assert/strict';
|
|
4
|
-
import fs from 'node:fs';
|
|
5
|
-
import path from 'node:path';
|
|
6
|
-
|
|
7
|
-
const root = process.cwd();
|
|
8
|
-
|
|
9
|
-
function read(relativePath) {
|
|
10
|
-
return fs.readFileSync(path.join(root, relativePath), 'utf8');
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
function assertIncludes(file, pattern, message) {
|
|
14
|
-
assert.match(file, pattern instanceof RegExp ? pattern : new RegExp(pattern), message);
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
const page = read('src/components/control-room/ControlRoomPage.tsx');
|
|
18
|
-
const primitives = read('src/components/control-room/ControlRoomPrimitives.tsx');
|
|
19
|
-
const motion = read('src/lib/animations.ts');
|
|
20
|
-
|
|
21
|
-
for (const symbol of [
|
|
22
|
-
'CONTROL_ROOM_GROUPS',
|
|
23
|
-
'OVERVIEW_CARDS',
|
|
24
|
-
'operations',
|
|
25
|
-
'people',
|
|
26
|
-
'access',
|
|
27
|
-
'security',
|
|
28
|
-
'insights',
|
|
29
|
-
]) {
|
|
30
|
-
assertIncludes(page, new RegExp(symbol), `Control Room should include grouped ${symbol} IA.`);
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
const overviewCards = page.match(/id: 'overview-card-/g) || [];
|
|
34
|
-
assert.equal(overviewCards.length, 6, 'Control Room first screen should expose exactly six overview action cards.');
|
|
35
|
-
|
|
36
|
-
assert.doesNotMatch(page, /const sections:/, 'Control Room should not use the old flat feature-wall section list.');
|
|
37
|
-
assert.doesNotMatch(page, /v1\.46 launch surface/, 'Control Room should not frame the UI as a release inventory.');
|
|
38
|
-
assert.doesNotMatch(page, /activeSection === 'admin'[\s\S]{0,3000}activeSection === 'team'/, 'Control Room should group people/admin concepts instead of flat adjacent sections.');
|
|
39
|
-
|
|
40
|
-
for (const primitive of [
|
|
41
|
-
'CommandCard',
|
|
42
|
-
'SummaryCard',
|
|
43
|
-
'ControlRoomPanel',
|
|
44
|
-
'ContextDrawer',
|
|
45
|
-
'GuidanceCard',
|
|
46
|
-
'EmptyGuidance',
|
|
47
|
-
'TimelineItem',
|
|
48
|
-
'ActionRow',
|
|
49
|
-
'ResponsiveDataList',
|
|
50
|
-
'AdvancedDisclosure',
|
|
51
|
-
]) {
|
|
52
|
-
assertIncludes(primitives, new RegExp(`function ${primitive}|const ${primitive}`), `Control Room primitives should include ${primitive}.`);
|
|
53
|
-
assertIncludes(page, new RegExp(primitive), `ControlRoomPage should use ${primitive}.`);
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
for (const copy of [
|
|
57
|
-
'What needs attention',
|
|
58
|
-
'Running now',
|
|
59
|
-
'Team and access',
|
|
60
|
-
'Security and secrets',
|
|
61
|
-
'Usage and evaluations',
|
|
62
|
-
'Run timeline',
|
|
63
|
-
'Recommended next step',
|
|
64
|
-
'What this means',
|
|
65
|
-
]) {
|
|
66
|
-
assertIncludes(page, copy.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'), `Control Room should include plain-language copy: ${copy}.`);
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
assertIncludes(page, /lg:grid-cols-\[220px_minmax\(0,1fr\)_320px\]/, 'Desktop Control Room should use a master-detail grid.');
|
|
70
|
-
assertIncludes(page, /overflow-x-auto lg:hidden/, 'Mobile Control Room should use compact horizontal group navigation.');
|
|
71
|
-
assertIncludes(page, /min-h-\[44px\]|h-11/, 'Primary controls should keep mobile touch targets at or above 44px.');
|
|
72
|
-
assertIncludes(page, /AdvancedDisclosure/, 'Advanced forms should be hidden behind progressive disclosure.');
|
|
73
|
-
assertIncludes(page, /TimelineItem[\s\S]*actor[\s\S]*result/, 'Run timeline should expose readable actor/action/result fields.');
|
|
74
|
-
assertIncludes(page, /ContextDrawer/, 'Control Room should include a contextual detail rail or sheet.');
|
|
75
|
-
|
|
76
|
-
for (const motionSymbol of [
|
|
77
|
-
'surface',
|
|
78
|
-
'drawer',
|
|
79
|
-
'status',
|
|
80
|
-
'useGsapSurfaceTransition',
|
|
81
|
-
'useGsapListReveal',
|
|
82
|
-
'useGsapStatusHighlight',
|
|
83
|
-
'prefersReducedMotion',
|
|
84
|
-
]) {
|
|
85
|
-
assertIncludes(motion, new RegExp(motionSymbol), `Shared motion system should include ${motionSymbol}.`);
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
assertIncludes(motion, /prefers-reduced-motion: reduce/, 'Motion utilities should respect prefers-reduced-motion.');
|
|
89
|
-
assertIncludes(motion, /opacityOnly|reduced motion|Reduced/, 'Motion utilities should document or implement reduced-motion fallback.');
|
|
90
|
-
|
|
91
|
-
console.log('control room UX redesign smoke passed');
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
import fs from 'node:fs';
|
|
5
|
+
import path from 'node:path';
|
|
6
|
+
|
|
7
|
+
const root = process.cwd();
|
|
8
|
+
|
|
9
|
+
function read(relativePath) {
|
|
10
|
+
return fs.readFileSync(path.join(root, relativePath), 'utf8');
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
function assertIncludes(file, pattern, message) {
|
|
14
|
+
assert.match(file, pattern instanceof RegExp ? pattern : new RegExp(pattern), message);
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
const page = read('src/components/control-room/ControlRoomPage.tsx');
|
|
18
|
+
const primitives = read('src/components/control-room/ControlRoomPrimitives.tsx');
|
|
19
|
+
const motion = read('src/lib/animations.ts');
|
|
20
|
+
|
|
21
|
+
for (const symbol of [
|
|
22
|
+
'CONTROL_ROOM_GROUPS',
|
|
23
|
+
'OVERVIEW_CARDS',
|
|
24
|
+
'operations',
|
|
25
|
+
'people',
|
|
26
|
+
'access',
|
|
27
|
+
'security',
|
|
28
|
+
'insights',
|
|
29
|
+
]) {
|
|
30
|
+
assertIncludes(page, new RegExp(symbol), `Control Room should include grouped ${symbol} IA.`);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
const overviewCards = page.match(/id: 'overview-card-/g) || [];
|
|
34
|
+
assert.equal(overviewCards.length, 6, 'Control Room first screen should expose exactly six overview action cards.');
|
|
35
|
+
|
|
36
|
+
assert.doesNotMatch(page, /const sections:/, 'Control Room should not use the old flat feature-wall section list.');
|
|
37
|
+
assert.doesNotMatch(page, /v1\.46 launch surface/, 'Control Room should not frame the UI as a release inventory.');
|
|
38
|
+
assert.doesNotMatch(page, /activeSection === 'admin'[\s\S]{0,3000}activeSection === 'team'/, 'Control Room should group people/admin concepts instead of flat adjacent sections.');
|
|
39
|
+
|
|
40
|
+
for (const primitive of [
|
|
41
|
+
'CommandCard',
|
|
42
|
+
'SummaryCard',
|
|
43
|
+
'ControlRoomPanel',
|
|
44
|
+
'ContextDrawer',
|
|
45
|
+
'GuidanceCard',
|
|
46
|
+
'EmptyGuidance',
|
|
47
|
+
'TimelineItem',
|
|
48
|
+
'ActionRow',
|
|
49
|
+
'ResponsiveDataList',
|
|
50
|
+
'AdvancedDisclosure',
|
|
51
|
+
]) {
|
|
52
|
+
assertIncludes(primitives, new RegExp(`function ${primitive}|const ${primitive}`), `Control Room primitives should include ${primitive}.`);
|
|
53
|
+
assertIncludes(page, new RegExp(primitive), `ControlRoomPage should use ${primitive}.`);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
for (const copy of [
|
|
57
|
+
'What needs attention',
|
|
58
|
+
'Running now',
|
|
59
|
+
'Team and access',
|
|
60
|
+
'Security and secrets',
|
|
61
|
+
'Usage and evaluations',
|
|
62
|
+
'Run timeline',
|
|
63
|
+
'Recommended next step',
|
|
64
|
+
'What this means',
|
|
65
|
+
]) {
|
|
66
|
+
assertIncludes(page, copy.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'), `Control Room should include plain-language copy: ${copy}.`);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
assertIncludes(page, /lg:grid-cols-\[220px_minmax\(0,1fr\)_320px\]/, 'Desktop Control Room should use a master-detail grid.');
|
|
70
|
+
assertIncludes(page, /overflow-x-auto lg:hidden/, 'Mobile Control Room should use compact horizontal group navigation.');
|
|
71
|
+
assertIncludes(page, /min-h-\[44px\]|h-11/, 'Primary controls should keep mobile touch targets at or above 44px.');
|
|
72
|
+
assertIncludes(page, /AdvancedDisclosure/, 'Advanced forms should be hidden behind progressive disclosure.');
|
|
73
|
+
assertIncludes(page, /TimelineItem[\s\S]*actor[\s\S]*result/, 'Run timeline should expose readable actor/action/result fields.');
|
|
74
|
+
assertIncludes(page, /ContextDrawer/, 'Control Room should include a contextual detail rail or sheet.');
|
|
75
|
+
|
|
76
|
+
for (const motionSymbol of [
|
|
77
|
+
'surface',
|
|
78
|
+
'drawer',
|
|
79
|
+
'status',
|
|
80
|
+
'useGsapSurfaceTransition',
|
|
81
|
+
'useGsapListReveal',
|
|
82
|
+
'useGsapStatusHighlight',
|
|
83
|
+
'prefersReducedMotion',
|
|
84
|
+
]) {
|
|
85
|
+
assertIncludes(motion, new RegExp(motionSymbol), `Shared motion system should include ${motionSymbol}.`);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
assertIncludes(motion, /prefers-reduced-motion: reduce/, 'Motion utilities should respect prefers-reduced-motion.');
|
|
89
|
+
assertIncludes(motion, /opacityOnly|reduced motion|Reduced/, 'Motion utilities should document or implement reduced-motion fallback.');
|
|
90
|
+
|
|
91
|
+
console.log('control room UX redesign smoke passed');
|
|
@@ -1,20 +1,20 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
|
|
3
|
-
import assert from 'node:assert/strict';
|
|
4
|
-
import path from 'node:path';
|
|
5
|
-
|
|
6
|
-
import { resolveDaemonCliEntryPath } from '../../server/daemon-manager.js';
|
|
7
|
-
|
|
8
|
-
const appRoot = path.resolve('.');
|
|
9
|
-
const resolved = resolveDaemonCliEntryPath({
|
|
10
|
-
appRoot,
|
|
11
|
-
cliEntry: path.join(appRoot, 'server', 'cli.js'),
|
|
12
|
-
});
|
|
13
|
-
|
|
14
|
-
assert.equal(
|
|
15
|
-
resolved,
|
|
16
|
-
path.join(appRoot, 'dist-server', 'server', 'cli.js'),
|
|
17
|
-
`expected daemon entrypoint to prefer dist-server, got ${resolved}`,
|
|
18
|
-
);
|
|
19
|
-
|
|
20
|
-
console.log('daemon entrypoint smoke passed');
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
import path from 'node:path';
|
|
5
|
+
|
|
6
|
+
import { resolveDaemonCliEntryPath } from '../../server/daemon-manager.js';
|
|
7
|
+
|
|
8
|
+
const appRoot = path.resolve('.');
|
|
9
|
+
const resolved = resolveDaemonCliEntryPath({
|
|
10
|
+
appRoot,
|
|
11
|
+
cliEntry: path.join(appRoot, 'server', 'cli.js'),
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
assert.equal(
|
|
15
|
+
resolved,
|
|
16
|
+
path.join(appRoot, 'dist-server', 'server', 'cli.js'),
|
|
17
|
+
`expected daemon entrypoint to prefer dist-server, got ${resolved}`,
|
|
18
|
+
);
|
|
19
|
+
|
|
20
|
+
console.log('daemon entrypoint smoke passed');
|
|
@@ -1,33 +1,33 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
|
|
3
|
-
import assert from 'node:assert/strict';
|
|
4
|
-
import fs from 'node:fs';
|
|
5
|
-
import path from 'node:path';
|
|
6
|
-
|
|
7
|
-
const root = process.cwd();
|
|
8
|
-
|
|
9
|
-
function read(relativePath) {
|
|
10
|
-
return fs.readFileSync(path.join(root, relativePath), 'utf8');
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
const projectsState = read('src/hooks/useProjectsState.ts');
|
|
14
|
-
assert.match(projectsState, /NON_RESTORABLE_TABS/, 'Non-default utility tabs should be excluded from startup restore.');
|
|
15
|
-
assert.match(projectsState, /controlRoom/, 'Control Room should be treated as a non-restorable startup tab.');
|
|
16
|
-
assert.match(projectsState, /getPersistableTab/, 'Active tab persistence should normalize non-restorable tabs.');
|
|
17
|
-
assert.doesNotMatch(projectsState, /handleQuickStartTasks|quickStartIntoTab\('tasks'\)|setTasksEnabled/, 'Task quick-start routing should be removed.');
|
|
18
|
-
|
|
19
|
-
const appContent = read('src/components/app/AppContent.tsx');
|
|
20
|
-
assert.doesNotMatch(appContent, /handleQuickStartTasks|onQuickStartTasks/, 'App content should not pass Task quick-start callbacks.');
|
|
21
|
-
assert.match(appContent, /<VSCodeWorkbench/, 'App content should use VS Code workbench as the desktop default.');
|
|
22
|
-
|
|
23
|
-
const mainContentTypes = read('src/components/main-content/types/types.ts');
|
|
24
|
-
assert.doesNotMatch(mainContentTypes, /onQuickStartTasks/, 'Main content props should not include Task quick-start support.');
|
|
25
|
-
|
|
26
|
-
const mainContent = read('src/components/main-content/view/MainContent.tsx');
|
|
27
|
-
assert.doesNotMatch(mainContent, /shouldShowTasksTab|TaskMasterPanel|tasksEnabled/, 'Main content should not render the TaskMaster surface.');
|
|
28
|
-
|
|
29
|
-
const emptyState = read('src/components/main-content/view/subcomponents/MainContentStateView.tsx');
|
|
30
|
-
assert.doesNotMatch(emptyState, /onQuickStartTasks|ClipboardCheck|taskSystem/, 'Empty state should not expose TaskMaster landing cards.');
|
|
31
|
-
assert.match(emptyState, /pixcode:create-project/, 'Empty state should keep the generic create-project flow.');
|
|
32
|
-
|
|
33
|
-
console.log('default landing routing smoke passed');
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
import assert from 'node:assert/strict';
|
|
4
|
+
import fs from 'node:fs';
|
|
5
|
+
import path from 'node:path';
|
|
6
|
+
|
|
7
|
+
const root = process.cwd();
|
|
8
|
+
|
|
9
|
+
function read(relativePath) {
|
|
10
|
+
return fs.readFileSync(path.join(root, relativePath), 'utf8');
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
const projectsState = read('src/hooks/useProjectsState.ts');
|
|
14
|
+
assert.match(projectsState, /NON_RESTORABLE_TABS/, 'Non-default utility tabs should be excluded from startup restore.');
|
|
15
|
+
assert.match(projectsState, /controlRoom/, 'Control Room should be treated as a non-restorable startup tab.');
|
|
16
|
+
assert.match(projectsState, /getPersistableTab/, 'Active tab persistence should normalize non-restorable tabs.');
|
|
17
|
+
assert.doesNotMatch(projectsState, /handleQuickStartTasks|quickStartIntoTab\('tasks'\)|setTasksEnabled/, 'Task quick-start routing should be removed.');
|
|
18
|
+
|
|
19
|
+
const appContent = read('src/components/app/AppContent.tsx');
|
|
20
|
+
assert.doesNotMatch(appContent, /handleQuickStartTasks|onQuickStartTasks/, 'App content should not pass Task quick-start callbacks.');
|
|
21
|
+
assert.match(appContent, /<VSCodeWorkbench/, 'App content should use VS Code workbench as the desktop default.');
|
|
22
|
+
|
|
23
|
+
const mainContentTypes = read('src/components/main-content/types/types.ts');
|
|
24
|
+
assert.doesNotMatch(mainContentTypes, /onQuickStartTasks/, 'Main content props should not include Task quick-start support.');
|
|
25
|
+
|
|
26
|
+
const mainContent = read('src/components/main-content/view/MainContent.tsx');
|
|
27
|
+
assert.doesNotMatch(mainContent, /shouldShowTasksTab|TaskMasterPanel|tasksEnabled/, 'Main content should not render the TaskMaster surface.');
|
|
28
|
+
|
|
29
|
+
const emptyState = read('src/components/main-content/view/subcomponents/MainContentStateView.tsx');
|
|
30
|
+
assert.doesNotMatch(emptyState, /onQuickStartTasks|ClipboardCheck|taskSystem/, 'Empty state should not expose TaskMaster landing cards.');
|
|
31
|
+
assert.match(emptyState, /pixcode:create-project/, 'Empty state should keep the generic create-project flow.');
|
|
32
|
+
|
|
33
|
+
console.log('default landing routing smoke passed');
|
|
@@ -1,30 +1,30 @@
|
|
|
1
|
-
import fs from 'node:fs';
|
|
2
|
-
|
|
3
|
-
const mainPath = 'desktop/electron/main.cjs';
|
|
4
|
-
const preloadPath = 'desktop/electron/preload.cjs';
|
|
5
|
-
const localNotificationsPath = 'src/utils/localNotifications.ts';
|
|
6
|
-
const globalsPath = 'src/types/global.d.ts';
|
|
7
|
-
|
|
8
|
-
function assert(condition, message) {
|
|
9
|
-
if (!condition) {
|
|
10
|
-
console.error(`FAIL: ${message}`);
|
|
11
|
-
process.exit(1);
|
|
12
|
-
}
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
const main = fs.readFileSync(mainPath, 'utf8');
|
|
16
|
-
const localNotifications = fs.readFileSync(localNotificationsPath, 'utf8');
|
|
17
|
-
const globals = fs.readFileSync(globalsPath, 'utf8');
|
|
18
|
-
|
|
19
|
-
assert(fs.existsSync(preloadPath), 'desktop preload bridge exists');
|
|
20
|
-
const preload = fs.readFileSync(preloadPath, 'utf8');
|
|
21
|
-
|
|
22
|
-
assert(main.includes("preload: path.join(__dirname, 'preload.cjs')"), 'BrowserWindow loads the desktop preload');
|
|
23
|
-
assert(main.includes("ipcMain.handle('pixcode:desktop-notification'"), 'main process handles desktop notification IPC');
|
|
24
|
-
assert(preload.includes('contextBridge.exposeInMainWorld'), 'preload exposes a safe renderer bridge');
|
|
25
|
-
assert(preload.includes('pixcodeDesktop'), 'preload exposes window.pixcodeDesktop');
|
|
26
|
-
assert(preload.includes("ipcRenderer.invoke('pixcode:desktop-notification'"), 'preload invokes desktop notification IPC');
|
|
27
|
-
assert(localNotifications.includes('window.pixcodeDesktop?.notify'), 'local notifications try the desktop native bridge first');
|
|
28
|
-
assert(globals.includes('pixcodeDesktop?:'), 'global window type declares the desktop bridge');
|
|
29
|
-
|
|
30
|
-
console.log('desktop native notification bridge smoke passed');
|
|
1
|
+
import fs from 'node:fs';
|
|
2
|
+
|
|
3
|
+
const mainPath = 'desktop/electron/main.cjs';
|
|
4
|
+
const preloadPath = 'desktop/electron/preload.cjs';
|
|
5
|
+
const localNotificationsPath = 'src/utils/localNotifications.ts';
|
|
6
|
+
const globalsPath = 'src/types/global.d.ts';
|
|
7
|
+
|
|
8
|
+
function assert(condition, message) {
|
|
9
|
+
if (!condition) {
|
|
10
|
+
console.error(`FAIL: ${message}`);
|
|
11
|
+
process.exit(1);
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
const main = fs.readFileSync(mainPath, 'utf8');
|
|
16
|
+
const localNotifications = fs.readFileSync(localNotificationsPath, 'utf8');
|
|
17
|
+
const globals = fs.readFileSync(globalsPath, 'utf8');
|
|
18
|
+
|
|
19
|
+
assert(fs.existsSync(preloadPath), 'desktop preload bridge exists');
|
|
20
|
+
const preload = fs.readFileSync(preloadPath, 'utf8');
|
|
21
|
+
|
|
22
|
+
assert(main.includes("preload: path.join(__dirname, 'preload.cjs')"), 'BrowserWindow loads the desktop preload');
|
|
23
|
+
assert(main.includes("ipcMain.handle('pixcode:desktop-notification'"), 'main process handles desktop notification IPC');
|
|
24
|
+
assert(preload.includes('contextBridge.exposeInMainWorld'), 'preload exposes a safe renderer bridge');
|
|
25
|
+
assert(preload.includes('pixcodeDesktop'), 'preload exposes window.pixcodeDesktop');
|
|
26
|
+
assert(preload.includes("ipcRenderer.invoke('pixcode:desktop-notification'"), 'preload invokes desktop notification IPC');
|
|
27
|
+
assert(localNotifications.includes('window.pixcodeDesktop?.notify'), 'local notifications try the desktop native bridge first');
|
|
28
|
+
assert(globals.includes('pixcodeDesktop?:'), 'global window type declares the desktop bridge');
|
|
29
|
+
|
|
30
|
+
console.log('desktop native notification bridge smoke passed');
|
|
@@ -1,33 +1,33 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
import { readFileSync } from 'node:fs';
|
|
3
|
-
|
|
4
|
-
const main = readFileSync('desktop/electron/main.cjs', 'utf8');
|
|
5
|
-
|
|
6
|
-
function assert(condition, message) {
|
|
7
|
-
if (!condition) {
|
|
8
|
-
console.error(`desktop-tray-icon smoke failed: ${message}`);
|
|
9
|
-
process.exit(1);
|
|
10
|
-
}
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
assert(
|
|
14
|
-
main.includes('function normalizeTrayIcon'),
|
|
15
|
-
'tray icons should be normalized before creating the Electron Tray',
|
|
16
|
-
);
|
|
17
|
-
|
|
18
|
-
assert(
|
|
19
|
-
main.includes("process.platform === 'darwin'") && main.includes('width: 18') && main.includes('height: 18'),
|
|
20
|
-
'macOS menu bar icon should be resized to 18x18 instead of using the 1024 app icon',
|
|
21
|
-
);
|
|
22
|
-
|
|
23
|
-
assert(
|
|
24
|
-
main.includes('setTemplateImage(true)'),
|
|
25
|
-
'macOS menu bar icon should be marked as a template image',
|
|
26
|
-
);
|
|
27
|
-
|
|
28
|
-
assert(
|
|
29
|
-
main.includes('return normalizeTrayIcon(nativeImage.createFromPath(p))'),
|
|
30
|
-
'resolveTrayIcon should return the normalized image',
|
|
31
|
-
);
|
|
32
|
-
|
|
33
|
-
console.log('desktop-tray-icon smoke passed');
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { readFileSync } from 'node:fs';
|
|
3
|
+
|
|
4
|
+
const main = readFileSync('desktop/electron/main.cjs', 'utf8');
|
|
5
|
+
|
|
6
|
+
function assert(condition, message) {
|
|
7
|
+
if (!condition) {
|
|
8
|
+
console.error(`desktop-tray-icon smoke failed: ${message}`);
|
|
9
|
+
process.exit(1);
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
assert(
|
|
14
|
+
main.includes('function normalizeTrayIcon'),
|
|
15
|
+
'tray icons should be normalized before creating the Electron Tray',
|
|
16
|
+
);
|
|
17
|
+
|
|
18
|
+
assert(
|
|
19
|
+
main.includes("process.platform === 'darwin'") && main.includes('width: 18') && main.includes('height: 18'),
|
|
20
|
+
'macOS menu bar icon should be resized to 18x18 instead of using the 1024 app icon',
|
|
21
|
+
);
|
|
22
|
+
|
|
23
|
+
assert(
|
|
24
|
+
main.includes('setTemplateImage(true)'),
|
|
25
|
+
'macOS menu bar icon should be marked as a template image',
|
|
26
|
+
);
|
|
27
|
+
|
|
28
|
+
assert(
|
|
29
|
+
main.includes('return normalizeTrayIcon(nativeImage.createFromPath(p))'),
|
|
30
|
+
'resolveTrayIcon should return the normalized image',
|
|
31
|
+
);
|
|
32
|
+
|
|
33
|
+
console.log('desktop-tray-icon smoke passed');
|
|
@@ -1,24 +1,24 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
import fs from 'node:fs';
|
|
3
|
-
|
|
4
|
-
const workflow = fs.readFileSync('.github/workflows/discord-release.yml', 'utf8');
|
|
5
|
-
|
|
6
|
-
const requiredSnippets = [
|
|
7
|
-
'DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_WEBHOOK_URL }}',
|
|
8
|
-
"if: ${{ env.DISCORD_WEBHOOK_URL == '' }}",
|
|
9
|
-
"if: ${{ env.DISCORD_WEBHOOK_URL != '' }}",
|
|
10
|
-
'curl -fsS',
|
|
11
|
-
'GITHUB_EVENT_PATH',
|
|
12
|
-
];
|
|
13
|
-
|
|
14
|
-
for (const snippet of requiredSnippets) {
|
|
15
|
-
if (!workflow.includes(snippet)) {
|
|
16
|
-
throw new Error(`Discord release workflow is missing required guard/publish snippet: ${snippet}`);
|
|
17
|
-
}
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
if (workflow.includes('SethCohen/github-releases-to-discord')) {
|
|
21
|
-
throw new Error('Discord release workflow should not depend on the deprecated external Discord action.');
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
console.log('discord release workflow smoke passed');
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import fs from 'node:fs';
|
|
3
|
+
|
|
4
|
+
const workflow = fs.readFileSync('.github/workflows/discord-release.yml', 'utf8');
|
|
5
|
+
|
|
6
|
+
const requiredSnippets = [
|
|
7
|
+
'DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_WEBHOOK_URL }}',
|
|
8
|
+
"if: ${{ env.DISCORD_WEBHOOK_URL == '' }}",
|
|
9
|
+
"if: ${{ env.DISCORD_WEBHOOK_URL != '' }}",
|
|
10
|
+
'curl -fsS',
|
|
11
|
+
'GITHUB_EVENT_PATH',
|
|
12
|
+
];
|
|
13
|
+
|
|
14
|
+
for (const snippet of requiredSnippets) {
|
|
15
|
+
if (!workflow.includes(snippet)) {
|
|
16
|
+
throw new Error(`Discord release workflow is missing required guard/publish snippet: ${snippet}`);
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
if (workflow.includes('SethCohen/github-releases-to-discord')) {
|
|
21
|
+
throw new Error('Discord release workflow should not depend on the deprecated external Discord action.');
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
console.log('discord release workflow smoke passed');
|