@vellumai/assistant 0.7.2 → 0.7.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/ARCHITECTURE.md +16 -1
- package/docs/architecture/memory.md +5 -2
- package/node_modules/@vellumai/gateway-client/src/ipc-client.ts +13 -4
- package/node_modules/@vellumai/skill-host-contracts/src/assistant-event.ts +0 -9
- package/node_modules/@vellumai/slack-text/src/index.test.ts +18 -35
- package/node_modules/@vellumai/slack-text/src/index.ts +2 -48
- package/openapi.yaml +449 -22
- package/package.json +1 -1
- package/src/__tests__/app-control-flow.test.ts +21 -11
- package/src/__tests__/assistant-event-hub.test.ts +48 -0
- package/src/__tests__/assistant-event.test.ts +0 -10
- package/src/__tests__/assistant-events-sse-hardening.test.ts +2 -7
- package/src/__tests__/assistant-feature-flags-integration.test.ts +18 -0
- package/src/__tests__/auto-analysis-end-to-end.test.ts +62 -1
- package/src/__tests__/background-workers-disk-pressure.test.ts +268 -0
- package/src/__tests__/call-conversation-messages.test.ts +8 -2
- package/src/__tests__/channel-inbound-disk-pressure.test.ts +537 -0
- package/src/__tests__/channel-readiness-service.test.ts +4 -2
- package/src/__tests__/config-loader-backfill.test.ts +379 -0
- package/src/__tests__/config-schema.test.ts +1 -0
- package/src/__tests__/config-watcher-cleanup-throttle.test.ts +18 -9
- package/src/__tests__/config-watcher.test.ts +140 -69
- package/src/__tests__/context-search-agent-runner.test.ts +61 -3
- package/src/__tests__/context-search-conversations-source.test.ts +0 -24
- package/src/__tests__/context-search-fanout.test.ts +0 -1
- package/src/__tests__/context-search-memory-source.test.ts +3 -7
- package/src/__tests__/context-search-memory-v2-source.test.ts +0 -2
- package/src/__tests__/context-search-pkb-source.test.ts +0 -1
- package/src/__tests__/context-search-workspace-source.test.ts +0 -1
- package/src/__tests__/conversation-abort-tool-results.test.ts +6 -0
- package/src/__tests__/conversation-agent-loop-disk-pressure.test.ts +223 -0
- package/src/__tests__/conversation-agent-loop.test.ts +454 -5
- package/src/__tests__/conversation-error.test.ts +150 -3
- package/src/__tests__/conversation-process-callsite.test.ts +43 -0
- package/src/__tests__/conversation-provider-retry-repair.test.ts +6 -0
- package/src/__tests__/conversation-runtime-assembly.test.ts +65 -0
- package/src/__tests__/conversation-slash-unknown.test.ts +6 -0
- package/src/__tests__/conversation-speed-override.test.ts +0 -3
- package/src/__tests__/conversation-store.test.ts +0 -18
- package/src/__tests__/conversation-surfaces-app-control.test.ts +15 -4
- package/src/__tests__/conversation-surfaces-data-persist.test.ts +404 -0
- package/src/__tests__/conversation-tool-setup-app-refresh.test.ts +2 -5
- package/src/__tests__/conversation-workspace-injection.test.ts +6 -0
- package/src/__tests__/conversation-workspace-tool-tracking.test.ts +6 -0
- package/src/__tests__/credentials-cli.test.ts +7 -0
- package/src/__tests__/cu-unified-flow.test.ts +176 -10
- package/src/__tests__/date-context.test.ts +164 -2
- package/src/__tests__/disk-pressure-guard.test.ts +262 -0
- package/src/__tests__/disk-pressure-lifecycle.test.ts +168 -0
- package/src/__tests__/disk-pressure-policy.test.ts +241 -0
- package/src/__tests__/disk-pressure-routes.test.ts +379 -0
- package/src/__tests__/disk-pressure-tools.test.ts +277 -0
- package/src/__tests__/disk-usage.test.ts +150 -0
- package/src/__tests__/events-client-registration.test.ts +52 -0
- package/src/__tests__/events-dev-bypass-actor.test.ts +162 -0
- package/src/__tests__/file-write-tool.test.ts +4 -10
- package/src/__tests__/filing-service.test.ts +3 -4
- package/src/__tests__/heartbeat-disk-pressure.test.ts +183 -0
- package/src/__tests__/heartbeat-service.test.ts +260 -11
- package/src/__tests__/host-app-control-proxy.test.ts +195 -25
- package/src/__tests__/host-bash-proxy.test.ts +227 -34
- package/src/__tests__/host-bash-routes.test.ts +178 -13
- package/src/__tests__/host-cu-proxy.test.ts +210 -3
- package/src/__tests__/host-cu-routes-targeted.test.ts +141 -12
- package/src/__tests__/host-file-proxy-targeted.test.ts +48 -9
- package/src/__tests__/host-file-proxy.test.ts +268 -6
- package/src/__tests__/host-file-routes-targeted.test.ts +175 -17
- package/src/__tests__/host-transfer-proxy-targeted.test.ts +408 -59
- package/src/__tests__/host-transfer-routes-targeted.test.ts +232 -17
- package/src/__tests__/http-user-message-parity.test.ts +107 -1
- package/src/__tests__/injector-chain.test.ts +18 -6
- package/src/__tests__/injector-disk-pressure.test.ts +224 -0
- package/src/__tests__/managed-profile-guard.test.ts +18 -0
- package/src/__tests__/mcp-abort-signal.test.ts +130 -0
- package/src/__tests__/memory-admin-recall.test.ts +3 -11
- package/src/__tests__/memory-retrieval-pipeline.test.ts +22 -1
- package/src/__tests__/normalize-onboarding.test.ts +180 -0
- package/src/__tests__/oauth-connect-routes.test.ts +316 -0
- package/src/__tests__/oauth-provider-seed-logos.test.ts +24 -2
- package/src/__tests__/onboarding-persona-write.test.ts +308 -0
- package/src/__tests__/openai-provider.test.ts +45 -8
- package/src/__tests__/persist-onboarding-artifacts.test.ts +44 -64
- package/src/__tests__/platform-callback-registration.test.ts +21 -4
- package/src/__tests__/platform.test.ts +2 -1
- package/src/__tests__/playbook-execution.test.ts +0 -43
- package/src/__tests__/plugin-tool-contribution.test.ts +47 -0
- package/src/__tests__/prechat-onboarding-contract.test.ts +214 -27
- package/src/__tests__/provider-tool-name.test.ts +23 -0
- package/src/__tests__/relay-server.test.ts +15 -4
- package/src/__tests__/runtime-events-sse.test.ts +4 -8
- package/src/__tests__/scheduler-disk-pressure.test.ts +148 -0
- package/src/__tests__/secret-ingress-http.test.ts +0 -1
- package/src/__tests__/suggestion-routes.test.ts +46 -0
- package/src/__tests__/twilio-validation.test.ts +2 -2
- package/src/__tests__/workspace-migration-065-bump-stale-heartbeat-interval.test.ts +122 -0
- package/src/__tests__/workspace-migration-066-seed-heartbeat-callsite-cost-default.test.ts +285 -0
- package/src/__tests__/workspace-migration-068-release-notes-local-timezone.test.ts +90 -0
- package/src/__tests__/workspace-migration-safe-storage-limits-release.test.ts +90 -0
- package/src/approvals/guardian-decision-primitive.ts +13 -0
- package/src/approvals/guardian-request-resolvers.ts +16 -17
- package/src/backup/snapshot-lock.ts +2 -27
- package/src/bundler/compiler-tools.ts +3 -2
- package/src/calls/call-conversation-messages.ts +46 -10
- package/src/cli/commands/__tests__/webhooks.test.ts +0 -4
- package/src/cli/commands/bash.ts +35 -108
- package/src/cli/commands/contacts.ts +64 -25
- package/src/cli/commands/credentials.ts +56 -0
- package/src/cli/commands/memory-v2.ts +7 -6
- package/src/cli/commands/oauth/__tests__/connect.test.ts +437 -1
- package/src/cli/commands/oauth/connect.ts +127 -1
- package/src/cli/commands/platform/__tests__/callback-routes-list.test.ts +0 -3
- package/src/cli/commands/platform/__tests__/connect.test.ts +7 -1
- package/src/cli/commands/platform/__tests__/disconnect.test.ts +7 -1
- package/src/cli/commands/platform/__tests__/status.test.ts +103 -6
- package/src/cli/commands/platform/index.ts +16 -7
- package/src/cli/commands/status.ts +57 -0
- package/src/cli/program.ts +4 -2
- package/src/config/assistant-feature-flags.ts +13 -3
- package/src/config/bundled-skills/messaging/tools/messaging-analyze-style.ts +4 -3
- package/src/config/bundled-skills/phone-calls/references/TROUBLESHOOTING.md +13 -7
- package/src/config/bundled-skills/playbooks/tools/playbook-create.ts +2 -2
- package/src/config/bundled-skills/playbooks/tools/playbook-delete.ts +2 -2
- package/src/config/bundled-skills/playbooks/tools/playbook-list.ts +2 -2
- package/src/config/bundled-skills/playbooks/tools/playbook-update.ts +2 -2
- package/src/config/env.ts +0 -8
- package/src/config/feature-flag-registry.json +27 -3
- package/src/config/loader.ts +127 -8
- package/src/config/schemas/__tests__/memory-v2.test.ts +10 -5
- package/src/config/schemas/call-site-catalog.ts +14 -0
- package/src/config/schemas/channels.ts +0 -5
- package/src/config/schemas/heartbeat.ts +1 -1
- package/src/config/schemas/llm.ts +2 -0
- package/src/config/schemas/memory-lifecycle.ts +13 -0
- package/src/config/schemas/memory-v2.ts +75 -11
- package/src/config/schemas/platform.ts +43 -3
- package/src/config/schemas/services.ts +28 -0
- package/src/config/seed-inference-profiles.ts +230 -33
- package/src/contacts/contact-store.ts +0 -25
- package/src/daemon/__tests__/conversation-tool-setup.test.ts +86 -25
- package/src/daemon/assistant-attachments.ts +4 -4
- package/src/daemon/config-watcher.ts +85 -57
- package/src/daemon/conversation-agent-loop-handlers.ts +6 -0
- package/src/daemon/conversation-agent-loop.ts +170 -33
- package/src/daemon/conversation-error.ts +87 -15
- package/src/daemon/conversation-lifecycle.ts +1 -3
- package/src/daemon/conversation-process.ts +8 -0
- package/src/daemon/conversation-runtime-assembly.ts +26 -0
- package/src/daemon/conversation-store.ts +2 -2
- package/src/daemon/conversation-surfaces.ts +195 -15
- package/src/daemon/conversation-tool-setup.ts +57 -14
- package/src/daemon/conversation.ts +17 -22
- package/src/daemon/date-context.ts +71 -22
- package/src/daemon/disk-pressure-background-gate.ts +73 -0
- package/src/daemon/disk-pressure-guard.ts +343 -0
- package/src/daemon/disk-pressure-policy.ts +163 -0
- package/src/daemon/handlers/shared.ts +0 -1
- package/src/daemon/handlers/skills.ts +3 -4
- package/src/daemon/host-app-control-proxy.ts +137 -41
- package/src/daemon/host-bash-proxy.ts +46 -21
- package/src/daemon/host-cu-proxy.ts +49 -3
- package/src/daemon/host-file-proxy.ts +43 -7
- package/src/daemon/host-transfer-proxy.ts +95 -4
- package/src/daemon/lifecycle.ts +79 -28
- package/src/daemon/meet-host-supervisor.ts +4 -4
- package/src/daemon/meet-manifest-loader.ts +0 -1
- package/src/daemon/memory-v2-startup.ts +14 -4
- package/src/daemon/message-protocol.ts +3 -0
- package/src/daemon/message-types/conversations.ts +4 -0
- package/src/daemon/message-types/disk-pressure.ts +9 -0
- package/src/daemon/message-types/messages.ts +3 -0
- package/src/daemon/profiler-run-store.ts +5 -5
- package/src/daemon/tool-setup-types.ts +2 -2
- package/src/documents/document-store.ts +85 -0
- package/src/filing/filing-service.ts +30 -5
- package/src/heartbeat/__tests__/heartbeat-feed-event.test.ts +9 -16
- package/src/heartbeat/__tests__/heartbeat-run-store.test.ts +36 -0
- package/src/heartbeat/heartbeat-run-store.ts +13 -0
- package/src/heartbeat/heartbeat-service.ts +205 -31
- package/src/home/feed-scheduler.ts +18 -0
- package/src/inbound/platform-callback-registration.ts +8 -15
- package/src/ipc/__tests__/clients-list-ipc.test.ts +169 -0
- package/src/ipc/assistant-server.ts +56 -2
- package/src/ipc/gateway-client.ts +37 -3
- package/src/live-voice/live-voice-archive.ts +4 -4
- package/src/live-voice/protocol.ts +5 -7
- package/src/media/image-service.ts +1 -7
- package/src/memory/__tests__/fixtures/memory-v2-activation-fixtures.ts +21 -13
- package/src/memory/__tests__/jobs-worker-v2-schedule.test.ts +52 -22
- package/src/memory/__tests__/memory-v2-activation-log-store.test.ts +0 -6
- package/src/memory/__tests__/memory-v2-concept-frequency.test.ts +272 -0
- package/src/memory/admin.ts +5 -9
- package/src/memory/context-search/agent-runner.ts +19 -2
- package/src/memory/context-search/sources/conversations.ts +2 -11
- package/src/memory/context-search/sources/memory-v2.ts +5 -4
- package/src/memory/context-search/sources/memory.ts +0 -1
- package/src/memory/context-search/types.ts +0 -1
- package/src/memory/conversation-crud.ts +4 -12
- package/src/memory/db-init.ts +2 -0
- package/src/memory/embedding-runtime-manager.ts +119 -5
- package/src/memory/graph/__tests__/conversation-graph-memory-v2-routing.test.ts +32 -21
- package/src/memory/graph/conversation-graph-memory.ts +42 -54
- package/src/memory/graph/extraction.ts +1 -3
- package/src/memory/graph/graph-search.test.ts +10 -67
- package/src/memory/graph/graph-search.ts +1 -20
- package/src/memory/graph/retriever.test.ts +6 -0
- package/src/memory/graph/retriever.ts +6 -10
- package/src/memory/indexer.ts +54 -45
- package/src/memory/job-handlers/backfill.ts +2 -11
- package/src/memory/job-handlers/cleanup.ts +43 -0
- package/src/memory/job-handlers/embedding.ts +6 -8
- package/src/memory/job-handlers/summarization.ts +2 -7
- package/src/memory/jobs-store.ts +48 -0
- package/src/memory/jobs-worker.ts +81 -43
- package/src/memory/memory-v2-activation-log-store.ts +32 -14
- package/src/memory/memory-v2-concept-frequency.ts +169 -0
- package/src/memory/migrations/239-trace-events-created-at-index.ts +18 -0
- package/src/memory/migrations/index.ts +1 -0
- package/src/memory/pkb/pkb-search.test.ts +6 -0
- package/src/memory/qdrant-client.ts +0 -13
- package/src/memory/rerank-local.ts +374 -0
- package/src/memory/search/semantic.ts +6 -67
- package/src/memory/trace-event-store.ts +1 -17
- package/src/memory/v2/__tests__/activation.test.ts +311 -250
- package/src/memory/v2/__tests__/consolidation-job.test.ts +40 -8
- package/src/memory/v2/__tests__/injection.test.ts +157 -167
- package/src/memory/v2/__tests__/prompts-consolidation.test.ts +61 -2
- package/src/memory/v2/__tests__/qdrant.test.ts +16 -0
- package/src/memory/v2/__tests__/reranker.test.ts +338 -0
- package/src/memory/v2/__tests__/sim.test.ts +5 -199
- package/src/memory/v2/__tests__/skill-store.test.ts +71 -65
- package/src/memory/v2/__tests__/static-context.test.ts +76 -1
- package/src/memory/v2/activation.ts +149 -156
- package/src/memory/v2/consolidation-job.ts +62 -12
- package/src/memory/v2/injection.ts +47 -60
- package/src/memory/v2/prompts/consolidation.ts +36 -1
- package/src/memory/v2/qdrant.ts +99 -0
- package/src/memory/v2/reranker.ts +177 -0
- package/src/memory/v2/sim.ts +10 -84
- package/src/memory/v2/skill-content.ts +4 -3
- package/src/memory/v2/skill-store.ts +82 -59
- package/src/memory/v2/static-context.ts +22 -0
- package/src/memory/v2/types.ts +10 -10
- package/src/notifications/copy-composer.ts +13 -0
- package/src/notifications/signal.ts +4 -0
- package/src/oauth/AGENTS.md +3 -1
- package/src/oauth/__tests__/oauth-connect-state.test.ts +137 -0
- package/src/oauth/connect-orchestrator.ts +2 -0
- package/src/oauth/connection-resolver.test.ts +66 -1
- package/src/oauth/connection-resolver.ts +55 -1
- package/src/oauth/oauth-connect-state.ts +77 -0
- package/src/oauth/seed-providers.ts +58 -1
- package/src/plugins/defaults/injectors.ts +35 -2
- package/src/plugins/defaults/memory-retrieval.ts +5 -6
- package/src/plugins/types.ts +7 -0
- package/src/proactive-artifact/aux-message-injector.ts +74 -0
- package/src/proactive-artifact/decision.test.ts +226 -0
- package/src/proactive-artifact/decision.ts +165 -0
- package/src/proactive-artifact/index.ts +7 -0
- package/src/proactive-artifact/job.test.ts +867 -0
- package/src/proactive-artifact/job.ts +352 -0
- package/src/proactive-artifact/message-copy.ts +41 -0
- package/src/proactive-artifact/trigger-state.test.ts +277 -0
- package/src/proactive-artifact/trigger-state.ts +119 -0
- package/src/prompts/normalize-onboarding.ts +80 -0
- package/src/prompts/persona-resolver.ts +101 -9
- package/src/prompts/system-prompt.ts +21 -7
- package/src/prompts/templates/BOOTSTRAP.md +13 -5
- package/src/providers/__tests__/retry-callsite.test.ts +222 -1
- package/src/providers/model-intents.ts +7 -0
- package/src/providers/openrouter/client.ts +8 -0
- package/src/providers/retry.ts +50 -0
- package/src/providers/types.ts +1 -0
- package/src/runtime/__tests__/agent-wake.test.ts +456 -3
- package/src/runtime/agent-wake.ts +238 -100
- package/src/runtime/assistant-event-hub.ts +36 -6
- package/src/runtime/assistant-event.ts +0 -1
- package/src/runtime/auth/__tests__/route-policy.test.ts +64 -0
- package/src/runtime/auth/route-policy.ts +14 -1
- package/src/runtime/auth/same-actor.ts +216 -0
- package/src/runtime/channel-retry-sweep.ts +65 -1
- package/src/runtime/guardian-reply-router.ts +10 -0
- package/src/runtime/local-actor-identity.ts +52 -11
- package/src/runtime/pending-interactions.ts +8 -0
- package/src/runtime/routes/__tests__/client-routes.test.ts +155 -0
- package/src/runtime/routes/__tests__/conversation-query-routes.test.ts +0 -5
- package/src/runtime/routes/__tests__/heartbeat-routes.test.ts +1 -1
- package/src/runtime/routes/client-routes.ts +20 -2
- package/src/runtime/routes/contact-routes.ts +0 -25
- package/src/runtime/routes/conversation-routes.ts +35 -26
- package/src/runtime/routes/debug-bash-routes.ts +163 -0
- package/src/runtime/routes/disk-pressure-routes.ts +121 -0
- package/src/runtime/routes/document-pdf-renderer.ts +6 -2
- package/src/runtime/routes/documents-routes.ts +2 -75
- package/src/runtime/routes/events-routes.ts +41 -9
- package/src/runtime/routes/host-bash-routes.ts +23 -3
- package/src/runtime/routes/host-cu-routes.ts +33 -6
- package/src/runtime/routes/host-file-routes.ts +32 -6
- package/src/runtime/routes/host-transfer-routes.ts +79 -16
- package/src/runtime/routes/identity-routes.ts +7 -138
- package/src/runtime/routes/inbound-message-handler.ts +77 -12
- package/src/runtime/routes/inbound-stages/guardian-reply-intercept.ts +3 -0
- package/src/runtime/routes/index.ts +6 -0
- package/src/runtime/routes/memory-item-routes.test.ts +41 -15
- package/src/runtime/routes/memory-v2-routes.ts +33 -0
- package/src/runtime/routes/oauth-connect-routes.ts +153 -0
- package/src/runtime/verification-outbound-actions.ts +4 -4
- package/src/schedule/run-script.ts +37 -5
- package/src/schedule/scheduler.ts +20 -1
- package/src/security/encrypted-store.ts +2 -0
- package/src/security/secure-keys.ts +55 -0
- package/src/skills/remote-skill-policy.ts +4 -10
- package/src/subagent/index.ts +1 -7
- package/src/subagent/manager.ts +1 -15
- package/src/tasks/task-runner.ts +0 -1
- package/src/tasks/task-store.ts +0 -3
- package/src/tools/background-tool-registry.ts +17 -3
- package/src/tools/host-filesystem/edit.test.ts +151 -0
- package/src/tools/host-filesystem/edit.ts +43 -1
- package/src/tools/host-filesystem/read.test.ts +129 -0
- package/src/tools/host-filesystem/read.ts +43 -1
- package/src/tools/host-filesystem/transfer.test.ts +127 -2
- package/src/tools/host-filesystem/transfer.ts +56 -11
- package/src/tools/host-filesystem/write.test.ts +134 -0
- package/src/tools/host-filesystem/write.ts +43 -1
- package/src/tools/host-terminal/host-shell.ts +13 -6
- package/src/tools/mcp/mcp-tool-factory.ts +2 -1
- package/src/tools/memory/register.test.ts +12 -9
- package/src/tools/memory/register.ts +1 -2
- package/src/tools/provider-tool-name.ts +28 -0
- package/src/tools/registry.ts +30 -9
- package/src/tools/terminal/shell.ts +9 -1
- package/src/tools/tool-approval-handler.ts +31 -6
- package/src/tools/types.ts +24 -2
- package/src/tts/provider-catalog.ts +3 -5
- package/src/util/disk-usage.ts +138 -0
- package/src/util/platform.ts +21 -11
- package/src/util/process-liveness.ts +26 -0
- package/src/workspace/heartbeat-service.ts +19 -0
- package/src/workspace/migrations/065-bump-stale-heartbeat-interval.ts +60 -0
- package/src/workspace/migrations/066-seed-heartbeat-callsite-cost-default.ts +146 -0
- package/src/workspace/migrations/067-release-notes-safe-storage-limits.ts +72 -0
- package/src/workspace/migrations/068-release-notes-local-timezone.ts +65 -0
- package/src/workspace/migrations/registry.ts +8 -0
- package/src/__tests__/conversation-tool-setup-memory-scope.test.ts +0 -167
- package/src/memory/v2/__tests__/skill-qdrant.test.ts +0 -657
- package/src/memory/v2/skill-qdrant.ts +0 -404
- package/src/signals/bash.ts +0 -198
|
@@ -0,0 +1,226 @@
|
|
|
1
|
+
import { describe, expect, test } from "bun:test";
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
type DecisionOutput,
|
|
5
|
+
formatTranscript,
|
|
6
|
+
parseDecisionOutput,
|
|
7
|
+
} from "./decision.js";
|
|
8
|
+
|
|
9
|
+
describe("parseDecisionOutput", () => {
|
|
10
|
+
test("parses SHOULD_BUILD: yes with all fields correctly", () => {
|
|
11
|
+
const text = `SHOULD_BUILD: yes
|
|
12
|
+
ARTIFACT_TYPE: app
|
|
13
|
+
ARTIFACT_TITLE: Sarah's Marathon Training Pace Calculator
|
|
14
|
+
ARTIFACT_DESCRIPTION: An interactive pace calculator that accounts for Sarah's goal of a sub-4-hour marathon, her current 9:30/mile easy pace, and the hilly terrain of the Boston course.`;
|
|
15
|
+
|
|
16
|
+
const result = parseDecisionOutput(text);
|
|
17
|
+
expect(result).toEqual({
|
|
18
|
+
shouldBuild: true,
|
|
19
|
+
artifactType: "app",
|
|
20
|
+
artifactTitle: "Sarah's Marathon Training Pace Calculator",
|
|
21
|
+
artifactDescription:
|
|
22
|
+
"An interactive pace calculator that accounts for Sarah's goal of a sub-4-hour marathon, her current 9:30/mile easy pace, and the hilly terrain of the Boston course.",
|
|
23
|
+
});
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
test("parses SHOULD_BUILD: no with skip reason", () => {
|
|
27
|
+
const text = `SHOULD_BUILD: no
|
|
28
|
+
SKIP_REASON: The conversation is too early and generic — the user has only asked a factual question with no personal context.`;
|
|
29
|
+
|
|
30
|
+
const result = parseDecisionOutput(text);
|
|
31
|
+
expect(result).toEqual({
|
|
32
|
+
shouldBuild: false,
|
|
33
|
+
skipReason:
|
|
34
|
+
"The conversation is too early and generic — the user has only asked a factual question with no personal context.",
|
|
35
|
+
});
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
test("parses SHOULD_BUILD: no without skip reason defaults to 'no reason given'", () => {
|
|
39
|
+
const text = `SHOULD_BUILD: no`;
|
|
40
|
+
|
|
41
|
+
const result = parseDecisionOutput(text);
|
|
42
|
+
expect(result).toEqual({
|
|
43
|
+
shouldBuild: false,
|
|
44
|
+
skipReason: "no reason given",
|
|
45
|
+
});
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
test("returns null for missing SHOULD_BUILD line", () => {
|
|
49
|
+
const text = `ARTIFACT_TYPE: app
|
|
50
|
+
ARTIFACT_TITLE: Some Title
|
|
51
|
+
ARTIFACT_DESCRIPTION: Some description.`;
|
|
52
|
+
|
|
53
|
+
const result = parseDecisionOutput(text);
|
|
54
|
+
expect(result).toBeNull();
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
test("returns null when yes but ARTIFACT_TYPE is missing", () => {
|
|
58
|
+
const text = `SHOULD_BUILD: yes
|
|
59
|
+
ARTIFACT_TITLE: Some Title
|
|
60
|
+
ARTIFACT_DESCRIPTION: Some description.`;
|
|
61
|
+
|
|
62
|
+
const result = parseDecisionOutput(text);
|
|
63
|
+
expect(result).toBeNull();
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
test("returns null when yes but ARTIFACT_TYPE is invalid", () => {
|
|
67
|
+
const text = `SHOULD_BUILD: yes
|
|
68
|
+
ARTIFACT_TYPE: widget
|
|
69
|
+
ARTIFACT_TITLE: Some Title
|
|
70
|
+
ARTIFACT_DESCRIPTION: Some description.`;
|
|
71
|
+
|
|
72
|
+
const result = parseDecisionOutput(text);
|
|
73
|
+
expect(result).toBeNull();
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
test("returns null when yes but ARTIFACT_TITLE is missing", () => {
|
|
77
|
+
const text = `SHOULD_BUILD: yes
|
|
78
|
+
ARTIFACT_TYPE: document
|
|
79
|
+
ARTIFACT_DESCRIPTION: Some description.`;
|
|
80
|
+
|
|
81
|
+
const result = parseDecisionOutput(text);
|
|
82
|
+
expect(result).toBeNull();
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
test("returns null when yes but ARTIFACT_DESCRIPTION is missing", () => {
|
|
86
|
+
const text = `SHOULD_BUILD: yes
|
|
87
|
+
ARTIFACT_TYPE: document
|
|
88
|
+
ARTIFACT_TITLE: Some Title`;
|
|
89
|
+
|
|
90
|
+
const result = parseDecisionOutput(text);
|
|
91
|
+
expect(result).toBeNull();
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
test("returns null when yes but ARTIFACT_TITLE is empty", () => {
|
|
95
|
+
const text = `SHOULD_BUILD: yes
|
|
96
|
+
ARTIFACT_TYPE: app
|
|
97
|
+
ARTIFACT_TITLE:
|
|
98
|
+
ARTIFACT_DESCRIPTION: Some description.`;
|
|
99
|
+
|
|
100
|
+
const result = parseDecisionOutput(text);
|
|
101
|
+
expect(result).toBeNull();
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
test("handles multi-line ARTIFACT_DESCRIPTION", () => {
|
|
105
|
+
const text = `SHOULD_BUILD: yes
|
|
106
|
+
ARTIFACT_TYPE: document
|
|
107
|
+
ARTIFACT_TITLE: Jake's Q3 OKR Tracker
|
|
108
|
+
ARTIFACT_DESCRIPTION: A structured comparison table for Jake's three competing priorities:
|
|
109
|
+
scaling the data pipeline from 10M to 50M events/day,
|
|
110
|
+
hiring two senior engineers by September,
|
|
111
|
+
and reducing p99 latency below 200ms.`;
|
|
112
|
+
|
|
113
|
+
const result = parseDecisionOutput(text) as DecisionOutput & {
|
|
114
|
+
shouldBuild: true;
|
|
115
|
+
};
|
|
116
|
+
expect(result).not.toBeNull();
|
|
117
|
+
expect(result.shouldBuild).toBe(true);
|
|
118
|
+
expect(result.artifactType).toBe("document");
|
|
119
|
+
expect(result.artifactTitle).toBe("Jake's Q3 OKR Tracker");
|
|
120
|
+
expect(result.artifactDescription).toContain(
|
|
121
|
+
"A structured comparison table",
|
|
122
|
+
);
|
|
123
|
+
expect(result.artifactDescription).toContain(
|
|
124
|
+
"reducing p99 latency below 200ms",
|
|
125
|
+
);
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
test("handles case-insensitive SHOULD_BUILD value", () => {
|
|
129
|
+
const text = `SHOULD_BUILD: Yes
|
|
130
|
+
ARTIFACT_TYPE: app
|
|
131
|
+
ARTIFACT_TITLE: Test Title
|
|
132
|
+
ARTIFACT_DESCRIPTION: Test description.`;
|
|
133
|
+
|
|
134
|
+
const result = parseDecisionOutput(text);
|
|
135
|
+
expect(result).not.toBeNull();
|
|
136
|
+
expect(result!.shouldBuild).toBe(true);
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
test("handles case-insensitive ARTIFACT_TYPE value", () => {
|
|
140
|
+
const text = `SHOULD_BUILD: yes
|
|
141
|
+
ARTIFACT_TYPE: Document
|
|
142
|
+
ARTIFACT_TITLE: Test Title
|
|
143
|
+
ARTIFACT_DESCRIPTION: Test description.`;
|
|
144
|
+
|
|
145
|
+
const result = parseDecisionOutput(text);
|
|
146
|
+
expect(result).not.toBeNull();
|
|
147
|
+
expect((result as { artifactType: string }).artifactType).toBe("document");
|
|
148
|
+
});
|
|
149
|
+
});
|
|
150
|
+
|
|
151
|
+
describe("formatTranscript", () => {
|
|
152
|
+
test("formats plain text messages correctly", () => {
|
|
153
|
+
const messages = [
|
|
154
|
+
{ role: "user", content: "Hello, I need help with my project" },
|
|
155
|
+
{
|
|
156
|
+
role: "assistant",
|
|
157
|
+
content: "I'd be happy to help! What kind of project?",
|
|
158
|
+
},
|
|
159
|
+
{
|
|
160
|
+
role: "user",
|
|
161
|
+
content: "I'm building a fitness tracker for marathon training",
|
|
162
|
+
},
|
|
163
|
+
];
|
|
164
|
+
|
|
165
|
+
const result = formatTranscript(messages);
|
|
166
|
+
expect(result).toBe(
|
|
167
|
+
`[User]: Hello, I need help with my project
|
|
168
|
+
|
|
169
|
+
[Assistant]: I'd be happy to help! What kind of project?
|
|
170
|
+
|
|
171
|
+
[User]: I'm building a fitness tracker for marathon training`,
|
|
172
|
+
);
|
|
173
|
+
});
|
|
174
|
+
|
|
175
|
+
test("handles JSON content blocks", () => {
|
|
176
|
+
const messages = [
|
|
177
|
+
{
|
|
178
|
+
role: "user",
|
|
179
|
+
content: JSON.stringify([{ type: "text", text: "What is 2+2?" }]),
|
|
180
|
+
},
|
|
181
|
+
{
|
|
182
|
+
role: "assistant",
|
|
183
|
+
content: JSON.stringify([
|
|
184
|
+
{ type: "text", text: "The answer is 4." },
|
|
185
|
+
{ type: "text", text: "Would you like to know more?" },
|
|
186
|
+
]),
|
|
187
|
+
},
|
|
188
|
+
];
|
|
189
|
+
|
|
190
|
+
const result = formatTranscript(messages);
|
|
191
|
+
expect(result).toBe(
|
|
192
|
+
`[User]: What is 2+2?
|
|
193
|
+
|
|
194
|
+
[Assistant]: The answer is 4.
|
|
195
|
+
Would you like to know more?`,
|
|
196
|
+
);
|
|
197
|
+
});
|
|
198
|
+
|
|
199
|
+
test("handles mixed JSON and plain text messages", () => {
|
|
200
|
+
const messages = [
|
|
201
|
+
{
|
|
202
|
+
role: "user",
|
|
203
|
+
content: JSON.stringify([
|
|
204
|
+
{ type: "text", text: "Help me plan my week" },
|
|
205
|
+
]),
|
|
206
|
+
},
|
|
207
|
+
{ role: "assistant", content: "Sure, what do you have coming up?" },
|
|
208
|
+
];
|
|
209
|
+
|
|
210
|
+
const result = formatTranscript(messages);
|
|
211
|
+
expect(result).toBe(
|
|
212
|
+
`[User]: Help me plan my week
|
|
213
|
+
|
|
214
|
+
[Assistant]: Sure, what do you have coming up?`,
|
|
215
|
+
);
|
|
216
|
+
});
|
|
217
|
+
|
|
218
|
+
test("handles unknown roles", () => {
|
|
219
|
+
const messages = [
|
|
220
|
+
{ role: "system", content: "You are a helpful assistant" },
|
|
221
|
+
];
|
|
222
|
+
|
|
223
|
+
const result = formatTranscript(messages);
|
|
224
|
+
expect(result).toBe("[system]: You are a helpful assistant");
|
|
225
|
+
});
|
|
226
|
+
});
|
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
export type DecisionOutput =
|
|
2
|
+
| { shouldBuild: false; skipReason: string }
|
|
3
|
+
| {
|
|
4
|
+
shouldBuild: true;
|
|
5
|
+
artifactType: "app" | "document";
|
|
6
|
+
artifactTitle: string;
|
|
7
|
+
artifactDescription: string;
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
export function buildDecisionPrompt(transcript: string): string {
|
|
11
|
+
return `You are deciding whether to proactively build a personalized artifact for the user based on their conversation so far.
|
|
12
|
+
|
|
13
|
+
Read the conversation below carefully. Your job:
|
|
14
|
+
1. Identify what the user cares about — their goals, context, specific details they've shared.
|
|
15
|
+
2. Decide: should we build a small interactive app or document that would delight this specific user?
|
|
16
|
+
3. Quality test: Could you have built the same thing for any random person? If yes, too generic — output SHOULD_BUILD: no.
|
|
17
|
+
|
|
18
|
+
Rules:
|
|
19
|
+
- Only say yes if you can build something SPECIFIC to this user's situation, using details from their conversation.
|
|
20
|
+
- An "app" is a small interactive web application (calculator, tracker, visualizer, planner, etc.)
|
|
21
|
+
- A "document" is a structured reference (checklist, guide, comparison table, template, etc.)
|
|
22
|
+
- The title and description must reference specifics from the conversation — names, numbers, goals, constraints the user mentioned.
|
|
23
|
+
- Do NOT include a MESSAGE field.
|
|
24
|
+
|
|
25
|
+
Conversation:
|
|
26
|
+
${transcript}
|
|
27
|
+
|
|
28
|
+
Respond in EXACTLY this format (no extra text before or after):
|
|
29
|
+
|
|
30
|
+
SHOULD_BUILD: [yes|no]
|
|
31
|
+
SKIP_REASON: [required if no — why this conversation isn't a good fit]
|
|
32
|
+
ARTIFACT_TYPE: [app|document]
|
|
33
|
+
ARTIFACT_TITLE: [specific title seeded with user context]
|
|
34
|
+
ARTIFACT_DESCRIPTION: [1-3 sentence build spec with user-specific details]
|
|
35
|
+
|
|
36
|
+
If SHOULD_BUILD is no, omit ARTIFACT_TYPE, ARTIFACT_TITLE, and ARTIFACT_DESCRIPTION.
|
|
37
|
+
If SHOULD_BUILD is yes, omit SKIP_REASON.`;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
export function parseDecisionOutput(text: string): DecisionOutput | null {
|
|
41
|
+
const lines = text.trim().split("\n");
|
|
42
|
+
|
|
43
|
+
const shouldBuildLine = lines.find((line) =>
|
|
44
|
+
line.trim().startsWith("SHOULD_BUILD:"),
|
|
45
|
+
);
|
|
46
|
+
if (!shouldBuildLine) return null;
|
|
47
|
+
|
|
48
|
+
const shouldBuildValue = shouldBuildLine
|
|
49
|
+
.split(":")
|
|
50
|
+
.slice(1)
|
|
51
|
+
.join(":")
|
|
52
|
+
.trim()
|
|
53
|
+
.toLowerCase();
|
|
54
|
+
|
|
55
|
+
if (shouldBuildValue === "no") {
|
|
56
|
+
const skipReasonLine = lines.find((line) =>
|
|
57
|
+
line.trim().startsWith("SKIP_REASON:"),
|
|
58
|
+
);
|
|
59
|
+
const skipReason = skipReasonLine
|
|
60
|
+
? skipReasonLine.split(":").slice(1).join(":").trim()
|
|
61
|
+
: "no reason given";
|
|
62
|
+
return { shouldBuild: false, skipReason };
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
if (shouldBuildValue === "yes") {
|
|
66
|
+
const artifactTypeLine = lines.find((line) =>
|
|
67
|
+
line.trim().startsWith("ARTIFACT_TYPE:"),
|
|
68
|
+
);
|
|
69
|
+
if (!artifactTypeLine) return null;
|
|
70
|
+
const artifactType = artifactTypeLine
|
|
71
|
+
.split(":")
|
|
72
|
+
.slice(1)
|
|
73
|
+
.join(":")
|
|
74
|
+
.trim()
|
|
75
|
+
.toLowerCase();
|
|
76
|
+
if (artifactType !== "app" && artifactType !== "document") return null;
|
|
77
|
+
|
|
78
|
+
const artifactTitleLine = lines.find((line) =>
|
|
79
|
+
line.trim().startsWith("ARTIFACT_TITLE:"),
|
|
80
|
+
);
|
|
81
|
+
if (!artifactTitleLine) return null;
|
|
82
|
+
const artifactTitle = artifactTitleLine
|
|
83
|
+
.split(":")
|
|
84
|
+
.slice(1)
|
|
85
|
+
.join(":")
|
|
86
|
+
.trim();
|
|
87
|
+
if (!artifactTitle) return null;
|
|
88
|
+
|
|
89
|
+
const artifactDescriptionStartIndex = lines.findIndex((line) =>
|
|
90
|
+
line.trim().startsWith("ARTIFACT_DESCRIPTION:"),
|
|
91
|
+
);
|
|
92
|
+
if (artifactDescriptionStartIndex === -1) return null;
|
|
93
|
+
|
|
94
|
+
const firstDescLine = lines[artifactDescriptionStartIndex]
|
|
95
|
+
.split(":")
|
|
96
|
+
.slice(1)
|
|
97
|
+
.join(":")
|
|
98
|
+
.trim();
|
|
99
|
+
|
|
100
|
+
// Collect continuation lines (lines after ARTIFACT_DESCRIPTION that aren't other fields)
|
|
101
|
+
const descriptionParts = [firstDescLine];
|
|
102
|
+
for (let i = artifactDescriptionStartIndex + 1; i < lines.length; i++) {
|
|
103
|
+
const line = lines[i].trim();
|
|
104
|
+
if (
|
|
105
|
+
line.startsWith("SHOULD_BUILD:") ||
|
|
106
|
+
line.startsWith("SKIP_REASON:") ||
|
|
107
|
+
line.startsWith("ARTIFACT_TYPE:") ||
|
|
108
|
+
line.startsWith("ARTIFACT_TITLE:")
|
|
109
|
+
) {
|
|
110
|
+
break;
|
|
111
|
+
}
|
|
112
|
+
descriptionParts.push(line);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
const artifactDescription = descriptionParts.join("\n").trim();
|
|
116
|
+
if (!artifactDescription) return null;
|
|
117
|
+
|
|
118
|
+
return {
|
|
119
|
+
shouldBuild: true,
|
|
120
|
+
artifactType: artifactType as "app" | "document",
|
|
121
|
+
artifactTitle,
|
|
122
|
+
artifactDescription,
|
|
123
|
+
};
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
return null;
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
export function formatTranscript(
|
|
130
|
+
messages: Array<{ role: string; content: string }>,
|
|
131
|
+
): string {
|
|
132
|
+
return messages
|
|
133
|
+
.map((msg) => {
|
|
134
|
+
const label =
|
|
135
|
+
msg.role === "user"
|
|
136
|
+
? "[User]"
|
|
137
|
+
: msg.role === "assistant"
|
|
138
|
+
? "[Assistant]"
|
|
139
|
+
: `[${msg.role}]`;
|
|
140
|
+
const content = parseContent(msg.content);
|
|
141
|
+
return `${label}: ${content}`;
|
|
142
|
+
})
|
|
143
|
+
.join("\n\n");
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
function parseContent(content: string): string {
|
|
147
|
+
// Try to parse as JSON content block array
|
|
148
|
+
try {
|
|
149
|
+
const parsed = JSON.parse(content);
|
|
150
|
+
if (Array.isArray(parsed)) {
|
|
151
|
+
return parsed
|
|
152
|
+
.map((block) => {
|
|
153
|
+
if (typeof block === "string") return block;
|
|
154
|
+
if (block.type === "text" && typeof block.text === "string")
|
|
155
|
+
return block.text;
|
|
156
|
+
return "";
|
|
157
|
+
})
|
|
158
|
+
.filter(Boolean)
|
|
159
|
+
.join("\n");
|
|
160
|
+
}
|
|
161
|
+
} catch {
|
|
162
|
+
// Not JSON, treat as plain text
|
|
163
|
+
}
|
|
164
|
+
return content;
|
|
165
|
+
}
|