squish-memory 1.1.5 → 1.2.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.
Files changed (499) hide show
  1. package/.env.example +32 -16
  2. package/CHANGELOG.md +147 -0
  3. package/README.md +120 -78
  4. package/{scripts → bin}/dependency-manager.mjs +217 -217
  5. package/{scripts → bin}/detect-clients.mjs +78 -78
  6. package/bin/install-interactive.mjs +321 -0
  7. package/bin/squish-mcp.mjs +46 -0
  8. package/bin/squish.mjs +33 -0
  9. package/config/mcp-migration-map.json +1 -6
  10. package/config/mcp-mode-semantics.json +19 -23
  11. package/config/mcp-remote-auth.json +3 -26
  12. package/config/mcp-universal.schema.json +5 -35
  13. package/config/settings.json +107 -52
  14. package/config.js +5 -0
  15. package/config.ts +218 -0
  16. package/core/adapters/config/claude-code.ts +133 -0
  17. package/core/adapters/config/cursor.ts +90 -0
  18. package/core/adapters/config/opencode.ts +89 -0
  19. package/core/adapters/config/windsurf.ts +90 -0
  20. package/core/adapters/index.ts +102 -0
  21. package/core/adapters/timeline.ts +116 -0
  22. package/core/adapters/types.ts +166 -0
  23. package/core/agent-preferences.ts +140 -0
  24. package/core/algorithms/analytics/token-estimator.ts +216 -0
  25. package/core/algorithms/detection/hash-filters.ts +260 -0
  26. package/core/algorithms/detection/semantic-ranker.ts +194 -0
  27. package/core/algorithms/detection/two-stage-detector.ts +421 -0
  28. package/core/algorithms/handlers/approve-merge.ts +215 -0
  29. package/core/algorithms/handlers/detect-duplicates.ts +192 -0
  30. package/core/algorithms/handlers/get-stats.ts +132 -0
  31. package/core/algorithms/handlers/list-proposals.ts +130 -0
  32. package/core/algorithms/handlers/preview-merge.ts +139 -0
  33. package/core/algorithms/handlers/reject-merge.ts +93 -0
  34. package/core/algorithms/handlers/reverse-merge.ts +155 -0
  35. package/core/algorithms/index.ts +39 -0
  36. package/core/algorithms/operations/cache-maintenance.ts +182 -0
  37. package/core/algorithms/safety/safety-checks.ts +256 -0
  38. package/core/algorithms/strategies/merge-strategies.ts +381 -0
  39. package/core/algorithms/types.ts +140 -0
  40. package/core/algorithms/utils/response-builder.ts +61 -0
  41. package/core/associations.ts +363 -0
  42. package/core/beliefs/decay.ts +289 -0
  43. package/core/beliefs/extractor.ts +131 -0
  44. package/core/beliefs/store.ts +557 -0
  45. package/core/beliefs/types.ts +38 -0
  46. package/core/commands/mcp-server.ts +5 -0
  47. package/core/compression.ts +177 -0
  48. package/core/config.js +2 -0
  49. package/core/consolidation.ts +330 -0
  50. package/core/context/agent-context.ts +388 -0
  51. package/core/context/context-paging.ts +449 -0
  52. package/core/context/context-window.ts +234 -0
  53. package/core/context/context.ts +35 -0
  54. package/core/embeddings/embeddings.ts +616 -0
  55. package/core/embeddings/google-multimodal.ts +200 -0
  56. package/{dist/core/local-embeddings.js → core/embeddings/local-embeddings.ts} +12 -11
  57. package/core/embeddings/qmd-client.ts +495 -0
  58. package/core/embeddings/transformers-local.ts +261 -0
  59. package/core/embeddings.js +4 -0
  60. package/core/error-handling.ts +206 -0
  61. package/core/external +219 -0
  62. package/core/graph/entity-deduplicator.ts +232 -0
  63. package/core/graph/graph-builder.ts +257 -0
  64. package/core/graph/graph-traversal.ts +490 -0
  65. package/core/graph/index.ts +24 -0
  66. package/core/graph/llm-entity-extractor.ts +402 -0
  67. package/core/graph/multi-hop-retrieval.ts +317 -0
  68. package/core/graph/relationship-extractor.ts +465 -0
  69. package/core/hooks/agent-hooks.ts +653 -0
  70. package/core/hooks/auto-tagger.ts +149 -0
  71. package/core/hooks/capture-filter.ts +169 -0
  72. package/core/hot-cache.ts +388 -0
  73. package/core/index.ts +10 -0
  74. package/core/ingestion/agent-memory.ts +167 -0
  75. package/core/ingestion/core-memory.ts +326 -0
  76. package/core/ingestion/learnings.ts +260 -0
  77. package/core/ingestion/signal-engine.ts +266 -0
  78. package/core/integrations/obsidian-vault.ts +197 -0
  79. package/core/layers/generator.ts +115 -0
  80. package/core/lib/db-client.ts +168 -0
  81. package/core/lib/parse-embedding.ts +59 -0
  82. package/core/lib/schemas.ts +102 -0
  83. package/core/lib/types.ts +49 -0
  84. package/core/lib/utils.ts +151 -0
  85. package/core/lib/validation.ts +180 -0
  86. package/core/lifecycle.ts +353 -0
  87. package/core/logger.ts +59 -0
  88. package/core/memory/bridge-discovery.ts +395 -0
  89. package/core/memory/categorizer.ts +390 -0
  90. package/core/memory/conflict-detector.ts +62 -0
  91. package/core/memory/consolidation.ts +372 -0
  92. package/core/memory/context-collector.ts +75 -0
  93. package/core/memory/contradiction-resolver.ts +494 -0
  94. package/core/memory/edit-workflow.ts +174 -0
  95. package/core/memory/entity-extractor.ts +426 -0
  96. package/core/memory/entity-resolver.ts +89 -0
  97. package/core/memory/explain.ts +112 -0
  98. package/core/memory/fact-deriver.ts +300 -0
  99. package/core/memory/fact-extractor.ts +120 -0
  100. package/core/memory/feedback-tracker.ts +200 -0
  101. package/core/memory/hooks.ts +230 -0
  102. package/core/memory/hybrid-retrieval.ts +65 -0
  103. package/core/memory/hybrid-scorer.ts +325 -0
  104. package/core/memory/hybrid-search.ts +748 -0
  105. package/core/memory/importance.ts +319 -0
  106. package/core/memory/index.ts +11 -0
  107. package/core/memory/loader.ts +178 -0
  108. package/core/memory/markdown/markdown-storage.ts +318 -0
  109. package/core/memory/memories.ts +565 -0
  110. package/core/memory/memory-lifecycle.ts +51 -0
  111. package/core/memory/memory-manager.ts +53 -0
  112. package/core/memory/migrate.ts +173 -0
  113. package/core/memory/normalization.ts +30 -0
  114. package/core/memory/path-strengthener.ts +211 -0
  115. package/core/memory/progressive-disclosure.ts +392 -0
  116. package/core/memory/query-processor.ts +130 -0
  117. package/core/memory/query-rewriter.ts +153 -0
  118. package/core/memory/response-analyzer.ts +81 -0
  119. package/core/memory/retrieval-feedback.ts +276 -0
  120. package/core/memory/serialization.ts +83 -0
  121. package/core/memory/stale-cleaner.ts +147 -0
  122. package/core/memory/stats.ts +181 -0
  123. package/core/memory/telemetry.ts +392 -0
  124. package/core/memory/temporal-facts.ts +356 -0
  125. package/core/memory/temporal-parser.ts +477 -0
  126. package/core/memory/trigger-detector.ts +104 -0
  127. package/core/memory/write-gate.ts +288 -0
  128. package/core/places/index.ts +14 -0
  129. package/core/places/memory-places.ts +339 -0
  130. package/core/places/places.ts +406 -0
  131. package/core/places/rules.ts +308 -0
  132. package/core/places/walking.ts +192 -0
  133. package/core/projects +89 -0
  134. package/core/projects.ts +131 -0
  135. package/core/redis.ts +82 -0
  136. package/core/responses.ts +187 -0
  137. package/core/runtime/trust-report.ts +195 -0
  138. package/core/runtime/trust-state.ts +360 -0
  139. package/core/scheduler/cron-scheduler.ts +581 -0
  140. package/core/scheduler/heartbeat.ts +91 -0
  141. package/core/scheduler/index.ts +8 -0
  142. package/core/scheduler/job-runner.ts +197 -0
  143. package/core/search/conversations.ts +166 -0
  144. package/core/search/entities.ts +46 -0
  145. package/core/search/folder-context.ts +154 -0
  146. package/core/search/graph-boost.ts +22 -0
  147. package/core/search/index.ts +4 -0
  148. package/core/search/qmd-wrapper.ts +84 -0
  149. package/core/security/encrypt.ts +51 -0
  150. package/core/security/governance.ts +102 -0
  151. package/core/security/privacy.ts +108 -0
  152. package/core/security/secret-detector.ts +122 -0
  153. package/core/session/auto-load.ts +160 -0
  154. package/core/session/entity-tracker.ts +363 -0
  155. package/core/session/index.ts +7 -0
  156. package/core/session/reference-resolver.ts +158 -0
  157. package/core/session/self-iteration-job.ts +478 -0
  158. package/core/session/session-hooks.ts +69 -0
  159. package/core/session/types.ts +36 -0
  160. package/core/session/working-set.ts +275 -0
  161. package/core/snapshots/cleanup.ts +13 -0
  162. package/core/snapshots/comparison.ts +59 -0
  163. package/core/snapshots/creation.ts +139 -0
  164. package/core/snapshots/retrieval.ts +44 -0
  165. package/core/snapshots/stats.ts +63 -0
  166. package/core/storage/cache.ts +241 -0
  167. package/core/storage/database.ts +23 -0
  168. package/core/summarization/cleanup.ts +13 -0
  169. package/core/summarization/queries.ts +32 -0
  170. package/core/summarization/stats.ts +64 -0
  171. package/core/summarization/strategies.ts +52 -0
  172. package/core/summarization.ts +248 -0
  173. package/core/temporal-facts.ts +244 -0
  174. package/core/tracing/collector.ts +470 -0
  175. package/core/tracing/visualizer.ts +195 -0
  176. package/core/utils/cleanup-operations.ts +50 -0
  177. package/core/utils/content-extraction.ts +95 -0
  178. package/core/utils/filter-builder.ts +56 -0
  179. package/core/utils/history-traversal.ts +63 -0
  180. package/core/utils/memory-operations.ts +56 -0
  181. package/core/utils/query-operations.ts +83 -0
  182. package/core/utils/summarization-helpers.ts +45 -0
  183. package/core/utils/temporal-queries.ts +39 -0
  184. package/core/utils/vector-operations.ts +135 -0
  185. package/core/utils/version-management.ts +74 -0
  186. package/core/worker.ts +324 -0
  187. package/db/adapter.ts +215 -0
  188. package/db/bootstrap.ts +1055 -0
  189. package/db/drizzle/migrations/0000_needy_cerebro.sql +402 -0
  190. package/db/drizzle/migrations/meta/0000_snapshot.json +3451 -0
  191. package/db/drizzle/migrations/meta/_journal.json +13 -0
  192. package/db/drizzle/schema-sqlite.ts +1032 -0
  193. package/db/drizzle/schema.ts +1128 -0
  194. package/db/drizzle.config.ts +12 -0
  195. package/db/index.ts +83 -0
  196. package/db/init.sql +5 -0
  197. package/db/migrations/associations.ts +35 -0
  198. package/db/migrations/beliefs.ts +89 -0
  199. package/db/migrations/core-memory.ts +35 -0
  200. package/db/migrations/fts.ts +59 -0
  201. package/db/migrations/index.ts +54 -0
  202. package/db/migrations/indexes.ts +36 -0
  203. package/db/migrations/learnings.ts +34 -0
  204. package/db/migrations/maintenance.ts +68 -0
  205. package/db/migrations/memories.ts +22 -0
  206. package/db/migrations/memory-places.ts +35 -0
  207. package/db/migrations/places.ts +49 -0
  208. package/db/migrations/projects.ts +21 -0
  209. package/db/migrations/tier-conversion.ts +24 -0
  210. package/db/neon.ts +22 -0
  211. package/db/schema/beliefs.ts +50 -0
  212. package/db/schema/generator.ts +159 -0
  213. package/db/schema/index.ts +58 -0
  214. package/db/schema/learnings.ts +32 -0
  215. package/db/schema/memories.ts +83 -0
  216. package/db/schema/projects.ts +33 -0
  217. package/db/schema.ts +13 -0
  218. package/db/supabase.ts +27 -0
  219. package/dist/config.d.ts +40 -17
  220. package/dist/config.js +150 -198
  221. package/dist/core/adapters/types.d.ts +13 -33
  222. package/dist/core/adapters/types.js +1 -1
  223. package/dist/core/agent-preferences.d.ts +16 -0
  224. package/dist/core/agent-preferences.js +124 -0
  225. package/dist/core/algorithms/safety/safety-checks.d.ts +1 -5
  226. package/dist/core/algorithms/types.d.ts +0 -8
  227. package/dist/core/associations.d.ts +3 -1
  228. package/dist/core/associations.js +37 -1
  229. package/dist/core/beliefs/decay.d.ts +27 -0
  230. package/dist/core/beliefs/decay.js +217 -0
  231. package/dist/core/beliefs/extractor.d.ts +9 -0
  232. package/dist/core/beliefs/extractor.js +113 -0
  233. package/dist/core/beliefs/store.d.ts +46 -0
  234. package/dist/core/beliefs/store.js +466 -0
  235. package/dist/core/beliefs/types.d.ts +28 -0
  236. package/dist/core/beliefs/types.js +2 -0
  237. package/dist/core/commands/mcp-server.d.ts +0 -1
  238. package/dist/core/commands/mcp-server.js +4 -737
  239. package/dist/core/commands/remember.d.ts +24 -0
  240. package/dist/core/commands/remember.js +144 -0
  241. package/dist/core/{toon.d.ts → compression.d.ts} +6 -4
  242. package/dist/core/{toon.js → compression.js} +8 -8
  243. package/dist/core/context/agent-context.js +1 -1
  244. package/dist/core/embeddings/embeddings.d.ts +29 -0
  245. package/dist/core/embeddings/embeddings.js +546 -0
  246. package/dist/core/embeddings/google-multimodal.js +6 -2
  247. package/dist/core/{local-embeddings.d.ts → embeddings/local-embeddings.d.ts} +1 -1
  248. package/dist/core/embeddings/local-embeddings.js +11 -0
  249. package/dist/core/embeddings/qmd-client.js +1 -1
  250. package/dist/core/embeddings/transformers-local.d.ts +64 -0
  251. package/dist/core/embeddings/transformers-local.js +213 -0
  252. package/dist/core/embeddings.d.ts +1 -28
  253. package/dist/core/embeddings.js +2 -453
  254. package/dist/core/graph/entity-deduplicator.d.ts +24 -0
  255. package/dist/core/graph/entity-deduplicator.js +183 -0
  256. package/dist/core/graph/graph-builder.d.ts +46 -0
  257. package/dist/core/graph/graph-builder.js +174 -0
  258. package/dist/core/graph/graph-traversal.d.ts +80 -0
  259. package/dist/core/graph/graph-traversal.js +315 -0
  260. package/dist/core/graph/index.d.ts +19 -0
  261. package/dist/core/graph/index.js +13 -0
  262. package/dist/core/graph/llm-entity-extractor.d.ts +49 -0
  263. package/dist/core/graph/llm-entity-extractor.js +313 -0
  264. package/dist/core/graph/multi-hop-retrieval.d.ts +48 -0
  265. package/dist/core/graph/multi-hop-retrieval.js +215 -0
  266. package/dist/core/graph/relationship-extractor.d.ts +48 -0
  267. package/dist/core/graph/relationship-extractor.js +351 -0
  268. package/dist/core/hooks/agent-hooks.d.ts +10 -1
  269. package/dist/core/hooks/agent-hooks.js +301 -24
  270. package/dist/core/hot-cache.d.ts +86 -0
  271. package/dist/core/hot-cache.js +285 -0
  272. package/dist/core/index.d.ts +9 -9
  273. package/dist/core/index.js +9 -12
  274. package/dist/core/ingestion/core-memory.d.ts +2 -2
  275. package/dist/core/ingestion/core-memory.js +3 -3
  276. package/dist/core/ingestion/learnings.js +3 -0
  277. package/dist/core/ingestion/signal-engine.d.ts +41 -0
  278. package/dist/core/ingestion/signal-engine.js +201 -0
  279. package/dist/core/{obsidian-vault.d.ts → integrations/obsidian-vault.d.ts} +2 -1
  280. package/dist/core/{obsidian-vault.js → integrations/obsidian-vault.js} +69 -7
  281. package/dist/core/lib/parse-embedding.d.ts +9 -0
  282. package/dist/core/lib/parse-embedding.js +58 -0
  283. package/dist/core/lib/schemas.d.ts +57 -54
  284. package/dist/core/lib/types.d.ts +45 -0
  285. package/dist/core/lib/types.js +6 -0
  286. package/dist/core/lib/utils.d.ts +4 -0
  287. package/dist/core/lib/utils.js +55 -0
  288. package/dist/core/lifecycle.d.ts +0 -1
  289. package/dist/core/lifecycle.js +13 -23
  290. package/dist/core/logger.d.ts +1 -0
  291. package/dist/core/logger.js +14 -8
  292. package/dist/core/mcp/tools.d.ts +0 -2
  293. package/dist/core/mcp/tools.js +0 -87
  294. package/dist/core/mcp/types.d.ts +25 -253
  295. package/dist/core/mcp/types.js +2 -2
  296. package/dist/core/memory/categorizer.js +1 -0
  297. package/dist/core/memory/consolidation.js +2 -28
  298. package/dist/core/memory/entity-extractor.d.ts +4 -0
  299. package/dist/core/memory/entity-extractor.js +30 -16
  300. package/dist/core/memory/explain.d.ts +18 -0
  301. package/dist/core/memory/explain.js +92 -0
  302. package/dist/core/memory/fact-deriver.d.ts +31 -0
  303. package/dist/core/memory/fact-deriver.js +236 -0
  304. package/dist/core/memory/hybrid-retrieval.d.ts +14 -16
  305. package/dist/core/memory/hybrid-retrieval.js +25 -127
  306. package/dist/core/memory/hybrid-scorer.js +6 -23
  307. package/dist/core/memory/hybrid-search.d.ts +10 -7
  308. package/dist/core/memory/hybrid-search.js +458 -221
  309. package/dist/core/memory/importance.d.ts +0 -17
  310. package/dist/core/memory/importance.js +1 -58
  311. package/dist/core/memory/index.d.ts +1 -0
  312. package/dist/core/memory/index.js +1 -0
  313. package/dist/core/memory/memories.d.ts +13 -17
  314. package/dist/core/memory/memories.js +78 -75
  315. package/dist/core/memory/memory-lifecycle.d.ts +2 -2
  316. package/dist/core/memory/memory-lifecycle.js +10 -18
  317. package/dist/core/memory/normalization.d.ts +1 -16
  318. package/dist/core/memory/path-strengthener.d.ts +39 -0
  319. package/dist/core/memory/path-strengthener.js +150 -0
  320. package/dist/core/memory/query-processor.js +37 -3
  321. package/dist/core/memory/retrieval-feedback.d.ts +70 -0
  322. package/dist/core/memory/retrieval-feedback.js +213 -0
  323. package/dist/core/memory/stale-cleaner.d.ts +26 -0
  324. package/dist/core/memory/stale-cleaner.js +97 -0
  325. package/dist/core/memory/stats.d.ts +10 -0
  326. package/dist/core/memory/stats.js +8 -3
  327. package/dist/core/memory/trigger-detector.d.ts +8 -1
  328. package/dist/core/memory/trigger-detector.js +42 -5
  329. package/dist/core/places/index.d.ts +1 -1
  330. package/dist/core/places/index.js +1 -1
  331. package/dist/core/places/places.d.ts +13 -13
  332. package/dist/core/places/places.js +27 -27
  333. package/dist/core/places/rules.js +23 -23
  334. package/dist/core/places/walking.d.ts +3 -3
  335. package/dist/core/places/walking.js +7 -7
  336. package/dist/core/projects.js +8 -0
  337. package/dist/core/runtime/trust-report.d.ts +102 -0
  338. package/dist/core/runtime/trust-report.js +107 -0
  339. package/dist/core/runtime/trust-state.d.ts +12 -0
  340. package/dist/core/runtime/trust-state.js +309 -0
  341. package/dist/core/scheduler/cron-scheduler.d.ts +1 -1
  342. package/dist/core/scheduler/cron-scheduler.js +164 -3
  343. package/dist/core/scheduler/job-runner.js +1 -1
  344. package/dist/core/search/qmd-wrapper.d.ts +36 -0
  345. package/dist/core/search/qmd-wrapper.js +58 -0
  346. package/dist/core/session/auto-load.js +28 -3
  347. package/dist/core/session/entity-tracker.d.ts +62 -0
  348. package/dist/core/session/entity-tracker.js +287 -0
  349. package/dist/core/session/reference-resolver.d.ts +26 -0
  350. package/dist/core/session/reference-resolver.js +121 -0
  351. package/dist/core/session/self-iteration-job.d.ts +15 -0
  352. package/dist/core/session/self-iteration-job.js +163 -58
  353. package/dist/core/session/working-set.d.ts +50 -0
  354. package/dist/core/session/working-set.js +212 -0
  355. package/dist/core/snapshots/creation.d.ts +2 -8
  356. package/dist/core/snapshots/creation.js +3 -12
  357. package/dist/core/utils/summarization-helpers.d.ts +0 -4
  358. package/dist/core/utils/summarization-helpers.js +1 -6
  359. package/dist/db/bootstrap.d.ts +2 -0
  360. package/dist/db/bootstrap.js +229 -280
  361. package/dist/db/drizzle/schema-sqlite.d.ts +702 -1
  362. package/dist/db/drizzle/schema-sqlite.js +83 -4
  363. package/dist/db/drizzle/schema.d.ts +653 -1
  364. package/dist/db/drizzle/schema.js +93 -4
  365. package/dist/db/migrations/associations.d.ts +6 -0
  366. package/dist/db/migrations/associations.js +29 -0
  367. package/dist/db/migrations/beliefs.d.ts +10 -0
  368. package/dist/db/migrations/beliefs.js +76 -0
  369. package/dist/db/migrations/core-memory.d.ts +6 -0
  370. package/dist/db/migrations/core-memory.js +29 -0
  371. package/dist/db/migrations/fts.d.ts +6 -0
  372. package/dist/db/migrations/fts.js +52 -0
  373. package/dist/db/migrations/index.d.ts +25 -0
  374. package/dist/db/migrations/index.js +51 -0
  375. package/dist/db/migrations/indexes.d.ts +6 -0
  376. package/dist/db/migrations/indexes.js +30 -0
  377. package/dist/db/migrations/learnings.d.ts +7 -0
  378. package/dist/db/migrations/learnings.js +26 -0
  379. package/dist/db/migrations/maintenance.d.ts +6 -0
  380. package/dist/db/migrations/maintenance.js +61 -0
  381. package/dist/db/migrations/memories.d.ts +7 -0
  382. package/dist/db/migrations/memories.js +16 -0
  383. package/dist/db/migrations/memory-places.d.ts +6 -0
  384. package/dist/db/migrations/memory-places.js +29 -0
  385. package/dist/db/migrations/places.d.ts +6 -0
  386. package/dist/db/migrations/places.js +43 -0
  387. package/dist/db/migrations/projects.d.ts +3 -0
  388. package/dist/db/migrations/projects.js +13 -0
  389. package/dist/db/migrations/tier-conversion.d.ts +7 -0
  390. package/dist/db/migrations/tier-conversion.js +20 -0
  391. package/dist/db/schema/beliefs.d.ts +9 -0
  392. package/dist/db/schema/beliefs.js +46 -0
  393. package/dist/db/schema/generator.d.ts +38 -0
  394. package/dist/db/schema/generator.js +108 -0
  395. package/dist/db/schema/index.d.ts +19 -20
  396. package/dist/db/schema/index.js +25 -79
  397. package/dist/db/schema/learnings.d.ts +7 -0
  398. package/dist/db/schema/learnings.js +30 -0
  399. package/dist/db/schema/memories.d.ts +7 -0
  400. package/dist/db/schema/memories.js +81 -0
  401. package/dist/db/schema/projects.d.ts +4 -0
  402. package/dist/db/schema/projects.js +31 -0
  403. package/dist/packages/mcp/src/index.d.ts +3 -0
  404. package/dist/packages/mcp/src/index.js +733 -0
  405. package/mcp.json.example +8 -11
  406. package/package.json +57 -76
  407. package/packages/cli/package.json +22 -0
  408. package/packages/cli/src/commands/clean.ts +68 -0
  409. package/packages/cli/src/commands/context.ts +79 -0
  410. package/packages/cli/src/commands/doctor.ts +357 -0
  411. package/packages/cli/src/commands/forget.ts +72 -0
  412. package/packages/cli/src/commands/health.ts +36 -0
  413. package/packages/cli/src/commands/inspect.ts +41 -0
  414. package/packages/cli/src/commands/link.ts +50 -0
  415. package/packages/cli/src/commands/migrate.ts +93 -0
  416. package/packages/cli/src/commands/recall.ts +99 -0
  417. package/packages/cli/src/commands/recent.ts +57 -0
  418. package/packages/cli/src/commands/remember.ts +139 -0
  419. package/packages/cli/src/commands/run.ts +58 -0
  420. package/packages/cli/src/commands/stale.ts +43 -0
  421. package/packages/cli/src/commands/stats.ts +42 -0
  422. package/packages/cli/src/index.ts +57 -0
  423. package/packages/cli/tsconfig.json +24 -0
  424. package/packages/mcp/package.json +26 -0
  425. package/packages/mcp/src/index.ts +877 -0
  426. package/packages/mcp/tsconfig.json +20 -0
  427. package/skills/squish-memory/SKILL.md +38 -35
  428. package/skills/squish-memory/{scripts/install.sh → install.sh} +1 -1
  429. package/skills/squish-memory/references/claude-desktop.json +12 -0
  430. package/skills/squish-memory/references/openclaw.json +13 -0
  431. package/skills/squish-memory/references/opencode.json +14 -0
  432. package/config/hooks/claude-code-hooks.json +0 -39
  433. package/config/hooks/cursor-hooks.json +0 -30
  434. package/config/hooks/opencode-hooks.json +0 -30
  435. package/config/hooks/windsurf-hooks.json +0 -30
  436. package/config/mcp-cli-fallback-policy.json +0 -22
  437. package/config/mcp.json +0 -38
  438. package/config/plugin-manifest.json +0 -101
  439. package/config/plugin-manifest.schema.json +0 -244
  440. package/config/plugin.json +0 -32
  441. package/config/remote-memory-policy.json +0 -32
  442. package/core/commands/context-paging.md +0 -51
  443. package/core/commands/context-status.md +0 -22
  444. package/core/commands/context.md +0 -5
  445. package/core/commands/core-memory.md +0 -56
  446. package/core/commands/health.md +0 -5
  447. package/core/commands/init.md +0 -39
  448. package/core/commands/merge.md +0 -113
  449. package/core/commands/recall.md +0 -5
  450. package/core/commands/remember.md +0 -11
  451. package/core/commands/search.md +0 -10
  452. package/dist/core/commands/managed-sync.d.ts +0 -10
  453. package/dist/core/commands/managed-sync.js +0 -64
  454. package/dist/core/external-folder/index.d.ts +0 -102
  455. package/dist/core/external-folder/index.js +0 -294
  456. package/dist/core/namespaces/index.d.ts +0 -71
  457. package/dist/core/namespaces/index.js +0 -305
  458. package/dist/core/namespaces/uri-parser.d.ts +0 -31
  459. package/dist/core/namespaces/uri-parser.js +0 -74
  460. package/dist/core/search/qmd-search.d.ts +0 -61
  461. package/dist/core/search/qmd-search.js +0 -178
  462. package/dist/core/session-hooks/self-iteration-job.d.ts +0 -20
  463. package/dist/core/session-hooks/self-iteration-job.js +0 -282
  464. package/dist/core/session-hooks/session-hooks.d.ts +0 -18
  465. package/dist/core/session-hooks/session-hooks.js +0 -58
  466. package/dist/core/snapshots.d.ts +0 -29
  467. package/dist/core/snapshots.js +0 -220
  468. package/dist/core/sync/qmd-sync.d.ts +0 -94
  469. package/dist/core/sync/qmd-sync.js +0 -201
  470. package/dist/index.d.ts +0 -7
  471. package/dist/index.js +0 -1677
  472. package/dist/vendor/sql.js/sql-wasm.wasm +0 -0
  473. package/dist/webui/server.d.ts +0 -5
  474. package/dist/webui/server.js +0 -642
  475. package/generated/mcp/manifest.json +0 -23
  476. package/generated/mcp/mcp-servers.json +0 -25
  477. package/generated/mcp/mcporter.json +0 -34
  478. package/generated/mcp/openclaw-memory-qmd.json +0 -17
  479. package/generated/mcp/runtime.json +0 -12
  480. package/scripts/README.md +0 -60
  481. package/scripts/build-release.sh +0 -36
  482. package/scripts/check-secrets.js +0 -132
  483. package/scripts/copy-runtime-assets.mjs +0 -26
  484. package/scripts/generate-mcp.mjs +0 -264
  485. package/scripts/github-release.sh +0 -77
  486. package/scripts/init-dirs.mjs +0 -13
  487. package/scripts/install-claude-code.sh +0 -85
  488. package/scripts/install-cursor.sh +0 -56
  489. package/scripts/install-hooks.sh +0 -73
  490. package/scripts/install-interactive.mjs +0 -357
  491. package/scripts/install-opencode.sh +0 -75
  492. package/scripts/install-plugin.mjs +0 -415
  493. package/scripts/install-windsurf.sh +0 -67
  494. package/scripts/remote-preflight.mjs +0 -62
  495. package/scripts/squish-fallback.mjs +0 -132
  496. package/scripts/test-interactive.mjs +0 -131
  497. package/scripts/verify-mcp.mjs +0 -214
  498. package/skills/squish-memory/scripts/install.mjs +0 -335
  499. package/skills/squish-memory/write_skill.js +0 -2
