onbuzz 4.9.13 → 4.10.0
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/node_modules/glob/README.md +31 -5
- package/node_modules/glob/dist/commonjs/glob.d.ts +8 -0
- package/node_modules/glob/dist/commonjs/glob.d.ts.map +1 -1
- package/node_modules/glob/dist/commonjs/glob.js +2 -1
- package/node_modules/glob/dist/commonjs/glob.js.map +1 -1
- package/node_modules/glob/dist/commonjs/index.min.js +3 -3
- package/node_modules/glob/dist/commonjs/index.min.js.map +4 -4
- package/node_modules/glob/dist/commonjs/pattern.d.ts +3 -0
- package/node_modules/glob/dist/commonjs/pattern.d.ts.map +1 -1
- package/node_modules/glob/dist/commonjs/pattern.js +4 -0
- package/node_modules/glob/dist/commonjs/pattern.js.map +1 -1
- package/node_modules/glob/dist/esm/glob.d.ts +8 -0
- package/node_modules/glob/dist/esm/glob.d.ts.map +1 -1
- package/node_modules/glob/dist/esm/glob.js +2 -1
- package/node_modules/glob/dist/esm/glob.js.map +1 -1
- package/node_modules/glob/dist/esm/index.min.js +3 -3
- package/node_modules/glob/dist/esm/index.min.js.map +4 -4
- package/node_modules/glob/dist/esm/pattern.d.ts +3 -0
- package/node_modules/glob/dist/esm/pattern.d.ts.map +1 -1
- package/node_modules/glob/dist/esm/pattern.js +4 -0
- package/node_modules/glob/dist/esm/pattern.js.map +1 -1
- package/node_modules/{@isaacs → glob/node_modules}/balanced-match/README.md +7 -10
- package/node_modules/{@isaacs → glob/node_modules}/balanced-match/package.json +7 -18
- package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/README.md +3 -6
- package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/dist/commonjs/index.js +6 -4
- package/node_modules/glob/node_modules/brace-expansion/dist/commonjs/index.js.map +1 -0
- package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/dist/esm/index.js +6 -4
- package/node_modules/glob/node_modules/brace-expansion/dist/esm/index.js.map +1 -0
- package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/package.json +11 -7
- package/node_modules/glob/node_modules/minimatch/README.md +76 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/assert-valid-pattern.d.ts +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/assert-valid-pattern.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/assert-valid-pattern.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/ast.d.ts +4 -2
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/ast.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/ast.js +309 -55
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/ast.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/brace-expressions.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/brace-expressions.js +2 -4
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/brace-expressions.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/escape.d.ts +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/escape.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/escape.js +4 -4
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/escape.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/index.d.ts +81 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/index.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/index.js +232 -134
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/index.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/unescape.d.ts +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/unescape.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/unescape.js +8 -8
- package/node_modules/glob/node_modules/minimatch/dist/commonjs/unescape.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/assert-valid-pattern.d.ts +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/assert-valid-pattern.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/assert-valid-pattern.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/ast.d.ts +4 -2
- package/node_modules/glob/node_modules/minimatch/dist/esm/ast.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/ast.js +309 -55
- package/node_modules/glob/node_modules/minimatch/dist/esm/ast.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/brace-expressions.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/brace-expressions.js +2 -4
- package/node_modules/glob/node_modules/minimatch/dist/esm/brace-expressions.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/escape.d.ts +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/escape.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/escape.js +4 -4
- package/node_modules/glob/node_modules/minimatch/dist/esm/escape.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/index.d.ts +81 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/index.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/index.js +232 -134
- package/node_modules/glob/node_modules/minimatch/dist/esm/index.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/unescape.d.ts +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/unescape.d.ts.map +1 -1
- package/node_modules/glob/node_modules/minimatch/dist/esm/unescape.js +8 -8
- package/node_modules/glob/node_modules/minimatch/dist/esm/unescape.js.map +1 -1
- package/node_modules/glob/node_modules/minimatch/package.json +17 -11
- package/node_modules/glob/package.json +10 -13
- package/node_modules/minipass/LICENSE.md +55 -0
- package/node_modules/minipass/dist/commonjs/index.d.ts +12 -16
- package/node_modules/minipass/dist/commonjs/index.d.ts.map +1 -1
- package/node_modules/minipass/dist/commonjs/index.js +13 -3
- package/node_modules/minipass/dist/commonjs/index.js.map +1 -1
- package/node_modules/minipass/dist/esm/index.d.ts +12 -16
- package/node_modules/minipass/dist/esm/index.d.ts.map +1 -1
- package/node_modules/minipass/dist/esm/index.js +3 -1
- package/node_modules/minipass/dist/esm/index.js.map +1 -1
- package/node_modules/minipass/package.json +9 -14
- package/node_modules/path-scurry/node_modules/lru-cache/README.md +96 -10
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/diagnostics-channel-browser.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/diagnostics-channel-browser.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/diagnostics-channel.d.ts +5 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/diagnostics-channel.js +7 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/index.d.ts +1400 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/index.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/index.js +1726 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/index.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/index.min.js +2 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/index.min.js.map +7 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/perf.d.ts +12 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/perf.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/perf.js +10 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/browser/perf.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/diagnostics-channel-cjs.cjs.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/diagnostics-channel-cjs.d.cts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/diagnostics-channel.d.ts +5 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/diagnostics-channel.js +7 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/index.d.ts +109 -32
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/index.d.ts.map +1 -1
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/index.js +334 -197
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/index.js.map +1 -1
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/index.min.js +1 -1
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/index.min.js.map +4 -4
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/diagnostics-channel-node.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/diagnostics-channel-node.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/diagnostics-channel.d.ts +5 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/diagnostics-channel.js +9 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/index.d.ts +1400 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/index.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/index.js +1726 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/index.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/index.min.js +2 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/index.min.js.map +7 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/perf.d.ts +12 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/perf.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/perf.js +10 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/node/perf.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/perf.d.ts +12 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/perf.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/perf.js +10 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/commonjs/perf.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/diagnostics-channel-browser.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/diagnostics-channel-browser.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/diagnostics-channel.d.ts +5 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/diagnostics-channel.js +4 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/index.d.ts +1400 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/index.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/index.js +1722 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/index.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/index.min.js +2 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/index.min.js.map +7 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/perf.d.ts +12 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/perf.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/perf.js +7 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/browser/perf.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/diagnostics-channel-esm.d.mts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/diagnostics-channel-esm.mjs.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/diagnostics-channel.d.ts +5 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/diagnostics-channel.js +19 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/index.d.ts +109 -32
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/index.d.ts.map +1 -1
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/index.js +333 -196
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/index.js.map +1 -1
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/index.min.js +1 -1
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/index.min.js.map +4 -4
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/diagnostics-channel-node.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/diagnostics-channel-node.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/diagnostics-channel.d.ts +5 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/diagnostics-channel.js +6 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/index.d.ts +1400 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/index.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/index.js +1722 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/index.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/index.min.js +2 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/index.min.js.map +7 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/perf.d.ts +12 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/perf.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/perf.js +7 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/node/perf.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/perf.d.ts +12 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/perf.d.ts.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/perf.js +7 -0
- package/node_modules/path-scurry/node_modules/lru-cache/dist/esm/perf.js.map +1 -0
- package/node_modules/path-scurry/node_modules/lru-cache/package.json +71 -18
- package/node_modules/path-scurry/package.json +8 -24
- package/package.json +1 -1
- package/scripts/debug-balance-probe.mjs +35 -35
- package/scripts/push-image.sh +43 -43
- package/scripts/setup-acr.sh +65 -65
- package/scripts/verify-optional-deps.js +96 -1
- package/src/__tests__/composioCliFlags.test.js +239 -239
- package/src/analyzers/CSSAnalyzer.js +298 -297
- package/src/analyzers/ConfigValidator.js +691 -690
- package/src/analyzers/ESLintAnalyzer.js +320 -320
- package/src/analyzers/JavaScriptAnalyzer.js +260 -261
- package/src/analyzers/PrettierFormatter.js +246 -247
- package/src/analyzers/PythonAnalyzer.js +283 -283
- package/src/analyzers/SecurityAnalyzer.js +729 -729
- package/src/analyzers/SparrowAnalyzer.js +341 -341
- package/src/analyzers/TypeScriptAnalyzer.js +247 -247
- package/src/analyzers/__tests__/CSSAnalyzer.test.js +41 -41
- package/src/analyzers/__tests__/ConfigValidator.test.js +362 -362
- package/src/analyzers/__tests__/JavaScriptAnalyzer.test.js +40 -40
- package/src/analyzers/__tests__/PythonAnalyzer.test.js +205 -208
- package/src/analyzers/__tests__/SecurityAnalyzer.test.js +303 -303
- package/src/analyzers/__tests__/TypeScriptAnalyzer.test.js +187 -187
- package/src/analyzers/codeCloneDetector/analyzer.js +344 -344
- package/src/analyzers/codeCloneDetector/detector.js +250 -250
- package/src/analyzers/codeCloneDetector/index.js +194 -192
- package/src/analyzers/codeCloneDetector/parser.js +199 -199
- package/src/core/__tests__/agentPool.test.js +866 -866
- package/src/core/__tests__/agentPoolAutoResume.test.js +209 -209
- package/src/core/__tests__/agentPoolWakeOnMessage.test.js +315 -315
- package/src/core/__tests__/agentScheduler.emptyResponseChatStall.test.js +213 -213
- package/src/core/__tests__/agentScheduler.errorCategorisation.test.js +246 -246
- package/src/core/__tests__/agentScheduler.firstChunkTimeout.test.js +138 -138
- package/src/core/__tests__/agentScheduler.modeTransitions.test.js +233 -233
- package/src/core/__tests__/agentScheduler.nativePromptPick.test.js +319 -319
- package/src/core/__tests__/agentScheduler.taskLifecycleInstruction.test.js +78 -78
- package/src/core/__tests__/agentScheduler.visualizer.test.js +258 -258
- package/src/core/__tests__/flowCheckpointStore.test.js +140 -140
- package/src/core/__tests__/flowEndToEnd.test.js +565 -565
- package/src/core/__tests__/flowFieldMapping.test.js +188 -189
- package/src/core/__tests__/flowLintClientMirror.test.js +96 -98
- package/src/core/__tests__/flowSavePayload.test.js +170 -169
- package/src/core/__tests__/flowTemplates.test.js +311 -311
- package/src/core/__tests__/flowVersionStore.test.js +123 -123
- package/src/core/__tests__/messageProcessor.test.js +669 -669
- package/src/core/__tests__/stateManager.test.js +0 -1
- package/src/core/agentPool.js +2474 -2475
- package/src/core/agentScheduler.js +1 -4
- package/src/core/contextManager.js +708 -708
- package/src/core/flowExecutor.js +1510 -1510
- package/src/core/flowFieldMapping.js +136 -138
- package/src/core/messageProcessor.js +953 -954
- package/src/core/orchestrator.js +593 -595
- package/src/core/stateManager.js +1765 -1752
- package/src/index.js +1221 -1221
- package/src/interfaces/__tests__/archivedAgentDelete.test.js +207 -207
- package/src/interfaces/__tests__/bulkAgentRoute.test.js +361 -361
- package/src/interfaces/__tests__/imageServing.test.js +228 -228
- package/src/interfaces/__tests__/remoteSessionAuth.test.js +308 -308
- package/src/interfaces/__tests__/videoJobsRoutes.test.js +178 -179
- package/src/interfaces/__tests__/webServer.marketplace.test.js +629 -629
- package/src/interfaces/schedulerRoutes.js +50 -50
- package/src/interfaces/terminal/__tests__/smoke/connection.test.js +341 -350
- package/src/interfaces/terminal/__tests__/smoke/enhancements.test.js +156 -156
- package/src/interfaces/terminal/__tests__/smoke/imports.test.js +325 -330
- package/src/interfaces/terminal/__tests__/smoke/tools.test.js +385 -388
- package/src/interfaces/terminal/api/session.js +265 -266
- package/src/interfaces/terminal/api/websocket.js +496 -497
- package/src/interfaces/terminal/components/AgentCreator.js +691 -705
- package/src/interfaces/terminal/components/AgentEditor.js +676 -678
- package/src/interfaces/terminal/components/AgentSwitcher.js +331 -330
- package/src/interfaces/terminal/components/ErrorPanel.js +263 -264
- package/src/interfaces/terminal/components/Header.js +28 -28
- package/src/interfaces/terminal/components/Layout.js +598 -603
- package/src/interfaces/terminal/components/MessageList.js +280 -281
- package/src/interfaces/terminal/components/SettingsPanel.js +410 -415
- package/src/interfaces/terminal/components/StatusBar.js +2 -0
- package/src/interfaces/terminal/index.js +168 -168
- package/src/interfaces/terminal/state/useAgentControl.js +496 -496
- package/src/interfaces/terminal/state/useAgents.js +537 -537
- package/src/interfaces/terminal/state/useMessages.js +629 -630
- package/src/interfaces/terminal/state/useTools.js +554 -554
- package/src/interfaces/terminal/utils/debugLogger.js +44 -44
- package/src/interfaces/terminal/utils/settingsStorage.js +232 -232
- package/src/interfaces/webServer.js +7578 -7579
- package/src/interfaces/webServer.js.bak +7046 -7046
- package/src/modules/fileExplorer/__tests__/zipDownload.test.js +237 -237
- package/src/modules/fileExplorer/controller.js +470 -469
- package/src/modules/fileExplorer/routes.js +285 -286
- package/src/modules/widget/__tests__/isDisabled.test.js +41 -41
- package/src/modules/widget/__tests__/routes.test.js +677 -678
- package/src/modules/widget/__tests__/runtime.test.js +401 -401
- package/src/modules/widget/__tests__/versioning.test.js +309 -309
- package/src/modules/widget/__tests__/webComponentRuntime.test.js +565 -565
- package/src/modules/widget/__tests__/widgetTool.test.js +316 -316
- package/src/modules/widget/routes.js +435 -435
- package/src/modules/widget/runtime/bundle.js +640 -640
- package/src/modules/widget/runtime/webComponentBundle.js +470 -470
- package/src/modules/widget/schema.js +182 -181
- package/src/modules/widget/widgetTool.js +1389 -1389
- package/src/services/__tests__/agentActivityService.test.js +401 -402
- package/src/services/__tests__/benchmarkService.test.js +184 -184
- package/src/services/__tests__/contextInjectionService.test.js +246 -246
- package/src/services/__tests__/conversationQuery.test.js +721 -723
- package/src/services/__tests__/credentialVault.test.js +469 -469
- package/src/services/__tests__/discordService.integration.test.js +638 -639
- package/src/services/__tests__/flowContextService.test.js +590 -590
- package/src/services/__tests__/memoryService.test.js +1 -1
- package/src/services/__tests__/messageSource.test.js +380 -380
- package/src/services/__tests__/modelRouterNaming.test.js +111 -111
- package/src/services/__tests__/projectDetector.test.js +34 -34
- package/src/services/__tests__/promptService.test.js +242 -242
- package/src/services/__tests__/telegramService.test.js +941 -941
- package/src/services/__tests__/tokenCountingService.test.js +48 -48
- package/src/services/agentActivityService.js +419 -420
- package/src/services/aiService.js +2997 -3001
- package/src/services/apiKeyManager.js +359 -359
- package/src/services/benchmarkService.js +196 -196
- package/src/services/codebaseKnowledgeService.js +2 -2
- package/src/services/composioService.js +738 -738
- package/src/services/conversationCompactionService.js +1258 -1257
- package/src/services/credentialVault.js +685 -685
- package/src/services/discordService.js +792 -793
- package/src/services/embeddings/__tests__/azureCustomProvider.test.js +232 -232
- package/src/services/embeddings/__tests__/embeddingService.test.js +417 -417
- package/src/services/embeddings/__tests__/localProvider.test.js +263 -263
- package/src/services/embeddings/autoRecall.js +218 -219
- package/src/services/embeddings/indexers/__tests__/agentIndexer.test.js +232 -232
- package/src/services/embeddings/indexers/__tests__/memoryIndexer.test.js +418 -418
- package/src/services/embeddings/indexers/__tests__/reminisceIndexer.test.js +356 -357
- package/src/services/embeddings/indexers/__tests__/skillsIndexer.test.js +145 -145
- package/src/services/embeddings/indexers/__tests__/taskIndexer.test.js +146 -146
- package/src/services/embeddings/indexers/composioIndexer.js +279 -279
- package/src/services/embeddings/providerInterface.js +206 -206
- package/src/services/embeddings/providers/localProvider.js +11 -7
- package/src/services/embeddings/providers/openaiProvider.js +101 -101
- package/src/services/embeddings/vectorStore/inMemoryJsonStore.js +356 -356
- package/src/services/errorHandler.js +809 -809
- package/src/services/flowContextService.js +586 -586
- package/src/services/grounding/MockAdapter.js +125 -125
- package/src/services/modelRouterService.js +26 -31
- package/src/services/modelsService.js +322 -322
- package/src/services/ollamaService.js +452 -452
- package/src/services/projectDetector.js +403 -404
- package/src/services/promptService.js +418 -418
- package/src/services/qualityInspector.js +795 -795
- package/src/services/scheduleService.js +726 -726
- package/src/services/serviceRegistry.js +386 -386
- package/src/services/telegrafBot.js +174 -174
- package/src/services/telegramService.js +1972 -1972
- package/src/services/visualEditorBridge.js +1033 -1033
- package/src/services/visualEditorServer.js +1769 -1774
- package/src/services/whatsappService.js +667 -668
- package/src/tools/__tests__/agentCommunicationTool.findAgent.test.js +226 -226
- package/src/tools/__tests__/agentCommunicationTool.test.js +3 -3
- package/src/tools/__tests__/agentDelayTool.test.js +342 -342
- package/src/tools/__tests__/baseTool.test.js +3 -3
- package/src/tools/__tests__/codeMapTool.test.js +915 -915
- package/src/tools/__tests__/fileContentReplaceTool.test.js +309 -309
- package/src/tools/__tests__/fileTreeTool.test.js +274 -274
- package/src/tools/__tests__/filesystemTool.test.js +815 -815
- package/src/tools/__tests__/foundryWebSearchTool.test.js +252 -252
- package/src/tools/__tests__/imageTool.validator.test.js +194 -194
- package/src/tools/__tests__/jobDoneTool.test.js +580 -581
- package/src/tools/__tests__/memoryTool.forgetStale.test.js +272 -272
- package/src/tools/__tests__/memoryTool.reminisce.test.js +2 -2
- package/src/tools/__tests__/memoryTool.reminisceSemanticSearch.test.js +301 -301
- package/src/tools/__tests__/memoryTool.semanticSearch.test.js +405 -405
- package/src/tools/__tests__/memoryTool.teamPool.test.js +293 -293
- package/src/tools/__tests__/memoryTool.test.js +1 -1
- package/src/tools/__tests__/seekTool.test.js +282 -282
- package/src/tools/__tests__/skillsTool.search.test.js +164 -164
- package/src/tools/__tests__/skillsTool.test.js +226 -226
- package/src/tools/__tests__/staticAnalysisTool.test.js +509 -509
- package/src/tools/__tests__/taskManagerTool.discipline.test.js +137 -137
- package/src/tools/__tests__/taskManagerTool.search.test.js +143 -143
- package/src/tools/__tests__/taskManagerTool.test.js +866 -866
- package/src/tools/__tests__/terminalTool.test.js +448 -448
- package/src/tools/__tests__/toolShapeForgiveness.test.js +259 -260
- package/src/tools/__tests__/userPromptTool.test.js +297 -297
- package/src/tools/__tests__/videoTool.jobs.test.js +147 -147
- package/src/tools/__tests__/webTool.e2e.test.js +609 -603
- package/src/tools/__tests__/webTool.unit.test.js +195 -195
- package/src/tools/__tests__/webTool.visionModel.test.js +75 -75
- package/src/tools/agentCommunicationTool.js +8 -10
- package/src/tools/agentDelayTool.js +496 -497
- package/src/tools/asyncToolManager.js +602 -603
- package/src/tools/baseTool.js +12 -11
- package/src/tools/cloneDetectionTool.js +576 -581
- package/src/tools/codeMapTool.js +0 -6
- package/src/tools/composioTool.js +617 -617
- package/src/tools/dependencyResolverTool.js +1211 -1212
- package/src/tools/desktop/DesktopTool.js +629 -638
- package/src/tools/desktop/__tests__/DesktopTool.e2e.test.js +306 -306
- package/src/tools/desktop/__tests__/DesktopTool.test.js +507 -507
- package/src/tools/desktop/__tests__/osController.test.js +364 -364
- package/src/tools/desktop/osController.js +491 -491
- package/src/tools/docxTool.js +623 -623
- package/src/tools/excelTool.js +636 -636
- package/src/tools/fileContentReplaceTool.js +5 -7
- package/src/tools/fileSystemTool.js +12 -19
- package/src/tools/fileTreeTool.js +840 -840
- package/src/tools/foundryWebSearchTool.js +273 -273
- package/src/tools/helpTool.js +198 -198
- package/src/tools/imageTool.js +1397 -1397
- package/src/tools/importAnalyzerTool.js +1056 -1056
- package/src/tools/jobDoneTool.js +495 -495
- package/src/tools/memoryTool.js +1 -1
- package/src/tools/office/pres/__tests__/presSystem.test.js +365 -365
- package/src/tools/office/pres/archetypes/agenda.js +61 -61
- package/src/tools/office/pres/archetypes/bentoGrid.js +218 -219
- package/src/tools/office/pres/archetypes/bigStat.js +140 -142
- package/src/tools/office/pres/archetypes/closing.js +70 -70
- package/src/tools/office/pres/archetypes/hero.js +70 -70
- package/src/tools/office/pres/archetypes/productHero.js +93 -94
- package/src/tools/office/pres/archetypes/table.js +73 -74
- package/src/tools/office/pres/backgrounds/orb.js +66 -66
- package/src/tools/office/pres/components.js +422 -423
- package/src/tools/officeTool.js +441 -441
- package/src/tools/pdfTool.js +625 -627
- package/src/tools/platformControlTool.js +1081 -1081
- package/src/tools/seekTool.js +917 -918
- package/src/tools/skillsTool.js +1 -1
- package/src/tools/staticAnalysisTool.js +2143 -2146
- package/src/tools/taskManagerTool.js +3324 -3324
- package/src/tools/terminalTool.js +2615 -2618
- package/src/tools/videoTool.js +1303 -1303
- package/src/tools/visionTool.js +508 -508
- package/src/tools/visualEditorTool.js +1289 -1290
- package/src/tools/webTool.js +3368 -3368
- package/src/tools/whatsappTool.js +464 -464
- package/src/types/__tests__/agent.test.js +499 -499
- package/src/types/__tests__/contextReference.test.js +606 -606
- package/src/types/__tests__/conversation.test.js +555 -555
- package/src/types/__tests__/toolCommand.test.js +584 -584
- package/src/types/contextReference.js +974 -971
- package/src/types/conversation.js +729 -729
- package/src/types/toolCommand.js +746 -746
- package/src/utilities/__tests__/attachmentValidator.test.js +80 -80
- package/src/utilities/__tests__/auditReport.test.js +328 -328
- package/src/utilities/__tests__/directoryAccessManager.test.js +388 -388
- package/src/utilities/__tests__/jsonRepair.test.js +103 -104
- package/src/utilities/__tests__/modeTransitionReasons.test.js +105 -105
- package/src/utilities/__tests__/platformUtils.test.js +80 -87
- package/src/utilities/__tests__/structuredFileValidator.test.js +261 -263
- package/src/utilities/__tests__/toolConstants.test.js +92 -94
- package/src/utilities/__tests__/useIsTouchDevice.detect.test.js +114 -114
- package/src/utilities/__tests__/webUiUtilSync.test.js +117 -117
- package/src/utilities/attachmentValidator.js +284 -288
- package/src/utilities/authCache.js.backup-1779570472481 +121 -121
- package/src/utilities/browserStealth.js +631 -630
- package/src/utilities/configManager.js +616 -617
- package/src/utilities/directoryAccessManager.js +564 -565
- package/src/utilities/fileProcessor.js +308 -307
- package/src/utilities/humanBehavior.js +454 -453
- package/src/utilities/logger.js +479 -479
- package/src/utilities/structuredFileValidator.js +696 -699
- package/src/utilities/tagParser.js +5 -10
- package/src/utilities/userDataDir.js +308 -308
- package/node_modules/@isaacs/brace-expansion/dist/commonjs/index.js.map +0 -1
- package/node_modules/@isaacs/brace-expansion/dist/esm/index.js.map +0 -1
- package/node_modules/minipass/LICENSE +0 -15
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/LICENSE.md +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/commonjs/index.d.ts +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/commonjs/index.d.ts.map +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/commonjs/index.js +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/commonjs/index.js.map +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/commonjs/package.json +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/esm/index.d.ts +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/esm/index.d.ts.map +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/esm/index.js +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/esm/index.js.map +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/balanced-match/dist/esm/package.json +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/LICENSE +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/dist/commonjs/index.d.ts +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/dist/commonjs/index.d.ts.map +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/dist/commonjs/package.json +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/dist/esm/index.d.ts +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/dist/esm/index.d.ts.map +0 -0
- /package/node_modules/{@isaacs → glob/node_modules}/brace-expansion/dist/esm/package.json +0 -0
|
@@ -1,365 +1,365 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* presSystem.test.js — system-level tests for the themed presentation
|
|
3
|
-
* builder. Locks the invariants that make the design system reliable.
|
|
4
|
-
*
|
|
5
|
-
* Coverage:
|
|
6
|
-
* - Every archetype × every theme builds without throwing.
|
|
7
|
-
* - Theme resolver returns a complete shape no matter what the
|
|
8
|
-
* agent passes (empty spec, unknown theme, unknown palette).
|
|
9
|
-
* - Palette has all 8 semantic roles for every named palette.
|
|
10
|
-
* - Deck context tracks sections correctly across mixed slide types.
|
|
11
|
-
* - Background dispatch returns the right function for every named
|
|
12
|
-
* style (and falls back without throwing on unknown ones).
|
|
13
|
-
* - buildThemedPres produces a real .pptx that pres.read can parse.
|
|
14
|
-
*
|
|
15
|
-
* These tests are CHEAP — they build to a Buffer in memory and never
|
|
16
|
-
* touch disk except where a real round-trip matters. Total runtime
|
|
17
|
-
* should stay under 5s so this can run on every commit.
|
|
18
|
-
*/
|
|
19
|
-
import { describe, it, expect, beforeAll } from '@jest/globals';
|
|
20
|
-
import { tmpdir } from 'os';
|
|
21
|
-
import path from 'path';
|
|
22
|
-
import fs from 'fs/promises';
|
|
23
|
-
|
|
24
|
-
import { resolveTheme, listThemes } from '../themes/index.js';
|
|
25
|
-
import { listArchetypes, renderArchetype } from '../archetypes/index.js';
|
|
26
|
-
import { listBackgrounds, applyBackground } from '../backgrounds/index.js';
|
|
27
|
-
import { PALETTES, resolvePalette, CHART_SERIES } from '../palettes.js';
|
|
28
|
-
import { resolveSpacing } from '../grid.js';
|
|
29
|
-
import { resolveShape } from '../shapes.js';
|
|
30
|
-
import { buildDeckContext,
|
|
31
|
-
import { buildThemedPres, isThemedSpec, listSystem } from '../index.js';
|
|
32
|
-
import { readPres } from '../edit/read.js';
|
|
33
|
-
|
|
34
|
-
/* ─── slide spy — captures every pptxgenjs call without writing ──── */
|
|
35
|
-
function makeSlideSpy() {
|
|
36
|
-
const calls = [];
|
|
37
|
-
const slide = {
|
|
38
|
-
_calls: calls,
|
|
39
|
-
background: null,
|
|
40
|
-
addText: (...args) => { calls.push(['addText', args]); },
|
|
41
|
-
addShape: (...args) => { calls.push(['addShape', args]); },
|
|
42
|
-
addImage: (...args) => { calls.push(['addImage', args]); },
|
|
43
|
-
addTable: (...args) => { calls.push(['addTable', args]); },
|
|
44
|
-
addChart: (...args) => { calls.push(['addChart', args]); },
|
|
45
|
-
addNotes: (...args) => { calls.push(['addNotes', args]); },
|
|
46
|
-
};
|
|
47
|
-
return slide;
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
/* ─── minimal pptx stub for chart-dependent archetypes ───────────── */
|
|
51
|
-
const fakePptx = {
|
|
52
|
-
ChartType: { bar: 'bar', doughnut: 'doughnut', pie: 'pie', line: 'line', area: 'area' },
|
|
53
|
-
};
|
|
54
|
-
|
|
55
|
-
/* ─── one example slide spec per archetype ───────────────────────── */
|
|
56
|
-
const SLIDE_FIXTURES = {
|
|
57
|
-
hero: { title: 'Hello', subtitle: 'World', eyebrow: 'Q4', author: 'Loxia', date: '2026-05-29' },
|
|
58
|
-
agenda: { title: 'Agenda', items: ['One', 'Two', 'Three'] },
|
|
59
|
-
sectionHeader: { number: '01', title: 'Section A', subtitle: 'Where we begin' },
|
|
60
|
-
headlineMessage: { eyebrow: 'Tagline', message: 'A single sentence carries the whole slide.', footnote: 'Source: us' },
|
|
61
|
-
productHero: { eyebrow: 'New', title: 'Product Name', subtitle: 'A short tagline.', imagePosition: 'bottom' },
|
|
62
|
-
statCallout: { eyebrow: 'KPI', stat: '47%', label: 'YoY', footnote: 'Source' },
|
|
63
|
-
bigStat: { eyebrow: 'Headline', stat: '143', unit: '%', label: 'Net retention', delta: { value: '+8pp', direction: 'up' },
|
|
64
|
-
tags: [{ label: 'tag1', kind: 'highlight' }], footnote: 'Source' },
|
|
65
|
-
statTrio: { title: 'KPIs', stats: [
|
|
66
|
-
{ label: 'ARR', value: '$8.3', unit: 'M', delta: '+47% YoY', deltaDirection: 'up' },
|
|
67
|
-
{ label: 'NRR', value: '143', unit: '%', delta: '+8pp', deltaDirection: 'up' },
|
|
68
|
-
{ label: 'Burn', value: '14', unit: 'mo', delta: '-3mo', deltaDirection: 'down' },
|
|
69
|
-
] },
|
|
70
|
-
comparisonGrid: { title: 'A vs B', cells: [
|
|
71
|
-
{ eyebrow: 'A', headline: 'Old', body: '...' },
|
|
72
|
-
{ eyebrow: 'B', headline: 'New', body: '...' },
|
|
73
|
-
] },
|
|
74
|
-
bentoGrid: { title: 'Inside', tiles: [
|
|
75
|
-
{ colStart: 1, colSpan: 8, rowStart: 1, rowSpan: 2, kind: 'stat', value: '143', unit: '%', label: 'NRR' },
|
|
76
|
-
{ colStart: 9, colSpan: 4, rowStart: 1, rowSpan: 1, kind: 'stat', value: '$8.3', unit: 'M', label: 'ARR' },
|
|
77
|
-
{ colStart: 9, colSpan: 4, rowStart: 2, rowSpan: 1, kind: 'stat', value: '14', unit: 'mo', label: 'Burn' },
|
|
78
|
-
{ colStart: 1, colSpan: 12, rowStart: 3, rowSpan: 1, kind: 'feature', headline: 'A thing happened', body: '...' },
|
|
79
|
-
] },
|
|
80
|
-
bullets: { title: 'Wins', items: ['A', 'B', 'C'] },
|
|
81
|
-
twoColText: { title: 'Two cols', left: { headline: 'L', body: '...' }, right: { headline: 'R', body: '...' } },
|
|
82
|
-
twoColChart: { title: 'Chart', leftText: { headline: 'L', body: '...' },
|
|
83
|
-
chart: { type: 'donut', data: [{ label: 'A', value: 60 }, { label: 'B', value: 40 }] } },
|
|
84
|
-
quote: { quote: 'A sentence', attribution: 'Someone' },
|
|
85
|
-
imageFullBleed: { title: 'Title', eyebrow: 'Eyebrow' /* no image — fallback panel */ },
|
|
86
|
-
table: { title: 'Data', headers: ['A', 'B'], rows: [['1', '2'], ['3', '4']] },
|
|
87
|
-
closing: { eyebrow: 'Thanks', title: 'Bye', contact: [{ label: 'Email', value: 'x@y' }] },
|
|
88
|
-
freeform: { title: 'Freeform', items: [{ type: 'text', text: 'hi', x: 0, y: 0, w: 1, h: 0.1 }] },
|
|
89
|
-
};
|
|
90
|
-
|
|
91
|
-
const THEME_IDS = ['editorial-warm', 'tech-bold', 'data-heavy', 'mono-keynote'];
|
|
92
|
-
|
|
93
|
-
/* ─── tests ──────────────────────────────────────────────────────── */
|
|
94
|
-
describe('pres — every archetype × every theme renders without throwing', () => {
|
|
95
|
-
const archetypeIds = listArchetypes();
|
|
96
|
-
|
|
97
|
-
it('every archetype has a fixture in SLIDE_FIXTURES (catch missing test coverage)', () => {
|
|
98
|
-
for (const id of archetypeIds) {
|
|
99
|
-
expect(SLIDE_FIXTURES[id]).toBeDefined();
|
|
100
|
-
}
|
|
101
|
-
});
|
|
102
|
-
|
|
103
|
-
for (const themeId of THEME_IDS) {
|
|
104
|
-
describe(`theme=${themeId}`, () => {
|
|
105
|
-
const resolved = resolveTheme({ theme: themeId });
|
|
106
|
-
const ctx = {
|
|
107
|
-
slideIndex: 0, totalSlides: 1,
|
|
108
|
-
sectionName: 'TEST', sectionIdx: 0, sectionTotal: 1,
|
|
109
|
-
brand: 'TestBrand', isSectionStart: false,
|
|
110
|
-
};
|
|
111
|
-
|
|
112
|
-
for (const archetypeId of archetypeIds) {
|
|
113
|
-
it(`renders ${archetypeId}`, () => {
|
|
114
|
-
const slide = makeSlideSpy();
|
|
115
|
-
expect(() => {
|
|
116
|
-
renderArchetype(slide, resolved, { archetype: archetypeId, ...SLIDE_FIXTURES[archetypeId] }, fakePptx, ctx);
|
|
117
|
-
}).not.toThrow();
|
|
118
|
-
// Smoke: the archetype must add SOMETHING to the slide.
|
|
119
|
-
expect(slide._calls.length).toBeGreaterThan(0);
|
|
120
|
-
});
|
|
121
|
-
}
|
|
122
|
-
});
|
|
123
|
-
}
|
|
124
|
-
});
|
|
125
|
-
|
|
126
|
-
describe('pres/themes — resolver always returns a complete shape', () => {
|
|
127
|
-
it('returns valid resolved object for an empty spec', () => {
|
|
128
|
-
const r = resolveTheme({});
|
|
129
|
-
expect(r.theme.id).toBeTruthy();
|
|
130
|
-
expect(r.palette.bg).toMatch(/^[0-9A-F]{6}$/i);
|
|
131
|
-
expect(r.spacing.safe).toMatchObject({ x: expect.any(Number), y: expect.any(Number), w: expect.any(Number), h: expect.any(Number) });
|
|
132
|
-
expect(r.shape.cornerRadius).toBeDefined();
|
|
133
|
-
expect(typeof r.bg).toBe('string');
|
|
134
|
-
});
|
|
135
|
-
|
|
136
|
-
it('falls back to editorial-warm for unknown theme id', () => {
|
|
137
|
-
const r = resolveTheme({ theme: 'this-is-not-a-real-theme' });
|
|
138
|
-
expect(r.theme.id).toBe('editorial-warm');
|
|
139
|
-
});
|
|
140
|
-
|
|
141
|
-
it('falls back to family default for unknown palette override', () => {
|
|
142
|
-
const r = resolveTheme({ theme: 'tech-bold', palette: 'unknown-palette' });
|
|
143
|
-
expect(r.palette.name).toBe('electric-night'); // tech-bold's family default
|
|
144
|
-
});
|
|
145
|
-
|
|
146
|
-
it('listThemes returns the 4 registered themes', () => {
|
|
147
|
-
const names = listThemes().map(t => t.id).sort();
|
|
148
|
-
expect(names).toEqual(['data-heavy', 'editorial-warm', 'mono-keynote', 'tech-bold']);
|
|
149
|
-
});
|
|
150
|
-
|
|
151
|
-
it('mono-keynote resolves to the mono-light palette by default', () => {
|
|
152
|
-
const r = resolveTheme({ theme: 'mono-keynote' });
|
|
153
|
-
expect(r.theme.family).toBe('mono');
|
|
154
|
-
expect(r.palette.name).toBe('mono-light');
|
|
155
|
-
expect(r.palette.primary).toBe('0070F3'); // Geist blue
|
|
156
|
-
expect(r.bg).toBe('solid'); // mono is restrained
|
|
157
|
-
});
|
|
158
|
-
|
|
159
|
-
it('mono-keynote with mono-dark palette inverts surfaces correctly', () => {
|
|
160
|
-
const r = resolveTheme({ theme: 'mono-keynote', palette: 'mono-dark' });
|
|
161
|
-
expect(r.palette.bg).toBe('000000');
|
|
162
|
-
expect(r.palette.text).toBe('FAFAFA');
|
|
163
|
-
});
|
|
164
|
-
});
|
|
165
|
-
|
|
166
|
-
describe('pres/palettes — every palette has all 8 semantic roles', () => {
|
|
167
|
-
it.each(Object.keys(PALETTES))('palette "%s" has every role', (name) => {
|
|
168
|
-
const p = PALETTES[name];
|
|
169
|
-
for (const role of ['bg', 'surface', 'text', 'subtle', 'primary', 'accent', 'ink', 'highlight']) {
|
|
170
|
-
expect(p[role]).toMatch(/^[0-9A-F]{6}$/i);
|
|
171
|
-
}
|
|
172
|
-
});
|
|
173
|
-
|
|
174
|
-
it.each(Object.keys(PALETTES))('palette "%s" has a CHART_SERIES entry', (name) => {
|
|
175
|
-
expect(Array.isArray(CHART_SERIES[name])).toBe(true);
|
|
176
|
-
expect(CHART_SERIES[name].length).toBeGreaterThanOrEqual(3);
|
|
177
|
-
});
|
|
178
|
-
|
|
179
|
-
it('resolvePalette always returns a usable palette object', () => {
|
|
180
|
-
const r = resolvePalette('nonsense-name-xyz', 'editorial');
|
|
181
|
-
expect(r.name).toBe('cream-charcoal');
|
|
182
|
-
expect(r.bg).toMatch(/^[0-9A-F]{6}$/);
|
|
183
|
-
});
|
|
184
|
-
});
|
|
185
|
-
|
|
186
|
-
describe('pres/grid + shapes — token resolvers', () => {
|
|
187
|
-
it('resolveSpacing returns identical shape for unknown preset (defaults to cozy)', () => {
|
|
188
|
-
const a = resolveSpacing('nonsense');
|
|
189
|
-
const b = resolveSpacing('cozy');
|
|
190
|
-
expect(Object.keys(a).sort()).toEqual(Object.keys(b).sort());
|
|
191
|
-
expect(a.pageMargin).toBe(b.pageMargin);
|
|
192
|
-
});
|
|
193
|
-
|
|
194
|
-
it('resolveShape returns identical shape for unknown preset (defaults to soft)', () => {
|
|
195
|
-
const a = resolveShape('nonsense');
|
|
196
|
-
const b = resolveShape('soft');
|
|
197
|
-
expect(a.cornerRadius).toBe(b.cornerRadius);
|
|
198
|
-
});
|
|
199
|
-
|
|
200
|
-
it('resolveSpacing.col() produces non-overlapping frames within the safe area', () => {
|
|
201
|
-
const s = resolveSpacing('cozy');
|
|
202
|
-
const left = s.col(1, 6);
|
|
203
|
-
const right = s.col(7, 6);
|
|
204
|
-
expect(left.x + left.w).toBeLessThanOrEqual(right.x);
|
|
205
|
-
expect(right.x + right.w).toBeLessThanOrEqual(s.safe.x + s.safe.w + 0.001);
|
|
206
|
-
});
|
|
207
|
-
});
|
|
208
|
-
|
|
209
|
-
describe('pres/master — deck context tracks sections across slide list', () => {
|
|
210
|
-
it('assigns each non-section slide to the most recent sectionHeader', () => {
|
|
211
|
-
const { context } = buildDeckContext({
|
|
212
|
-
title: 'Deck',
|
|
213
|
-
slides: [
|
|
214
|
-
{ archetype: 'hero' },
|
|
215
|
-
{ archetype: 'sectionHeader', title: 'A' },
|
|
216
|
-
{ archetype: 'bullets' },
|
|
217
|
-
{ archetype: 'bullets' },
|
|
218
|
-
{ archetype: 'sectionHeader', title: 'B' },
|
|
219
|
-
{ archetype: 'closing' },
|
|
220
|
-
],
|
|
221
|
-
});
|
|
222
|
-
expect(context.map(c => c.sectionName)).toEqual([
|
|
223
|
-
null, // hero before any section
|
|
224
|
-
'A', // section A starts
|
|
225
|
-
'A', 'A', // inside section A
|
|
226
|
-
'B', // section B starts
|
|
227
|
-
'B', // closing inside section B
|
|
228
|
-
]);
|
|
229
|
-
expect(context.map(c => c.sectionIdx)).toEqual([0, 0, 0, 0, 1, 1]);
|
|
230
|
-
expect(context[0].totalSlides).toBe(6);
|
|
231
|
-
expect(context[0].sectionTotal).toBe(2);
|
|
232
|
-
});
|
|
233
|
-
|
|
234
|
-
it('isSectionStart is true ONLY on sectionHeader slides', () => {
|
|
235
|
-
const { context } = buildDeckContext({
|
|
236
|
-
slides: [
|
|
237
|
-
{ archetype: 'hero' },
|
|
238
|
-
{ archetype: 'sectionHeader' },
|
|
239
|
-
{ archetype: 'bullets' },
|
|
240
|
-
],
|
|
241
|
-
});
|
|
242
|
-
expect(context.map(c => c.isSectionStart)).toEqual([false, true, false]);
|
|
243
|
-
});
|
|
244
|
-
|
|
245
|
-
it('usesChrome returns false for bare archetypes', () => {
|
|
246
|
-
expect(usesChrome('hero')).toBe(false);
|
|
247
|
-
expect(usesChrome('sectionHeader')).toBe(false);
|
|
248
|
-
expect(usesChrome('closing')).toBe(false);
|
|
249
|
-
expect(usesChrome('imageFullBleed')).toBe(false);
|
|
250
|
-
expect(usesChrome('headlineMessage')).toBe(false);
|
|
251
|
-
expect(usesChrome('productHero')).toBe(false);
|
|
252
|
-
// Default = chrome for everything else.
|
|
253
|
-
expect(usesChrome('bullets')).toBe(true);
|
|
254
|
-
expect(usesChrome('bentoGrid')).toBe(true);
|
|
255
|
-
});
|
|
256
|
-
});
|
|
257
|
-
|
|
258
|
-
describe('pres/components — glass-card primitive', () => {
|
|
259
|
-
it('renders without throwing on every theme', async () => {
|
|
260
|
-
const { glassCard } = await import('../components.js');
|
|
261
|
-
for (const themeId of THEME_IDS) {
|
|
262
|
-
const resolved = resolveTheme({ theme: themeId });
|
|
263
|
-
const slide = makeSlideSpy();
|
|
264
|
-
expect(() => glassCard(slide, resolved, { x: 0.5, y: 0.5, w: 4, h: 2 })).not.toThrow();
|
|
265
|
-
expect(slide._calls.length).toBeGreaterThan(0);
|
|
266
|
-
}
|
|
267
|
-
});
|
|
268
|
-
|
|
269
|
-
it('paints two shapes when tinted (base + tint wash)', async () => {
|
|
270
|
-
const { glassCard } = await import('../components.js');
|
|
271
|
-
const resolved = resolveTheme({ theme: 'tech-bold' });
|
|
272
|
-
const slide = makeSlideSpy();
|
|
273
|
-
glassCard(slide, resolved, { x: 0.5, y: 0.5, w: 4, h: 2, tint: 'primary' });
|
|
274
|
-
const shapes = slide._calls.filter(c => c[0] === 'addShape');
|
|
275
|
-
expect(shapes.length).toBe(2);
|
|
276
|
-
});
|
|
277
|
-
|
|
278
|
-
it('paints only the base shape when not tinted', async () => {
|
|
279
|
-
const { glassCard } = await import('../components.js');
|
|
280
|
-
const resolved = resolveTheme({ theme: 'tech-bold' });
|
|
281
|
-
const slide = makeSlideSpy();
|
|
282
|
-
glassCard(slide, resolved, { x: 0.5, y: 0.5, w: 4, h: 2 });
|
|
283
|
-
const shapes = slide._calls.filter(c => c[0] === 'addShape');
|
|
284
|
-
expect(shapes.length).toBe(1);
|
|
285
|
-
});
|
|
286
|
-
});
|
|
287
|
-
|
|
288
|
-
describe('pres/backgrounds — dispatch is total', () => {
|
|
289
|
-
it('every named style resolves to a renderer', () => {
|
|
290
|
-
const styles = listBackgrounds();
|
|
291
|
-
expect(styles).toEqual(expect.arrayContaining(['solid', 'gradient', 'mesh', 'noise', 'geometric', 'grid', 'void', 'orb']));
|
|
292
|
-
});
|
|
293
|
-
|
|
294
|
-
it('every background renders against every theme without throwing', () => {
|
|
295
|
-
const styles = listBackgrounds();
|
|
296
|
-
for (const themeId of THEME_IDS) {
|
|
297
|
-
const resolved = resolveTheme({ theme: themeId });
|
|
298
|
-
for (const style of styles) {
|
|
299
|
-
const slide = makeSlideSpy();
|
|
300
|
-
expect(() => applyBackground(slide, resolved, style)).not.toThrow();
|
|
301
|
-
}
|
|
302
|
-
}
|
|
303
|
-
});
|
|
304
|
-
|
|
305
|
-
it('unknown background style falls back to solid (no throw)', () => {
|
|
306
|
-
const resolved = resolveTheme({ theme: 'tech-bold' });
|
|
307
|
-
const slide = makeSlideSpy();
|
|
308
|
-
expect(() => applyBackground(slide, resolved, 'this-doesnt-exist')).not.toThrow();
|
|
309
|
-
// Some shapes should have been added (or at minimum the bg colour set).
|
|
310
|
-
expect(slide.background || slide._calls.length > 0).toBeTruthy();
|
|
311
|
-
});
|
|
312
|
-
});
|
|
313
|
-
|
|
314
|
-
describe('pres — public API surface', () => {
|
|
315
|
-
it('isThemedSpec discriminates between new and legacy shape', () => {
|
|
316
|
-
expect(isThemedSpec({ theme: 'tech-bold' })).toBe(true);
|
|
317
|
-
expect(isThemedSpec({ theme: { primaryColor: '#000' } })).toBe(false);
|
|
318
|
-
expect(isThemedSpec({})).toBe(false);
|
|
319
|
-
expect(isThemedSpec(null)).toBe(false);
|
|
320
|
-
});
|
|
321
|
-
|
|
322
|
-
it('listSystem returns themes / archetypes / backgrounds discovery', () => {
|
|
323
|
-
const sys = listSystem();
|
|
324
|
-
expect(Array.isArray(sys.themes)).toBe(true);
|
|
325
|
-
expect(Array.isArray(sys.archetypes)).toBe(true);
|
|
326
|
-
expect(Array.isArray(sys.backgrounds)).toBe(true);
|
|
327
|
-
expect(sys.themes.length).toBe(4);
|
|
328
|
-
expect(sys.archetypes.length).toBeGreaterThanOrEqual(17);
|
|
329
|
-
expect(sys.backgrounds.length).toBeGreaterThanOrEqual(8);
|
|
330
|
-
});
|
|
331
|
-
});
|
|
332
|
-
|
|
333
|
-
describe('pres — end-to-end: build + read round-trip', () => {
|
|
334
|
-
let tmpFile;
|
|
335
|
-
beforeAll(async () => {
|
|
336
|
-
tmpFile = path.join(tmpdir(), `pres-test-${Date.now()}.pptx`);
|
|
337
|
-
await buildThemedPres({
|
|
338
|
-
title: 'Test deck',
|
|
339
|
-
theme: 'tech-bold',
|
|
340
|
-
slides: [
|
|
341
|
-
{ archetype: 'hero', title: 'Hello' },
|
|
342
|
-
{ archetype: 'sectionHeader', number: '01', title: 'Part one' },
|
|
343
|
-
{ archetype: 'bigStat', stat: '143', unit: '%', label: 'NRR' },
|
|
344
|
-
{ archetype: 'bullets', title: 'Wins', items: ['A', 'B', 'C'] },
|
|
345
|
-
{ archetype: 'closing', title: 'Bye' },
|
|
346
|
-
],
|
|
347
|
-
}, tmpFile);
|
|
348
|
-
});
|
|
349
|
-
|
|
350
|
-
it('produces a real .pptx file', async () => {
|
|
351
|
-
const stat = await fs.stat(tmpFile);
|
|
352
|
-
expect(stat.size).toBeGreaterThan(10_000);
|
|
353
|
-
});
|
|
354
|
-
|
|
355
|
-
it('pres.read can parse the deck and count slides', async () => {
|
|
356
|
-
const outline = await readPres(tmpFile);
|
|
357
|
-
expect(outline.slideCount).toBe(5);
|
|
358
|
-
});
|
|
359
|
-
|
|
360
|
-
it('round-trip preserves the deck — tokens scan finds zero unresolved tokens', async () => {
|
|
361
|
-
const outline = await readPres(tmpFile);
|
|
362
|
-
// No tokens were authored — the result should be an empty array.
|
|
363
|
-
expect(outline.tokens).toEqual([]);
|
|
364
|
-
});
|
|
365
|
-
});
|
|
1
|
+
/**
|
|
2
|
+
* presSystem.test.js — system-level tests for the themed presentation
|
|
3
|
+
* builder. Locks the invariants that make the design system reliable.
|
|
4
|
+
*
|
|
5
|
+
* Coverage:
|
|
6
|
+
* - Every archetype × every theme builds without throwing.
|
|
7
|
+
* - Theme resolver returns a complete shape no matter what the
|
|
8
|
+
* agent passes (empty spec, unknown theme, unknown palette).
|
|
9
|
+
* - Palette has all 8 semantic roles for every named palette.
|
|
10
|
+
* - Deck context tracks sections correctly across mixed slide types.
|
|
11
|
+
* - Background dispatch returns the right function for every named
|
|
12
|
+
* style (and falls back without throwing on unknown ones).
|
|
13
|
+
* - buildThemedPres produces a real .pptx that pres.read can parse.
|
|
14
|
+
*
|
|
15
|
+
* These tests are CHEAP — they build to a Buffer in memory and never
|
|
16
|
+
* touch disk except where a real round-trip matters. Total runtime
|
|
17
|
+
* should stay under 5s so this can run on every commit.
|
|
18
|
+
*/
|
|
19
|
+
import { describe, it, expect, beforeAll } from '@jest/globals';
|
|
20
|
+
import { tmpdir } from 'os';
|
|
21
|
+
import path from 'path';
|
|
22
|
+
import fs from 'fs/promises';
|
|
23
|
+
|
|
24
|
+
import { resolveTheme, listThemes } from '../themes/index.js';
|
|
25
|
+
import { listArchetypes, renderArchetype } from '../archetypes/index.js';
|
|
26
|
+
import { listBackgrounds, applyBackground } from '../backgrounds/index.js';
|
|
27
|
+
import { PALETTES, resolvePalette, CHART_SERIES } from '../palettes.js';
|
|
28
|
+
import { resolveSpacing } from '../grid.js';
|
|
29
|
+
import { resolveShape } from '../shapes.js';
|
|
30
|
+
import { buildDeckContext, usesChrome } from '../master.js';
|
|
31
|
+
import { buildThemedPres, isThemedSpec, listSystem } from '../index.js';
|
|
32
|
+
import { readPres } from '../edit/read.js';
|
|
33
|
+
|
|
34
|
+
/* ─── slide spy — captures every pptxgenjs call without writing ──── */
|
|
35
|
+
function makeSlideSpy() {
|
|
36
|
+
const calls = [];
|
|
37
|
+
const slide = {
|
|
38
|
+
_calls: calls,
|
|
39
|
+
background: null,
|
|
40
|
+
addText: (...args) => { calls.push(['addText', args]); },
|
|
41
|
+
addShape: (...args) => { calls.push(['addShape', args]); },
|
|
42
|
+
addImage: (...args) => { calls.push(['addImage', args]); },
|
|
43
|
+
addTable: (...args) => { calls.push(['addTable', args]); },
|
|
44
|
+
addChart: (...args) => { calls.push(['addChart', args]); },
|
|
45
|
+
addNotes: (...args) => { calls.push(['addNotes', args]); },
|
|
46
|
+
};
|
|
47
|
+
return slide;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
/* ─── minimal pptx stub for chart-dependent archetypes ───────────── */
|
|
51
|
+
const fakePptx = {
|
|
52
|
+
ChartType: { bar: 'bar', doughnut: 'doughnut', pie: 'pie', line: 'line', area: 'area' },
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
/* ─── one example slide spec per archetype ───────────────────────── */
|
|
56
|
+
const SLIDE_FIXTURES = {
|
|
57
|
+
hero: { title: 'Hello', subtitle: 'World', eyebrow: 'Q4', author: 'Loxia', date: '2026-05-29' },
|
|
58
|
+
agenda: { title: 'Agenda', items: ['One', 'Two', 'Three'] },
|
|
59
|
+
sectionHeader: { number: '01', title: 'Section A', subtitle: 'Where we begin' },
|
|
60
|
+
headlineMessage: { eyebrow: 'Tagline', message: 'A single sentence carries the whole slide.', footnote: 'Source: us' },
|
|
61
|
+
productHero: { eyebrow: 'New', title: 'Product Name', subtitle: 'A short tagline.', imagePosition: 'bottom' },
|
|
62
|
+
statCallout: { eyebrow: 'KPI', stat: '47%', label: 'YoY', footnote: 'Source' },
|
|
63
|
+
bigStat: { eyebrow: 'Headline', stat: '143', unit: '%', label: 'Net retention', delta: { value: '+8pp', direction: 'up' },
|
|
64
|
+
tags: [{ label: 'tag1', kind: 'highlight' }], footnote: 'Source' },
|
|
65
|
+
statTrio: { title: 'KPIs', stats: [
|
|
66
|
+
{ label: 'ARR', value: '$8.3', unit: 'M', delta: '+47% YoY', deltaDirection: 'up' },
|
|
67
|
+
{ label: 'NRR', value: '143', unit: '%', delta: '+8pp', deltaDirection: 'up' },
|
|
68
|
+
{ label: 'Burn', value: '14', unit: 'mo', delta: '-3mo', deltaDirection: 'down' },
|
|
69
|
+
] },
|
|
70
|
+
comparisonGrid: { title: 'A vs B', cells: [
|
|
71
|
+
{ eyebrow: 'A', headline: 'Old', body: '...' },
|
|
72
|
+
{ eyebrow: 'B', headline: 'New', body: '...' },
|
|
73
|
+
] },
|
|
74
|
+
bentoGrid: { title: 'Inside', tiles: [
|
|
75
|
+
{ colStart: 1, colSpan: 8, rowStart: 1, rowSpan: 2, kind: 'stat', value: '143', unit: '%', label: 'NRR' },
|
|
76
|
+
{ colStart: 9, colSpan: 4, rowStart: 1, rowSpan: 1, kind: 'stat', value: '$8.3', unit: 'M', label: 'ARR' },
|
|
77
|
+
{ colStart: 9, colSpan: 4, rowStart: 2, rowSpan: 1, kind: 'stat', value: '14', unit: 'mo', label: 'Burn' },
|
|
78
|
+
{ colStart: 1, colSpan: 12, rowStart: 3, rowSpan: 1, kind: 'feature', headline: 'A thing happened', body: '...' },
|
|
79
|
+
] },
|
|
80
|
+
bullets: { title: 'Wins', items: ['A', 'B', 'C'] },
|
|
81
|
+
twoColText: { title: 'Two cols', left: { headline: 'L', body: '...' }, right: { headline: 'R', body: '...' } },
|
|
82
|
+
twoColChart: { title: 'Chart', leftText: { headline: 'L', body: '...' },
|
|
83
|
+
chart: { type: 'donut', data: [{ label: 'A', value: 60 }, { label: 'B', value: 40 }] } },
|
|
84
|
+
quote: { quote: 'A sentence', attribution: 'Someone' },
|
|
85
|
+
imageFullBleed: { title: 'Title', eyebrow: 'Eyebrow' /* no image — fallback panel */ },
|
|
86
|
+
table: { title: 'Data', headers: ['A', 'B'], rows: [['1', '2'], ['3', '4']] },
|
|
87
|
+
closing: { eyebrow: 'Thanks', title: 'Bye', contact: [{ label: 'Email', value: 'x@y' }] },
|
|
88
|
+
freeform: { title: 'Freeform', items: [{ type: 'text', text: 'hi', x: 0, y: 0, w: 1, h: 0.1 }] },
|
|
89
|
+
};
|
|
90
|
+
|
|
91
|
+
const THEME_IDS = ['editorial-warm', 'tech-bold', 'data-heavy', 'mono-keynote'];
|
|
92
|
+
|
|
93
|
+
/* ─── tests ──────────────────────────────────────────────────────── */
|
|
94
|
+
describe('pres — every archetype × every theme renders without throwing', () => {
|
|
95
|
+
const archetypeIds = listArchetypes();
|
|
96
|
+
|
|
97
|
+
it('every archetype has a fixture in SLIDE_FIXTURES (catch missing test coverage)', () => {
|
|
98
|
+
for (const id of archetypeIds) {
|
|
99
|
+
expect(SLIDE_FIXTURES[id]).toBeDefined();
|
|
100
|
+
}
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
for (const themeId of THEME_IDS) {
|
|
104
|
+
describe(`theme=${themeId}`, () => {
|
|
105
|
+
const resolved = resolveTheme({ theme: themeId });
|
|
106
|
+
const ctx = {
|
|
107
|
+
slideIndex: 0, totalSlides: 1,
|
|
108
|
+
sectionName: 'TEST', sectionIdx: 0, sectionTotal: 1,
|
|
109
|
+
brand: 'TestBrand', isSectionStart: false,
|
|
110
|
+
};
|
|
111
|
+
|
|
112
|
+
for (const archetypeId of archetypeIds) {
|
|
113
|
+
it(`renders ${archetypeId}`, () => {
|
|
114
|
+
const slide = makeSlideSpy();
|
|
115
|
+
expect(() => {
|
|
116
|
+
renderArchetype(slide, resolved, { archetype: archetypeId, ...SLIDE_FIXTURES[archetypeId] }, fakePptx, ctx);
|
|
117
|
+
}).not.toThrow();
|
|
118
|
+
// Smoke: the archetype must add SOMETHING to the slide.
|
|
119
|
+
expect(slide._calls.length).toBeGreaterThan(0);
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
});
|
|
123
|
+
}
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
describe('pres/themes — resolver always returns a complete shape', () => {
|
|
127
|
+
it('returns valid resolved object for an empty spec', () => {
|
|
128
|
+
const r = resolveTheme({});
|
|
129
|
+
expect(r.theme.id).toBeTruthy();
|
|
130
|
+
expect(r.palette.bg).toMatch(/^[0-9A-F]{6}$/i);
|
|
131
|
+
expect(r.spacing.safe).toMatchObject({ x: expect.any(Number), y: expect.any(Number), w: expect.any(Number), h: expect.any(Number) });
|
|
132
|
+
expect(r.shape.cornerRadius).toBeDefined();
|
|
133
|
+
expect(typeof r.bg).toBe('string');
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
it('falls back to editorial-warm for unknown theme id', () => {
|
|
137
|
+
const r = resolveTheme({ theme: 'this-is-not-a-real-theme' });
|
|
138
|
+
expect(r.theme.id).toBe('editorial-warm');
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
it('falls back to family default for unknown palette override', () => {
|
|
142
|
+
const r = resolveTheme({ theme: 'tech-bold', palette: 'unknown-palette' });
|
|
143
|
+
expect(r.palette.name).toBe('electric-night'); // tech-bold's family default
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
it('listThemes returns the 4 registered themes', () => {
|
|
147
|
+
const names = listThemes().map(t => t.id).sort();
|
|
148
|
+
expect(names).toEqual(['data-heavy', 'editorial-warm', 'mono-keynote', 'tech-bold']);
|
|
149
|
+
});
|
|
150
|
+
|
|
151
|
+
it('mono-keynote resolves to the mono-light palette by default', () => {
|
|
152
|
+
const r = resolveTheme({ theme: 'mono-keynote' });
|
|
153
|
+
expect(r.theme.family).toBe('mono');
|
|
154
|
+
expect(r.palette.name).toBe('mono-light');
|
|
155
|
+
expect(r.palette.primary).toBe('0070F3'); // Geist blue
|
|
156
|
+
expect(r.bg).toBe('solid'); // mono is restrained
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
it('mono-keynote with mono-dark palette inverts surfaces correctly', () => {
|
|
160
|
+
const r = resolveTheme({ theme: 'mono-keynote', palette: 'mono-dark' });
|
|
161
|
+
expect(r.palette.bg).toBe('000000');
|
|
162
|
+
expect(r.palette.text).toBe('FAFAFA');
|
|
163
|
+
});
|
|
164
|
+
});
|
|
165
|
+
|
|
166
|
+
describe('pres/palettes — every palette has all 8 semantic roles', () => {
|
|
167
|
+
it.each(Object.keys(PALETTES))('palette "%s" has every role', (name) => {
|
|
168
|
+
const p = PALETTES[name];
|
|
169
|
+
for (const role of ['bg', 'surface', 'text', 'subtle', 'primary', 'accent', 'ink', 'highlight']) {
|
|
170
|
+
expect(p[role]).toMatch(/^[0-9A-F]{6}$/i);
|
|
171
|
+
}
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
it.each(Object.keys(PALETTES))('palette "%s" has a CHART_SERIES entry', (name) => {
|
|
175
|
+
expect(Array.isArray(CHART_SERIES[name])).toBe(true);
|
|
176
|
+
expect(CHART_SERIES[name].length).toBeGreaterThanOrEqual(3);
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
it('resolvePalette always returns a usable palette object', () => {
|
|
180
|
+
const r = resolvePalette('nonsense-name-xyz', 'editorial');
|
|
181
|
+
expect(r.name).toBe('cream-charcoal');
|
|
182
|
+
expect(r.bg).toMatch(/^[0-9A-F]{6}$/);
|
|
183
|
+
});
|
|
184
|
+
});
|
|
185
|
+
|
|
186
|
+
describe('pres/grid + shapes — token resolvers', () => {
|
|
187
|
+
it('resolveSpacing returns identical shape for unknown preset (defaults to cozy)', () => {
|
|
188
|
+
const a = resolveSpacing('nonsense');
|
|
189
|
+
const b = resolveSpacing('cozy');
|
|
190
|
+
expect(Object.keys(a).sort()).toEqual(Object.keys(b).sort());
|
|
191
|
+
expect(a.pageMargin).toBe(b.pageMargin);
|
|
192
|
+
});
|
|
193
|
+
|
|
194
|
+
it('resolveShape returns identical shape for unknown preset (defaults to soft)', () => {
|
|
195
|
+
const a = resolveShape('nonsense');
|
|
196
|
+
const b = resolveShape('soft');
|
|
197
|
+
expect(a.cornerRadius).toBe(b.cornerRadius);
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
it('resolveSpacing.col() produces non-overlapping frames within the safe area', () => {
|
|
201
|
+
const s = resolveSpacing('cozy');
|
|
202
|
+
const left = s.col(1, 6);
|
|
203
|
+
const right = s.col(7, 6);
|
|
204
|
+
expect(left.x + left.w).toBeLessThanOrEqual(right.x);
|
|
205
|
+
expect(right.x + right.w).toBeLessThanOrEqual(s.safe.x + s.safe.w + 0.001);
|
|
206
|
+
});
|
|
207
|
+
});
|
|
208
|
+
|
|
209
|
+
describe('pres/master — deck context tracks sections across slide list', () => {
|
|
210
|
+
it('assigns each non-section slide to the most recent sectionHeader', () => {
|
|
211
|
+
const { context } = buildDeckContext({
|
|
212
|
+
title: 'Deck',
|
|
213
|
+
slides: [
|
|
214
|
+
{ archetype: 'hero' },
|
|
215
|
+
{ archetype: 'sectionHeader', title: 'A' },
|
|
216
|
+
{ archetype: 'bullets' },
|
|
217
|
+
{ archetype: 'bullets' },
|
|
218
|
+
{ archetype: 'sectionHeader', title: 'B' },
|
|
219
|
+
{ archetype: 'closing' },
|
|
220
|
+
],
|
|
221
|
+
});
|
|
222
|
+
expect(context.map(c => c.sectionName)).toEqual([
|
|
223
|
+
null, // hero before any section
|
|
224
|
+
'A', // section A starts
|
|
225
|
+
'A', 'A', // inside section A
|
|
226
|
+
'B', // section B starts
|
|
227
|
+
'B', // closing inside section B
|
|
228
|
+
]);
|
|
229
|
+
expect(context.map(c => c.sectionIdx)).toEqual([0, 0, 0, 0, 1, 1]);
|
|
230
|
+
expect(context[0].totalSlides).toBe(6);
|
|
231
|
+
expect(context[0].sectionTotal).toBe(2);
|
|
232
|
+
});
|
|
233
|
+
|
|
234
|
+
it('isSectionStart is true ONLY on sectionHeader slides', () => {
|
|
235
|
+
const { context } = buildDeckContext({
|
|
236
|
+
slides: [
|
|
237
|
+
{ archetype: 'hero' },
|
|
238
|
+
{ archetype: 'sectionHeader' },
|
|
239
|
+
{ archetype: 'bullets' },
|
|
240
|
+
],
|
|
241
|
+
});
|
|
242
|
+
expect(context.map(c => c.isSectionStart)).toEqual([false, true, false]);
|
|
243
|
+
});
|
|
244
|
+
|
|
245
|
+
it('usesChrome returns false for bare archetypes', () => {
|
|
246
|
+
expect(usesChrome('hero')).toBe(false);
|
|
247
|
+
expect(usesChrome('sectionHeader')).toBe(false);
|
|
248
|
+
expect(usesChrome('closing')).toBe(false);
|
|
249
|
+
expect(usesChrome('imageFullBleed')).toBe(false);
|
|
250
|
+
expect(usesChrome('headlineMessage')).toBe(false);
|
|
251
|
+
expect(usesChrome('productHero')).toBe(false);
|
|
252
|
+
// Default = chrome for everything else.
|
|
253
|
+
expect(usesChrome('bullets')).toBe(true);
|
|
254
|
+
expect(usesChrome('bentoGrid')).toBe(true);
|
|
255
|
+
});
|
|
256
|
+
});
|
|
257
|
+
|
|
258
|
+
describe('pres/components — glass-card primitive', () => {
|
|
259
|
+
it('renders without throwing on every theme', async () => {
|
|
260
|
+
const { glassCard } = await import('../components.js');
|
|
261
|
+
for (const themeId of THEME_IDS) {
|
|
262
|
+
const resolved = resolveTheme({ theme: themeId });
|
|
263
|
+
const slide = makeSlideSpy();
|
|
264
|
+
expect(() => glassCard(slide, resolved, { x: 0.5, y: 0.5, w: 4, h: 2 })).not.toThrow();
|
|
265
|
+
expect(slide._calls.length).toBeGreaterThan(0);
|
|
266
|
+
}
|
|
267
|
+
});
|
|
268
|
+
|
|
269
|
+
it('paints two shapes when tinted (base + tint wash)', async () => {
|
|
270
|
+
const { glassCard } = await import('../components.js');
|
|
271
|
+
const resolved = resolveTheme({ theme: 'tech-bold' });
|
|
272
|
+
const slide = makeSlideSpy();
|
|
273
|
+
glassCard(slide, resolved, { x: 0.5, y: 0.5, w: 4, h: 2, tint: 'primary' });
|
|
274
|
+
const shapes = slide._calls.filter(c => c[0] === 'addShape');
|
|
275
|
+
expect(shapes.length).toBe(2);
|
|
276
|
+
});
|
|
277
|
+
|
|
278
|
+
it('paints only the base shape when not tinted', async () => {
|
|
279
|
+
const { glassCard } = await import('../components.js');
|
|
280
|
+
const resolved = resolveTheme({ theme: 'tech-bold' });
|
|
281
|
+
const slide = makeSlideSpy();
|
|
282
|
+
glassCard(slide, resolved, { x: 0.5, y: 0.5, w: 4, h: 2 });
|
|
283
|
+
const shapes = slide._calls.filter(c => c[0] === 'addShape');
|
|
284
|
+
expect(shapes.length).toBe(1);
|
|
285
|
+
});
|
|
286
|
+
});
|
|
287
|
+
|
|
288
|
+
describe('pres/backgrounds — dispatch is total', () => {
|
|
289
|
+
it('every named style resolves to a renderer', () => {
|
|
290
|
+
const styles = listBackgrounds();
|
|
291
|
+
expect(styles).toEqual(expect.arrayContaining(['solid', 'gradient', 'mesh', 'noise', 'geometric', 'grid', 'void', 'orb']));
|
|
292
|
+
});
|
|
293
|
+
|
|
294
|
+
it('every background renders against every theme without throwing', () => {
|
|
295
|
+
const styles = listBackgrounds();
|
|
296
|
+
for (const themeId of THEME_IDS) {
|
|
297
|
+
const resolved = resolveTheme({ theme: themeId });
|
|
298
|
+
for (const style of styles) {
|
|
299
|
+
const slide = makeSlideSpy();
|
|
300
|
+
expect(() => applyBackground(slide, resolved, style)).not.toThrow();
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
});
|
|
304
|
+
|
|
305
|
+
it('unknown background style falls back to solid (no throw)', () => {
|
|
306
|
+
const resolved = resolveTheme({ theme: 'tech-bold' });
|
|
307
|
+
const slide = makeSlideSpy();
|
|
308
|
+
expect(() => applyBackground(slide, resolved, 'this-doesnt-exist')).not.toThrow();
|
|
309
|
+
// Some shapes should have been added (or at minimum the bg colour set).
|
|
310
|
+
expect(slide.background || slide._calls.length > 0).toBeTruthy();
|
|
311
|
+
});
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
describe('pres — public API surface', () => {
|
|
315
|
+
it('isThemedSpec discriminates between new and legacy shape', () => {
|
|
316
|
+
expect(isThemedSpec({ theme: 'tech-bold' })).toBe(true);
|
|
317
|
+
expect(isThemedSpec({ theme: { primaryColor: '#000' } })).toBe(false);
|
|
318
|
+
expect(isThemedSpec({})).toBe(false);
|
|
319
|
+
expect(isThemedSpec(null)).toBe(false);
|
|
320
|
+
});
|
|
321
|
+
|
|
322
|
+
it('listSystem returns themes / archetypes / backgrounds discovery', () => {
|
|
323
|
+
const sys = listSystem();
|
|
324
|
+
expect(Array.isArray(sys.themes)).toBe(true);
|
|
325
|
+
expect(Array.isArray(sys.archetypes)).toBe(true);
|
|
326
|
+
expect(Array.isArray(sys.backgrounds)).toBe(true);
|
|
327
|
+
expect(sys.themes.length).toBe(4);
|
|
328
|
+
expect(sys.archetypes.length).toBeGreaterThanOrEqual(17);
|
|
329
|
+
expect(sys.backgrounds.length).toBeGreaterThanOrEqual(8);
|
|
330
|
+
});
|
|
331
|
+
});
|
|
332
|
+
|
|
333
|
+
describe('pres — end-to-end: build + read round-trip', () => {
|
|
334
|
+
let tmpFile;
|
|
335
|
+
beforeAll(async () => {
|
|
336
|
+
tmpFile = path.join(tmpdir(), `pres-test-${Date.now()}.pptx`);
|
|
337
|
+
await buildThemedPres({
|
|
338
|
+
title: 'Test deck',
|
|
339
|
+
theme: 'tech-bold',
|
|
340
|
+
slides: [
|
|
341
|
+
{ archetype: 'hero', title: 'Hello' },
|
|
342
|
+
{ archetype: 'sectionHeader', number: '01', title: 'Part one' },
|
|
343
|
+
{ archetype: 'bigStat', stat: '143', unit: '%', label: 'NRR' },
|
|
344
|
+
{ archetype: 'bullets', title: 'Wins', items: ['A', 'B', 'C'] },
|
|
345
|
+
{ archetype: 'closing', title: 'Bye' },
|
|
346
|
+
],
|
|
347
|
+
}, tmpFile);
|
|
348
|
+
});
|
|
349
|
+
|
|
350
|
+
it('produces a real .pptx file', async () => {
|
|
351
|
+
const stat = await fs.stat(tmpFile);
|
|
352
|
+
expect(stat.size).toBeGreaterThan(10_000);
|
|
353
|
+
});
|
|
354
|
+
|
|
355
|
+
it('pres.read can parse the deck and count slides', async () => {
|
|
356
|
+
const outline = await readPres(tmpFile);
|
|
357
|
+
expect(outline.slideCount).toBe(5);
|
|
358
|
+
});
|
|
359
|
+
|
|
360
|
+
it('round-trip preserves the deck — tokens scan finds zero unresolved tokens', async () => {
|
|
361
|
+
const outline = await readPres(tmpFile);
|
|
362
|
+
// No tokens were authored — the result should be an empty array.
|
|
363
|
+
expect(outline.tokens).toEqual([]);
|
|
364
|
+
});
|
|
365
|
+
});
|