@elevasis/core 0.7.0 → 0.8.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/package.json +3 -3
- package/src/README.md +41 -41
- package/src/__tests__/publish.test.ts +18 -18
- package/src/__tests__/{template-foundations-compatibility.test.ts → template-core-compatibility.test.ts} +99 -99
- package/src/_gen/__tests__/__snapshots__/contracts.md.snap +1135 -1131
- package/src/_gen/__tests__/scaffold-contracts.test.ts +53 -53
- package/src/_gen/scaffold-contracts.ts +45 -45
- package/src/auth/multi-tenancy/credentials/README.md +38 -38
- package/src/auth/multi-tenancy/credentials/index.ts +6 -6
- package/src/auth/multi-tenancy/credentials/server/encryption.ts +39 -39
- package/src/auth/multi-tenancy/credentials/server/service.ts +60 -60
- package/src/auth/multi-tenancy/index.ts +17 -17
- package/src/auth/multi-tenancy/invitations/api-schemas.ts +107 -107
- package/src/auth/multi-tenancy/invitations/index.ts +37 -37
- package/src/auth/multi-tenancy/invitations/invitation.ts +86 -86
- package/src/auth/multi-tenancy/invitations/server/index.ts +25 -25
- package/src/auth/multi-tenancy/invitations/server/transforms.ts +24 -24
- package/src/auth/multi-tenancy/invitations/server/workos.ts +24 -24
- package/src/auth/multi-tenancy/invitations/supabase.ts +50 -50
- package/src/auth/multi-tenancy/memberships/api-schemas.ts +126 -126
- package/src/auth/multi-tenancy/memberships/index.ts +21 -21
- package/src/auth/multi-tenancy/memberships/membership.ts +138 -138
- package/src/auth/multi-tenancy/memberships/server/index.ts +15 -15
- package/src/auth/multi-tenancy/memberships/server/transforms.ts +32 -32
- package/src/auth/multi-tenancy/memberships/server/workos.ts +21 -21
- package/src/auth/multi-tenancy/memberships/supabase.ts +46 -46
- package/src/auth/multi-tenancy/organizations/api-schemas.ts +128 -128
- package/src/auth/multi-tenancy/organizations/index.ts +23 -23
- package/src/auth/multi-tenancy/organizations/organization.ts +24 -24
- package/src/auth/multi-tenancy/organizations/server/index.ts +10 -10
- package/src/auth/multi-tenancy/organizations/server/transforms.ts +35 -35
- package/src/auth/multi-tenancy/organizations/server/workos.ts +20 -20
- package/src/auth/multi-tenancy/types.ts +83 -83
- package/src/auth/multi-tenancy/users/api-schemas.ts +194 -194
- package/src/auth/multi-tenancy/users/index.ts +27 -27
- package/src/auth/multi-tenancy/users/server/index.ts +19 -19
- package/src/auth/multi-tenancy/users/server/transforms.ts +21 -21
- package/src/auth/multi-tenancy/users/server/workos.ts +16 -16
- package/src/auth/multi-tenancy/users/user.ts +65 -65
- package/src/business/README.md +52 -52
- package/src/business/__tests__/entities-published.test.ts +33 -33
- package/src/business/acquisition/api-schemas.ts +759 -759
- package/src/business/acquisition/index.ts +109 -109
- package/src/business/acquisition/types.ts +402 -402
- package/src/business/base-entities.test.ts +481 -481
- package/src/business/base-entities.ts +241 -241
- package/src/business/entities-published.ts +24 -24
- package/src/business/index.ts +15 -15
- package/src/business/pdf/browser/pdfmake-browser.ts +229 -229
- package/src/business/pdf/index.ts +10 -10
- package/src/business/pdf/server/index.ts +21 -21
- package/src/business/pdf/server/themes/default.ts +8 -8
- package/src/business/pdf/server/themes/index.ts +9 -9
- package/src/business/pdf/server/themes/types.ts +8 -8
- package/src/business/pdf/types.ts +272 -272
- package/src/business/projects/index.ts +2 -1
- package/src/business/projects/sse-events.ts +21 -0
- package/src/business/projects/types.ts +89 -89
- package/src/business/sales/api-schemas.ts +75 -75
- package/src/business/seo/__tests__/linking.test.ts +549 -549
- package/src/business/seo/__tests__/types.test.ts +404 -404
- package/src/business/seo/index.ts +2 -2
- package/src/business/seo/linking.ts +281 -281
- package/src/business/seo/types.ts +199 -199
- package/src/commands/queue/index.ts +3 -3
- package/src/commands/queue/schemas.test.ts +593 -593
- package/src/commands/queue/schemas.ts +125 -125
- package/src/commands/queue/sse-events.ts +61 -61
- package/src/commands/queue/types/action.ts +52 -52
- package/src/commands/queue/types/checkpoint.ts +44 -44
- package/src/commands/queue/types/index.ts +7 -7
- package/src/commands/queue/types/task.ts +116 -116
- package/src/commands/queue/types.ts +14 -14
- package/src/content/distribution-metadata.ts +61 -61
- package/src/content/index.ts +10 -10
- package/src/deployments/index.ts +22 -22
- package/src/execution/core/__tests__/archived-logs.test.ts +72 -72
- package/src/execution/core/index.ts +11 -11
- package/src/execution/core/runner-types.ts +80 -80
- package/src/execution/core/server/environment.ts +31 -31
- package/src/execution/core/sse-executions.ts +119 -119
- package/src/execution/core/types.ts +29 -29
- package/src/execution/engine/__tests__/fixtures/test-agents.ts +4 -4
- package/src/execution/engine/__tests__/timeout.test.ts +565 -565
- package/src/execution/engine/agent/__tests__/errors.test.ts +508 -508
- package/src/execution/engine/agent/actions/__tests__/processor.test.ts +531 -531
- package/src/execution/engine/agent/actions/executor.ts +205 -205
- package/src/execution/engine/agent/actions/navigate-knowledge-executor.ts +230 -230
- package/src/execution/engine/agent/actions/processor.ts +116 -116
- package/src/execution/engine/agent/actions/types.ts +70 -70
- package/src/execution/engine/agent/core/agent.ts +810 -810
- package/src/execution/engine/agent/core/types.ts +155 -155
- package/src/execution/engine/agent/errors.ts +251 -251
- package/src/execution/engine/agent/index.ts +78 -78
- package/src/execution/engine/agent/knowledge-map/types.ts +106 -106
- package/src/execution/engine/agent/knowledge-map/utils.ts +101 -101
- package/src/execution/engine/agent/memory/__tests__/manager.test.ts +754 -754
- package/src/execution/engine/agent/memory/domains.ts +99 -99
- package/src/execution/engine/agent/memory/manager.ts +365 -365
- package/src/execution/engine/agent/memory/processor.ts +66 -66
- package/src/execution/engine/agent/memory/types.ts +90 -90
- package/src/execution/engine/agent/memory/utils.ts +134 -134
- package/src/execution/engine/agent/observability/logging.ts +467 -467
- package/src/execution/engine/agent/observability/types.ts +64 -64
- package/src/execution/engine/agent/reasoning/adapters/agent-adapter-helpers.ts +349 -349
- package/src/execution/engine/agent/reasoning/processor.ts +92 -92
- package/src/execution/engine/agent/reasoning/prompt-sections/base-actions.ts +134 -134
- package/src/execution/engine/agent/reasoning/prompt-sections/completion.ts +49 -49
- package/src/execution/engine/agent/reasoning/prompt-sections/knowledge-map.ts +93 -93
- package/src/execution/engine/agent/reasoning/prompt-sections/memory.ts +65 -65
- package/src/execution/engine/agent/reasoning/prompt-sections/tools.ts +44 -44
- package/src/execution/engine/agent/reasoning/request-builder.ts +169 -169
- package/src/execution/engine/agent/reasoning/types.ts +18 -18
- package/src/execution/engine/base/errors.ts +118 -118
- package/src/execution/engine/base/index.ts +2 -2
- package/src/execution/engine/base/logging.ts +31 -31
- package/src/execution/engine/base/serialization.ts +324 -324
- package/src/execution/engine/base/types.ts +126 -126
- package/src/execution/engine/base/utils.ts +41 -41
- package/src/execution/engine/index.ts +434 -434
- package/src/execution/engine/interface/index.ts +1 -1
- package/src/execution/engine/interface/types.ts +62 -62
- package/src/execution/engine/llm/__tests__/model-info.test.ts +50 -50
- package/src/execution/engine/llm/__tests__/model-validation.test.ts +321 -321
- package/src/execution/engine/llm/__tests__/response-schema-validator.test.ts +115 -115
- package/src/execution/engine/llm/adapters/__tests__/adapter-factory.test.ts +375 -375
- package/src/execution/engine/llm/adapters/__tests__/anthropic-adapter.test.ts +463 -463
- package/src/execution/engine/llm/adapters/__tests__/anthropic.integration.test.ts +177 -177
- package/src/execution/engine/llm/adapters/__tests__/google-adapter.test.ts +722 -722
- package/src/execution/engine/llm/adapters/__tests__/google.integration.test.ts +376 -376
- package/src/execution/engine/llm/adapters/__tests__/openai-adapter.test.ts +551 -551
- package/src/execution/engine/llm/adapters/__tests__/openrouter-adapter.test.ts +563 -563
- package/src/execution/engine/llm/adapters/__tests__/openrouter.integration.test.ts +105 -105
- package/src/execution/engine/llm/adapters/__tests__/universal-adapter.test.ts +537 -537
- package/src/execution/engine/llm/adapters/circuit-breaker.ts +147 -147
- package/src/execution/engine/llm/adapters/index.ts +17 -17
- package/src/execution/engine/llm/adapters/mock-adapter.ts +116 -116
- package/src/execution/engine/llm/adapters/server/adapter-factory.ts +130 -130
- package/src/execution/engine/llm/adapters/server/anthropic.ts +137 -137
- package/src/execution/engine/llm/adapters/server/google.ts +283 -283
- package/src/execution/engine/llm/adapters/server/index.ts +12 -12
- package/src/execution/engine/llm/adapters/server/openai.ts +206 -206
- package/src/execution/engine/llm/adapters/server/openrouter.ts +235 -235
- package/src/execution/engine/llm/adapters/universal-adapter.ts +230 -230
- package/src/execution/engine/llm/errors.ts +186 -186
- package/src/execution/engine/llm/model-info.ts +332 -332
- package/src/execution/engine/llm/response-schema-validator.ts +113 -113
- package/src/execution/engine/llm/types.ts +86 -86
- package/src/execution/engine/test-utils/index.ts +6 -6
- package/src/execution/engine/test-utils/mocks.ts +56 -56
- package/src/execution/engine/tools/integration/base-integration-adapter.ts +50 -50
- package/src/execution/engine/tools/integration/index.ts +53 -53
- package/src/execution/engine/tools/integration/server/adapters/anymailfinder/anymailfinder-adapter.ts +73 -73
- package/src/execution/engine/tools/integration/server/adapters/anymailfinder/anymailfinder-tools.ts +209 -209
- package/src/execution/engine/tools/integration/server/adapters/anymailfinder/fetch/find-company-email/index.ts +82 -82
- package/src/execution/engine/tools/integration/server/adapters/anymailfinder/fetch/find-decision-maker-email/index.ts +122 -122
- package/src/execution/engine/tools/integration/server/adapters/anymailfinder/fetch/find-person-email/index.ts +89 -89
- package/src/execution/engine/tools/integration/server/adapters/anymailfinder/fetch/verify-email/index.ts +84 -84
- package/src/execution/engine/tools/integration/server/adapters/anymailfinder/index.ts +16 -16
- package/src/execution/engine/tools/integration/server/adapters/apify/__tests__/apify-run-actor.integration.test.ts +293 -293
- package/src/execution/engine/tools/integration/server/adapters/apify/apify-adapter.ts +100 -100
- package/src/execution/engine/tools/integration/server/adapters/apify/apify-tools.ts +217 -217
- package/src/execution/engine/tools/integration/server/adapters/apify/fetch/get-dataset-items/index.ts +92 -92
- package/src/execution/engine/tools/integration/server/adapters/apify/fetch/run-actor/index.ts +218 -218
- package/src/execution/engine/tools/integration/server/adapters/apify/fetch/start-actor/index.ts +87 -87
- package/src/execution/engine/tools/integration/server/adapters/apify/index.ts +11 -11
- package/src/execution/engine/tools/integration/server/adapters/attio/__tests__/attio-crud.integration.test.ts +361 -361
- package/src/execution/engine/tools/integration/server/adapters/attio/attio-adapter.ts +162 -162
- package/src/execution/engine/tools/integration/server/adapters/attio/attio-tools.ts +594 -594
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/create-attribute/index.ts +214 -214
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/create-note/index.ts +152 -152
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/create-record/index.ts +141 -141
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/delete-note/index.ts +86 -86
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/delete-record/index.ts +105 -105
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/get-record/index.ts +118 -118
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/list-attributes/index.ts +165 -165
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/list-notes/index.ts +96 -96
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/list-objects/index.ts +104 -104
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/list-records/index.ts +156 -156
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/update-attribute/index.ts +220 -220
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/update-record/index.ts +140 -140
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/utils/types.ts +146 -146
- package/src/execution/engine/tools/integration/server/adapters/attio/index.ts +31 -31
- package/src/execution/engine/tools/integration/server/adapters/gmail/gmail-adapter.ts +210 -210
- package/src/execution/engine/tools/integration/server/adapters/gmail/gmail-tools.ts +104 -104
- package/src/execution/engine/tools/integration/server/adapters/google-sheets/__tests__/google-sheets.integration.test.ts +261 -261
- package/src/execution/engine/tools/integration/server/adapters/google-sheets/google-sheets-adapter.ts +1189 -1189
- package/src/execution/engine/tools/integration/server/adapters/google-sheets/google-sheets-tools.ts +641 -641
- package/src/execution/engine/tools/integration/server/adapters/google-sheets/index.ts +18 -18
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/activate-campaign/index.ts +86 -86
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/add-to-campaign/__tests__/index.test.ts +289 -289
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/add-to-campaign/index.ts +154 -154
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/bulk-add-leads/__tests__/index.test.ts +325 -325
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/bulk-add-leads/index.ts +153 -153
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/bulk-delete-leads/index.ts +84 -84
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/create-campaign/index.ts +125 -125
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/create-inbox-test/index.ts +107 -107
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/delete-campaign/index.ts +85 -85
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/get-account-health/index.ts +91 -91
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/get-campaign/index.ts +92 -92
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/get-campaign-analytics/__tests__/index.test.ts +195 -195
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/get-campaign-analytics/index.ts +113 -113
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/get-daily-campaign-analytics/index.ts +104 -104
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/get-emails/index.ts +155 -155
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/get-step-analytics/__tests__/index.test.ts +196 -196
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/get-step-analytics/index.ts +102 -102
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/list-campaigns/__tests__/index.test.ts +189 -189
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/list-campaigns/index.ts +87 -87
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/list-leads/index.ts +112 -112
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/patch-lead/index.ts +76 -76
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/pause-campaign/index.ts +86 -86
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/remove-from-subsequence/index.ts +98 -98
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/send-reply/index.ts +126 -126
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/update-campaign/__tests__/index.test.ts +193 -193
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/update-campaign/index.ts +99 -99
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/update-interest-status/__tests__/index.test.ts +621 -621
- package/src/execution/engine/tools/integration/server/adapters/instantly/fetch/update-interest-status/index.ts +125 -125
- package/src/execution/engine/tools/integration/server/adapters/instantly/index.ts +29 -29
- package/src/execution/engine/tools/integration/server/adapters/instantly/instantly-adapter.ts +178 -178
- package/src/execution/engine/tools/integration/server/adapters/instantly/instantly-tools.ts +1473 -1473
- package/src/execution/engine/tools/integration/server/adapters/millionverifier/fetch/check-credits/index.ts +59 -59
- package/src/execution/engine/tools/integration/server/adapters/millionverifier/fetch/verify-email/index.ts +102 -102
- package/src/execution/engine/tools/integration/server/adapters/millionverifier/index.ts +17 -17
- package/src/execution/engine/tools/integration/server/adapters/millionverifier/millionverifier-adapter.ts +80 -80
- package/src/execution/engine/tools/integration/server/adapters/millionverifier/millionverifier-tools.ts +102 -102
- package/src/execution/engine/tools/integration/server/adapters/resend/fetch/get-email/index.ts +102 -102
- package/src/execution/engine/tools/integration/server/adapters/resend/fetch/send-email/index.ts +134 -134
- package/src/execution/engine/tools/integration/server/adapters/resend/fetch/utils/types.ts +75 -75
- package/src/execution/engine/tools/integration/server/adapters/resend/index.ts +27 -27
- package/src/execution/engine/tools/integration/server/adapters/resend/resend-adapter.ts +108 -108
- package/src/execution/engine/tools/integration/server/adapters/resend/resend-tools.ts +132 -132
- package/src/execution/engine/tools/integration/server/adapters/signature-api/fetch/create-envelope/index.ts +274 -274
- package/src/execution/engine/tools/integration/server/adapters/signature-api/fetch/download-document/index.ts +230 -230
- package/src/execution/engine/tools/integration/server/adapters/signature-api/fetch/get-envelope/index.ts +133 -133
- package/src/execution/engine/tools/integration/server/adapters/signature-api/fetch/void-envelope/index.ts +90 -90
- package/src/execution/engine/tools/integration/server/adapters/stripe/fetch/utils/types.ts +210 -210
- package/src/execution/engine/tools/integration/server/adapters/stripe/stripe-adapter.ts +517 -517
- package/src/execution/engine/tools/integration/server/adapters/stripe/stripe-tools.ts +309 -309
- package/src/execution/engine/tools/integration/server/adapters/tomba/fetch/domain-search/index.ts +133 -133
- package/src/execution/engine/tools/integration/server/adapters/tomba/fetch/email-finder/index.ts +122 -122
- package/src/execution/engine/tools/integration/server/adapters/tomba/fetch/email-verifier/index.ts +111 -111
- package/src/execution/engine/tools/integration/server/adapters/tomba/index.ts +11 -11
- package/src/execution/engine/tools/integration/server/adapters/tomba/tomba-adapter.ts +78 -78
- package/src/execution/engine/tools/integration/server/adapters/tomba/tomba-tools.ts +222 -222
- package/src/execution/engine/tools/integration/server/index.ts +61 -61
- package/src/execution/engine/tools/integration/service.ts +161 -161
- package/src/execution/engine/tools/integration/tool.ts +253 -253
- package/src/execution/engine/tools/integration/types/anymailfinder.ts +74 -74
- package/src/execution/engine/tools/integration/types/apify.ts +92 -92
- package/src/execution/engine/tools/integration/types/index.ts +19 -19
- package/src/execution/engine/tools/integration/types/instantly.ts +557 -557
- package/src/execution/engine/tools/integration/types/millionverifier.ts +56 -56
- package/src/execution/engine/tools/integration/types/stripe.ts +162 -162
- package/src/execution/engine/tools/integration/types/tomba.ts +94 -94
- package/src/execution/engine/tools/lead-service-types.ts +884 -884
- package/src/execution/engine/tools/llm/index.ts +11 -11
- package/src/execution/engine/tools/llm/server/index.ts +8 -8
- package/src/execution/engine/tools/llm/server/llm-call-tool.ts +118 -118
- package/src/execution/engine/tools/platform/__tests__/pdf.test.ts +441 -441
- package/src/execution/engine/tools/platform/acquisition/company-tools.ts +248 -248
- package/src/execution/engine/tools/platform/acquisition/contact-tools.ts +319 -319
- package/src/execution/engine/tools/platform/acquisition/index.ts +43 -43
- package/src/execution/engine/tools/platform/acquisition/list-tools.ts +148 -148
- package/src/execution/engine/tools/platform/acquisition/types.ts +260 -260
- package/src/execution/engine/tools/platform/email/index.ts +122 -122
- package/src/execution/engine/tools/platform/email/types.ts +96 -96
- package/src/execution/engine/tools/platform/index.ts +157 -157
- package/src/execution/engine/tools/platform/notification.ts +81 -81
- package/src/execution/engine/tools/platform/pdf/index.ts +110 -110
- package/src/execution/engine/tools/platform/pdf/types.ts +77 -77
- package/src/execution/engine/tools/platform/scheduler.ts +87 -87
- package/src/execution/engine/tools/platform/storage/index.ts +370 -370
- package/src/execution/engine/tools/platform/types.ts +148 -148
- package/src/execution/engine/tools/registry.ts +700 -699
- package/src/execution/engine/tools/tool-maps.ts +786 -786
- package/src/execution/engine/tools/types.ts +233 -233
- package/src/execution/engine/workflow/__tests__/errors.test.ts +139 -139
- package/src/execution/engine/workflow/errors.ts +63 -63
- package/src/execution/engine/workflow/helpers/index.ts +11 -11
- package/src/execution/engine/workflow/helpers/server/index.ts +8 -8
- package/src/execution/engine/workflow/helpers/server/llm-call.ts +93 -93
- package/src/execution/engine/workflow/index.ts +19 -19
- package/src/execution/engine/workflow/log-truncate.ts +26 -26
- package/src/execution/engine/workflow/logging.ts +191 -191
- package/src/execution/engine/workflow/types.ts +182 -182
- package/src/execution/engine/workflow/utils.ts +280 -280
- package/src/execution/engine/workflow/workflow.ts +168 -168
- package/src/execution/index.ts +3 -3
- package/src/execution/scheduler/__tests__/api-schemas.test.ts +733 -733
- package/src/execution/scheduler/__tests__/utils.test.ts +1009 -1009
- package/src/execution/scheduler/api-schemas.ts +296 -296
- package/src/execution/scheduler/index.ts +50 -50
- package/src/execution/scheduler/schemas.ts +264 -264
- package/src/execution/scheduler/types.ts +111 -111
- package/src/execution/scheduler/utils.ts +364 -364
- package/src/forms/index.ts +7 -7
- package/src/forms/schemas.ts +69 -69
- package/src/forms/types.ts +70 -70
- package/src/index.ts +71 -60
- package/src/integrations/credentials/__tests__/schemas.test.ts +82 -82
- package/src/integrations/credentials/__tests__/utils.test.ts +144 -144
- package/src/integrations/credentials/api-schemas.ts +143 -143
- package/src/integrations/credentials/index.ts +32 -32
- package/src/integrations/credentials/schemas.ts +164 -164
- package/src/integrations/credentials/utils.ts +59 -59
- package/src/integrations/oauth/__tests__/provider-registry.test.ts +59 -59
- package/src/integrations/oauth/api-schemas.ts +92 -92
- package/src/integrations/oauth/index.ts +19 -19
- package/src/integrations/oauth/provider-registry.ts +61 -61
- package/src/integrations/oauth/server/__tests__/refresh-concurrent.test.ts +183 -183
- package/src/integrations/oauth/server/__tests__/refresh.test.ts +577 -577
- package/src/integrations/oauth/server/credentials.ts +39 -39
- package/src/integrations/oauth/server/refresh.ts +214 -214
- package/src/integrations/oauth/types.ts +34 -34
- package/src/integrations/webhook-endpoints/__tests__/api-schemas.test.ts +318 -318
- package/src/integrations/webhook-endpoints/api-schemas.ts +102 -102
- package/src/integrations/webhook-endpoints/index.ts +28 -28
- package/src/integrations/webhook-endpoints/types.ts +51 -51
- package/src/operations/activities/api-schemas.ts +79 -79
- package/src/operations/activities/index.ts +9 -9
- package/src/operations/activities/sse-events.ts +30 -30
- package/src/operations/activities/types.ts +63 -63
- package/src/operations/debug-logs/client.ts +60 -60
- package/src/operations/debug-logs/debug-logger.ts +83 -83
- package/src/operations/debug-logs/index.ts +8 -8
- package/src/operations/debug-logs/server.ts +19 -19
- package/src/operations/debug-logs/types.ts +33 -33
- package/src/operations/index.ts +50 -50
- package/src/operations/notifications/api-schemas.ts +91 -91
- package/src/operations/notifications/index.ts +3 -3
- package/src/operations/notifications/sse-events.ts +21 -21
- package/src/operations/notifications/types.ts +47 -47
- package/src/operations/observability/__tests__/openrouter-cost-flow.test.ts +297 -297
- package/src/operations/observability/__tests__/utils.test.ts +54 -54
- package/src/operations/observability/ai-usage-collector.ts +64 -64
- package/src/operations/observability/index.ts +13 -13
- package/src/operations/observability/metrics-collector.ts +49 -49
- package/src/operations/observability/schemas.ts +39 -39
- package/src/operations/observability/types.ts +463 -463
- package/src/operations/observability/utils.ts +77 -77
- package/src/operations/sessions/__tests__/manager.test.ts +821 -821
- package/src/operations/sessions/index.ts +26 -26
- package/src/operations/sessions/server/manager.ts +90 -90
- package/src/operations/sessions/server/session.ts +180 -180
- package/src/operations/sessions/types.ts +98 -98
- package/src/operations/triggers/index.ts +12 -12
- package/src/operations/triggers/webhook/definitions/instantly-account-error.ts +44 -44
- package/src/operations/triggers/webhook/definitions/instantly-auto-reply-received.ts +51 -51
- package/src/operations/triggers/webhook/definitions/instantly-campaign-completed.ts +45 -45
- package/src/operations/triggers/webhook/definitions/instantly-email-bounced.ts +49 -49
- package/src/operations/triggers/webhook/definitions/instantly-lead-unsubscribed.ts +45 -45
- package/src/operations/triggers/webhook/definitions/instantly-reply-received.ts +54 -54
- package/src/operations/triggers/webhook/index.ts +35 -35
- package/src/operations/triggers/webhook/types.ts +74 -74
- package/src/organization-model/README.md +97 -97
- package/src/organization-model/__tests__/defaults.test.ts +175 -175
- package/src/organization-model/__tests__/domains/customers.test.ts +295 -295
- package/src/organization-model/__tests__/domains/goals.test.ts +479 -479
- package/src/organization-model/__tests__/domains/identity.test.ts +279 -279
- package/src/organization-model/__tests__/domains/navigation.test.ts +212 -212
- package/src/organization-model/__tests__/domains/offerings.test.ts +419 -419
- package/src/organization-model/__tests__/domains/operations.test.ts +203 -203
- package/src/organization-model/__tests__/domains/resource-mappings.test.ts +362 -362
- package/src/organization-model/__tests__/domains/roles.test.ts +347 -347
- package/src/organization-model/__tests__/domains/statuses.test.ts +243 -243
- package/src/organization-model/__tests__/foundation.test.ts +105 -105
- package/src/organization-model/__tests__/graph.test.ts +894 -894
- package/src/organization-model/__tests__/resolve.test.ts +690 -690
- package/src/organization-model/__tests__/schema.test.ts +407 -407
- package/src/organization-model/contracts.ts +14 -14
- package/src/organization-model/defaults.ts +148 -148
- package/src/organization-model/domains/branding.ts +22 -22
- package/src/organization-model/domains/customers.ts +75 -75
- package/src/organization-model/domains/features.ts +22 -22
- package/src/organization-model/domains/goals.ts +80 -80
- package/src/organization-model/domains/identity.ts +94 -94
- package/src/organization-model/domains/navigation.ts +391 -391
- package/src/organization-model/domains/offerings.ts +66 -66
- package/src/organization-model/domains/operations.ts +85 -85
- package/src/organization-model/domains/projects.ts +48 -48
- package/src/organization-model/domains/prospecting.ts +33 -33
- package/src/organization-model/domains/roles.ts +55 -55
- package/src/organization-model/domains/sales.ts +94 -94
- package/src/organization-model/domains/shared.ts +62 -62
- package/src/organization-model/domains/statuses.ts +130 -130
- package/src/organization-model/foundation.ts +97 -97
- package/src/organization-model/graph/build.ts +399 -399
- package/src/organization-model/graph/index.ts +4 -4
- package/src/organization-model/graph/schema.ts +48 -48
- package/src/organization-model/graph/types.ts +40 -40
- package/src/organization-model/index.ts +13 -13
- package/src/organization-model/organization-graph.mdx +272 -272
- package/src/organization-model/organization-model.mdx +320 -320
- package/src/organization-model/published.ts +85 -85
- package/src/organization-model/resolve.ts +66 -66
- package/src/organization-model/schema.ts +287 -287
- package/src/organization-model/types.ts +46 -46
- package/src/platform/api/index.ts +1 -1
- package/src/platform/api/types.ts +35 -35
- package/src/platform/constants/http.ts +37 -37
- package/src/platform/constants/index.ts +5 -5
- package/src/platform/constants/limits.ts +32 -32
- package/src/platform/constants/resilience.ts +51 -51
- package/src/platform/constants/timeouts.ts +20 -20
- package/src/platform/constants/versions.ts +3 -3
- package/src/platform/registry/__tests__/resource-registry-static.test.ts +347 -347
- package/src/platform/registry/__tests__/resource-registry.integration.test.ts +1028 -1028
- package/src/platform/registry/__tests__/resource-registry.list-executable.test.ts +393 -393
- package/src/platform/registry/__tests__/resource-registry.test.ts +2005 -2005
- package/src/platform/registry/__tests__/serialization.test.ts +1127 -1127
- package/src/platform/registry/command-view.ts +180 -180
- package/src/platform/registry/domains.ts +165 -165
- package/src/platform/registry/index.ts +93 -93
- package/src/platform/registry/reserved.ts +24 -24
- package/src/platform/registry/resource-metadata.ts +59 -59
- package/src/platform/registry/resource-registry.command-queue-groups.test.ts +129 -129
- package/src/platform/registry/resource-registry.ts +876 -876
- package/src/platform/registry/serialization.ts +273 -273
- package/src/platform/registry/serialized-types.ts +231 -231
- package/src/platform/registry/stats-types.ts +66 -66
- package/src/platform/registry/types.ts +404 -404
- package/src/platform/registry/validation.ts +513 -513
- package/src/platform/resilience/__tests__/rate-limiter.test.ts +471 -471
- package/src/platform/resilience/circuit-breaker.ts +164 -164
- package/src/platform/resilience/errors.ts +68 -68
- package/src/platform/resilience/http-error-mapper.ts +129 -129
- package/src/platform/resilience/index.ts +93 -93
- package/src/platform/resilience/rate-limiter-types.ts +46 -46
- package/src/platform/resilience/rate-limiter.ts +140 -140
- package/src/platform/resilience/retry.ts +89 -89
- package/src/platform/resilience/timeout.ts +63 -63
- package/src/platform/sse/events.ts +37 -34
- package/src/platform/sse/index.ts +7 -7
- package/src/platform/utils/__tests__/validation.test.ts +1083 -1083
- package/src/platform/utils/currency.ts +96 -96
- package/src/platform/utils/debounce.ts +52 -52
- package/src/platform/utils/error.ts +41 -41
- package/src/platform/utils/hmac.test.ts +97 -97
- package/src/platform/utils/index.ts +32 -32
- package/src/platform/utils/server/betterstack-logger.ts +210 -210
- package/src/platform/utils/server/hmac.ts +44 -44
- package/src/platform/utils/server/unsubscribe.ts +111 -111
- package/src/platform/utils/token-counter.ts +96 -96
- package/src/platform/utils/validation.ts +425 -425
- package/src/projects/api-schemas.ts +268 -268
- package/src/published.ts +1 -1
- package/src/reference/_generated/contracts.md +611 -607
- package/src/reference/glossary.md +105 -105
- package/src/requests/__tests__/api-schemas.test.ts +277 -277
- package/src/requests/api-schemas.ts +83 -83
- package/src/requests/index.ts +1 -1
- package/src/scaffold-registry/__tests__/index.test.ts +17 -0
- package/src/scaffold-registry/__tests__/schema.test.ts +329 -230
- package/src/scaffold-registry/index.ts +205 -189
- package/src/scaffold-registry/schema.ts +196 -128
- package/src/server.ts +272 -272
- package/src/supabase/database.types.ts +2719 -2719
- package/src/supabase/helpers.ts +20 -20
- package/src/supabase/index.ts +52 -52
- package/src/supabase/server/client.ts +58 -58
- package/src/test-utils/README.md +38 -38
- package/src/test-utils/browser-mocks.ts +54 -54
- package/src/test-utils/fixtures/api-keys.ts +52 -52
- package/src/test-utils/fixtures/index.ts +4 -4
- package/src/test-utils/fixtures/memberships.ts +80 -80
- package/src/test-utils/fixtures/organizations.ts +69 -69
- package/src/test-utils/fixtures/users.ts +79 -79
- package/src/test-utils/index.ts +11 -11
- package/src/test-utils/mocks/index.ts +2 -2
- package/src/test-utils/mocks/supabase.ts +142 -142
- package/src/test-utils/mocks/workos.ts +108 -108
- package/src/test-utils/rls/RLSTestContext.ts +556 -556
- package/src/test-utils/rls/index.ts +1 -1
|
@@ -1,754 +1,754 @@
|
|
|
1
|
-
import { describe, it, expect, vi } from 'vitest'
|
|
2
|
-
import { MemoryManager, truncateToolResult } from '../manager'
|
|
3
|
-
import type { AgentMemory } from '../types'
|
|
4
|
-
import type { AgentScopedLogger } from '../../observability/logging'
|
|
5
|
-
|
|
6
|
-
describe('MemoryManager', () => {
|
|
7
|
-
const createMockLogger = (): AgentScopedLogger => ({
|
|
8
|
-
action: vi.fn(),
|
|
9
|
-
lifecycle: vi.fn(),
|
|
10
|
-
memory: vi.fn(),
|
|
11
|
-
tool: vi.fn(),
|
|
12
|
-
reasoning: vi.fn()
|
|
13
|
-
})
|
|
14
|
-
|
|
15
|
-
const createEmptyMemory = (): AgentMemory => ({
|
|
16
|
-
sessionMemory: {},
|
|
17
|
-
history: []
|
|
18
|
-
})
|
|
19
|
-
|
|
20
|
-
describe('set() - Agent operations', () => {
|
|
21
|
-
it('sets session memory entry', () => {
|
|
22
|
-
const memory = createEmptyMemory()
|
|
23
|
-
const manager = new MemoryManager(memory)
|
|
24
|
-
|
|
25
|
-
manager.set('task', 'Build user registration flow')
|
|
26
|
-
|
|
27
|
-
expect(memory.sessionMemory['task']).toBeDefined()
|
|
28
|
-
expect(memory.sessionMemory['task'].type).toBe('context')
|
|
29
|
-
expect(memory.sessionMemory['task'].content).toBe('Build user registration flow')
|
|
30
|
-
})
|
|
31
|
-
|
|
32
|
-
it('adds timestamp to entry', () => {
|
|
33
|
-
const memory = createEmptyMemory()
|
|
34
|
-
const manager = new MemoryManager(memory)
|
|
35
|
-
|
|
36
|
-
const beforeTimestamp = Date.now()
|
|
37
|
-
manager.set('note', 'Test note')
|
|
38
|
-
const afterTimestamp = Date.now()
|
|
39
|
-
|
|
40
|
-
expect(memory.sessionMemory['note'].timestamp).toBeGreaterThanOrEqual(beforeTimestamp)
|
|
41
|
-
expect(memory.sessionMemory['note'].timestamp).toBeLessThanOrEqual(afterTimestamp)
|
|
42
|
-
})
|
|
43
|
-
|
|
44
|
-
it('overwrites existing key with same name', () => {
|
|
45
|
-
const memory = createEmptyMemory()
|
|
46
|
-
const manager = new MemoryManager(memory)
|
|
47
|
-
|
|
48
|
-
manager.set('counter', '1')
|
|
49
|
-
manager.set('counter', '2')
|
|
50
|
-
|
|
51
|
-
expect(memory.sessionMemory['counter'].content).toBe('2')
|
|
52
|
-
expect(Object.keys(memory.sessionMemory)).toHaveLength(1)
|
|
53
|
-
})
|
|
54
|
-
|
|
55
|
-
it('truncates entry exceeding single entry token limit', () => {
|
|
56
|
-
const memory = createEmptyMemory()
|
|
57
|
-
const logger = createMockLogger()
|
|
58
|
-
const manager = new MemoryManager(memory, {}, logger)
|
|
59
|
-
|
|
60
|
-
// 2000 tokens * 4 chars/token = 8000 chars, create 10000 chars
|
|
61
|
-
const largeContent = 'a'.repeat(10000)
|
|
62
|
-
manager.set('large', largeContent)
|
|
63
|
-
|
|
64
|
-
expect(memory.sessionMemory['large'].content).toContain('[truncated]')
|
|
65
|
-
expect(memory.sessionMemory['large'].content.length).toBeLessThan(largeContent.length)
|
|
66
|
-
expect(logger.action).toHaveBeenCalledWith(
|
|
67
|
-
'memory-truncate',
|
|
68
|
-
expect.stringContaining('Single entry exceeds token limit'),
|
|
69
|
-
expect.anything(),
|
|
70
|
-
expect.anything(),
|
|
71
|
-
expect.anything(),
|
|
72
|
-
expect.anything()
|
|
73
|
-
)
|
|
74
|
-
})
|
|
75
|
-
|
|
76
|
-
it('handles multiple session memory keys', () => {
|
|
77
|
-
const memory = createEmptyMemory()
|
|
78
|
-
const manager = new MemoryManager(memory)
|
|
79
|
-
|
|
80
|
-
manager.set('task', 'Build feature')
|
|
81
|
-
manager.set('context', 'User story...')
|
|
82
|
-
manager.set('progress', 'Step 1 done')
|
|
83
|
-
|
|
84
|
-
expect(Object.keys(memory.sessionMemory)).toHaveLength(3)
|
|
85
|
-
expect(memory.sessionMemory['task']).toBeDefined()
|
|
86
|
-
expect(memory.sessionMemory['context']).toBeDefined()
|
|
87
|
-
expect(memory.sessionMemory['progress']).toBeDefined()
|
|
88
|
-
})
|
|
89
|
-
})
|
|
90
|
-
|
|
91
|
-
describe('delete() - Agent operations', () => {
|
|
92
|
-
it('deletes existing session memory entry', () => {
|
|
93
|
-
const memory = createEmptyMemory()
|
|
94
|
-
const manager = new MemoryManager(memory)
|
|
95
|
-
|
|
96
|
-
manager.set('temp', 'temporary data')
|
|
97
|
-
const result = manager.delete('temp')
|
|
98
|
-
|
|
99
|
-
expect(result).toBe(true)
|
|
100
|
-
expect(memory.sessionMemory['temp']).toBeUndefined()
|
|
101
|
-
})
|
|
102
|
-
|
|
103
|
-
it('returns false for non-existent key', () => {
|
|
104
|
-
const memory = createEmptyMemory()
|
|
105
|
-
const manager = new MemoryManager(memory)
|
|
106
|
-
|
|
107
|
-
const result = manager.delete('nonexistent')
|
|
108
|
-
|
|
109
|
-
expect(result).toBe(false)
|
|
110
|
-
})
|
|
111
|
-
|
|
112
|
-
it('does not affect other keys when deleting', () => {
|
|
113
|
-
const memory = createEmptyMemory()
|
|
114
|
-
const manager = new MemoryManager(memory)
|
|
115
|
-
|
|
116
|
-
manager.set('keep1', 'data1')
|
|
117
|
-
manager.set('delete-me', 'data2')
|
|
118
|
-
manager.set('keep2', 'data3')
|
|
119
|
-
|
|
120
|
-
manager.delete('delete-me')
|
|
121
|
-
|
|
122
|
-
expect(Object.keys(memory.sessionMemory)).toHaveLength(2)
|
|
123
|
-
expect(memory.sessionMemory['keep1']).toBeDefined()
|
|
124
|
-
expect(memory.sessionMemory['keep2']).toBeDefined()
|
|
125
|
-
})
|
|
126
|
-
})
|
|
127
|
-
|
|
128
|
-
describe('addToHistory() - Framework operations', () => {
|
|
129
|
-
it('adds entry to history with iteration number', () => {
|
|
130
|
-
const memory = createEmptyMemory()
|
|
131
|
-
const manager = new MemoryManager(memory)
|
|
132
|
-
|
|
133
|
-
const entry = {
|
|
134
|
-
type: 'reasoning' as const,
|
|
135
|
-
content: 'Thinking about the problem...',
|
|
136
|
-
timestamp: Date.now()
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
manager.addToHistory({ ...entry, turnNumber: null, iterationNumber: 1 })
|
|
140
|
-
|
|
141
|
-
expect(memory.history).toHaveLength(1)
|
|
142
|
-
expect(memory.history[0].type).toBe('reasoning')
|
|
143
|
-
expect(memory.history[0].content).toBe('Thinking about the problem...')
|
|
144
|
-
expect(memory.history[0].iterationNumber).toBe(1)
|
|
145
|
-
})
|
|
146
|
-
|
|
147
|
-
it('appends multiple entries in order', () => {
|
|
148
|
-
const memory = createEmptyMemory()
|
|
149
|
-
const manager = new MemoryManager(memory)
|
|
150
|
-
|
|
151
|
-
const entry1 = { type: 'input' as const, content: 'User request', timestamp: Date.now() }
|
|
152
|
-
const entry2 = { type: 'reasoning' as const, content: 'Agent thinking', timestamp: Date.now() }
|
|
153
|
-
const entry3 = { type: 'tool-result' as const, content: 'Tool output', timestamp: Date.now() }
|
|
154
|
-
|
|
155
|
-
manager.addToHistory({ ...entry1, turnNumber: null, iterationNumber: 0 })
|
|
156
|
-
manager.addToHistory({ ...entry2, turnNumber: null, iterationNumber: 1 })
|
|
157
|
-
manager.addToHistory({ ...entry3, turnNumber: null, iterationNumber: 1 })
|
|
158
|
-
|
|
159
|
-
expect(memory.history).toHaveLength(3)
|
|
160
|
-
expect(memory.history[0].type).toBe('input')
|
|
161
|
-
expect(memory.history[1].type).toBe('reasoning')
|
|
162
|
-
expect(memory.history[2].type).toBe('tool-result')
|
|
163
|
-
})
|
|
164
|
-
|
|
165
|
-
it('triggers auto-compact when at 100% token budget', () => {
|
|
166
|
-
const memory = createEmptyMemory()
|
|
167
|
-
const manager = new MemoryManager(memory, { maxMemoryTokens: 100 })
|
|
168
|
-
|
|
169
|
-
// Add first entry (will be preserved as anchor)
|
|
170
|
-
manager.addToHistory({ type: 'input', content: 'First', turnNumber: null, iterationNumber: 0 })
|
|
171
|
-
|
|
172
|
-
// Add many entries to exceed token budget
|
|
173
|
-
for (let i = 0; i < 50; i++) {
|
|
174
|
-
manager.addToHistory({
|
|
175
|
-
type: 'reasoning',
|
|
176
|
-
content: 'a'.repeat(100), // ~30 tokens each
|
|
177
|
-
turnNumber: null,
|
|
178
|
-
iterationNumber: 1
|
|
179
|
-
})
|
|
180
|
-
}
|
|
181
|
-
|
|
182
|
-
// Should be compacted to first + last 10
|
|
183
|
-
expect(memory.history.length).toBe(11)
|
|
184
|
-
expect(memory.history[0].content).toBe('First')
|
|
185
|
-
})
|
|
186
|
-
})
|
|
187
|
-
|
|
188
|
-
describe('autoCompact()', () => {
|
|
189
|
-
it('does not compact when below 100% token budget', () => {
|
|
190
|
-
const memory = createEmptyMemory()
|
|
191
|
-
const manager = new MemoryManager(memory, { maxMemoryTokens: 10000 })
|
|
192
|
-
|
|
193
|
-
// Add a few entries (well below budget)
|
|
194
|
-
for (let i = 0; i < 5; i++) {
|
|
195
|
-
manager.addToHistory({
|
|
196
|
-
type: 'reasoning',
|
|
197
|
-
content: 'Small entry',
|
|
198
|
-
turnNumber: null,
|
|
199
|
-
iterationNumber: 1
|
|
200
|
-
})
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
expect(memory.history).toHaveLength(5)
|
|
204
|
-
})
|
|
205
|
-
|
|
206
|
-
it('compacts when at exactly 100% token budget', () => {
|
|
207
|
-
const memory = createEmptyMemory()
|
|
208
|
-
const logger = createMockLogger()
|
|
209
|
-
const manager = new MemoryManager(memory, { maxMemoryTokens: 10 }, logger)
|
|
210
|
-
|
|
211
|
-
// Add enough entries to reach 100%
|
|
212
|
-
for (let i = 0; i < 20; i++) {
|
|
213
|
-
manager.addToHistory({
|
|
214
|
-
type: 'reasoning',
|
|
215
|
-
content: 'Entry that takes tokens',
|
|
216
|
-
turnNumber: null,
|
|
217
|
-
iterationNumber: 1
|
|
218
|
-
})
|
|
219
|
-
}
|
|
220
|
-
|
|
221
|
-
expect(memory.history.length).toBe(11) // First + last 10
|
|
222
|
-
expect(logger.action).toHaveBeenCalledWith(
|
|
223
|
-
'memory-auto-compact',
|
|
224
|
-
expect.stringContaining('Auto-compacted'),
|
|
225
|
-
expect.anything(),
|
|
226
|
-
expect.anything(),
|
|
227
|
-
expect.anything(),
|
|
228
|
-
expect.anything()
|
|
229
|
-
)
|
|
230
|
-
})
|
|
231
|
-
|
|
232
|
-
it('preserves first entry (anchor)', () => {
|
|
233
|
-
const memory = createEmptyMemory()
|
|
234
|
-
const manager = new MemoryManager(memory, { maxMemoryTokens: 10 })
|
|
235
|
-
|
|
236
|
-
const firstEntry = { type: 'input' as const, content: 'FIRST ENTRY', timestamp: Date.now() }
|
|
237
|
-
manager.addToHistory({ ...firstEntry, turnNumber: null, iterationNumber: 0 })
|
|
238
|
-
|
|
239
|
-
// Add many more entries
|
|
240
|
-
for (let i = 0; i < 50; i++) {
|
|
241
|
-
manager.addToHistory({
|
|
242
|
-
type: 'reasoning',
|
|
243
|
-
content: 'Later entry...',
|
|
244
|
-
turnNumber: null,
|
|
245
|
-
iterationNumber: 1
|
|
246
|
-
})
|
|
247
|
-
}
|
|
248
|
-
|
|
249
|
-
expect(memory.history[0].type).toBe(firstEntry.type)
|
|
250
|
-
expect(memory.history[0].content).toBe('FIRST ENTRY')
|
|
251
|
-
})
|
|
252
|
-
|
|
253
|
-
it('preserves last 10 entries', () => {
|
|
254
|
-
const memory = createEmptyMemory()
|
|
255
|
-
const manager = new MemoryManager(memory, { maxMemoryTokens: 10 })
|
|
256
|
-
|
|
257
|
-
manager.addToHistory({ type: 'input', content: 'First', turnNumber: null, iterationNumber: 0 })
|
|
258
|
-
|
|
259
|
-
const lastEntries: { type: 'reasoning'; content: string; timestamp: number }[] = []
|
|
260
|
-
// Add middle entries that will be removed
|
|
261
|
-
for (let i = 0; i < 20; i++) {
|
|
262
|
-
manager.addToHistory({ type: 'reasoning', content: `Middle ${i}`, turnNumber: null, iterationNumber: 1 })
|
|
263
|
-
}
|
|
264
|
-
|
|
265
|
-
// Add last 10 entries to preserve
|
|
266
|
-
for (let i = 0; i < 10; i++) {
|
|
267
|
-
const entry = { type: 'reasoning' as const, content: `Last ${i}`, timestamp: Date.now() }
|
|
268
|
-
lastEntries.push(entry)
|
|
269
|
-
manager.addToHistory({ ...entry, turnNumber: null, iterationNumber: 2 })
|
|
270
|
-
}
|
|
271
|
-
|
|
272
|
-
expect(memory.history).toHaveLength(11) // First + 10
|
|
273
|
-
// Verify last 10 are preserved
|
|
274
|
-
for (let i = 0; i < 10; i++) {
|
|
275
|
-
expect(memory.history[i + 1].content).toBe(lastEntries[i].content)
|
|
276
|
-
expect(memory.history[i + 1].type).toBe(lastEntries[i].type)
|
|
277
|
-
}
|
|
278
|
-
})
|
|
279
|
-
})
|
|
280
|
-
|
|
281
|
-
describe('enforceHardLimits()', () => {
|
|
282
|
-
it('removes oldest session memory entries when exceeding limit', () => {
|
|
283
|
-
const memory = createEmptyMemory()
|
|
284
|
-
const logger = createMockLogger()
|
|
285
|
-
const manager = new MemoryManager(memory, { maxSessionMemoryKeys: 3 }, logger)
|
|
286
|
-
|
|
287
|
-
// Add entries with specific timestamps
|
|
288
|
-
manager.set('key1', 'oldest')
|
|
289
|
-
manager.set('key2', 'middle')
|
|
290
|
-
manager.set('key3', 'recent')
|
|
291
|
-
manager.set('key4', 'newest')
|
|
292
|
-
|
|
293
|
-
manager.enforceHardLimits()
|
|
294
|
-
|
|
295
|
-
expect(Object.keys(memory.sessionMemory)).toHaveLength(3)
|
|
296
|
-
expect(memory.sessionMemory['key1']).toBeUndefined() // Oldest removed
|
|
297
|
-
expect(memory.sessionMemory['key2']).toBeDefined()
|
|
298
|
-
expect(memory.sessionMemory['key3']).toBeDefined()
|
|
299
|
-
expect(memory.sessionMemory['key4']).toBeDefined()
|
|
300
|
-
})
|
|
301
|
-
|
|
302
|
-
it('logs when session memory limit exceeded', () => {
|
|
303
|
-
const memory = createEmptyMemory()
|
|
304
|
-
const logger = createMockLogger()
|
|
305
|
-
const manager = new MemoryManager(memory, { maxSessionMemoryKeys: 2 }, logger)
|
|
306
|
-
|
|
307
|
-
manager.set('key1', 'data1')
|
|
308
|
-
manager.set('key2', 'data2')
|
|
309
|
-
manager.set('key3', 'data3')
|
|
310
|
-
|
|
311
|
-
manager.enforceHardLimits()
|
|
312
|
-
|
|
313
|
-
expect(logger.action).toHaveBeenCalledWith(
|
|
314
|
-
'memory-limit-exceeded',
|
|
315
|
-
expect.stringContaining('Session memory exceeds hard limit'),
|
|
316
|
-
expect.anything(),
|
|
317
|
-
expect.anything(),
|
|
318
|
-
expect.anything(),
|
|
319
|
-
expect.anything()
|
|
320
|
-
)
|
|
321
|
-
})
|
|
322
|
-
|
|
323
|
-
it('performs emergency compaction when exceeding total token budget', () => {
|
|
324
|
-
const memory = createEmptyMemory()
|
|
325
|
-
const logger = createMockLogger()
|
|
326
|
-
const manager = new MemoryManager(memory, { maxMemoryTokens: 20 }, logger)
|
|
327
|
-
|
|
328
|
-
// Add first entry (will be preserved)
|
|
329
|
-
manager.addToHistory({ type: 'input', content: 'First', turnNumber: null, iterationNumber: 0 })
|
|
330
|
-
|
|
331
|
-
// Add many entries to exceed budget
|
|
332
|
-
for (let i = 0; i < 100; i++) {
|
|
333
|
-
manager.addToHistory({
|
|
334
|
-
type: 'reasoning',
|
|
335
|
-
content: 'a'.repeat(200),
|
|
336
|
-
turnNumber: null,
|
|
337
|
-
iterationNumber: 1
|
|
338
|
-
})
|
|
339
|
-
}
|
|
340
|
-
|
|
341
|
-
manager.enforceHardLimits()
|
|
342
|
-
|
|
343
|
-
// Emergency compaction: first + last 5 = 6 total
|
|
344
|
-
expect(memory.history.length).toBe(6)
|
|
345
|
-
// Check that memory-emergency was called at some point
|
|
346
|
-
const emergencyCalls = (logger.action as unknown as { mock: { calls: unknown[][] } }).mock.calls.filter(
|
|
347
|
-
(call: unknown[]) => call[0] === 'memory-emergency'
|
|
348
|
-
)
|
|
349
|
-
expect(emergencyCalls.length).toBeGreaterThan(0)
|
|
350
|
-
expect(emergencyCalls[0][1]).toContain('Total memory exceeds token budget')
|
|
351
|
-
})
|
|
352
|
-
|
|
353
|
-
it('uses default hard limits when no constraints provided', () => {
|
|
354
|
-
const memory = createEmptyMemory()
|
|
355
|
-
const manager = new MemoryManager(memory)
|
|
356
|
-
|
|
357
|
-
// Default MAX_SESSION_MEMORY_KEYS is 25
|
|
358
|
-
for (let i = 0; i < 27; i++) {
|
|
359
|
-
manager.set(`key${i}`, `data${i}`)
|
|
360
|
-
}
|
|
361
|
-
|
|
362
|
-
manager.enforceHardLimits()
|
|
363
|
-
|
|
364
|
-
expect(Object.keys(memory.sessionMemory)).toHaveLength(25)
|
|
365
|
-
})
|
|
366
|
-
})
|
|
367
|
-
|
|
368
|
-
describe('getHistoryLength()', () => {
|
|
369
|
-
it('returns 0 for empty history', () => {
|
|
370
|
-
const memory = createEmptyMemory()
|
|
371
|
-
const manager = new MemoryManager(memory)
|
|
372
|
-
|
|
373
|
-
expect(manager.getHistoryLength()).toBe(0)
|
|
374
|
-
})
|
|
375
|
-
|
|
376
|
-
it('returns correct count after adding entries', () => {
|
|
377
|
-
const memory = createEmptyMemory()
|
|
378
|
-
const manager = new MemoryManager(memory)
|
|
379
|
-
|
|
380
|
-
manager.addToHistory({ type: 'input', content: 'Entry 1', turnNumber: null, iterationNumber: 0 })
|
|
381
|
-
expect(manager.getHistoryLength()).toBe(1)
|
|
382
|
-
|
|
383
|
-
manager.addToHistory({ type: 'reasoning', content: 'Entry 2', turnNumber: null, iterationNumber: 1 })
|
|
384
|
-
expect(manager.getHistoryLength()).toBe(2)
|
|
385
|
-
|
|
386
|
-
manager.addToHistory({ type: 'tool-result', content: 'Entry 3', turnNumber: null, iterationNumber: 1 })
|
|
387
|
-
expect(manager.getHistoryLength()).toBe(3)
|
|
388
|
-
})
|
|
389
|
-
})
|
|
390
|
-
|
|
391
|
-
describe('getStatus()', () => {
|
|
392
|
-
it('returns status with all fields', () => {
|
|
393
|
-
const memory = createEmptyMemory()
|
|
394
|
-
const manager = new MemoryManager(memory)
|
|
395
|
-
|
|
396
|
-
const status = manager.getStatus()
|
|
397
|
-
|
|
398
|
-
expect(status).toHaveProperty('sessionMemoryKeys')
|
|
399
|
-
expect(status).toHaveProperty('sessionMemoryLimit')
|
|
400
|
-
expect(status).toHaveProperty('currentKeys')
|
|
401
|
-
expect(status).toHaveProperty('historyPercent')
|
|
402
|
-
expect(status).toHaveProperty('historyTokens')
|
|
403
|
-
expect(status).toHaveProperty('tokenBudget')
|
|
404
|
-
})
|
|
405
|
-
|
|
406
|
-
it('calculates session memory key count correctly', () => {
|
|
407
|
-
const memory = createEmptyMemory()
|
|
408
|
-
const manager = new MemoryManager(memory)
|
|
409
|
-
|
|
410
|
-
manager.set('key1', 'data1')
|
|
411
|
-
manager.set('key2', 'data2')
|
|
412
|
-
manager.set('key3', 'data3')
|
|
413
|
-
|
|
414
|
-
const status = manager.getStatus()
|
|
415
|
-
|
|
416
|
-
expect(status.sessionMemoryKeys).toBe(3)
|
|
417
|
-
expect(status.currentKeys).toEqual(['key1', 'key2', 'key3'])
|
|
418
|
-
})
|
|
419
|
-
|
|
420
|
-
it('calculates token usage percentage', () => {
|
|
421
|
-
const memory = createEmptyMemory()
|
|
422
|
-
const manager = new MemoryManager(memory, { maxMemoryTokens: 100 })
|
|
423
|
-
|
|
424
|
-
// Add entry that takes ~50 tokens (50 chars * 4 = ~12 tokens * 1.2 = ~15 tokens)
|
|
425
|
-
manager.addToHistory({
|
|
426
|
-
type: 'input',
|
|
427
|
-
content: 'a'.repeat(200), // ~60 tokens
|
|
428
|
-
turnNumber: null,
|
|
429
|
-
iterationNumber: 0
|
|
430
|
-
})
|
|
431
|
-
|
|
432
|
-
const status = manager.getStatus()
|
|
433
|
-
|
|
434
|
-
expect(status.historyPercent).toBeGreaterThan(0)
|
|
435
|
-
expect(status.historyPercent).toBeLessThanOrEqual(100)
|
|
436
|
-
expect(status.tokenBudget).toBe(100)
|
|
437
|
-
})
|
|
438
|
-
|
|
439
|
-
it('includes session memory and history tokens in total', () => {
|
|
440
|
-
const memory = createEmptyMemory()
|
|
441
|
-
const manager = new MemoryManager(memory, { maxMemoryTokens: 1000 })
|
|
442
|
-
|
|
443
|
-
manager.set('persistent1', 'a'.repeat(100))
|
|
444
|
-
manager.addToHistory({ type: 'input', content: 'a'.repeat(100), turnNumber: null, iterationNumber: 0 })
|
|
445
|
-
|
|
446
|
-
const status = manager.getStatus()
|
|
447
|
-
|
|
448
|
-
expect(status.historyTokens).toBeGreaterThan(0)
|
|
449
|
-
})
|
|
450
|
-
})
|
|
451
|
-
|
|
452
|
-
describe('toSnapshot() and getSnapshot()', () => {
|
|
453
|
-
it('creates deep copy of memory', () => {
|
|
454
|
-
const memory = createEmptyMemory()
|
|
455
|
-
const manager = new MemoryManager(memory)
|
|
456
|
-
|
|
457
|
-
manager.set('key', 'value')
|
|
458
|
-
manager.addToHistory({ type: 'input', content: 'test', turnNumber: null, iterationNumber: 0 })
|
|
459
|
-
|
|
460
|
-
const snapshot = manager.toSnapshot()
|
|
461
|
-
|
|
462
|
-
// Modify original
|
|
463
|
-
manager.set('key', 'modified')
|
|
464
|
-
manager.addToHistory({ type: 'reasoning', content: 'new', turnNumber: null, iterationNumber: 1 })
|
|
465
|
-
|
|
466
|
-
// Snapshot should not be affected
|
|
467
|
-
expect(snapshot.sessionMemory['key'].content).toBe('value')
|
|
468
|
-
expect(snapshot.history).toHaveLength(1)
|
|
469
|
-
})
|
|
470
|
-
|
|
471
|
-
it('caches snapshot for later retrieval', () => {
|
|
472
|
-
const memory = createEmptyMemory()
|
|
473
|
-
const manager = new MemoryManager(memory)
|
|
474
|
-
|
|
475
|
-
manager.set('data', 'test')
|
|
476
|
-
|
|
477
|
-
const snapshot1 = manager.toSnapshot()
|
|
478
|
-
const snapshot2 = manager.getSnapshot()
|
|
479
|
-
|
|
480
|
-
expect(snapshot2).toBe(snapshot1)
|
|
481
|
-
})
|
|
482
|
-
|
|
483
|
-
it('returns undefined if toSnapshot not called yet', () => {
|
|
484
|
-
const memory = createEmptyMemory()
|
|
485
|
-
const manager = new MemoryManager(memory)
|
|
486
|
-
|
|
487
|
-
expect(manager.getSnapshot()).toBeUndefined()
|
|
488
|
-
})
|
|
489
|
-
})
|
|
490
|
-
|
|
491
|
-
describe('toContext()', () => {
|
|
492
|
-
it('generates formatted context string with iteration filtering', () => {
|
|
493
|
-
const memory = createEmptyMemory()
|
|
494
|
-
const manager = new MemoryManager(memory)
|
|
495
|
-
|
|
496
|
-
const context = manager.toContext(1)
|
|
497
|
-
|
|
498
|
-
expect(context).toContain('MEMORY STATUS')
|
|
499
|
-
expect(context).toContain('SESSION MEMORY')
|
|
500
|
-
expect(context).toContain('ITERATION 1 - CURRENT CONTEXT')
|
|
501
|
-
expect(context).toContain('EARLIER CONTEXT')
|
|
502
|
-
})
|
|
503
|
-
|
|
504
|
-
it('includes session memory entries with keys', () => {
|
|
505
|
-
const memory = createEmptyMemory()
|
|
506
|
-
const manager = new MemoryManager(memory)
|
|
507
|
-
|
|
508
|
-
manager.set('task', 'Build feature X')
|
|
509
|
-
manager.set('context', 'User requested...')
|
|
510
|
-
|
|
511
|
-
const context = manager.toContext(1)
|
|
512
|
-
|
|
513
|
-
expect(context).toContain('[SESSION:task]')
|
|
514
|
-
expect(context).toContain('Build feature X')
|
|
515
|
-
expect(context).toContain('[SESSION:context]')
|
|
516
|
-
expect(context).toContain('User requested...')
|
|
517
|
-
})
|
|
518
|
-
|
|
519
|
-
it('filters history by iteration - current context reversed', () => {
|
|
520
|
-
const memory = createEmptyMemory()
|
|
521
|
-
const manager = new MemoryManager(memory)
|
|
522
|
-
|
|
523
|
-
manager.addToHistory({ type: 'input', content: 'User input', turnNumber: null, iterationNumber: 0 })
|
|
524
|
-
manager.addToHistory({ type: 'reasoning', content: 'First reasoning', turnNumber: null, iterationNumber: 1 })
|
|
525
|
-
manager.addToHistory({ type: 'tool-result', content: 'Tool result', turnNumber: null, iterationNumber: 1 })
|
|
526
|
-
|
|
527
|
-
const context = manager.toContext(1)
|
|
528
|
-
|
|
529
|
-
// Current iteration (1) entries should appear reversed (tool result first)
|
|
530
|
-
const currentSection = context
|
|
531
|
-
.split('=== ITERATION 1 - CURRENT CONTEXT ===')[1]
|
|
532
|
-
.split('=== EARLIER CONTEXT ===')[0]
|
|
533
|
-
expect(currentSection.indexOf('TOOL-RESULT')).toBeLessThan(currentSection.indexOf('REASONING'))
|
|
534
|
-
|
|
535
|
-
// Earlier context (iteration 0) should appear after
|
|
536
|
-
const earlierSection = context.split('=== EARLIER CONTEXT ===')[1]
|
|
537
|
-
expect(earlierSection).toContain('[INPUT]')
|
|
538
|
-
expect(earlierSection).toContain('User input')
|
|
539
|
-
})
|
|
540
|
-
|
|
541
|
-
it('shows empty message when no session memory entries', () => {
|
|
542
|
-
const memory = createEmptyMemory()
|
|
543
|
-
const manager = new MemoryManager(memory)
|
|
544
|
-
|
|
545
|
-
const context = manager.toContext(1)
|
|
546
|
-
|
|
547
|
-
expect(context).toContain('(empty)')
|
|
548
|
-
})
|
|
549
|
-
|
|
550
|
-
it('shows no earlier context message for iteration 0', () => {
|
|
551
|
-
const memory = createEmptyMemory()
|
|
552
|
-
const manager = new MemoryManager(memory)
|
|
553
|
-
|
|
554
|
-
manager.addToHistory({ type: 'input', content: 'test', turnNumber: null, iterationNumber: 0 })
|
|
555
|
-
|
|
556
|
-
const context = manager.toContext(0)
|
|
557
|
-
|
|
558
|
-
expect(context).toContain('(no earlier context)')
|
|
559
|
-
})
|
|
560
|
-
|
|
561
|
-
it('handles turn-scoped iteration filtering correctly', () => {
|
|
562
|
-
const memory = createEmptyMemory()
|
|
563
|
-
const manager = new MemoryManager(memory)
|
|
564
|
-
|
|
565
|
-
// Turn 1 entries
|
|
566
|
-
manager.addToHistory({ type: 'input', turnNumber: 1, iterationNumber: 0, content: 'Turn 1 input' })
|
|
567
|
-
manager.addToHistory({ type: 'reasoning', turnNumber: 1, iterationNumber: 1, content: 'Turn 1 reasoning' })
|
|
568
|
-
|
|
569
|
-
// Turn 2 entries
|
|
570
|
-
manager.addToHistory({ type: 'input', turnNumber: 2, iterationNumber: 0, content: 'Turn 2 input' })
|
|
571
|
-
manager.addToHistory({ type: 'reasoning', turnNumber: 2, iterationNumber: 1, content: 'Turn 2 reasoning' })
|
|
572
|
-
|
|
573
|
-
// Turn 2, Iteration 1 - should ONLY see Turn 2's context
|
|
574
|
-
const context = manager.toContext(1, 2)
|
|
575
|
-
|
|
576
|
-
// Assert: Current context contains ONLY Turn 2's iteration 1
|
|
577
|
-
expect(context).toContain('Turn 2 reasoning')
|
|
578
|
-
expect(context).not.toContain('Turn 1 reasoning')
|
|
579
|
-
|
|
580
|
-
// Assert: Earlier context contains ONLY Turn 2's iteration 0
|
|
581
|
-
expect(context).toContain('Turn 2 input')
|
|
582
|
-
expect(context).not.toContain('Turn 1 input')
|
|
583
|
-
})
|
|
584
|
-
|
|
585
|
-
it('handles one-off executions without turnNumber', () => {
|
|
586
|
-
const memory = createEmptyMemory()
|
|
587
|
-
const manager = new MemoryManager(memory)
|
|
588
|
-
|
|
589
|
-
manager.addToHistory({ type: 'input', turnNumber: null, iterationNumber: 0, content: 'Input' })
|
|
590
|
-
manager.addToHistory({ type: 'reasoning', turnNumber: null, iterationNumber: 1, content: 'Reasoning' })
|
|
591
|
-
|
|
592
|
-
// One-off execution - toContext() called without currentTurn parameter
|
|
593
|
-
const context = manager.toContext(1)
|
|
594
|
-
|
|
595
|
-
// Should work as before (no turn filtering)
|
|
596
|
-
expect(context).toContain('Reasoning')
|
|
597
|
-
expect(context).toContain('Input')
|
|
598
|
-
})
|
|
599
|
-
|
|
600
|
-
it('handles legacy entries without turnNumber gracefully', () => {
|
|
601
|
-
const memory: AgentMemory = {
|
|
602
|
-
sessionMemory: {},
|
|
603
|
-
history: [
|
|
604
|
-
// Legacy entries (pre-migration) without turnNumber
|
|
605
|
-
{
|
|
606
|
-
type: 'input',
|
|
607
|
-
content: 'Legacy input',
|
|
608
|
-
timestamp: 1,
|
|
609
|
-
turnNumber: undefined as unknown as null,
|
|
610
|
-
iterationNumber: 0
|
|
611
|
-
},
|
|
612
|
-
{
|
|
613
|
-
type: 'reasoning',
|
|
614
|
-
content: 'Legacy reasoning',
|
|
615
|
-
timestamp: 2,
|
|
616
|
-
turnNumber: undefined as unknown as null,
|
|
617
|
-
iterationNumber: 1
|
|
618
|
-
},
|
|
619
|
-
// New entries with turnNumber
|
|
620
|
-
{ type: 'input', content: 'New input', timestamp: 3, turnNumber: 2, iterationNumber: 0 },
|
|
621
|
-
{ type: 'reasoning', content: 'New reasoning', timestamp: 4, turnNumber: 2, iterationNumber: 1 }
|
|
622
|
-
]
|
|
623
|
-
}
|
|
624
|
-
|
|
625
|
-
const manager = new MemoryManager(memory)
|
|
626
|
-
|
|
627
|
-
// Turn 2, Iteration 1 - should see both legacy (undefined treated as valid) and Turn 2 entries
|
|
628
|
-
const context = manager.toContext(1, 2)
|
|
629
|
-
|
|
630
|
-
// Should include Turn 2's entries
|
|
631
|
-
expect(context).toContain('New reasoning')
|
|
632
|
-
|
|
633
|
-
// Should also include legacy entries (backward compatibility)
|
|
634
|
-
expect(context).toContain('Legacy reasoning')
|
|
635
|
-
})
|
|
636
|
-
})
|
|
637
|
-
})
|
|
638
|
-
|
|
639
|
-
describe('truncateToolResult()', () => {
|
|
640
|
-
it('returns content unchanged when under the token limit', () => {
|
|
641
|
-
const content = 'short result'
|
|
642
|
-
const result = truncateToolResult(content, 4_000)
|
|
643
|
-
expect(result).toBe(content)
|
|
644
|
-
})
|
|
645
|
-
|
|
646
|
-
it('truncates content that exceeds the token limit', () => {
|
|
647
|
-
// 4000 tokens * 3.5 chars/token = 14000 chars; create 20000 chars to exceed
|
|
648
|
-
const content = 'x'.repeat(20_000)
|
|
649
|
-
const result = truncateToolResult(content, 4_000)
|
|
650
|
-
|
|
651
|
-
expect(result.length).toBeLessThan(content.length)
|
|
652
|
-
expect(result).toContain('[Response truncated')
|
|
653
|
-
expect(result).toContain('tokens omitted')
|
|
654
|
-
})
|
|
655
|
-
|
|
656
|
-
it('includes the number of omitted tokens in the truncation notice', () => {
|
|
657
|
-
const content = 'a'.repeat(21_000) // ~6000 tokens at 3.5 chars/token
|
|
658
|
-
const result = truncateToolResult(content, 4_000)
|
|
659
|
-
|
|
660
|
-
// Should mention roughly 2000 omitted tokens
|
|
661
|
-
expect(result).toContain('tokens omitted')
|
|
662
|
-
})
|
|
663
|
-
|
|
664
|
-
it('returns content unchanged when exactly at the limit', () => {
|
|
665
|
-
// 4000 tokens * 3.5 = 14000 chars exactly
|
|
666
|
-
const content = 'b'.repeat(14_000)
|
|
667
|
-
const result = truncateToolResult(content, 4_000)
|
|
668
|
-
expect(result).toBe(content)
|
|
669
|
-
})
|
|
670
|
-
})
|
|
671
|
-
|
|
672
|
-
describe('MemoryManager - tool result truncation in addToHistory', () => {
|
|
673
|
-
const createMockLogger = (): AgentScopedLogger =>
|
|
674
|
-
({
|
|
675
|
-
action: vi.fn(),
|
|
676
|
-
lifecycle: vi.fn(),
|
|
677
|
-
memory: vi.fn(),
|
|
678
|
-
tool: vi.fn(),
|
|
679
|
-
reasoning: vi.fn()
|
|
680
|
-
}) as unknown as AgentScopedLogger
|
|
681
|
-
|
|
682
|
-
const createEmptyMemory = (): AgentMemory => ({
|
|
683
|
-
sessionMemory: {},
|
|
684
|
-
history: []
|
|
685
|
-
})
|
|
686
|
-
|
|
687
|
-
it('truncates tool-result entries exceeding MAX_TOOL_RESULT_TOKENS', () => {
|
|
688
|
-
const memory = createEmptyMemory()
|
|
689
|
-
const logger = createMockLogger()
|
|
690
|
-
const manager = new MemoryManager(memory, {}, logger)
|
|
691
|
-
|
|
692
|
-
// 4000 tokens * 3.5 = 14000 chars; create 28000 chars (~8000 tokens)
|
|
693
|
-
const largeContent = 'z'.repeat(28_000)
|
|
694
|
-
|
|
695
|
-
manager.addToHistory({
|
|
696
|
-
type: 'tool-result',
|
|
697
|
-
content: largeContent,
|
|
698
|
-
turnNumber: null,
|
|
699
|
-
iterationNumber: 1
|
|
700
|
-
})
|
|
701
|
-
|
|
702
|
-
expect(memory.history).toHaveLength(1)
|
|
703
|
-
expect(memory.history[0].content.length).toBeLessThan(largeContent.length)
|
|
704
|
-
expect(memory.history[0].content).toContain('[Response truncated')
|
|
705
|
-
expect(logger.action).toHaveBeenCalledWith(
|
|
706
|
-
'memory-tool-result-truncate',
|
|
707
|
-
expect.stringContaining('Tool result truncated'),
|
|
708
|
-
expect.anything(),
|
|
709
|
-
expect.anything(),
|
|
710
|
-
expect.anything(),
|
|
711
|
-
expect.anything()
|
|
712
|
-
)
|
|
713
|
-
})
|
|
714
|
-
|
|
715
|
-
it('does not truncate tool-result entries under the limit', () => {
|
|
716
|
-
const memory = createEmptyMemory()
|
|
717
|
-
const manager = new MemoryManager(memory)
|
|
718
|
-
|
|
719
|
-
const smallContent = '{"status": "ok"}'
|
|
720
|
-
manager.addToHistory({
|
|
721
|
-
type: 'tool-result',
|
|
722
|
-
content: smallContent,
|
|
723
|
-
turnNumber: null,
|
|
724
|
-
iterationNumber: 1
|
|
725
|
-
})
|
|
726
|
-
|
|
727
|
-
expect(memory.history[0].content).toBe(smallContent)
|
|
728
|
-
})
|
|
729
|
-
|
|
730
|
-
it('does not truncate non-tool-result entries regardless of size', () => {
|
|
731
|
-
const memory = createEmptyMemory()
|
|
732
|
-
const manager = new MemoryManager(memory, { maxMemoryTokens: 100_000 })
|
|
733
|
-
|
|
734
|
-
const largeReasoning = 'r'.repeat(28_000)
|
|
735
|
-
manager.addToHistory({
|
|
736
|
-
type: 'reasoning',
|
|
737
|
-
content: largeReasoning,
|
|
738
|
-
turnNumber: null,
|
|
739
|
-
iterationNumber: 1
|
|
740
|
-
})
|
|
741
|
-
|
|
742
|
-
expect(memory.history[0].content).toBe(largeReasoning)
|
|
743
|
-
|
|
744
|
-
const largeInput = 'i'.repeat(28_000)
|
|
745
|
-
manager.addToHistory({
|
|
746
|
-
type: 'input',
|
|
747
|
-
content: largeInput,
|
|
748
|
-
turnNumber: null,
|
|
749
|
-
iterationNumber: 0
|
|
750
|
-
})
|
|
751
|
-
|
|
752
|
-
expect(memory.history[1].content).toBe(largeInput)
|
|
753
|
-
})
|
|
754
|
-
})
|
|
1
|
+
import { describe, it, expect, vi } from 'vitest'
|
|
2
|
+
import { MemoryManager, truncateToolResult } from '../manager'
|
|
3
|
+
import type { AgentMemory } from '../types'
|
|
4
|
+
import type { AgentScopedLogger } from '../../observability/logging'
|
|
5
|
+
|
|
6
|
+
describe('MemoryManager', () => {
|
|
7
|
+
const createMockLogger = (): AgentScopedLogger => ({
|
|
8
|
+
action: vi.fn(),
|
|
9
|
+
lifecycle: vi.fn(),
|
|
10
|
+
memory: vi.fn(),
|
|
11
|
+
tool: vi.fn(),
|
|
12
|
+
reasoning: vi.fn()
|
|
13
|
+
})
|
|
14
|
+
|
|
15
|
+
const createEmptyMemory = (): AgentMemory => ({
|
|
16
|
+
sessionMemory: {},
|
|
17
|
+
history: []
|
|
18
|
+
})
|
|
19
|
+
|
|
20
|
+
describe('set() - Agent operations', () => {
|
|
21
|
+
it('sets session memory entry', () => {
|
|
22
|
+
const memory = createEmptyMemory()
|
|
23
|
+
const manager = new MemoryManager(memory)
|
|
24
|
+
|
|
25
|
+
manager.set('task', 'Build user registration flow')
|
|
26
|
+
|
|
27
|
+
expect(memory.sessionMemory['task']).toBeDefined()
|
|
28
|
+
expect(memory.sessionMemory['task'].type).toBe('context')
|
|
29
|
+
expect(memory.sessionMemory['task'].content).toBe('Build user registration flow')
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
it('adds timestamp to entry', () => {
|
|
33
|
+
const memory = createEmptyMemory()
|
|
34
|
+
const manager = new MemoryManager(memory)
|
|
35
|
+
|
|
36
|
+
const beforeTimestamp = Date.now()
|
|
37
|
+
manager.set('note', 'Test note')
|
|
38
|
+
const afterTimestamp = Date.now()
|
|
39
|
+
|
|
40
|
+
expect(memory.sessionMemory['note'].timestamp).toBeGreaterThanOrEqual(beforeTimestamp)
|
|
41
|
+
expect(memory.sessionMemory['note'].timestamp).toBeLessThanOrEqual(afterTimestamp)
|
|
42
|
+
})
|
|
43
|
+
|
|
44
|
+
it('overwrites existing key with same name', () => {
|
|
45
|
+
const memory = createEmptyMemory()
|
|
46
|
+
const manager = new MemoryManager(memory)
|
|
47
|
+
|
|
48
|
+
manager.set('counter', '1')
|
|
49
|
+
manager.set('counter', '2')
|
|
50
|
+
|
|
51
|
+
expect(memory.sessionMemory['counter'].content).toBe('2')
|
|
52
|
+
expect(Object.keys(memory.sessionMemory)).toHaveLength(1)
|
|
53
|
+
})
|
|
54
|
+
|
|
55
|
+
it('truncates entry exceeding single entry token limit', () => {
|
|
56
|
+
const memory = createEmptyMemory()
|
|
57
|
+
const logger = createMockLogger()
|
|
58
|
+
const manager = new MemoryManager(memory, {}, logger)
|
|
59
|
+
|
|
60
|
+
// 2000 tokens * 4 chars/token = 8000 chars, create 10000 chars
|
|
61
|
+
const largeContent = 'a'.repeat(10000)
|
|
62
|
+
manager.set('large', largeContent)
|
|
63
|
+
|
|
64
|
+
expect(memory.sessionMemory['large'].content).toContain('[truncated]')
|
|
65
|
+
expect(memory.sessionMemory['large'].content.length).toBeLessThan(largeContent.length)
|
|
66
|
+
expect(logger.action).toHaveBeenCalledWith(
|
|
67
|
+
'memory-truncate',
|
|
68
|
+
expect.stringContaining('Single entry exceeds token limit'),
|
|
69
|
+
expect.anything(),
|
|
70
|
+
expect.anything(),
|
|
71
|
+
expect.anything(),
|
|
72
|
+
expect.anything()
|
|
73
|
+
)
|
|
74
|
+
})
|
|
75
|
+
|
|
76
|
+
it('handles multiple session memory keys', () => {
|
|
77
|
+
const memory = createEmptyMemory()
|
|
78
|
+
const manager = new MemoryManager(memory)
|
|
79
|
+
|
|
80
|
+
manager.set('task', 'Build feature')
|
|
81
|
+
manager.set('context', 'User story...')
|
|
82
|
+
manager.set('progress', 'Step 1 done')
|
|
83
|
+
|
|
84
|
+
expect(Object.keys(memory.sessionMemory)).toHaveLength(3)
|
|
85
|
+
expect(memory.sessionMemory['task']).toBeDefined()
|
|
86
|
+
expect(memory.sessionMemory['context']).toBeDefined()
|
|
87
|
+
expect(memory.sessionMemory['progress']).toBeDefined()
|
|
88
|
+
})
|
|
89
|
+
})
|
|
90
|
+
|
|
91
|
+
describe('delete() - Agent operations', () => {
|
|
92
|
+
it('deletes existing session memory entry', () => {
|
|
93
|
+
const memory = createEmptyMemory()
|
|
94
|
+
const manager = new MemoryManager(memory)
|
|
95
|
+
|
|
96
|
+
manager.set('temp', 'temporary data')
|
|
97
|
+
const result = manager.delete('temp')
|
|
98
|
+
|
|
99
|
+
expect(result).toBe(true)
|
|
100
|
+
expect(memory.sessionMemory['temp']).toBeUndefined()
|
|
101
|
+
})
|
|
102
|
+
|
|
103
|
+
it('returns false for non-existent key', () => {
|
|
104
|
+
const memory = createEmptyMemory()
|
|
105
|
+
const manager = new MemoryManager(memory)
|
|
106
|
+
|
|
107
|
+
const result = manager.delete('nonexistent')
|
|
108
|
+
|
|
109
|
+
expect(result).toBe(false)
|
|
110
|
+
})
|
|
111
|
+
|
|
112
|
+
it('does not affect other keys when deleting', () => {
|
|
113
|
+
const memory = createEmptyMemory()
|
|
114
|
+
const manager = new MemoryManager(memory)
|
|
115
|
+
|
|
116
|
+
manager.set('keep1', 'data1')
|
|
117
|
+
manager.set('delete-me', 'data2')
|
|
118
|
+
manager.set('keep2', 'data3')
|
|
119
|
+
|
|
120
|
+
manager.delete('delete-me')
|
|
121
|
+
|
|
122
|
+
expect(Object.keys(memory.sessionMemory)).toHaveLength(2)
|
|
123
|
+
expect(memory.sessionMemory['keep1']).toBeDefined()
|
|
124
|
+
expect(memory.sessionMemory['keep2']).toBeDefined()
|
|
125
|
+
})
|
|
126
|
+
})
|
|
127
|
+
|
|
128
|
+
describe('addToHistory() - Framework operations', () => {
|
|
129
|
+
it('adds entry to history with iteration number', () => {
|
|
130
|
+
const memory = createEmptyMemory()
|
|
131
|
+
const manager = new MemoryManager(memory)
|
|
132
|
+
|
|
133
|
+
const entry = {
|
|
134
|
+
type: 'reasoning' as const,
|
|
135
|
+
content: 'Thinking about the problem...',
|
|
136
|
+
timestamp: Date.now()
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
manager.addToHistory({ ...entry, turnNumber: null, iterationNumber: 1 })
|
|
140
|
+
|
|
141
|
+
expect(memory.history).toHaveLength(1)
|
|
142
|
+
expect(memory.history[0].type).toBe('reasoning')
|
|
143
|
+
expect(memory.history[0].content).toBe('Thinking about the problem...')
|
|
144
|
+
expect(memory.history[0].iterationNumber).toBe(1)
|
|
145
|
+
})
|
|
146
|
+
|
|
147
|
+
it('appends multiple entries in order', () => {
|
|
148
|
+
const memory = createEmptyMemory()
|
|
149
|
+
const manager = new MemoryManager(memory)
|
|
150
|
+
|
|
151
|
+
const entry1 = { type: 'input' as const, content: 'User request', timestamp: Date.now() }
|
|
152
|
+
const entry2 = { type: 'reasoning' as const, content: 'Agent thinking', timestamp: Date.now() }
|
|
153
|
+
const entry3 = { type: 'tool-result' as const, content: 'Tool output', timestamp: Date.now() }
|
|
154
|
+
|
|
155
|
+
manager.addToHistory({ ...entry1, turnNumber: null, iterationNumber: 0 })
|
|
156
|
+
manager.addToHistory({ ...entry2, turnNumber: null, iterationNumber: 1 })
|
|
157
|
+
manager.addToHistory({ ...entry3, turnNumber: null, iterationNumber: 1 })
|
|
158
|
+
|
|
159
|
+
expect(memory.history).toHaveLength(3)
|
|
160
|
+
expect(memory.history[0].type).toBe('input')
|
|
161
|
+
expect(memory.history[1].type).toBe('reasoning')
|
|
162
|
+
expect(memory.history[2].type).toBe('tool-result')
|
|
163
|
+
})
|
|
164
|
+
|
|
165
|
+
it('triggers auto-compact when at 100% token budget', () => {
|
|
166
|
+
const memory = createEmptyMemory()
|
|
167
|
+
const manager = new MemoryManager(memory, { maxMemoryTokens: 100 })
|
|
168
|
+
|
|
169
|
+
// Add first entry (will be preserved as anchor)
|
|
170
|
+
manager.addToHistory({ type: 'input', content: 'First', turnNumber: null, iterationNumber: 0 })
|
|
171
|
+
|
|
172
|
+
// Add many entries to exceed token budget
|
|
173
|
+
for (let i = 0; i < 50; i++) {
|
|
174
|
+
manager.addToHistory({
|
|
175
|
+
type: 'reasoning',
|
|
176
|
+
content: 'a'.repeat(100), // ~30 tokens each
|
|
177
|
+
turnNumber: null,
|
|
178
|
+
iterationNumber: 1
|
|
179
|
+
})
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
// Should be compacted to first + last 10
|
|
183
|
+
expect(memory.history.length).toBe(11)
|
|
184
|
+
expect(memory.history[0].content).toBe('First')
|
|
185
|
+
})
|
|
186
|
+
})
|
|
187
|
+
|
|
188
|
+
describe('autoCompact()', () => {
|
|
189
|
+
it('does not compact when below 100% token budget', () => {
|
|
190
|
+
const memory = createEmptyMemory()
|
|
191
|
+
const manager = new MemoryManager(memory, { maxMemoryTokens: 10000 })
|
|
192
|
+
|
|
193
|
+
// Add a few entries (well below budget)
|
|
194
|
+
for (let i = 0; i < 5; i++) {
|
|
195
|
+
manager.addToHistory({
|
|
196
|
+
type: 'reasoning',
|
|
197
|
+
content: 'Small entry',
|
|
198
|
+
turnNumber: null,
|
|
199
|
+
iterationNumber: 1
|
|
200
|
+
})
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
expect(memory.history).toHaveLength(5)
|
|
204
|
+
})
|
|
205
|
+
|
|
206
|
+
it('compacts when at exactly 100% token budget', () => {
|
|
207
|
+
const memory = createEmptyMemory()
|
|
208
|
+
const logger = createMockLogger()
|
|
209
|
+
const manager = new MemoryManager(memory, { maxMemoryTokens: 10 }, logger)
|
|
210
|
+
|
|
211
|
+
// Add enough entries to reach 100%
|
|
212
|
+
for (let i = 0; i < 20; i++) {
|
|
213
|
+
manager.addToHistory({
|
|
214
|
+
type: 'reasoning',
|
|
215
|
+
content: 'Entry that takes tokens',
|
|
216
|
+
turnNumber: null,
|
|
217
|
+
iterationNumber: 1
|
|
218
|
+
})
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
expect(memory.history.length).toBe(11) // First + last 10
|
|
222
|
+
expect(logger.action).toHaveBeenCalledWith(
|
|
223
|
+
'memory-auto-compact',
|
|
224
|
+
expect.stringContaining('Auto-compacted'),
|
|
225
|
+
expect.anything(),
|
|
226
|
+
expect.anything(),
|
|
227
|
+
expect.anything(),
|
|
228
|
+
expect.anything()
|
|
229
|
+
)
|
|
230
|
+
})
|
|
231
|
+
|
|
232
|
+
it('preserves first entry (anchor)', () => {
|
|
233
|
+
const memory = createEmptyMemory()
|
|
234
|
+
const manager = new MemoryManager(memory, { maxMemoryTokens: 10 })
|
|
235
|
+
|
|
236
|
+
const firstEntry = { type: 'input' as const, content: 'FIRST ENTRY', timestamp: Date.now() }
|
|
237
|
+
manager.addToHistory({ ...firstEntry, turnNumber: null, iterationNumber: 0 })
|
|
238
|
+
|
|
239
|
+
// Add many more entries
|
|
240
|
+
for (let i = 0; i < 50; i++) {
|
|
241
|
+
manager.addToHistory({
|
|
242
|
+
type: 'reasoning',
|
|
243
|
+
content: 'Later entry...',
|
|
244
|
+
turnNumber: null,
|
|
245
|
+
iterationNumber: 1
|
|
246
|
+
})
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
expect(memory.history[0].type).toBe(firstEntry.type)
|
|
250
|
+
expect(memory.history[0].content).toBe('FIRST ENTRY')
|
|
251
|
+
})
|
|
252
|
+
|
|
253
|
+
it('preserves last 10 entries', () => {
|
|
254
|
+
const memory = createEmptyMemory()
|
|
255
|
+
const manager = new MemoryManager(memory, { maxMemoryTokens: 10 })
|
|
256
|
+
|
|
257
|
+
manager.addToHistory({ type: 'input', content: 'First', turnNumber: null, iterationNumber: 0 })
|
|
258
|
+
|
|
259
|
+
const lastEntries: { type: 'reasoning'; content: string; timestamp: number }[] = []
|
|
260
|
+
// Add middle entries that will be removed
|
|
261
|
+
for (let i = 0; i < 20; i++) {
|
|
262
|
+
manager.addToHistory({ type: 'reasoning', content: `Middle ${i}`, turnNumber: null, iterationNumber: 1 })
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
// Add last 10 entries to preserve
|
|
266
|
+
for (let i = 0; i < 10; i++) {
|
|
267
|
+
const entry = { type: 'reasoning' as const, content: `Last ${i}`, timestamp: Date.now() }
|
|
268
|
+
lastEntries.push(entry)
|
|
269
|
+
manager.addToHistory({ ...entry, turnNumber: null, iterationNumber: 2 })
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
expect(memory.history).toHaveLength(11) // First + 10
|
|
273
|
+
// Verify last 10 are preserved
|
|
274
|
+
for (let i = 0; i < 10; i++) {
|
|
275
|
+
expect(memory.history[i + 1].content).toBe(lastEntries[i].content)
|
|
276
|
+
expect(memory.history[i + 1].type).toBe(lastEntries[i].type)
|
|
277
|
+
}
|
|
278
|
+
})
|
|
279
|
+
})
|
|
280
|
+
|
|
281
|
+
describe('enforceHardLimits()', () => {
|
|
282
|
+
it('removes oldest session memory entries when exceeding limit', () => {
|
|
283
|
+
const memory = createEmptyMemory()
|
|
284
|
+
const logger = createMockLogger()
|
|
285
|
+
const manager = new MemoryManager(memory, { maxSessionMemoryKeys: 3 }, logger)
|
|
286
|
+
|
|
287
|
+
// Add entries with specific timestamps
|
|
288
|
+
manager.set('key1', 'oldest')
|
|
289
|
+
manager.set('key2', 'middle')
|
|
290
|
+
manager.set('key3', 'recent')
|
|
291
|
+
manager.set('key4', 'newest')
|
|
292
|
+
|
|
293
|
+
manager.enforceHardLimits()
|
|
294
|
+
|
|
295
|
+
expect(Object.keys(memory.sessionMemory)).toHaveLength(3)
|
|
296
|
+
expect(memory.sessionMemory['key1']).toBeUndefined() // Oldest removed
|
|
297
|
+
expect(memory.sessionMemory['key2']).toBeDefined()
|
|
298
|
+
expect(memory.sessionMemory['key3']).toBeDefined()
|
|
299
|
+
expect(memory.sessionMemory['key4']).toBeDefined()
|
|
300
|
+
})
|
|
301
|
+
|
|
302
|
+
it('logs when session memory limit exceeded', () => {
|
|
303
|
+
const memory = createEmptyMemory()
|
|
304
|
+
const logger = createMockLogger()
|
|
305
|
+
const manager = new MemoryManager(memory, { maxSessionMemoryKeys: 2 }, logger)
|
|
306
|
+
|
|
307
|
+
manager.set('key1', 'data1')
|
|
308
|
+
manager.set('key2', 'data2')
|
|
309
|
+
manager.set('key3', 'data3')
|
|
310
|
+
|
|
311
|
+
manager.enforceHardLimits()
|
|
312
|
+
|
|
313
|
+
expect(logger.action).toHaveBeenCalledWith(
|
|
314
|
+
'memory-limit-exceeded',
|
|
315
|
+
expect.stringContaining('Session memory exceeds hard limit'),
|
|
316
|
+
expect.anything(),
|
|
317
|
+
expect.anything(),
|
|
318
|
+
expect.anything(),
|
|
319
|
+
expect.anything()
|
|
320
|
+
)
|
|
321
|
+
})
|
|
322
|
+
|
|
323
|
+
it('performs emergency compaction when exceeding total token budget', () => {
|
|
324
|
+
const memory = createEmptyMemory()
|
|
325
|
+
const logger = createMockLogger()
|
|
326
|
+
const manager = new MemoryManager(memory, { maxMemoryTokens: 20 }, logger)
|
|
327
|
+
|
|
328
|
+
// Add first entry (will be preserved)
|
|
329
|
+
manager.addToHistory({ type: 'input', content: 'First', turnNumber: null, iterationNumber: 0 })
|
|
330
|
+
|
|
331
|
+
// Add many entries to exceed budget
|
|
332
|
+
for (let i = 0; i < 100; i++) {
|
|
333
|
+
manager.addToHistory({
|
|
334
|
+
type: 'reasoning',
|
|
335
|
+
content: 'a'.repeat(200),
|
|
336
|
+
turnNumber: null,
|
|
337
|
+
iterationNumber: 1
|
|
338
|
+
})
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
manager.enforceHardLimits()
|
|
342
|
+
|
|
343
|
+
// Emergency compaction: first + last 5 = 6 total
|
|
344
|
+
expect(memory.history.length).toBe(6)
|
|
345
|
+
// Check that memory-emergency was called at some point
|
|
346
|
+
const emergencyCalls = (logger.action as unknown as { mock: { calls: unknown[][] } }).mock.calls.filter(
|
|
347
|
+
(call: unknown[]) => call[0] === 'memory-emergency'
|
|
348
|
+
)
|
|
349
|
+
expect(emergencyCalls.length).toBeGreaterThan(0)
|
|
350
|
+
expect(emergencyCalls[0][1]).toContain('Total memory exceeds token budget')
|
|
351
|
+
})
|
|
352
|
+
|
|
353
|
+
it('uses default hard limits when no constraints provided', () => {
|
|
354
|
+
const memory = createEmptyMemory()
|
|
355
|
+
const manager = new MemoryManager(memory)
|
|
356
|
+
|
|
357
|
+
// Default MAX_SESSION_MEMORY_KEYS is 25
|
|
358
|
+
for (let i = 0; i < 27; i++) {
|
|
359
|
+
manager.set(`key${i}`, `data${i}`)
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
manager.enforceHardLimits()
|
|
363
|
+
|
|
364
|
+
expect(Object.keys(memory.sessionMemory)).toHaveLength(25)
|
|
365
|
+
})
|
|
366
|
+
})
|
|
367
|
+
|
|
368
|
+
describe('getHistoryLength()', () => {
|
|
369
|
+
it('returns 0 for empty history', () => {
|
|
370
|
+
const memory = createEmptyMemory()
|
|
371
|
+
const manager = new MemoryManager(memory)
|
|
372
|
+
|
|
373
|
+
expect(manager.getHistoryLength()).toBe(0)
|
|
374
|
+
})
|
|
375
|
+
|
|
376
|
+
it('returns correct count after adding entries', () => {
|
|
377
|
+
const memory = createEmptyMemory()
|
|
378
|
+
const manager = new MemoryManager(memory)
|
|
379
|
+
|
|
380
|
+
manager.addToHistory({ type: 'input', content: 'Entry 1', turnNumber: null, iterationNumber: 0 })
|
|
381
|
+
expect(manager.getHistoryLength()).toBe(1)
|
|
382
|
+
|
|
383
|
+
manager.addToHistory({ type: 'reasoning', content: 'Entry 2', turnNumber: null, iterationNumber: 1 })
|
|
384
|
+
expect(manager.getHistoryLength()).toBe(2)
|
|
385
|
+
|
|
386
|
+
manager.addToHistory({ type: 'tool-result', content: 'Entry 3', turnNumber: null, iterationNumber: 1 })
|
|
387
|
+
expect(manager.getHistoryLength()).toBe(3)
|
|
388
|
+
})
|
|
389
|
+
})
|
|
390
|
+
|
|
391
|
+
describe('getStatus()', () => {
|
|
392
|
+
it('returns status with all fields', () => {
|
|
393
|
+
const memory = createEmptyMemory()
|
|
394
|
+
const manager = new MemoryManager(memory)
|
|
395
|
+
|
|
396
|
+
const status = manager.getStatus()
|
|
397
|
+
|
|
398
|
+
expect(status).toHaveProperty('sessionMemoryKeys')
|
|
399
|
+
expect(status).toHaveProperty('sessionMemoryLimit')
|
|
400
|
+
expect(status).toHaveProperty('currentKeys')
|
|
401
|
+
expect(status).toHaveProperty('historyPercent')
|
|
402
|
+
expect(status).toHaveProperty('historyTokens')
|
|
403
|
+
expect(status).toHaveProperty('tokenBudget')
|
|
404
|
+
})
|
|
405
|
+
|
|
406
|
+
it('calculates session memory key count correctly', () => {
|
|
407
|
+
const memory = createEmptyMemory()
|
|
408
|
+
const manager = new MemoryManager(memory)
|
|
409
|
+
|
|
410
|
+
manager.set('key1', 'data1')
|
|
411
|
+
manager.set('key2', 'data2')
|
|
412
|
+
manager.set('key3', 'data3')
|
|
413
|
+
|
|
414
|
+
const status = manager.getStatus()
|
|
415
|
+
|
|
416
|
+
expect(status.sessionMemoryKeys).toBe(3)
|
|
417
|
+
expect(status.currentKeys).toEqual(['key1', 'key2', 'key3'])
|
|
418
|
+
})
|
|
419
|
+
|
|
420
|
+
it('calculates token usage percentage', () => {
|
|
421
|
+
const memory = createEmptyMemory()
|
|
422
|
+
const manager = new MemoryManager(memory, { maxMemoryTokens: 100 })
|
|
423
|
+
|
|
424
|
+
// Add entry that takes ~50 tokens (50 chars * 4 = ~12 tokens * 1.2 = ~15 tokens)
|
|
425
|
+
manager.addToHistory({
|
|
426
|
+
type: 'input',
|
|
427
|
+
content: 'a'.repeat(200), // ~60 tokens
|
|
428
|
+
turnNumber: null,
|
|
429
|
+
iterationNumber: 0
|
|
430
|
+
})
|
|
431
|
+
|
|
432
|
+
const status = manager.getStatus()
|
|
433
|
+
|
|
434
|
+
expect(status.historyPercent).toBeGreaterThan(0)
|
|
435
|
+
expect(status.historyPercent).toBeLessThanOrEqual(100)
|
|
436
|
+
expect(status.tokenBudget).toBe(100)
|
|
437
|
+
})
|
|
438
|
+
|
|
439
|
+
it('includes session memory and history tokens in total', () => {
|
|
440
|
+
const memory = createEmptyMemory()
|
|
441
|
+
const manager = new MemoryManager(memory, { maxMemoryTokens: 1000 })
|
|
442
|
+
|
|
443
|
+
manager.set('persistent1', 'a'.repeat(100))
|
|
444
|
+
manager.addToHistory({ type: 'input', content: 'a'.repeat(100), turnNumber: null, iterationNumber: 0 })
|
|
445
|
+
|
|
446
|
+
const status = manager.getStatus()
|
|
447
|
+
|
|
448
|
+
expect(status.historyTokens).toBeGreaterThan(0)
|
|
449
|
+
})
|
|
450
|
+
})
|
|
451
|
+
|
|
452
|
+
describe('toSnapshot() and getSnapshot()', () => {
|
|
453
|
+
it('creates deep copy of memory', () => {
|
|
454
|
+
const memory = createEmptyMemory()
|
|
455
|
+
const manager = new MemoryManager(memory)
|
|
456
|
+
|
|
457
|
+
manager.set('key', 'value')
|
|
458
|
+
manager.addToHistory({ type: 'input', content: 'test', turnNumber: null, iterationNumber: 0 })
|
|
459
|
+
|
|
460
|
+
const snapshot = manager.toSnapshot()
|
|
461
|
+
|
|
462
|
+
// Modify original
|
|
463
|
+
manager.set('key', 'modified')
|
|
464
|
+
manager.addToHistory({ type: 'reasoning', content: 'new', turnNumber: null, iterationNumber: 1 })
|
|
465
|
+
|
|
466
|
+
// Snapshot should not be affected
|
|
467
|
+
expect(snapshot.sessionMemory['key'].content).toBe('value')
|
|
468
|
+
expect(snapshot.history).toHaveLength(1)
|
|
469
|
+
})
|
|
470
|
+
|
|
471
|
+
it('caches snapshot for later retrieval', () => {
|
|
472
|
+
const memory = createEmptyMemory()
|
|
473
|
+
const manager = new MemoryManager(memory)
|
|
474
|
+
|
|
475
|
+
manager.set('data', 'test')
|
|
476
|
+
|
|
477
|
+
const snapshot1 = manager.toSnapshot()
|
|
478
|
+
const snapshot2 = manager.getSnapshot()
|
|
479
|
+
|
|
480
|
+
expect(snapshot2).toBe(snapshot1)
|
|
481
|
+
})
|
|
482
|
+
|
|
483
|
+
it('returns undefined if toSnapshot not called yet', () => {
|
|
484
|
+
const memory = createEmptyMemory()
|
|
485
|
+
const manager = new MemoryManager(memory)
|
|
486
|
+
|
|
487
|
+
expect(manager.getSnapshot()).toBeUndefined()
|
|
488
|
+
})
|
|
489
|
+
})
|
|
490
|
+
|
|
491
|
+
describe('toContext()', () => {
|
|
492
|
+
it('generates formatted context string with iteration filtering', () => {
|
|
493
|
+
const memory = createEmptyMemory()
|
|
494
|
+
const manager = new MemoryManager(memory)
|
|
495
|
+
|
|
496
|
+
const context = manager.toContext(1)
|
|
497
|
+
|
|
498
|
+
expect(context).toContain('MEMORY STATUS')
|
|
499
|
+
expect(context).toContain('SESSION MEMORY')
|
|
500
|
+
expect(context).toContain('ITERATION 1 - CURRENT CONTEXT')
|
|
501
|
+
expect(context).toContain('EARLIER CONTEXT')
|
|
502
|
+
})
|
|
503
|
+
|
|
504
|
+
it('includes session memory entries with keys', () => {
|
|
505
|
+
const memory = createEmptyMemory()
|
|
506
|
+
const manager = new MemoryManager(memory)
|
|
507
|
+
|
|
508
|
+
manager.set('task', 'Build feature X')
|
|
509
|
+
manager.set('context', 'User requested...')
|
|
510
|
+
|
|
511
|
+
const context = manager.toContext(1)
|
|
512
|
+
|
|
513
|
+
expect(context).toContain('[SESSION:task]')
|
|
514
|
+
expect(context).toContain('Build feature X')
|
|
515
|
+
expect(context).toContain('[SESSION:context]')
|
|
516
|
+
expect(context).toContain('User requested...')
|
|
517
|
+
})
|
|
518
|
+
|
|
519
|
+
it('filters history by iteration - current context reversed', () => {
|
|
520
|
+
const memory = createEmptyMemory()
|
|
521
|
+
const manager = new MemoryManager(memory)
|
|
522
|
+
|
|
523
|
+
manager.addToHistory({ type: 'input', content: 'User input', turnNumber: null, iterationNumber: 0 })
|
|
524
|
+
manager.addToHistory({ type: 'reasoning', content: 'First reasoning', turnNumber: null, iterationNumber: 1 })
|
|
525
|
+
manager.addToHistory({ type: 'tool-result', content: 'Tool result', turnNumber: null, iterationNumber: 1 })
|
|
526
|
+
|
|
527
|
+
const context = manager.toContext(1)
|
|
528
|
+
|
|
529
|
+
// Current iteration (1) entries should appear reversed (tool result first)
|
|
530
|
+
const currentSection = context
|
|
531
|
+
.split('=== ITERATION 1 - CURRENT CONTEXT ===')[1]
|
|
532
|
+
.split('=== EARLIER CONTEXT ===')[0]
|
|
533
|
+
expect(currentSection.indexOf('TOOL-RESULT')).toBeLessThan(currentSection.indexOf('REASONING'))
|
|
534
|
+
|
|
535
|
+
// Earlier context (iteration 0) should appear after
|
|
536
|
+
const earlierSection = context.split('=== EARLIER CONTEXT ===')[1]
|
|
537
|
+
expect(earlierSection).toContain('[INPUT]')
|
|
538
|
+
expect(earlierSection).toContain('User input')
|
|
539
|
+
})
|
|
540
|
+
|
|
541
|
+
it('shows empty message when no session memory entries', () => {
|
|
542
|
+
const memory = createEmptyMemory()
|
|
543
|
+
const manager = new MemoryManager(memory)
|
|
544
|
+
|
|
545
|
+
const context = manager.toContext(1)
|
|
546
|
+
|
|
547
|
+
expect(context).toContain('(empty)')
|
|
548
|
+
})
|
|
549
|
+
|
|
550
|
+
it('shows no earlier context message for iteration 0', () => {
|
|
551
|
+
const memory = createEmptyMemory()
|
|
552
|
+
const manager = new MemoryManager(memory)
|
|
553
|
+
|
|
554
|
+
manager.addToHistory({ type: 'input', content: 'test', turnNumber: null, iterationNumber: 0 })
|
|
555
|
+
|
|
556
|
+
const context = manager.toContext(0)
|
|
557
|
+
|
|
558
|
+
expect(context).toContain('(no earlier context)')
|
|
559
|
+
})
|
|
560
|
+
|
|
561
|
+
it('handles turn-scoped iteration filtering correctly', () => {
|
|
562
|
+
const memory = createEmptyMemory()
|
|
563
|
+
const manager = new MemoryManager(memory)
|
|
564
|
+
|
|
565
|
+
// Turn 1 entries
|
|
566
|
+
manager.addToHistory({ type: 'input', turnNumber: 1, iterationNumber: 0, content: 'Turn 1 input' })
|
|
567
|
+
manager.addToHistory({ type: 'reasoning', turnNumber: 1, iterationNumber: 1, content: 'Turn 1 reasoning' })
|
|
568
|
+
|
|
569
|
+
// Turn 2 entries
|
|
570
|
+
manager.addToHistory({ type: 'input', turnNumber: 2, iterationNumber: 0, content: 'Turn 2 input' })
|
|
571
|
+
manager.addToHistory({ type: 'reasoning', turnNumber: 2, iterationNumber: 1, content: 'Turn 2 reasoning' })
|
|
572
|
+
|
|
573
|
+
// Turn 2, Iteration 1 - should ONLY see Turn 2's context
|
|
574
|
+
const context = manager.toContext(1, 2)
|
|
575
|
+
|
|
576
|
+
// Assert: Current context contains ONLY Turn 2's iteration 1
|
|
577
|
+
expect(context).toContain('Turn 2 reasoning')
|
|
578
|
+
expect(context).not.toContain('Turn 1 reasoning')
|
|
579
|
+
|
|
580
|
+
// Assert: Earlier context contains ONLY Turn 2's iteration 0
|
|
581
|
+
expect(context).toContain('Turn 2 input')
|
|
582
|
+
expect(context).not.toContain('Turn 1 input')
|
|
583
|
+
})
|
|
584
|
+
|
|
585
|
+
it('handles one-off executions without turnNumber', () => {
|
|
586
|
+
const memory = createEmptyMemory()
|
|
587
|
+
const manager = new MemoryManager(memory)
|
|
588
|
+
|
|
589
|
+
manager.addToHistory({ type: 'input', turnNumber: null, iterationNumber: 0, content: 'Input' })
|
|
590
|
+
manager.addToHistory({ type: 'reasoning', turnNumber: null, iterationNumber: 1, content: 'Reasoning' })
|
|
591
|
+
|
|
592
|
+
// One-off execution - toContext() called without currentTurn parameter
|
|
593
|
+
const context = manager.toContext(1)
|
|
594
|
+
|
|
595
|
+
// Should work as before (no turn filtering)
|
|
596
|
+
expect(context).toContain('Reasoning')
|
|
597
|
+
expect(context).toContain('Input')
|
|
598
|
+
})
|
|
599
|
+
|
|
600
|
+
it('handles legacy entries without turnNumber gracefully', () => {
|
|
601
|
+
const memory: AgentMemory = {
|
|
602
|
+
sessionMemory: {},
|
|
603
|
+
history: [
|
|
604
|
+
// Legacy entries (pre-migration) without turnNumber
|
|
605
|
+
{
|
|
606
|
+
type: 'input',
|
|
607
|
+
content: 'Legacy input',
|
|
608
|
+
timestamp: 1,
|
|
609
|
+
turnNumber: undefined as unknown as null,
|
|
610
|
+
iterationNumber: 0
|
|
611
|
+
},
|
|
612
|
+
{
|
|
613
|
+
type: 'reasoning',
|
|
614
|
+
content: 'Legacy reasoning',
|
|
615
|
+
timestamp: 2,
|
|
616
|
+
turnNumber: undefined as unknown as null,
|
|
617
|
+
iterationNumber: 1
|
|
618
|
+
},
|
|
619
|
+
// New entries with turnNumber
|
|
620
|
+
{ type: 'input', content: 'New input', timestamp: 3, turnNumber: 2, iterationNumber: 0 },
|
|
621
|
+
{ type: 'reasoning', content: 'New reasoning', timestamp: 4, turnNumber: 2, iterationNumber: 1 }
|
|
622
|
+
]
|
|
623
|
+
}
|
|
624
|
+
|
|
625
|
+
const manager = new MemoryManager(memory)
|
|
626
|
+
|
|
627
|
+
// Turn 2, Iteration 1 - should see both legacy (undefined treated as valid) and Turn 2 entries
|
|
628
|
+
const context = manager.toContext(1, 2)
|
|
629
|
+
|
|
630
|
+
// Should include Turn 2's entries
|
|
631
|
+
expect(context).toContain('New reasoning')
|
|
632
|
+
|
|
633
|
+
// Should also include legacy entries (backward compatibility)
|
|
634
|
+
expect(context).toContain('Legacy reasoning')
|
|
635
|
+
})
|
|
636
|
+
})
|
|
637
|
+
})
|
|
638
|
+
|
|
639
|
+
describe('truncateToolResult()', () => {
|
|
640
|
+
it('returns content unchanged when under the token limit', () => {
|
|
641
|
+
const content = 'short result'
|
|
642
|
+
const result = truncateToolResult(content, 4_000)
|
|
643
|
+
expect(result).toBe(content)
|
|
644
|
+
})
|
|
645
|
+
|
|
646
|
+
it('truncates content that exceeds the token limit', () => {
|
|
647
|
+
// 4000 tokens * 3.5 chars/token = 14000 chars; create 20000 chars to exceed
|
|
648
|
+
const content = 'x'.repeat(20_000)
|
|
649
|
+
const result = truncateToolResult(content, 4_000)
|
|
650
|
+
|
|
651
|
+
expect(result.length).toBeLessThan(content.length)
|
|
652
|
+
expect(result).toContain('[Response truncated')
|
|
653
|
+
expect(result).toContain('tokens omitted')
|
|
654
|
+
})
|
|
655
|
+
|
|
656
|
+
it('includes the number of omitted tokens in the truncation notice', () => {
|
|
657
|
+
const content = 'a'.repeat(21_000) // ~6000 tokens at 3.5 chars/token
|
|
658
|
+
const result = truncateToolResult(content, 4_000)
|
|
659
|
+
|
|
660
|
+
// Should mention roughly 2000 omitted tokens
|
|
661
|
+
expect(result).toContain('tokens omitted')
|
|
662
|
+
})
|
|
663
|
+
|
|
664
|
+
it('returns content unchanged when exactly at the limit', () => {
|
|
665
|
+
// 4000 tokens * 3.5 = 14000 chars exactly
|
|
666
|
+
const content = 'b'.repeat(14_000)
|
|
667
|
+
const result = truncateToolResult(content, 4_000)
|
|
668
|
+
expect(result).toBe(content)
|
|
669
|
+
})
|
|
670
|
+
})
|
|
671
|
+
|
|
672
|
+
describe('MemoryManager - tool result truncation in addToHistory', () => {
|
|
673
|
+
const createMockLogger = (): AgentScopedLogger =>
|
|
674
|
+
({
|
|
675
|
+
action: vi.fn(),
|
|
676
|
+
lifecycle: vi.fn(),
|
|
677
|
+
memory: vi.fn(),
|
|
678
|
+
tool: vi.fn(),
|
|
679
|
+
reasoning: vi.fn()
|
|
680
|
+
}) as unknown as AgentScopedLogger
|
|
681
|
+
|
|
682
|
+
const createEmptyMemory = (): AgentMemory => ({
|
|
683
|
+
sessionMemory: {},
|
|
684
|
+
history: []
|
|
685
|
+
})
|
|
686
|
+
|
|
687
|
+
it('truncates tool-result entries exceeding MAX_TOOL_RESULT_TOKENS', () => {
|
|
688
|
+
const memory = createEmptyMemory()
|
|
689
|
+
const logger = createMockLogger()
|
|
690
|
+
const manager = new MemoryManager(memory, {}, logger)
|
|
691
|
+
|
|
692
|
+
// 4000 tokens * 3.5 = 14000 chars; create 28000 chars (~8000 tokens)
|
|
693
|
+
const largeContent = 'z'.repeat(28_000)
|
|
694
|
+
|
|
695
|
+
manager.addToHistory({
|
|
696
|
+
type: 'tool-result',
|
|
697
|
+
content: largeContent,
|
|
698
|
+
turnNumber: null,
|
|
699
|
+
iterationNumber: 1
|
|
700
|
+
})
|
|
701
|
+
|
|
702
|
+
expect(memory.history).toHaveLength(1)
|
|
703
|
+
expect(memory.history[0].content.length).toBeLessThan(largeContent.length)
|
|
704
|
+
expect(memory.history[0].content).toContain('[Response truncated')
|
|
705
|
+
expect(logger.action).toHaveBeenCalledWith(
|
|
706
|
+
'memory-tool-result-truncate',
|
|
707
|
+
expect.stringContaining('Tool result truncated'),
|
|
708
|
+
expect.anything(),
|
|
709
|
+
expect.anything(),
|
|
710
|
+
expect.anything(),
|
|
711
|
+
expect.anything()
|
|
712
|
+
)
|
|
713
|
+
})
|
|
714
|
+
|
|
715
|
+
it('does not truncate tool-result entries under the limit', () => {
|
|
716
|
+
const memory = createEmptyMemory()
|
|
717
|
+
const manager = new MemoryManager(memory)
|
|
718
|
+
|
|
719
|
+
const smallContent = '{"status": "ok"}'
|
|
720
|
+
manager.addToHistory({
|
|
721
|
+
type: 'tool-result',
|
|
722
|
+
content: smallContent,
|
|
723
|
+
turnNumber: null,
|
|
724
|
+
iterationNumber: 1
|
|
725
|
+
})
|
|
726
|
+
|
|
727
|
+
expect(memory.history[0].content).toBe(smallContent)
|
|
728
|
+
})
|
|
729
|
+
|
|
730
|
+
it('does not truncate non-tool-result entries regardless of size', () => {
|
|
731
|
+
const memory = createEmptyMemory()
|
|
732
|
+
const manager = new MemoryManager(memory, { maxMemoryTokens: 100_000 })
|
|
733
|
+
|
|
734
|
+
const largeReasoning = 'r'.repeat(28_000)
|
|
735
|
+
manager.addToHistory({
|
|
736
|
+
type: 'reasoning',
|
|
737
|
+
content: largeReasoning,
|
|
738
|
+
turnNumber: null,
|
|
739
|
+
iterationNumber: 1
|
|
740
|
+
})
|
|
741
|
+
|
|
742
|
+
expect(memory.history[0].content).toBe(largeReasoning)
|
|
743
|
+
|
|
744
|
+
const largeInput = 'i'.repeat(28_000)
|
|
745
|
+
manager.addToHistory({
|
|
746
|
+
type: 'input',
|
|
747
|
+
content: largeInput,
|
|
748
|
+
turnNumber: null,
|
|
749
|
+
iterationNumber: 0
|
|
750
|
+
})
|
|
751
|
+
|
|
752
|
+
expect(memory.history[1].content).toBe(largeInput)
|
|
753
|
+
})
|
|
754
|
+
})
|