@@ -0,0 +1,581 @@
1
+ /** Cron Scheduler - Persistent cron-based job scheduling with fallback support */
2
+
3
+ import cron from 'node-cron';
4
+ import { selfIterationHandler } from '../session/self-iteration-job.js';
5
+ import { runLifecycleMaintenance } from '../lifecycle.js';
6
+ import { logger } from '../logger.js';
7
+ import { config } from '../../config.js';
8
+ import { getDb } from '../../db/index.js';
9
+ import { maintenanceJobs, maintenanceJobHistory } from '../../db/drizzle/schema-sqlite.js';
10
+ import { eq } from 'drizzle-orm';
11
+
12
+ export type JobType = 'nightly' | 'weekly' | 'hourly' | 'daily';
13
+ export type JobStatus = 'success' | 'failed' | 'skipped';
14
+
15
+ export interface ScheduledJob {
16
+ id: string;
17
+ jobName: string;
18
+ jobType: JobType;
19
+ cronExpression: string;
20
+ enabled: boolean;
21
+ lastRunAt: Date | null;
22
+ nextRunAt: Date | null;
23
+ jobConfig: Record<string, unknown>;
24
+ }
25
+
26
+ export interface JobExecutionContext {
27
+ jobId: string;
28
+ jobName: string;
29
+ jobType: JobType;
30
+ config: Record<string, unknown>;
31
+ startedAt: Date;
32
+ }
33
+
34
+ export type JobHandler = (context: JobExecutionContext) => Promise<{ recordsProcessed: number; summary: Record<string, unknown> }>;
35
+
36
+ const jobHandlers = new Map<string, JobHandler>();
37
+ const activeTasks = new Map<string, any>(); // node-cron ScheduledTask type
38
+
39
+ // Job interval by type (in ms) - used for catch-up detection
40
+ const JOB_INTERVALS: Record<JobType, number> = {
41
+ hourly: 60 * 60 * 1000, // 1 hour
42
+ daily: 24 * 60 * 60 * 1000, // 24 hours
43
+ nightly: 24 * 60 * 60 * 1000, // 24 hours (same as daily)
44
+ weekly: 7 * 24 * 60 * 60 * 1000, // 7 days
45
+ };
46
+
47
+ export function registerJobHandler(jobName: string, handler: JobHandler): void {
48
+ jobHandlers.set(jobName, handler);
49
+ logger.info(`[Scheduler] Registered handler for job: ${jobName}`);
50
+ }
51
+
52
+ // Register self-iteration job handler
53
+ registerJobHandler('self_iteration', selfIterationHandler);
54
+
55
+ // Decay job handler - runs lifecycle maintenance (decay, tier updates, eviction)
56
+ const decayHandler = async (context: JobExecutionContext) => {
57
+ const stats = await runLifecycleMaintenance();
58
+ return {
59
+ recordsProcessed: stats.decayed + stats.expired + stats.evicted,
60
+ summary: {
61
+ decayed: stats.decayed,
62
+ expired: stats.expired,
63
+ evicted: stats.evicted,
64
+ tierChanges: stats.tierChanges,
65
+ },
66
+ };
67
+ };
68
+ registerJobHandler('decay_maintenance', decayHandler);
69
+
70
+ // Belief decay handler - applies confidence decay to beliefs
71
+ const beliefDecayHandler = async (context: JobExecutionContext) => {
72
+ const { applyBeliefDecay } = await import('../beliefs/decay.js');
73
+ const stats = await applyBeliefDecay();
74
+ return {
75
+ recordsProcessed: stats.decayed + stats.sourceCountUpdated,
76
+ summary: {
77
+ beliefDecayed: stats.decayed,
78
+ sourceCountUpdated: stats.sourceCountUpdated,
79
+ errors: stats.errors,
80
+ },
81
+ };
82
+ };
83
+ registerJobHandler('belief_decay', beliefDecayHandler);
84
+
85
+ // Auto-clean handler - deletes stale memories automatically
86
+ const autoCleanHandler = async (context: JobExecutionContext) => {
87
+ const { getStaleMemories, deleteMemoryPermanently } = await import('../memory/stale-cleaner.js');
88
+ const { getAllProjects } = await import('../projects.js');
89
+
90
+ const jobConfig = context.config as {
91
+ enabled?: boolean;
92
+ olderThanDays?: number;
93
+ confidenceLevel?: string[];
94
+ minImportance?: number;
95
+ dryRun?: boolean;
96
+ };
97
+
98
+ if (jobConfig.enabled === false) {
99
+ return { recordsProcessed: 0, summary: { skipped: true, reason: 'auto-clean disabled' } };
100
+ }
101
+
102
+ const olderThanDays = jobConfig.olderThanDays || 30;
103
+ const confidenceLevels = jobConfig.confidenceLevel || ['outdated', 'speculative'];
104
+ const minImportance = jobConfig.minImportance || 40;
105
+ const dryRun = jobConfig.dryRun !== undefined ? jobConfig.dryRun : false; // Default to actual delete for safety
106
+
107
+ const projects = await getAllProjects();
108
+ let totalStale = 0;
109
+ let totalDeleted = 0;
110
+
111
+ for (const project of projects) {
112
+ const stale = await getStaleMemories({
113
+ olderThanDays,
114
+ confidenceLevels,
115
+ minImportance,
116
+ projectId: project.id,
117
+ });
118
+
119
+ totalStale += stale.length;
120
+
121
+ if (dryRun) {
122
+ logger.info(`[AutoClean] Would delete ${stale.length} stale memories in ${project.path}`);
123
+ } else {
124
+ for (const memory of stale) {
125
+ if (!memory.isPinned) {
126
+ await deleteMemoryPermanently(memory.id);
127
+ totalDeleted++;
128
+ }
129
+ }
130
+ logger.info(`[AutoClean] Deleted ${stale.length} stale memories in ${project.path}`);
131
+ }
132
+ }
133
+
134
+ return {
135
+ recordsProcessed: dryRun ? totalStale : totalDeleted,
136
+ summary: {
137
+ mode: dryRun ? 'dry-run' : 'deleted',
138
+ projectsScanned: projects.length,
139
+ memoriesAffected: dryRun ? totalStale : totalDeleted,
140
+ criteria: { olderThanDays, confidenceLevels, minImportance },
141
+ },
142
+ };
143
+ };
144
+ registerJobHandler('auto_clean', autoCleanHandler);
145
+
146
+ export async function initializeScheduler(): Promise<void> {
147
+ if (!config.cronEnabled) {
148
+ logger.info('[Scheduler] Cron scheduling disabled, using heartbeat fallback');
149
+ return;
150
+ }
151
+
152
+ const db = await getDb();
153
+ if (!db) {
154
+ logger.warn('[Scheduler] Database not available, scheduler disabled');
155
+ return;
156
+ }
157
+
158
+ try {
159
+ await ensureDefaultJobs(db);
160
+
161
+ // Check for missed jobs (catch-up after machine wake from sleep)
162
+ await checkMissedJobs();
163
+
164
+ const sqliteDb = db as any;
165
+ const jobs = await sqliteDb
166
+ .select()
167
+ .from(maintenanceJobs)
168
+ .where(eq(maintenanceJobs.enabled, true));
169
+
170
+ for (const job of jobs) {
171
+ await scheduleJob(job as unknown as ScheduledJob);
172
+ }
173
+
174
+ logger.info(`[Scheduler] Initialized with ${jobs.length} scheduled jobs`);
175
+ } catch (error) {
176
+ logger.error('[Scheduler] Failed to initialize:', error);
177
+ }
178
+ }
179
+
180
+ /**
181
+ * Check for missed jobs and execute catch-up if needed
182
+ * Called on scheduler initialization (including after machine wake from sleep)
183
+ */
184
+ async function checkMissedJobs(): Promise<void> {
185
+ try {
186
+ const db = await getDb();
187
+ if (!db) return;
188
+
189
+ const sqliteDb = db as any;
190
+ const jobs = await sqliteDb
191
+ .select()
192
+ .from(maintenanceJobs)
193
+ .where(eq(maintenanceJobs.enabled, true));
194
+
195
+ const now = Date.now();
196
+
197
+ for (const job of jobs) {
198
+ const intervalMs = JOB_INTERVALS[job.jobType as JobType];
199
+ if (!intervalMs) continue;
200
+
201
+ const lastRun = job.lastRunAt ? new Date(job.lastRunAt).getTime() : 0;
202
+ const elapsed = lastRun > 0 ? now - lastRun : intervalMs * 2; // If never run, treat as overdue
203
+ const gracePeriod = intervalMs * 1.5; // 1.5x interval grace
204
+
205
+ if (elapsed > gracePeriod) {
206
+ logger.info(`[Scheduler] Catch-up needed for ${job.jobName}, elapsed ${Math.round(elapsed / (60 * 60 * 1000))}h (grace: ${Math.round(gracePeriod / (60 * 60 * 1000))}h)`);
207
+
208
+ // Execute catch-up
209
+ const handler = jobHandlers.get(job.jobName);
210
+ if (handler) {
211
+ try {
212
+ const startedAt = new Date();
213
+ const context: JobExecutionContext = {
214
+ jobId: job.id,
215
+ jobName: job.jobName,
216
+ jobType: job.jobType as JobType,
217
+ config: typeof job.jobConfig === 'string'
218
+ ? JSON.parse(job.jobConfig)
219
+ : (job.jobConfig as Record<string, unknown>) ?? {},
220
+ startedAt,
221
+ };
222
+
223
+ await handler(context);
224
+
225
+ // Record successful catch-up run
226
+ const completedAt = new Date();
227
+ await sqliteDb
228
+ .update(maintenanceJobs)
229
+ .set({
230
+ lastRunAt: completedAt,
231
+ lastRunStatus: 'success' as const,
232
+ lastRunDuration: completedAt.getTime() - startedAt.getTime(),
233
+ })
234
+ .where(eq(maintenanceJobs.id, job.id));
235
+
236
+ logger.info(`[Scheduler] Catch-up completed for ${job.jobName}`);
237
+ } catch (catchError) {
238
+ const msg = catchError instanceof Error ? catchError.message : String(catchError);
239
+ logger.error(`[Scheduler] Catch-up failed for ${job.jobName}:`, msg);
240
+
241
+ await sqliteDb
242
+ .update(maintenanceJobs)
243
+ .set({
244
+ lastRunAt: new Date(),
245
+ lastRunStatus: 'failed' as const,
246
+ lastRunError: msg,
247
+ })
248
+ .where(eq(maintenanceJobs.id, job.id));
249
+ }
250
+ }
251
+ }
252
+ }
253
+ } catch (error) {
254
+ const msg = error instanceof Error ? error.message : String(error);
255
+ logger.error('[Scheduler] Error checking missed jobs:', msg);
256
+ }
257
+ }
258
+
259
+ async function ensureDefaultJobs(db: any): Promise<void> {
260
+ const defaultJobs = [
261
+ {
262
+ jobName: 'decay_maintenance',
263
+ jobType: 'hourly' as JobType,
264
+ cronExpression: '0 * * * *', // Run every hour at :00
265
+ enabled: true,
266
+ jobConfig: { applyDecay: true, updateTiers: true, evictOld: true },
267
+ },
268
+ {
269
+ jobName: 'belief_decay',
270
+ jobType: 'daily' as JobType,
271
+ cronExpression: '0 4 * * *', // Run daily at 4 AM
272
+ enabled: true,
273
+ jobConfig: { applyBeliefDecay: true },
274
+ },
275
+ {
276
+ jobName: 'nightly_maintenance',
277
+ jobType: 'nightly' as JobType,
278
+ cronExpression: '0 2 * * *',
279
+ enabled: true,
280
+ jobConfig: { mergeDuplicates: true, boostAccessed: true, decayScores: true },
281
+ },
282
+ {
283
+ jobName: 'weekly_maintenance',
284
+ jobType: 'weekly' as JobType,
285
+ cronExpression: '0 3 * * 0',
286
+ enabled: true,
287
+ jobConfig: { regenerateSummaries: true, archiveStale: true, cleanupOrphaned: true },
288
+ },
289
+ {
290
+ jobName: 'self_iteration',
291
+ jobType: 'hourly' as JobType,
292
+ cronExpression: '30 * * * *', // Run every hour at :30
293
+ enabled: true,
294
+ jobConfig: { minMessageCount: 5, maxMessagesToProcess: 50 },
295
+ },
296
+ {
297
+ jobName: 'auto_clean',
298
+ jobType: 'daily' as JobType,
299
+ cronExpression: '0 3 * * *', // Run daily at 3 AM
300
+ enabled: true,
301
+ jobConfig: {
302
+ enabled: true,
303
+ olderThanDays: 30,
304
+ confidenceLevel: ['outdated', 'speculative'],
305
+ minImportance: 40,
306
+ dryRun: true, // Start with dry-run for safety
307
+ },
308
+ },
309
+ ];
310
+
311
+ for (const job of defaultJobs) {
312
+ let existing;
313
+ try {
314
+ existing = await db
315
+ .select()
316
+ .from(maintenanceJobs)
317
+ .where(eq((maintenanceJobs as any).jobName, job.jobName))
318
+ .limit(1);
319
+ } catch (queryError: any) {
320
+ logger.error(`[Scheduler] Query failed for job ${job.jobName}:`, queryError.message);
321
+ // Try raw SQL fallback
322
+ try {
323
+ const rawDb = (db as any).$client;
324
+ if (rawDb && typeof rawDb.prepare === 'function') {
325
+ existing = rawDb.prepare('SELECT * FROM maintenance_jobs WHERE job_name = ?').all(job.jobName);
326
+ }
327
+ } catch (fallbackError: any) {
328
+ logger.error(`[Scheduler] Fallback query also failed:`, fallbackError.message);
329
+ throw queryError;
330
+ }
331
+ }
332
+
333
+ if (existing.length === 0) {
334
+ try {
335
+ await db.insert(maintenanceJobs).values({
336
+ jobName: job.jobName,
337
+ jobType: job.jobType,
338
+ cronExpression: job.cronExpression,
339
+ enabled: job.enabled,
340
+ jobConfig: job.jobConfig,
341
+ totalRuns: 0,
342
+ successCount: 0,
343
+ failureCount: 0,
344
+ lastRunAt: null,
345
+ nextRunAt: null,
346
+ lastRunDuration: null,
347
+ lastRunStatus: null,
348
+ lastRunError: null,
349
+ });
350
+ } catch (insertError: any) {
351
+ // Fallback to raw SQL if drizzle insert fails
352
+ logger.warn(`[Scheduler] Drizzle insert failed, using raw SQL: ${insertError.message}`);
353
+ const rawDb = (db as any).$client;
354
+ if (rawDb && typeof rawDb.prepare === 'function') {
355
+ const stmt = rawDb.prepare(`
356
+ INSERT INTO maintenance_jobs
357
+ (id, job_name, job_type, cron_expression, enabled, job_config,
358
+ total_runs, success_count, failure_count, last_run_at, next_run_at,
359
+ last_run_duration, last_run_status, last_run_error)
360
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
361
+ `);
362
+ stmt.run(
363
+ crypto.randomUUID(),
364
+ job.jobName,
365
+ job.jobType,
366
+ job.cronExpression,
367
+ job.enabled ? 1 : 0,
368
+ JSON.stringify(job.jobConfig),
369
+ 0, 0, 0,
370
+ null, null, null, null, null
371
+ );
372
+ }
373
+ }
374
+ logger.info(`[Scheduler] Created default job: ${job.jobName}`);
375
+
376
+ // Register self-iteration handler
377
+ if (job.jobName === 'self_iteration') {
378
+ registerJobHandler('self_iteration', selfIterationHandler);
379
+ }
380
+ }
381
+ }
382
+ }
383
+
384
+ export async function scheduleJob(job: ScheduledJob): Promise<void> {
385
+ const existingTask = activeTasks.get(job.jobName);
386
+ if (existingTask) {
387
+ existingTask.stop();
388
+ activeTasks.delete(job.jobName);
389
+ }
390
+
391
+ if (!job.enabled || !job.cronExpression) {
392
+ logger.debug(`[Scheduler] Job ${job.jobName} is disabled or has no cron expression`);
393
+ return;
394
+ }
395
+
396
+ if (!cron.validate(job.cronExpression)) {
397
+ logger.error(`[Scheduler] Invalid cron expression for ${job.jobName}: ${job.cronExpression}`);
398
+ return;
399
+ }
400
+
401
+ const task = cron.schedule(job.cronExpression, async () => {
402
+ await executeJob(job);
403
+ }, {
404
+ timezone: 'UTC',
405
+ });
406
+
407
+ activeTasks.set(job.jobName, task);
408
+
409
+ const nextRun = getNextRunTime(job.cronExpression);
410
+ const db = await getDb();
411
+ if (db) {
412
+ const sqliteDb = db as any;
413
+ await sqliteDb
414
+ .update(maintenanceJobs)
415
+ .set({ nextRunAt: nextRun })
416
+ .where(eq(maintenanceJobs.id, job.id));
417
+ }
418
+
419
+ logger.info(`[Scheduler] Scheduled ${job.jobName} with cron: ${job.cronExpression}${nextRun ? `, next run: ${nextRun.toISOString()}` : ''}`);
420
+ }
421
+
422
+ export async function executeJob(job: ScheduledJob): Promise<void> {
423
+ const db = await getDb();
424
+ const handler = jobHandlers.get(job.jobName);
425
+
426
+ if (!handler) {
427
+ logger.warn(`[Scheduler] No handler registered for job: ${job.jobName}`);
428
+ return;
429
+ }
430
+
431
+ const startedAt = new Date();
432
+ let status: JobStatus = 'success';
433
+ let error: string | null = null;
434
+ let recordsProcessed = 0;
435
+ let summary: Record<string, unknown> = {};
436
+
437
+ try {
438
+ logger.info(`[Scheduler] Executing job: ${job.jobName}`);
439
+
440
+ const result = await handler({
441
+ jobId: job.id,
442
+ jobName: job.jobName,
443
+ jobType: job.jobType,
444
+ config: job.jobConfig || {},
445
+ startedAt,
446
+ });
447
+
448
+ recordsProcessed = result.recordsProcessed;
449
+ summary = result.summary;
450
+
451
+ logger.info(`[Scheduler] Job ${job.jobName} completed: ${recordsProcessed} records processed`);
452
+ } catch (err) {
453
+ status = 'failed';
454
+ error = err instanceof Error ? err.message : String(err);
455
+ logger.error(`[Scheduler] Job ${job.jobName} failed:`, error);
456
+ }
457
+
458
+ const completedAt = new Date();
459
+
460
+ if (db) {
461
+ const sqliteDb = db as any;
462
+ const [currentJob] = await sqliteDb
463
+ .select()
464
+ .from(maintenanceJobs)
465
+ .where(eq(maintenanceJobs.id, job.id));
466
+
467
+ await sqliteDb
468
+ .update(maintenanceJobs)
469
+ .set({
470
+ lastRunAt: startedAt,
471
+ lastRunStatus: status,
472
+ lastRunError: error,
473
+ lastRunDuration: completedAt.getTime() - startedAt.getTime(),
474
+ totalRuns: (currentJob?.totalRuns ?? 0) + 1,
475
+ successCount: status === 'success' ? (currentJob?.successCount ?? 0) + 1 : currentJob?.successCount,
476
+ failureCount: status === 'failed' ? (currentJob?.failureCount ?? 0) + 1 : currentJob?.failureCount,
477
+ nextRunAt: job.cronExpression ? getNextRunTime(job.cronExpression) : null,
478
+ })
479
+ .where(eq(maintenanceJobs.id, job.id));
480
+
481
+ await sqliteDb.insert(maintenanceJobHistory).values({
482
+ jobId: job.id,
483
+ startedAt,
484
+ completedAt,
485
+ duration: completedAt.getTime() - startedAt.getTime(),
486
+ status,
487
+ error,
488
+ recordsProcessed,
489
+ resultSummary: summary,
490
+ });
491
+ }
492
+ }
493
+
494
+ function getNextRunTime(cronExpression: string): Date | null {
495
+ try {
496
+ const now = new Date();
497
+ const parts = cronExpression.split(' ');
498
+
499
+ // Daily jobs: MM HH * * *
500
+ if (parts[2] === '*' && parts[3] === '*' && parts[4] === '*' && parts[1] !== '*') {
501
+ const next = new Date(now);
502
+ next.setHours(parseInt(parts[1]), parseInt(parts[0]), 0, 0);
503
+ if (next <= now) next.setDate(next.getDate() + 1);
504
+ return next;
505
+ }
506
+
507
+ // Hourly jobs: MM * * * *
508
+ if (parts[1] === '*' && parts[2] === '*' && parts[3] === '*' && parts[4] === '*') {
509
+ const next = new Date(now);
510
+ next.setMinutes(parseInt(parts[0]), 0, 0);
511
+ if (next <= now) next.setHours(next.getHours() + 1);
512
+ return next;
513
+ }
514
+
515
+ // Weekly jobs: MM HH * * D
516
+ if (parts[4] !== '*') {
517
+ const dayOfWeek = parseInt(parts[4]);
518
+ const next = new Date(now);
519
+ next.setHours(parseInt(parts[1]), parseInt(parts[0]), 0, 0);
520
+ const daysUntil = (dayOfWeek - next.getDay() + 7) % 7;
521
+ next.setDate(next.getDate() + (daysUntil === 0 && next > now ? 7 : daysUntil));
522
+ return next;
523
+ }
524
+
525
+ return null;
526
+ } catch {
527
+ return null;
528
+ }
529
+ }
530
+
531
+ export async function getScheduledJobs(): Promise<ScheduledJob[]> {
532
+ const db = await getDb();
533
+ if (!db) return [];
534
+
535
+ const sqliteDb = db as any;
536
+ const jobs = await sqliteDb.select().from(maintenanceJobs);
537
+ return jobs.map((job: typeof maintenanceJobs.$inferSelect) => ({
538
+ id: job.id,
539
+ jobName: job.jobName,
540
+ jobType: job.jobType as JobType,
541
+ cronExpression: job.cronExpression || '',
542
+ enabled: job.enabled ?? true,
543
+ lastRunAt: job.lastRunAt ? new Date(job.lastRunAt) : null,
544
+ nextRunAt: job.nextRunAt ? new Date(job.nextRunAt) : null,
545
+ jobConfig: (job.jobConfig as Record<string, unknown>) || {},
546
+ }));
547
+ }
548
+
549
+ export async function getOverdueJobs(): Promise<ScheduledJob[]> {
550
+ const db = await getDb();
551
+ if (!db) return [];
552
+
553
+ const now = new Date();
554
+ const sqliteDb = db as any;
555
+ const jobs = await sqliteDb.select().from(maintenanceJobs);
556
+
557
+ return jobs
558
+ .filter((job: typeof maintenanceJobs.$inferSelect) => {
559
+ if (!job.enabled) return false;
560
+ if (!job.nextRunAt) return true;
561
+ return new Date(job.nextRunAt) < now;
562
+ })
563
+ .map((job: typeof maintenanceJobs.$inferSelect) => ({
564
+ id: job.id,
565
+ jobName: job.jobName,
566
+ jobType: job.jobType as JobType,
567
+ cronExpression: job.cronExpression || '',
568
+ enabled: job.enabled ?? true,
569
+ lastRunAt: job.lastRunAt ? new Date(job.lastRunAt) : null,
570
+ nextRunAt: job.nextRunAt ? new Date(job.nextRunAt) : null,
571
+ jobConfig: (job.jobConfig as Record<string, unknown>) || {},
572
+ }));
573
+ }
574
+
575
+ export function stopAllJobs(): void {
576
+ for (const [name, task] of activeTasks) {
577
+ task.stop();
578
+ logger.info(`[Scheduler] Stopped job: ${name}`);
579
+ }
580
+ activeTasks.clear();
581
+ }
@@ -0,0 +1,91 @@
1
+ /** Heartbeat - Fallback job execution via heartbeat checking */
2
+
3
+ import { logger } from '../logger.js';
4
+ import { config } from '../../config.js';
5
+ import { getOverdueJobs, executeJob } from './cron-scheduler.js';
6
+
7
+ let lastHeartbeat: Date | null = null;
8
+ let heartbeatInterval: NodeJS.Timeout | null = null;
9
+
10
+ export async function heartbeat(): Promise<{
11
+ checked: boolean;
12
+ overdueCount: number;
13
+ executedJobs: string[];
14
+ }> {
15
+ const result = {
16
+ checked: false,
17
+ overdueCount: 0,
18
+ executedJobs: [] as string[],
19
+ };
20
+
21
+ if (config.schedulerMode === 'cron' && config.cronEnabled) {
22
+ logger.debug('[Heartbeat] Cron mode active, skipping heartbeat check');
23
+ return result;
24
+ }
25
+
26
+ lastHeartbeat = new Date();
27
+ result.checked = true;
28
+
29
+ try {
30
+ const overdueJobs = await getOverdueJobs();
31
+ result.overdueCount = overdueJobs.length;
32
+
33
+ if (overdueJobs.length === 0) {
34
+ logger.debug('[Heartbeat] No overdue jobs');
35
+ return result;
36
+ }
37
+
38
+ logger.info(`[Heartbeat] Found ${overdueJobs.length} overdue jobs, executing...`);
39
+
40
+ for (const job of overdueJobs) {
41
+ try {
42
+ await executeJob(job);
43
+ result.executedJobs.push(job.jobName);
44
+ logger.info(`[Heartbeat] Executed overdue job: ${job.jobName}`);
45
+ } catch (error) {
46
+ logger.error(`[Heartbeat] Failed to execute job ${job.jobName}:`, error);
47
+ }
48
+ }
49
+
50
+ return result;
51
+ } catch (error) {
52
+ logger.error('[Heartbeat] Failed to check overdue jobs:', error);
53
+ return result;
54
+ }
55
+ }
56
+
57
+ export function startHeartbeatChecking(): void {
58
+ if (config.schedulerMode === 'cron' && config.cronEnabled) {
59
+ logger.info('[Heartbeat] Cron mode active, heartbeat checking disabled');
60
+ return;
61
+ }
62
+
63
+ stopHeartbeatChecking();
64
+
65
+ heartbeatInterval = setInterval(async () => {
66
+ await heartbeat();
67
+ }, config.heartbeatInterval);
68
+
69
+ logger.info(`[Heartbeat] Started periodic checking (interval: ${config.heartbeatInterval}ms)`);
70
+
71
+ heartbeat().catch(err => {
72
+ logger.error('[Heartbeat] Initial heartbeat failed:', err);
73
+ });
74
+ }
75
+
76
+ export function stopHeartbeatChecking(): void {
77
+ if (heartbeatInterval) {
78
+ clearInterval(heartbeatInterval);
79
+ heartbeatInterval = null;
80
+ logger.info('[Heartbeat] Stopped periodic checking');
81
+ }
82
+ }
83
+
84
+ export function getLastHeartbeat(): Date | null {
85
+ return lastHeartbeat;
86
+ }
87
+
88
+ export function isHeartbeatDue(): boolean {
89
+ if (!lastHeartbeat) return true;
90
+ return Date.now() - lastHeartbeat.getTime() > config.heartbeatInterval;
91
+ }
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Scheduler Module
3
+ * Cron-based job scheduling with heartbeat fallback
4
+ */
5
+
6
+ export * from './cron-scheduler.js';
7
+ export * from './heartbeat.js';
8
+ export * from './job-runner.js';