squish-memory 1.0.1 → 1.1.5

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 (601) hide show
  1. package/.env.example +130 -0
  2. package/CHANGELOG.md +55 -0
  3. package/README.md +155 -274
  4. package/config/hooks/claude-code-hooks.json +39 -0
  5. package/config/hooks/cursor-hooks.json +30 -0
  6. package/config/hooks/opencode-hooks.json +30 -0
  7. package/config/hooks/windsurf-hooks.json +30 -0
  8. package/config/mcp-mode-semantics.json +23 -21
  9. package/config/plugin-manifest.json +101 -152
  10. package/{plugin.json → config/plugin.json} +2 -2
  11. package/config/settings.json +52 -51
  12. package/{commands → core/commands}/init.md +39 -39
  13. package/dist/config.d.ts +28 -4
  14. package/dist/config.js +97 -29
  15. package/dist/core/adapters/config/claude-code.d.ts +45 -0
  16. package/dist/core/adapters/config/claude-code.js +113 -0
  17. package/dist/core/adapters/config/cursor.d.ts +26 -0
  18. package/dist/core/adapters/config/cursor.js +74 -0
  19. package/dist/core/adapters/config/opencode.d.ts +23 -0
  20. package/dist/core/adapters/config/opencode.js +73 -0
  21. package/dist/core/adapters/config/windsurf.d.ts +26 -0
  22. package/dist/core/adapters/config/windsurf.js +74 -0
  23. package/dist/core/adapters/index.d.ts +45 -0
  24. package/dist/core/adapters/index.js +84 -0
  25. package/dist/core/adapters/scripts/install-adapter.d.ts +19 -0
  26. package/dist/core/adapters/scripts/install-adapter.js +149 -0
  27. package/dist/core/adapters/timeline.d.ts +23 -0
  28. package/dist/core/adapters/timeline.js +88 -0
  29. package/dist/core/adapters/types.d.ts +157 -0
  30. package/dist/core/adapters/types.js +50 -0
  31. package/dist/{algorithms → core/algorithms}/analytics/token-estimator.d.ts +1 -1
  32. package/dist/{algorithms → core/algorithms}/analytics/token-estimator.js +3 -3
  33. package/dist/{algorithms → core/algorithms}/detection/semantic-ranker.d.ts +1 -1
  34. package/dist/{algorithms → core/algorithms}/detection/semantic-ranker.js +1 -1
  35. package/dist/{algorithms → core/algorithms}/detection/two-stage-detector.d.ts +1 -1
  36. package/dist/{algorithms → core/algorithms}/detection/two-stage-detector.js +7 -10
  37. package/dist/{algorithms → core/algorithms}/handlers/approve-merge.js +4 -4
  38. package/dist/{algorithms → core/algorithms}/handlers/detect-duplicates.js +3 -3
  39. package/dist/{algorithms → core/algorithms}/handlers/get-stats.js +3 -3
  40. package/dist/{algorithms → core/algorithms}/handlers/list-proposals.js +3 -3
  41. package/dist/{algorithms → core/algorithms}/handlers/preview-merge.js +3 -3
  42. package/dist/{algorithms → core/algorithms}/handlers/reject-merge.js +3 -3
  43. package/dist/{algorithms → core/algorithms}/handlers/reverse-merge.js +3 -3
  44. package/dist/core/algorithms/index.d.ts +21 -0
  45. package/dist/core/algorithms/index.js +26 -0
  46. package/dist/core/algorithms/operations/cache-maintenance.d.ts +12 -0
  47. package/dist/core/algorithms/operations/cache-maintenance.js +157 -0
  48. package/dist/{algorithms → core/algorithms}/safety/safety-checks.d.ts +1 -1
  49. package/dist/{algorithms → core/algorithms}/strategies/merge-strategies.d.ts +19 -1
  50. package/dist/{algorithms → core/algorithms}/strategies/merge-strategies.js +74 -123
  51. package/dist/core/algorithms/types.d.ts +133 -0
  52. package/dist/core/algorithms/types.js +5 -0
  53. package/dist/core/associations.d.ts +1 -2
  54. package/dist/core/associations.js +1 -2
  55. package/dist/core/autosave.d.ts +19 -0
  56. package/dist/core/autosave.js +16 -0
  57. package/dist/{commands → core/commands}/managed-sync.js +5 -5
  58. package/dist/core/commands/mcp-server.js +739 -0
  59. package/dist/core/context/agent-context.d.ts +106 -0
  60. package/dist/core/context/agent-context.js +274 -0
  61. package/dist/core/{context-paging.d.ts → context/context-paging.d.ts} +2 -12
  62. package/dist/core/{context-paging.js → context/context-paging.js} +19 -39
  63. package/dist/core/context/context-window.d.ts +40 -0
  64. package/dist/core/context/context-window.js +177 -0
  65. package/dist/core/context/context.js +22 -0
  66. package/dist/core/embeddings.d.ts +1 -1
  67. package/dist/core/embeddings.js +54 -2
  68. package/dist/core/error-handling.d.ts +63 -0
  69. package/dist/core/error-handling.js +173 -0
  70. package/dist/core/external-folder/index.d.ts +102 -0
  71. package/dist/core/external-folder/index.js +294 -0
  72. package/dist/core/hooks/agent-hooks.d.ts +74 -0
  73. package/dist/core/hooks/agent-hooks.js +244 -0
  74. package/dist/core/hooks/auto-tagger.d.ts +19 -0
  75. package/dist/core/hooks/auto-tagger.js +155 -0
  76. package/dist/core/hooks/capture-filter.d.ts +41 -0
  77. package/dist/core/hooks/capture-filter.js +128 -0
  78. package/dist/core/index.d.ts +6 -6
  79. package/dist/core/index.js +6 -6
  80. package/dist/core/{agent-memory.js → ingestion/agent-memory.js} +5 -7
  81. package/dist/core/{core-memory.js → ingestion/core-memory.js} +4 -4
  82. package/dist/core/ingestion/learnings.d.ts +57 -0
  83. package/dist/core/ingestion/learnings.js +202 -0
  84. package/dist/core/lib/db-client.d.ts +114 -0
  85. package/dist/core/lib/db-client.js +130 -0
  86. package/dist/core/lib/schemas.d.ts +129 -0
  87. package/dist/core/lib/schemas.js +87 -0
  88. package/dist/core/{utils.d.ts → lib/utils.d.ts} +1 -0
  89. package/dist/core/{utils.js → lib/utils.js} +31 -15
  90. package/dist/core/lib/validation.d.ts +38 -0
  91. package/dist/core/lib/validation.js +151 -0
  92. package/dist/core/lifecycle.d.ts +7 -0
  93. package/dist/core/lifecycle.js +140 -20
  94. package/dist/core/local-embeddings.d.ts +6 -1
  95. package/dist/core/local-embeddings.js +6 -15
  96. package/dist/core/logger.js +7 -1
  97. package/dist/core/mcp/server.js +27 -1
  98. package/dist/core/mcp/tools.js +35 -3
  99. package/dist/core/mcp/types.d.ts +4 -4
  100. package/dist/core/memory/categorizer.js +1 -0
  101. package/dist/core/memory/conflict-detector.js +1 -1
  102. package/dist/core/memory/consolidation.d.ts +1 -10
  103. package/dist/core/memory/consolidation.js +2 -11
  104. package/dist/core/memory/context-collector.js +1 -1
  105. package/dist/core/memory/edit-workflow.js +1 -1
  106. package/dist/core/memory/entity-resolver.js +7 -7
  107. package/dist/core/memory/fact-extractor.js +12 -12
  108. package/dist/core/memory/feedback-tracker.js +1 -1
  109. package/dist/core/memory/hooks.d.ts +88 -0
  110. package/dist/core/memory/hooks.js +174 -0
  111. package/dist/core/memory/hybrid-retrieval.js +2 -2
  112. package/dist/core/memory/hybrid-search.d.ts +1 -6
  113. package/dist/core/memory/hybrid-search.js +70 -84
  114. package/dist/core/memory/importance.d.ts +8 -13
  115. package/dist/core/memory/importance.js +47 -74
  116. package/dist/core/memory/loader.d.ts +31 -0
  117. package/dist/core/memory/loader.js +141 -0
  118. package/dist/core/memory/markdown/markdown-storage.d.ts +72 -0
  119. package/dist/core/memory/markdown/markdown-storage.js +243 -0
  120. package/dist/core/memory/memories.d.ts +12 -4
  121. package/dist/core/memory/memories.js +196 -181
  122. package/dist/core/memory/memory-lifecycle.d.ts +8 -0
  123. package/dist/core/memory/memory-lifecycle.js +55 -0
  124. package/dist/core/memory/migrate.d.ts +21 -0
  125. package/dist/core/memory/migrate.js +134 -0
  126. package/dist/core/memory/normalization.d.ts +22 -0
  127. package/dist/core/memory/normalization.js +26 -0
  128. package/dist/core/memory/progressive-disclosure.js +1 -1
  129. package/dist/core/memory/query-rewriter.js +9 -9
  130. package/dist/core/memory/serialization.d.ts +4 -0
  131. package/dist/core/memory/serialization.js +49 -0
  132. package/dist/core/memory/stats.d.ts +5 -0
  133. package/dist/core/memory/stats.js +63 -12
  134. package/dist/core/memory/temporal-facts.js +21 -0
  135. package/dist/core/memory/write-gate.js +1 -1
  136. package/dist/core/obsidian-vault.d.ts +30 -0
  137. package/dist/core/obsidian-vault.js +94 -0
  138. package/dist/core/places/index.d.ts +14 -0
  139. package/dist/core/places/index.js +14 -0
  140. package/dist/core/places/memory-places.d.ts +68 -0
  141. package/dist/core/places/memory-places.js +261 -0
  142. package/dist/core/places/places.d.ts +88 -0
  143. package/dist/core/places/places.js +314 -0
  144. package/dist/core/places/rules.d.ts +74 -0
  145. package/dist/core/places/rules.js +240 -0
  146. package/dist/core/places/walking.d.ts +56 -0
  147. package/dist/core/places/walking.js +121 -0
  148. package/dist/core/projects.d.ts +5 -0
  149. package/dist/core/projects.js +39 -18
  150. package/dist/core/responses.d.ts +96 -0
  151. package/dist/core/responses.js +122 -0
  152. package/dist/core/scheduler/cron-scheduler.js +89 -14
  153. package/dist/core/scheduler/index.d.ts +1 -1
  154. package/dist/core/scheduler/index.js +1 -1
  155. package/dist/core/scheduler/job-runner.js +1 -1
  156. package/dist/core/search/conversations.js +40 -42
  157. package/dist/core/search/entities.js +6 -9
  158. package/dist/core/search/graph-boost.d.ts +7 -0
  159. package/dist/core/search/graph-boost.js +23 -0
  160. package/dist/core/search/qmd-search.js +4 -4
  161. package/dist/core/security/encrypt.d.ts +6 -0
  162. package/dist/core/security/encrypt.js +47 -0
  163. package/dist/core/{governance.d.ts → security/governance.d.ts} +6 -1
  164. package/dist/core/security/governance.js +79 -0
  165. package/dist/core/session/auto-load.js +6 -6
  166. package/dist/core/session/index.d.ts +1 -1
  167. package/dist/core/session/index.js +1 -1
  168. package/dist/core/session/self-iteration-job.d.ts +20 -0
  169. package/dist/core/session/self-iteration-job.js +282 -0
  170. package/dist/core/session/session-hooks.d.ts +18 -0
  171. package/dist/core/session/session-hooks.js +58 -0
  172. package/dist/core/session-hooks/self-iteration-job.js +35 -35
  173. package/dist/core/{cache.js → storage/cache.js} +2 -2
  174. package/dist/core/sync/qmd-sync.d.ts +1 -13
  175. package/dist/core/sync/qmd-sync.js +1 -13
  176. package/dist/core/toon.d.ts +43 -0
  177. package/dist/core/toon.js +160 -0
  178. package/dist/core/utils/memory-operations.js +1 -1
  179. package/dist/core/utils/vector-operations.d.ts +71 -0
  180. package/dist/core/utils/vector-operations.js +129 -0
  181. package/dist/db/adapter.d.ts +3 -3
  182. package/dist/db/adapter.js +115 -36
  183. package/dist/db/bootstrap.js +927 -555
  184. package/dist/{drizzle → db/drizzle}/schema-sqlite.d.ts +74 -25
  185. package/dist/{drizzle → db/drizzle}/schema-sqlite.js +91 -24
  186. package/dist/{drizzle → db/drizzle}/schema.d.ts +79 -32
  187. package/dist/{drizzle → db/drizzle}/schema.js +106 -35
  188. package/dist/db/drizzle.config.d.ts +3 -0
  189. package/dist/db/drizzle.config.js +12 -0
  190. package/dist/db/index.d.ts +1 -5
  191. package/dist/db/index.js +51 -8
  192. package/dist/db/neon.d.ts +8 -0
  193. package/dist/db/neon.js +20 -0
  194. package/dist/db/schema/index.d.ts +40 -0
  195. package/dist/db/schema/index.js +105 -0
  196. package/dist/db/schema/tables/context-sessions.d.ts +9 -0
  197. package/dist/db/schema/tables/context-sessions.js +37 -0
  198. package/dist/db/schema/tables/conversations.d.ts +9 -0
  199. package/dist/db/schema/tables/conversations.js +47 -0
  200. package/dist/db/schema/tables/core-memory.d.ts +9 -0
  201. package/dist/db/schema/tables/core-memory.js +41 -0
  202. package/dist/db/schema/tables/entities.d.ts +9 -0
  203. package/dist/db/schema/tables/entities.js +39 -0
  204. package/dist/db/schema/tables/entity-relations.d.ts +9 -0
  205. package/dist/db/schema/tables/entity-relations.js +31 -0
  206. package/dist/db/schema/tables/learnings.d.ts +9 -0
  207. package/dist/db/schema/tables/learnings.js +66 -0
  208. package/dist/db/schema/tables/memories.d.ts +9 -0
  209. package/dist/db/schema/tables/memories.js +161 -0
  210. package/dist/db/schema/tables/memory-associations.d.ts +9 -0
  211. package/dist/db/schema/tables/memory-associations.js +39 -0
  212. package/dist/db/schema/tables/memory-hash-cache.d.ts +9 -0
  213. package/dist/db/schema/tables/memory-hash-cache.js +29 -0
  214. package/dist/db/schema/tables/memory-merge-history.d.ts +9 -0
  215. package/dist/db/schema/tables/memory-merge-history.js +33 -0
  216. package/dist/db/schema/tables/memory-merge-proposals.d.ts +9 -0
  217. package/dist/db/schema/tables/memory-merge-proposals.js +39 -0
  218. package/dist/db/schema/tables/messages.d.ts +9 -0
  219. package/dist/db/schema/tables/messages.js +41 -0
  220. package/dist/db/schema/tables/namespaces.d.ts +9 -0
  221. package/dist/db/schema/tables/namespaces.js +37 -0
  222. package/dist/db/schema/tables/projects.d.ts +9 -0
  223. package/dist/db/schema/tables/projects.js +31 -0
  224. package/dist/db/schema/tables/users.d.ts +9 -0
  225. package/dist/db/schema/tables/users.js +27 -0
  226. package/dist/db/schema.d.ts +1 -1
  227. package/dist/db/schema.js +2 -2
  228. package/dist/db/supabase.d.ts +9 -0
  229. package/dist/db/supabase.js +24 -0
  230. package/dist/index.d.ts +2 -14
  231. package/dist/index.js +1457 -631
  232. package/dist/vendor/sql.js/sql-wasm.wasm +0 -0
  233. package/dist/webui/server.d.ts +5 -0
  234. package/dist/{api/web/web.js → webui/server.js} +538 -509
  235. package/generated/mcp/manifest.json +23 -23
  236. package/{.mcp.json → mcp.json.example} +1 -1
  237. package/package.json +159 -169
  238. package/scripts/README.md +60 -0
  239. package/scripts/copy-runtime-assets.mjs +26 -0
  240. package/scripts/generate-mcp.mjs +264 -264
  241. package/scripts/github-release.sh +4 -4
  242. package/scripts/install-claude-code.sh +85 -0
  243. package/scripts/install-cursor.sh +56 -0
  244. package/scripts/install-hooks.sh +73 -0
  245. package/scripts/install-interactive.mjs +357 -674
  246. package/scripts/install-opencode.sh +75 -0
  247. package/scripts/install-windsurf.sh +67 -0
  248. package/skills/squish-memory/SKILL.md +104 -100
  249. package/skills/squish-memory/{install.mjs → scripts/install.mjs} +2 -2
  250. package/skills/squish-memory/{install.sh → scripts/install.sh} +2 -2
  251. package/skills/squish-memory/write_skill.js +2 -0
  252. package/.claude-plugin/marketplace.json +0 -20
  253. package/.claude-plugin/plugin.json +0 -32
  254. package/.env.mcp.example +0 -56
  255. package/QUICK-START.md +0 -71
  256. package/bin/squish-add.mjs +0 -32
  257. package/bin/squish-rm.mjs +0 -21
  258. package/commands/observe.md +0 -5
  259. package/dist/algorithms/analytics/token-estimator.d.ts.map +0 -1
  260. package/dist/algorithms/analytics/token-estimator.js.map +0 -1
  261. package/dist/algorithms/detection/hash-filters.d.ts.map +0 -1
  262. package/dist/algorithms/detection/hash-filters.js.map +0 -1
  263. package/dist/algorithms/detection/semantic-ranker.d.ts.map +0 -1
  264. package/dist/algorithms/detection/semantic-ranker.js.map +0 -1
  265. package/dist/algorithms/detection/two-stage-detector.d.ts.map +0 -1
  266. package/dist/algorithms/detection/two-stage-detector.js.map +0 -1
  267. package/dist/algorithms/handlers/approve-merge.d.ts.map +0 -1
  268. package/dist/algorithms/handlers/approve-merge.js.map +0 -1
  269. package/dist/algorithms/handlers/detect-duplicates.d.ts.map +0 -1
  270. package/dist/algorithms/handlers/detect-duplicates.js.map +0 -1
  271. package/dist/algorithms/handlers/get-stats.d.ts.map +0 -1
  272. package/dist/algorithms/handlers/get-stats.js.map +0 -1
  273. package/dist/algorithms/handlers/list-proposals.d.ts.map +0 -1
  274. package/dist/algorithms/handlers/list-proposals.js.map +0 -1
  275. package/dist/algorithms/handlers/preview-merge.d.ts.map +0 -1
  276. package/dist/algorithms/handlers/preview-merge.js.map +0 -1
  277. package/dist/algorithms/handlers/reject-merge.d.ts.map +0 -1
  278. package/dist/algorithms/handlers/reject-merge.js.map +0 -1
  279. package/dist/algorithms/handlers/reverse-merge.d.ts.map +0 -1
  280. package/dist/algorithms/handlers/reverse-merge.js.map +0 -1
  281. package/dist/algorithms/safety/safety-checks.d.ts.map +0 -1
  282. package/dist/algorithms/safety/safety-checks.js.map +0 -1
  283. package/dist/algorithms/strategies/merge-strategies.d.ts.map +0 -1
  284. package/dist/algorithms/strategies/merge-strategies.js.map +0 -1
  285. package/dist/algorithms/utils/response-builder.d.ts.map +0 -1
  286. package/dist/algorithms/utils/response-builder.js.map +0 -1
  287. package/dist/api/web/index.d.ts +0 -3
  288. package/dist/api/web/index.d.ts.map +0 -1
  289. package/dist/api/web/index.js +0 -4
  290. package/dist/api/web/index.js.map +0 -1
  291. package/dist/api/web/web-server.d.ts +0 -3
  292. package/dist/api/web/web-server.d.ts.map +0 -1
  293. package/dist/api/web/web-server.js +0 -6
  294. package/dist/api/web/web-server.js.map +0 -1
  295. package/dist/api/web/web.d.ts +0 -4
  296. package/dist/api/web/web.d.ts.map +0 -1
  297. package/dist/api/web/web.js.map +0 -1
  298. package/dist/commands/managed-sync.d.ts.map +0 -1
  299. package/dist/commands/managed-sync.js.map +0 -1
  300. package/dist/commands/mcp-server.d.ts.map +0 -1
  301. package/dist/commands/mcp-server.js +0 -393
  302. package/dist/commands/mcp-server.js.map +0 -1
  303. package/dist/config.d.ts.map +0 -1
  304. package/dist/config.js.map +0 -1
  305. package/dist/core/agent-memory.d.ts.map +0 -1
  306. package/dist/core/agent-memory.js.map +0 -1
  307. package/dist/core/associations.d.ts.map +0 -1
  308. package/dist/core/associations.js.map +0 -1
  309. package/dist/core/cache.d.ts.map +0 -1
  310. package/dist/core/cache.js.map +0 -1
  311. package/dist/core/consolidation.d.ts.map +0 -1
  312. package/dist/core/consolidation.js.map +0 -1
  313. package/dist/core/context-paging.d.ts.map +0 -1
  314. package/dist/core/context-paging.js.map +0 -1
  315. package/dist/core/context.d.ts.map +0 -1
  316. package/dist/core/context.js +0 -24
  317. package/dist/core/context.js.map +0 -1
  318. package/dist/core/core-memory.d.ts.map +0 -1
  319. package/dist/core/core-memory.js.map +0 -1
  320. package/dist/core/database.d.ts.map +0 -1
  321. package/dist/core/database.js.map +0 -1
  322. package/dist/core/embeddings/google-multimodal.d.ts.map +0 -1
  323. package/dist/core/embeddings/google-multimodal.js.map +0 -1
  324. package/dist/core/embeddings/qmd-client.d.ts.map +0 -1
  325. package/dist/core/embeddings/qmd-client.js.map +0 -1
  326. package/dist/core/embeddings.d.ts.map +0 -1
  327. package/dist/core/embeddings.js.map +0 -1
  328. package/dist/core/governance.d.ts.map +0 -1
  329. package/dist/core/governance.js +0 -64
  330. package/dist/core/governance.js.map +0 -1
  331. package/dist/core/index.d.ts.map +0 -1
  332. package/dist/core/index.js.map +0 -1
  333. package/dist/core/layers/generator.d.ts.map +0 -1
  334. package/dist/core/layers/generator.js.map +0 -1
  335. package/dist/core/lifecycle.d.ts.map +0 -1
  336. package/dist/core/lifecycle.js.map +0 -1
  337. package/dist/core/local-embeddings.d.ts.map +0 -1
  338. package/dist/core/local-embeddings.js.map +0 -1
  339. package/dist/core/logger.d.ts.map +0 -1
  340. package/dist/core/logger.js.map +0 -1
  341. package/dist/core/mcp/client.d.ts.map +0 -1
  342. package/dist/core/mcp/client.js.map +0 -1
  343. package/dist/core/mcp/index.d.ts.map +0 -1
  344. package/dist/core/mcp/index.js.map +0 -1
  345. package/dist/core/mcp/server.d.ts.map +0 -1
  346. package/dist/core/mcp/server.js.map +0 -1
  347. package/dist/core/mcp/standalone-server.d.ts.map +0 -1
  348. package/dist/core/mcp/standalone-server.js.map +0 -1
  349. package/dist/core/mcp/tools.d.ts.map +0 -1
  350. package/dist/core/mcp/tools.js.map +0 -1
  351. package/dist/core/mcp/types.d.ts.map +0 -1
  352. package/dist/core/mcp/types.js.map +0 -1
  353. package/dist/core/memory/bridge-discovery.d.ts.map +0 -1
  354. package/dist/core/memory/bridge-discovery.js.map +0 -1
  355. package/dist/core/memory/categorizer.d.ts.map +0 -1
  356. package/dist/core/memory/categorizer.js.map +0 -1
  357. package/dist/core/memory/conflict-detector.d.ts.map +0 -1
  358. package/dist/core/memory/conflict-detector.js.map +0 -1
  359. package/dist/core/memory/consolidation.d.ts.map +0 -1
  360. package/dist/core/memory/consolidation.js.map +0 -1
  361. package/dist/core/memory/context-collector.d.ts.map +0 -1
  362. package/dist/core/memory/context-collector.js.map +0 -1
  363. package/dist/core/memory/contradiction-resolver.d.ts.map +0 -1
  364. package/dist/core/memory/contradiction-resolver.js.map +0 -1
  365. package/dist/core/memory/edit-workflow.d.ts.map +0 -1
  366. package/dist/core/memory/edit-workflow.js.map +0 -1
  367. package/dist/core/memory/entity-extractor.d.ts.map +0 -1
  368. package/dist/core/memory/entity-extractor.js.map +0 -1
  369. package/dist/core/memory/entity-resolver.d.ts.map +0 -1
  370. package/dist/core/memory/entity-resolver.js.map +0 -1
  371. package/dist/core/memory/fact-extractor.d.ts.map +0 -1
  372. package/dist/core/memory/fact-extractor.js.map +0 -1
  373. package/dist/core/memory/feedback-tracker.d.ts.map +0 -1
  374. package/dist/core/memory/feedback-tracker.js.map +0 -1
  375. package/dist/core/memory/hybrid-retrieval.d.ts.map +0 -1
  376. package/dist/core/memory/hybrid-retrieval.js.map +0 -1
  377. package/dist/core/memory/hybrid-scorer.d.ts.map +0 -1
  378. package/dist/core/memory/hybrid-scorer.js.map +0 -1
  379. package/dist/core/memory/hybrid-search.d.ts.map +0 -1
  380. package/dist/core/memory/hybrid-search.js.map +0 -1
  381. package/dist/core/memory/importance.d.ts.map +0 -1
  382. package/dist/core/memory/importance.js.map +0 -1
  383. package/dist/core/memory/index.d.ts.map +0 -1
  384. package/dist/core/memory/index.js.map +0 -1
  385. package/dist/core/memory/memories.d.ts.map +0 -1
  386. package/dist/core/memory/memories.js.map +0 -1
  387. package/dist/core/memory/memory-manager.d.ts.map +0 -1
  388. package/dist/core/memory/memory-manager.js.map +0 -1
  389. package/dist/core/memory/progressive-disclosure.d.ts.map +0 -1
  390. package/dist/core/memory/progressive-disclosure.js.map +0 -1
  391. package/dist/core/memory/query-processor.d.ts.map +0 -1
  392. package/dist/core/memory/query-processor.js.map +0 -1
  393. package/dist/core/memory/query-rewriter.d.ts.map +0 -1
  394. package/dist/core/memory/query-rewriter.js.map +0 -1
  395. package/dist/core/memory/response-analyzer.d.ts.map +0 -1
  396. package/dist/core/memory/response-analyzer.js.map +0 -1
  397. package/dist/core/memory/serialization.d.ts.map +0 -1
  398. package/dist/core/memory/serialization.js.map +0 -1
  399. package/dist/core/memory/stats.d.ts.map +0 -1
  400. package/dist/core/memory/stats.js.map +0 -1
  401. package/dist/core/memory/telemetry.d.ts.map +0 -1
  402. package/dist/core/memory/telemetry.js.map +0 -1
  403. package/dist/core/memory/temporal-facts.d.ts.map +0 -1
  404. package/dist/core/memory/temporal-facts.js.map +0 -1
  405. package/dist/core/memory/temporal-parser.d.ts.map +0 -1
  406. package/dist/core/memory/temporal-parser.js.map +0 -1
  407. package/dist/core/memory/trigger-detector.d.ts.map +0 -1
  408. package/dist/core/memory/trigger-detector.js.map +0 -1
  409. package/dist/core/memory/write-gate.d.ts.map +0 -1
  410. package/dist/core/memory/write-gate.js.map +0 -1
  411. package/dist/core/namespaces/index.d.ts.map +0 -1
  412. package/dist/core/namespaces/index.js.map +0 -1
  413. package/dist/core/namespaces/uri-parser.d.ts.map +0 -1
  414. package/dist/core/namespaces/uri-parser.js.map +0 -1
  415. package/dist/core/observations.d.ts +0 -26
  416. package/dist/core/observations.d.ts.map +0 -1
  417. package/dist/core/observations.js +0 -110
  418. package/dist/core/observations.js.map +0 -1
  419. package/dist/core/privacy.d.ts.map +0 -1
  420. package/dist/core/privacy.js.map +0 -1
  421. package/dist/core/projects.d.ts.map +0 -1
  422. package/dist/core/projects.js.map +0 -1
  423. package/dist/core/redis.d.ts.map +0 -1
  424. package/dist/core/redis.js.map +0 -1
  425. package/dist/core/requirements.d.ts +0 -20
  426. package/dist/core/requirements.d.ts.map +0 -1
  427. package/dist/core/requirements.js +0 -35
  428. package/dist/core/requirements.js.map +0 -1
  429. package/dist/core/scheduler/cron-scheduler.d.ts.map +0 -1
  430. package/dist/core/scheduler/cron-scheduler.js.map +0 -1
  431. package/dist/core/scheduler/heartbeat.d.ts.map +0 -1
  432. package/dist/core/scheduler/heartbeat.js.map +0 -1
  433. package/dist/core/scheduler/index.d.ts.map +0 -1
  434. package/dist/core/scheduler/index.js.map +0 -1
  435. package/dist/core/scheduler/job-runner.d.ts.map +0 -1
  436. package/dist/core/scheduler/job-runner.js.map +0 -1
  437. package/dist/core/search/conversations.d.ts.map +0 -1
  438. package/dist/core/search/conversations.js.map +0 -1
  439. package/dist/core/search/entities.d.ts.map +0 -1
  440. package/dist/core/search/entities.js.map +0 -1
  441. package/dist/core/search/folder-context.d.ts.map +0 -1
  442. package/dist/core/search/folder-context.js.map +0 -1
  443. package/dist/core/search/index.d.ts.map +0 -1
  444. package/dist/core/search/index.js.map +0 -1
  445. package/dist/core/search/qmd-search.d.ts.map +0 -1
  446. package/dist/core/search/qmd-search.js.map +0 -1
  447. package/dist/core/secret-detector.d.ts.map +0 -1
  448. package/dist/core/secret-detector.js.map +0 -1
  449. package/dist/core/session/auto-load.d.ts.map +0 -1
  450. package/dist/core/session/auto-load.js.map +0 -1
  451. package/dist/core/session/index.d.ts.map +0 -1
  452. package/dist/core/session/index.js.map +0 -1
  453. package/dist/core/session/types.d.ts.map +0 -1
  454. package/dist/core/session/types.js.map +0 -1
  455. package/dist/core/session-hooks/self-iteration-job.d.ts.map +0 -1
  456. package/dist/core/session-hooks/self-iteration-job.js.map +0 -1
  457. package/dist/core/session-hooks/session-hooks.d.ts.map +0 -1
  458. package/dist/core/session-hooks/session-hooks.js.map +0 -1
  459. package/dist/core/snapshots/cleanup.d.ts.map +0 -1
  460. package/dist/core/snapshots/cleanup.js.map +0 -1
  461. package/dist/core/snapshots/comparison.d.ts.map +0 -1
  462. package/dist/core/snapshots/comparison.js.map +0 -1
  463. package/dist/core/snapshots/creation.d.ts.map +0 -1
  464. package/dist/core/snapshots/creation.js.map +0 -1
  465. package/dist/core/snapshots/retrieval.d.ts.map +0 -1
  466. package/dist/core/snapshots/retrieval.js.map +0 -1
  467. package/dist/core/snapshots/stats.d.ts.map +0 -1
  468. package/dist/core/snapshots/stats.js.map +0 -1
  469. package/dist/core/snapshots.d.ts.map +0 -1
  470. package/dist/core/snapshots.js.map +0 -1
  471. package/dist/core/summarization/cleanup.d.ts.map +0 -1
  472. package/dist/core/summarization/cleanup.js.map +0 -1
  473. package/dist/core/summarization/queries.d.ts.map +0 -1
  474. package/dist/core/summarization/queries.js.map +0 -1
  475. package/dist/core/summarization/stats.d.ts.map +0 -1
  476. package/dist/core/summarization/stats.js.map +0 -1
  477. package/dist/core/summarization/strategies.d.ts.map +0 -1
  478. package/dist/core/summarization/strategies.js.map +0 -1
  479. package/dist/core/summarization.d.ts.map +0 -1
  480. package/dist/core/summarization.js.map +0 -1
  481. package/dist/core/sync/qmd-sync.d.ts.map +0 -1
  482. package/dist/core/sync/qmd-sync.js.map +0 -1
  483. package/dist/core/temporal-facts.d.ts.map +0 -1
  484. package/dist/core/temporal-facts.js.map +0 -1
  485. package/dist/core/tracing/collector.d.ts.map +0 -1
  486. package/dist/core/tracing/collector.js.map +0 -1
  487. package/dist/core/tracing/visualizer.d.ts.map +0 -1
  488. package/dist/core/tracing/visualizer.js.map +0 -1
  489. package/dist/core/utils/cleanup-operations.d.ts.map +0 -1
  490. package/dist/core/utils/cleanup-operations.js.map +0 -1
  491. package/dist/core/utils/content-extraction.d.ts.map +0 -1
  492. package/dist/core/utils/content-extraction.js.map +0 -1
  493. package/dist/core/utils/filter-builder.d.ts.map +0 -1
  494. package/dist/core/utils/filter-builder.js.map +0 -1
  495. package/dist/core/utils/history-traversal.d.ts.map +0 -1
  496. package/dist/core/utils/history-traversal.js.map +0 -1
  497. package/dist/core/utils/memory-operations.d.ts.map +0 -1
  498. package/dist/core/utils/memory-operations.js.map +0 -1
  499. package/dist/core/utils/query-operations.d.ts.map +0 -1
  500. package/dist/core/utils/query-operations.js.map +0 -1
  501. package/dist/core/utils/summarization-helpers.d.ts.map +0 -1
  502. package/dist/core/utils/summarization-helpers.js.map +0 -1
  503. package/dist/core/utils/temporal-queries.d.ts.map +0 -1
  504. package/dist/core/utils/temporal-queries.js.map +0 -1
  505. package/dist/core/utils/version-management.d.ts.map +0 -1
  506. package/dist/core/utils/version-management.js.map +0 -1
  507. package/dist/core/utils.d.ts.map +0 -1
  508. package/dist/core/utils.js.map +0 -1
  509. package/dist/core/worker.d.ts.map +0 -1
  510. package/dist/core/worker.js.map +0 -1
  511. package/dist/db/adapter.d.ts.map +0 -1
  512. package/dist/db/adapter.js.map +0 -1
  513. package/dist/db/bootstrap.d.ts.map +0 -1
  514. package/dist/db/bootstrap.js.map +0 -1
  515. package/dist/db/index.d.ts.map +0 -1
  516. package/dist/db/index.js.map +0 -1
  517. package/dist/db/schema.d.ts.map +0 -1
  518. package/dist/db/schema.js.map +0 -1
  519. package/dist/drizzle/schema-sqlite.d.ts.map +0 -1
  520. package/dist/drizzle/schema-sqlite.js.map +0 -1
  521. package/dist/drizzle/schema.d.ts.map +0 -1
  522. package/dist/drizzle/schema.js.map +0 -1
  523. package/dist/index.d.ts.map +0 -1
  524. package/dist/index.js.map +0 -1
  525. package/hooks/hooks.json +0 -52
  526. package/hooks/post-tool-use.js +0 -26
  527. package/hooks/session-end.js +0 -28
  528. package/hooks/session-start.js +0 -33
  529. package/hooks/user-prompt-submit.js +0 -26
  530. package/hooks/utils.js +0 -153
  531. package/npx-installer.js +0 -208
  532. package/packages/plugin-claude-code/README.md +0 -73
  533. package/packages/plugin-claude-code/dist/plugin-wrapper.d.ts +0 -35
  534. package/packages/plugin-claude-code/dist/plugin-wrapper.d.ts.map +0 -1
  535. package/packages/plugin-claude-code/dist/plugin-wrapper.js +0 -191
  536. package/packages/plugin-claude-code/dist/plugin-wrapper.js.map +0 -1
  537. package/packages/plugin-claude-code/package.json +0 -31
  538. package/packages/plugin-openclaw/README.md +0 -70
  539. package/packages/plugin-openclaw/dist/index.d.ts +0 -49
  540. package/packages/plugin-openclaw/dist/index.d.ts.map +0 -1
  541. package/packages/plugin-openclaw/dist/index.js +0 -262
  542. package/packages/plugin-openclaw/dist/index.js.map +0 -1
  543. package/packages/plugin-openclaw/openclaw.plugin.json +0 -94
  544. package/packages/plugin-openclaw/package.json +0 -31
  545. package/packages/plugin-opencode/install.mjs +0 -217
  546. package/packages/plugin-opencode/package.json +0 -21
  547. package/scripts/db/check-db.mjs +0 -88
  548. package/scripts/db/fix-all-columns.mjs +0 -52
  549. package/scripts/db/fix-schema-all.mjs +0 -55
  550. package/scripts/db/fix-schema-full.mjs +0 -46
  551. package/scripts/db/fix-schema.mjs +0 -38
  552. package/scripts/db/init-db.mjs +0 -13
  553. package/scripts/db/recreate-db.mjs +0 -14
  554. package/scripts/install-mcp.mjs +0 -116
  555. package/scripts/install-web.sh +0 -120
  556. package/scripts/install.mjs +0 -340
  557. package/scripts/openclaw-bootstrap.mjs +0 -127
  558. package/scripts/package-release.sh +0 -71
  559. package/scripts/test/test-all-systems.mjs +0 -139
  560. package/scripts/test/test-memory-system.mjs +0 -139
  561. package/scripts/test/test-v0.5.0.mjs +0 -210
  562. package/skills/memory-guide/SKILL.md +0 -256
  563. package/skills/squish-cli/SKILL.md +0 -200
  564. package/skills/squish-mcp/SKILL.md +0 -311
  565. package/skills/squish-memory/claude-desktop.json +0 -12
  566. package/skills/squish-memory/openclaw.json +0 -13
  567. package/skills/squish-memory/opencode.json +0 -14
  568. package/skills/squish-memory/skill.json +0 -32
  569. /package/{commands → core/commands}/context-paging.md +0 -0
  570. /package/{commands → core/commands}/context-status.md +0 -0
  571. /package/{commands → core/commands}/context.md +0 -0
  572. /package/{commands → core/commands}/core-memory.md +0 -0
  573. /package/{commands → core/commands}/health.md +0 -0
  574. /package/{commands → core/commands}/merge.md +0 -0
  575. /package/{commands → core/commands}/recall.md +0 -0
  576. /package/{commands → core/commands}/remember.md +0 -0
  577. /package/{commands → core/commands}/search.md +0 -0
  578. /package/dist/{algorithms → core/algorithms}/detection/hash-filters.d.ts +0 -0
  579. /package/dist/{algorithms → core/algorithms}/detection/hash-filters.js +0 -0
  580. /package/dist/{algorithms → core/algorithms}/handlers/approve-merge.d.ts +0 -0
  581. /package/dist/{algorithms → core/algorithms}/handlers/detect-duplicates.d.ts +0 -0
  582. /package/dist/{algorithms → core/algorithms}/handlers/get-stats.d.ts +0 -0
  583. /package/dist/{algorithms → core/algorithms}/handlers/list-proposals.d.ts +0 -0
  584. /package/dist/{algorithms → core/algorithms}/handlers/preview-merge.d.ts +0 -0
  585. /package/dist/{algorithms → core/algorithms}/handlers/reject-merge.d.ts +0 -0
  586. /package/dist/{algorithms → core/algorithms}/handlers/reverse-merge.d.ts +0 -0
  587. /package/dist/{algorithms → core/algorithms}/safety/safety-checks.js +0 -0
  588. /package/dist/{algorithms → core/algorithms}/utils/response-builder.d.ts +0 -0
  589. /package/dist/{algorithms → core/algorithms}/utils/response-builder.js +0 -0
  590. /package/dist/{commands → core/commands}/managed-sync.d.ts +0 -0
  591. /package/dist/{commands → core/commands}/mcp-server.d.ts +0 -0
  592. /package/dist/core/{context.d.ts → context/context.d.ts} +0 -0
  593. /package/dist/core/{agent-memory.d.ts → ingestion/agent-memory.d.ts} +0 -0
  594. /package/dist/core/{core-memory.d.ts → ingestion/core-memory.d.ts} +0 -0
  595. /package/dist/core/{privacy.d.ts → security/privacy.d.ts} +0 -0
  596. /package/dist/core/{privacy.js → security/privacy.js} +0 -0
  597. /package/dist/core/{secret-detector.d.ts → security/secret-detector.d.ts} +0 -0
  598. /package/dist/core/{secret-detector.js → security/secret-detector.js} +0 -0
  599. /package/dist/core/{cache.d.ts → storage/cache.d.ts} +0 -0
  600. /package/dist/core/{database.d.ts → storage/database.d.ts} +0 -0
  601. /package/dist/core/{database.js → storage/database.js} +0 -0
@@ -1,42 +1,76 @@
1
1
  import express from 'express';
2
2
  import cors from 'cors';
3
- import { logger } from '../../core/logger.js';
4
- import { getRecentMemories } from '../../core/memory/memories.js';
5
- import { getObservationsForProject } from '../../core/observations.js';
6
- import { getAllProjects, getProjectByPath } from '../../core/projects.js';
7
- import { getDb } from '../../db/index.js';
3
+ import rateLimit from 'express-rate-limit';
4
+ import { logger } from '../core/logger.js';
5
+ import { getRecent } from '../core/memory/memories.js';
6
+ import { getObservations } from '../core/ingestion/learnings.js';
7
+ import { getAllProjects, requireProject } from '../core/projects.js';
8
+ import { checkDatabaseHealth } from '../db/index.js';
9
+ import { config } from '../config.js';
10
+ import { isDatabaseUnavailableError } from '../core/lib/utils.js';
11
+ import { validateLimit } from '../core/lib/validation.js';
8
12
  const app = express();
9
- const PORT = process.env.SQUISH_WEB_PORT || 37777;
10
- app.use(cors());
13
+ const PORT = Number(process.env.SQUISH_WEB_PORT || 37777);
14
+ const VERSION = '1.1.5';
15
+ const allowedOrigins = process.env.SQUISH_CORS_ORIGINS?.split(',').map(s => s.trim()) || ['http://localhost:*', 'http://127.0.0.1:*'];
16
+ const appCors = cors({
17
+ origin: (origin, callback) => {
18
+ if (!origin || allowedOrigins.some(allowed => {
19
+ if (allowed.endsWith(':*')) {
20
+ const prefix = allowed.slice(0, -1);
21
+ return origin.startsWith(prefix);
22
+ }
23
+ return origin === allowed;
24
+ })) {
25
+ callback(null, true);
26
+ }
27
+ else {
28
+ callback(new Error('Not allowed by CORS'));
29
+ }
30
+ }
31
+ });
32
+ const limiter = rateLimit({
33
+ windowMs: 15 * 60 * 1000,
34
+ max: 100,
35
+ message: { error: 'Too many requests, please try again later.' },
36
+ standardHeaders: true,
37
+ legacyHeaders: false,
38
+ });
39
+ app.use(appCors);
40
+ app.use(limiter);
11
41
  app.use(express.json());
12
42
  // Health check endpoint
13
43
  app.get('/api/health', async (req, res) => {
14
- let dbStatus = 'ok';
44
+ let dbStatus = 'error';
15
45
  let projectInfo = null;
16
46
  let allProjects = [];
47
+ let errorMessage = null;
17
48
  try {
18
- const db = await getDb();
19
- const sqliteDb = db;
20
- if (sqliteDb && typeof sqliteDb.prepare === 'function') {
21
- sqliteDb.prepare('SELECT 1').get();
22
- }
23
- // Get all projects from database
24
- allProjects = await getAllProjects();
25
- // If there are projects, use the first one as default (most recent)
26
- if (allProjects.length > 0) {
27
- projectInfo = { id: allProjects[0].id, name: allProjects[0].name, path: allProjects[0].path };
49
+ const healthy = await checkDatabaseHealth();
50
+ dbStatus = healthy ? 'ok' : 'error';
51
+ if (healthy) {
52
+ allProjects = await getAllProjects();
53
+ if (allProjects.length > 0) {
54
+ projectInfo = { id: allProjects[0].id, name: allProjects[0].name, path: allProjects[0].path };
55
+ }
28
56
  }
29
57
  }
30
58
  catch (error) {
31
- dbStatus = 'error';
32
- logger.error('Health check failed:', error.message);
59
+ errorMessage = error.message;
60
+ if (!isDatabaseUnavailableError(error)) {
61
+ logger.error('Health check failed:', error.message);
62
+ }
33
63
  }
34
64
  res.json({
35
- status: dbStatus === 'ok' ? 'ok' : 'error',
36
- version: '0.9.1',
65
+ ok: dbStatus === 'ok',
66
+ status: dbStatus,
67
+ version: VERSION,
37
68
  database: dbStatus,
69
+ cache: config.redisEnabled ? 'configured' : 'unavailable',
70
+ dataDirectory: config.dataDir,
38
71
  project: projectInfo || { id: 'unknown', name: 'No Project', path: '' },
39
72
  projects: allProjects,
73
+ error: errorMessage,
40
74
  timestamp: new Date().toISOString()
41
75
  });
42
76
  });
@@ -44,12 +78,9 @@ app.get('/api/health', async (req, res) => {
44
78
  app.get('/api/memories', async (req, res) => {
45
79
  try {
46
80
  const projectPath = req.query.projectPath || process.cwd();
47
- const limit = Math.min(Math.max(parseInt(req.query.limit) || 20, 1), 100);
48
- const project = await getProjectByPath(projectPath);
49
- if (!project) {
50
- return res.json({ status: 'ok', data: [], count: 0, message: 'Project not found' });
51
- }
52
- const memories = await getRecentMemories(projectPath, limit);
81
+ const limit = validateLimit(req.query.limit, 20, 1, 100);
82
+ const project = await requireProject(projectPath);
83
+ const memories = await getRecent(projectPath, limit);
53
84
  res.json({
54
85
  status: 'ok',
55
86
  data: memories,
@@ -58,20 +89,19 @@ app.get('/api/memories', async (req, res) => {
58
89
  });
59
90
  }
60
91
  catch (error) {
61
- logger.error('Failed to get memories:', error.message);
62
- res.status(500).json({ status: 'error', message: error.message });
92
+ if (!isDatabaseUnavailableError(error)) {
93
+ logger.error('Failed to get memories:', error.message);
94
+ }
95
+ res.status(isDatabaseUnavailableError(error) ? 503 : 500).json({ status: 'error', message: error.message });
63
96
  }
64
97
  });
65
98
  // Get observations for project
66
99
  app.get('/api/observations', async (req, res) => {
67
100
  try {
68
101
  const projectPath = req.query.projectPath || process.cwd();
69
- const limit = Math.min(Math.max(parseInt(req.query.limit) || 20, 1), 100);
70
- const project = await getProjectByPath(projectPath);
71
- if (!project) {
72
- return res.json({ status: 'ok', data: [], count: 0, message: 'Project not found' });
73
- }
74
- const observations = await getObservationsForProject(projectPath, limit);
102
+ const limit = validateLimit(req.query.limit, 20, 1, 100);
103
+ const project = await requireProject(projectPath);
104
+ const observations = await getObservations(projectPath, limit);
75
105
  res.json({
76
106
  status: 'ok',
77
107
  data: observations,
@@ -80,8 +110,10 @@ app.get('/api/observations', async (req, res) => {
80
110
  });
81
111
  }
82
112
  catch (error) {
83
- logger.error('Failed to get observations:', error.message);
84
- res.status(500).json({ status: 'error', message: error.message });
113
+ if (!isDatabaseUnavailableError(error)) {
114
+ logger.error('Failed to get observations:', error.message);
115
+ }
116
+ res.status(isDatabaseUnavailableError(error) ? 503 : 500).json({ status: 'error', message: error.message });
85
117
  }
86
118
  });
87
119
  // Get project context
@@ -105,20 +137,9 @@ app.get('/api/context', async (req, res) => {
105
137
  message: 'No projects found in database'
106
138
  });
107
139
  }
108
- const project = await getProjectByPath(projectPath);
109
- if (!project) {
110
- return res.json({
111
- status: 'ok',
112
- project: { id: 'unknown', name: 'Project Not Found', path: projectPath },
113
- projects: allProjects,
114
- memories: [],
115
- observations: [],
116
- totalCount: 0,
117
- message: 'Project not found in database'
118
- });
119
- }
120
- const memories = await getRecentMemories(projectPath, 20);
121
- const observations = await getObservationsForProject(projectPath, 20);
140
+ const project = await requireProject(projectPath);
141
+ const memories = await getRecent(projectPath, 20);
142
+ const observations = await getObservations(projectPath, 20);
122
143
  res.json({
123
144
  status: 'ok',
124
145
  project: { id: project.id, name: project.name, path: project.path },
@@ -129,8 +150,10 @@ app.get('/api/context', async (req, res) => {
129
150
  });
130
151
  }
131
152
  catch (error) {
132
- logger.error('Failed to get context:', error.message);
133
- res.status(500).json({ status: 'error', message: error.message });
153
+ if (!isDatabaseUnavailableError(error)) {
154
+ logger.error('Failed to get context:', error.message);
155
+ }
156
+ res.status(isDatabaseUnavailableError(error) ? 503 : 500).json({ status: 'error', message: error.message });
134
157
  }
135
158
  });
136
159
  // Get all projects
@@ -144,470 +167,476 @@ app.get('/api/projects', async (req, res) => {
144
167
  });
145
168
  }
146
169
  catch (error) {
147
- logger.error('Failed to get projects:', error.message);
148
- res.status(500).json({ status: 'error', message: error.message });
170
+ if (!isDatabaseUnavailableError(error)) {
171
+ logger.error('Failed to get projects:', error.message);
172
+ }
173
+ res.status(isDatabaseUnavailableError(error) ? 503 : 500).json({ status: 'error', message: error.message });
149
174
  }
150
175
  });
151
176
  // Web UI
152
177
  app.get('/', (req, res) => {
153
- const html = `<!DOCTYPE html>
154
- <html class="dark" lang="en"><head>
155
- <meta charset="utf-8"/>
156
- <meta content="width=device-width, initial-scale=1.0" name="viewport"/>
157
- <title>Squish Memory Viewer - Playful Dashboard</title>
158
- <link href="https://fonts.googleapis.com" rel="preconnect"/>
159
- <link crossorigin="" href="https://fonts.gstatic.com" rel="preconnect"/>
160
- <link href="https://fonts.googleapis.com/css2?family=Spline+Sans:wght@300;400;500;600;700;800&amp;display=swap" rel="stylesheet"/>
161
- <link href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:wght,FILL@100..700,0..1&amp;display=swap" rel="stylesheet"/>
162
- <script src="https://cdn.tailwindcss.com?plugins=forms,container-queries"></script>
163
- <script id="tailwind-config">
164
- tailwind.config = {
165
- darkMode: "class",
166
- theme: {
167
- extend: {
168
- colors: {
169
- "primary": "#00ffbf",
170
- "secondary": "#ff6392",
171
- "accent": "#ffcd26",
172
- "background-dark": "#0f172a",
173
- "card-bg": "#1e293b",
174
- "text-main": "#f8fafc",
175
- "text-muted": "#94a3b8",
176
- "alert-orange": "rgba(251, 146, 60, 0.1)"
177
- },
178
- fontFamily: {
179
- "display": ["Spline Sans", "sans-serif"]
180
- },
181
- borderRadius: {
182
- "pill": "2.5rem",
183
- "blob": "30% 70% 70% 30% / 30% 30% 70% 70%"
184
- }
185
- },
186
- },
187
- }
188
- </script>
189
- <style type="text/tailwindcss">
190
- @layer base {
191
- body { @apply font-display text-text-main bg-background-dark antialiased; }
192
- }
193
- .squish-pill {
194
- border-radius: 3rem;
195
- }
196
- .blob-alert {
197
- border-radius: 40% 60% 70% 30% / 40% 50% 60% 50%;
198
- }
199
- .squishy-hover {
200
- transition: transform 0.4s cubic-bezier(0.175, 0.885, 0.32, 1.275);
201
- }
202
- .squishy-hover:hover {
203
- transform: scale(1.02) translateY(-4px);
204
- }
205
- .pulse-red {
206
- animation: pulse 2s infinite;
207
- }
208
- @keyframes pulse {
209
- 0% { transform: scale(0.95); box-shadow: 0 0 0 0 rgba(239, 68, 68, 0.7); }
210
- 70% { transform: scale(1); box-shadow: 0 0 0 10px rgba(239, 68, 68, 0); }
211
- 100% { transform: scale(0.95); box-shadow: 0 0 0 0 rgba(239, 68, 68, 0); }
212
- }
213
- </style>
214
- </head>
215
- <body class="min-h-screen selection:bg-primary/30 pb-20">
216
- <header class="w-full px-6 py-8">
217
- <div class="max-w-6xl mx-auto flex items-center justify-between">
218
- <div class="flex items-center gap-4">
219
- <div class="relative size-12 flex items-center justify-center">
220
- <div class="absolute inset-0 bg-secondary/20 blur-xl rounded-full"></div>
221
- <span class="material-symbols-outlined text-secondary text-5xl relative z-10">psychology</span>
222
- </div>
223
- <h1 class="text-3xl font-black tracking-tight flex items-center gap-2">
224
- Squish <span class="text-primary italic">Memory Viewer</span>
225
- </h1>
226
- </div>
227
- <div class="flex items-center gap-4">
228
- <select id="project-select" onchange="changeProject(this.value)" class="bg-card-bg px-4 py-2 rounded-full border-2 border-slate-700/50 text-text-main text-sm font-medium focus:outline-none focus:border-primary">
229
- <option value="">Loading projects...</option>
230
- </select>
231
- <div class="bg-card-bg px-4 py-2 rounded-full border-2 border-slate-700/50 flex items-center gap-2">
232
- <div class="size-2 rounded-full bg-primary animate-pulse"></div>
233
- <span class="text-xs font-bold uppercase tracking-widest text-text-muted">Local Server: Online</span>
234
- </div>
235
- </div>
236
- </div>
237
- </header>
238
- <main class="max-w-6xl mx-auto px-6 space-y-12">
239
- <div class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 gap-6">
240
- <div class="bg-card-bg p-8 squish-pill border-2 border-slate-700/30 text-center squishy-hover shadow-xl">
241
- <p class="text-4xl font-black mb-1" id="memories-count">-</p>
242
- <p class="text-sm font-bold uppercase tracking-widest text-text-muted italic">Memories</p>
243
- </div>
244
- <div class="bg-card-bg p-8 squish-pill border-2 border-slate-700/30 text-center squishy-hover shadow-xl">
245
- <p class="text-4xl font-black mb-1" id="observations-count">-</p>
246
- <p class="text-sm font-bold uppercase tracking-widest text-text-muted italic">Observations</p>
247
- </div>
248
- <div class="bg-card-bg p-8 squish-pill border-2 border-slate-700/30 text-center squishy-hover shadow-xl">
249
- <p class="text-4xl font-black mb-1" id="total-count">-</p>
250
- <p class="text-sm font-bold uppercase tracking-widest text-text-muted italic">Total Items</p>
251
- </div>
252
- <div class="bg-card-bg p-8 squish-pill border-2 border-slate-700/30 flex flex-col items-center justify-center squishy-hover shadow-xl">
253
- <div class="flex items-center gap-3">
254
- <div class="size-4 bg-red-500 rounded-full pulse-red"></div>
255
- <p class="text-2xl font-black text-red-400 italic">Error</p>
256
- </div>
257
- <p class="text-sm font-bold uppercase tracking-widest text-text-muted italic mt-1">Status</p>
258
- </div>
259
- </div>
260
- <div class="relative py-6 px-10 bg-orange-500/10 border-2 border-orange-500/20 blob-alert flex items-center gap-6 overflow-hidden">
261
- <div class="absolute top-0 left-0 w-full h-full bg-orange-500/5 -z-10"></div>
262
- <span class="material-symbols-outlined text-orange-400 text-3xl">warning</span>
263
- <div>
264
- <h4 class="font-black text-orange-400 italic uppercase text-sm tracking-wider">Communication Breakdown</h4>
265
- <p class="text-orange-200/80 font-medium">Failed to load data: Unknown error. Is the blob server running?</p>
266
- </div>
267
- </div>
268
- <div class="grid grid-cols-1 lg:grid-cols-2 gap-12">
269
- <section class="space-y-6">
270
- <div class="flex items-center gap-4 border-b-4 border-primary/20 pb-4">
271
- <div class="size-10 bg-primary/20 rounded-full flex items-center justify-center border-2 border-primary">
272
- <span class="material-symbols-outlined text-primary font-bold">neurology</span>
273
- </div>
274
- <h2 class="text-2xl font-black italic text-primary uppercase">Recent Memories</h2>
275
- </div>
276
- <div class="space-y-4" id="memories">
277
- <div class="bg-card-bg/50 p-6 rounded-3xl border-2 border-slate-700/20 flex flex-col items-center justify-center py-16 opacity-60">
278
- <div class="size-12 border-4 border-primary border-t-transparent rounded-full animate-spin mb-4"></div>
279
- <p class="font-black italic text-text-muted">Loading memories...</p>
280
- </div>
281
- </div>
282
- </section>
283
- <section class="space-y-6">
284
- <div class="flex items-center gap-4 border-b-4 border-primary/20 pb-4">
285
- <div class="size-10 bg-primary/20 rounded-full flex items-center justify-center border-2 border-primary">
286
- <span class="material-symbols-outlined text-primary font-bold">visibility</span>
287
- </div>
288
- <h2 class="text-2xl font-black italic text-primary uppercase">Recent Observations</h2>
289
- </div>
290
- <div class="space-y-4" id="observations">
291
- <div class="bg-card-bg/50 p-6 rounded-3xl border-2 border-slate-700/20 flex flex-col items-center justify-center py-16 opacity-60">
292
- <div class="size-12 border-4 border-primary border-t-transparent rounded-full animate-spin mb-4"></div>
293
- <p class="font-black italic text-text-muted">Loading observations...</p>
294
- </div>
295
- </div>
296
- </section>
297
- </div>
298
- <div class="pt-12 flex justify-center">
299
- <div class="bg-card-bg border-4 border-slate-700/50 p-2 rounded-full flex gap-2">
300
- <button class="bg-primary text-black px-8 py-3 rounded-full font-black text-sm uppercase hover:scale-105 transition-transform flex items-center gap-2">
301
- <span class="material-symbols-outlined text-sm">refresh</span>
302
- Reconnect
303
- </button>
304
- <button class="text-text-main px-8 py-3 rounded-full font-black text-sm uppercase hover:bg-slate-700/50 transition-colors" onclick="openDocs()">
305
- Docs
306
- </button>
307
- <button class="text-text-main px-8 py-3 rounded-full font-black text-sm uppercase hover:bg-slate-700/50 transition-colors" onclick="openSettings()">
308
- Settings
309
- </button>
310
- </div>
311
- </div>
312
- </main>
313
- <footer class="mt-20 px-6 opacity-30">
314
- <div class="max-w-6xl mx-auto flex justify-between items-center py-8 border-t border-slate-700">
315
- <p class="text-xs font-black uppercase">© 2026 Squish-Memory Dashboard</p>
316
- <div class="flex gap-4">
317
- <span class="material-symbols-outlined text-sm">database</span>
318
- <span class="text-xs font-black uppercase italic">Local-First Engine v1.0</span>
319
- </div>
320
- </div>
321
- </footer>
322
- <script>
323
- let currentProjectPath = null;
324
-
325
- async function loadProjects() {
326
- try {
327
- const response = await fetch('/api/projects');
328
- const data = await response.json();
329
-
330
- if (data.status === 'ok' && data.data && data.data.length > 0) {
331
- const select = document.getElementById('project-select');
332
- if (select) {
333
- select.innerHTML = data.data.map(function(p) {
334
- return '<option value="' + escapeHtml(p.path) + '">' + escapeHtml(p.name || p.path) + '</option>';
335
- }).join('');
336
-
337
- // Try to select current directory
338
- const cwd = window.location.pathname === '/' ? '' : window.location.pathname;
339
- const defaultProject = data.data.find(function(p) { return p.path === cwd; }) || data.data[0];
340
- if (defaultProject) {
341
- currentProjectPath = defaultProject.path;
342
- select.value = defaultProject.path;
343
- }
344
- }
345
- } else {
346
- // No projects yet, use current directory
347
- currentProjectPath = window.location.pathname === '/' ? '' : window.location.pathname;
348
- }
349
- } catch (error) {
350
- console.error('Failed to load projects:', error);
351
- currentProjectPath = '';
352
- }
353
- }
354
-
355
- async function loadData() {
356
- try {
357
- const url = currentProjectPath ? '/api/context?projectPath=' + encodeURIComponent(currentProjectPath) : '/api/context';
358
- const response = await fetch(url);
359
- const data = await response.json();
360
-
361
- if (data.status === 'ok') {
362
- document.getElementById('memories-count').textContent = data.memories ? data.memories.length : 0;
363
- document.getElementById('observations-count').textContent = data.observations ? data.observations.length : 0;
364
- document.getElementById('total-count').textContent = data.totalCount || 0;
365
- updateStatus(data.memories && data.observations ? 'ok' : 'error');
366
-
367
- renderMemories(data.memories || []);
368
- renderObservations(data.observations || []);
369
-
370
- // Update project info
371
- if (data.project) {
372
- const projectInfo = document.getElementById('project-info');
373
- if (projectInfo) {
374
- projectInfo.textContent = data.project.name || data.project.path || 'Unknown';
375
- }
376
- }
377
-
378
- // Hide error alert if data loaded
379
- const errorAlert = document.querySelector('.blob-alert');
380
- if (errorAlert && (data.memories && data.memories.length > 0 || data.observations && data.observations.length > 0)) {
381
- errorAlert.style.display = 'none';
382
- }
383
-
384
- // Show error alert if message present
385
- if (data.message) {
386
- const errorAlert = document.querySelector('.blob-alert');
387
- if (errorAlert) {
388
- errorAlert.querySelector('p').textContent = data.message;
389
- errorAlert.style.display = 'flex';
390
- }
391
- }
392
- } else {
393
- throw new Error('API returned error status');
394
- }
395
- } catch (error) {
396
- updateStatus('error');
397
-
398
- // Show error alert
399
- const errorAlert = document.querySelector('.blob-alert');
400
- if (errorAlert) errorAlert.style.display = 'flex';
401
- }
402
- }
403
-
404
- function renderMemories(memories) {
405
- const container = document.getElementById('memories');
406
- if (!memories || memories.length === 0) {
407
- container.innerHTML = '<div class="bg-card-bg/50 p-6 rounded-3xl border-2 border-slate-700/20 flex flex-col items-center justify-center py-16 opacity-60"><p class="font-black italic text-text-muted">No memories found</p></div>';
408
- return;
409
- }
410
-
411
- container.innerHTML = memories.map(function(memory) {
412
- return '<div class="bg-card-bg p-6 rounded-3xl border-2 border-slate-700/20 squishy-hover">' +
413
- '<div class="flex items-start justify-between mb-4">' +
414
- '<span class="bg-primary text-black px-3 py-1 rounded-full text-xs font-bold uppercase">' + (memory.type || 'memory') + '</span>' +
415
- '<span class="text-text-muted text-sm">' + formatTime(memory.createdAt) + '</span>' +
416
- '</div>' +
417
- '<div class="text-text-main mb-4">' + escapeHtml(memory.content || memory.text || '') + '</div>' +
418
- '<div class="text-text-muted text-sm">' +
419
- 'Tags: ' + (memory.tags ? memory.tags.join(', ') : 'none') +
420
- '</div>' +
421
- '</div>';
422
- }).join('');
423
- }
424
-
425
- function renderObservations(observations) {
426
- const container = document.getElementById('observations');
427
- if (!observations || observations.length === 0) {
428
- container.innerHTML = '<div class="bg-card-bg/50 p-6 rounded-3xl border-2 border-slate-700/20 flex flex-col items-center justify-center py-16 opacity-60"><p class="font-black italic text-text-muted">No observations found</p></div>';
429
- return;
430
- }
431
-
432
- container.innerHTML = observations.map(function(obs) {
433
- return '<div class="bg-card-bg p-6 rounded-3xl border-2 border-slate-700/20 squishy-hover">' +
434
- '<div class="flex items-start justify-between mb-4">' +
435
- '<span class="bg-secondary text-black px-3 py-1 rounded-full text-xs font-bold uppercase">' + (obs.type || 'observation') + '</span>' +
436
- '<span class="text-text-muted text-sm">' + formatTime(obs.createdAt) + '</span>' +
437
- '</div>' +
438
- '<div class="text-text-main mb-4">' + escapeHtml(obs.summary || obs.content || '') + '</div>' +
439
- '<div class="text-text-muted text-sm">' +
440
- 'Action: ' + (obs.action || 'none') + ' | ' +
441
- 'Target: ' + (obs.target || 'none') +
442
- '</div>' +
443
- '</div>';
444
- }).join('');
445
- }
446
-
447
- function updateStatus(status) {
448
- const statusCard = document.querySelector('.squishy-hover.flex-col');
449
- if (!statusCard) return;
450
-
451
- if (status === 'ok') {
452
- statusCard.innerHTML = '<div class="flex items-center gap-3">' +
453
- '<div class="size-4 bg-primary rounded-full animate-pulse"></div>' +
454
- '<p class="text-2xl font-black text-primary italic">OK</p>' +
455
- '</div>' +
456
- '<p class="text-sm font-bold uppercase tracking-widest text-text-muted italic mt-1">Status</p>';
457
- } else {
458
- statusCard.innerHTML = '<div class="flex items-center gap-3">' +
459
- '<div class="size-4 bg-red-500 rounded-full pulse-red"></div>' +
460
- '<p class="text-2xl font-black text-red-400 italic">Error</p>' +
461
- '</div>' +
462
- '<p class="text-sm font-bold uppercase tracking-widest text-text-muted italic mt-1">Status</p>';
463
- }
464
- }
465
-
466
- function formatTime(timestamp) {
467
- if (!timestamp) return 'Unknown';
468
- const date = new Date(timestamp);
469
- return date.toLocaleString();
470
- }
471
-
472
- function escapeHtml(text) {
473
- const div = document.createElement('div');
474
- div.textContent = text;
475
- return div.innerHTML;
476
- }
477
-
478
- function openDocs() {
479
- // Create and show documentation modal
480
- const modal = document.createElement('div');
481
- modal.className = 'fixed inset-0 bg-black/50 flex items-center justify-center z-50';
482
- modal.innerHTML = '<div class="bg-card-bg p-8 rounded-3xl border-2 border-slate-700/50 max-w-2xl w-full mx-4 max-h-[80vh] overflow-y-auto">' +
483
- '<div class="flex justify-between items-center mb-6">' +
484
- '<h2 class="text-2xl font-black text-primary italic">Documentation</h2>' +
485
- '<button onclick="closeModal(this)" class="text-text-muted hover:text-text-main text-2xl">&times;</button>' +
486
- '</div>' +
487
- '<div class="space-y-4 text-text-main">' +
488
- '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
489
- '<h3 class="font-bold text-primary mb-2">🧠 Memory System</h3>' +
490
- '<p class="text-sm">The Squish Memory Plugin captures and stores conversations, tool usage, and project insights across sessions.</p>' +
491
- '</div>' +
492
- '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
493
- '<h3 class="font-bold text-primary mb-2">📊 API Endpoints</h3>' +
494
- '<ul class="text-sm space-y-1">' +
495
- '<li><code class="bg-slate-700/50 px-2 py-1 rounded">GET /api/health</code> - Service health status</li>' +
496
- '<li><code class="bg-slate-700/50 px-2 py-1 rounded">GET /api/memories</code> - Recent memories</li>' +
497
- '<li><code class="bg-slate-700/50 px-2 py-1 rounded">GET /api/observations</code> - Tool usage observations</li>' +
498
- '<li><code class="bg-slate-700/50 px-2 py-1 rounded">GET /api/context</code> - Combined data</li>' +
499
- '</ul>' +
500
- '</div>' +
501
- '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
502
- '<h3 class="font-bold text-primary mb-2">⚙️ Configuration</h3>' +
503
- '<p class="text-sm">Configure via environment variables: <code class="bg-slate-700/50 px-2 py-1 rounded">SQUISH_WEB_PORT</code>, <code class="bg-slate-700/50 px-2 py-1 rounded">DATABASE_URL</code>, etc.</p>' +
504
- '</div>' +
505
- '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
506
- '<h3 class="font-bold text-primary mb-2">🚀 Getting Started</h3>' +
507
- '<p class="text-sm">1. Install the plugin<br>2. Configure your database<br>3. Start the web UI<br>4. Access at http://localhost:37777</p>' +
508
- '</div>' +
509
- '</div>' +
510
- '</div>';
511
- document.body.appendChild(modal);
512
- }
513
-
514
- function openSettings() {
515
- // Create and show settings modal
516
- const modal = document.createElement('div');
517
- modal.className = 'fixed inset-0 bg-black/50 flex items-center justify-center z-50';
518
- modal.innerHTML = '<div class="bg-card-bg p-8 rounded-3xl border-2 border-slate-700/50 max-w-md w-full mx-4">' +
519
- '<div class="flex justify-between items-center mb-6">' +
520
- '<h2 class="text-2xl font-black text-primary italic">Settings</h2>' +
521
- '<button onclick="closeModal(this)" class="text-text-muted hover:text-text-main text-2xl">&times;</button>' +
522
- '</div>' +
523
- '<div class="space-y-4">' +
524
- '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
525
- '<label class="block text-sm font-bold text-text-main mb-2">Refresh Interval</label>' +
526
- '<select class="w-full bg-slate-700/50 border border-slate-600 rounded-lg px-3 py-2 text-text-main" onchange="changeRefreshInterval(this.value)">' +
527
- '<option value="10000">10 seconds</option>' +
528
- '<option value="30000" selected>30 seconds</option>' +
529
- '<option value="60000">1 minute</option>' +
530
- '<option value="300000">5 minutes</option>' +
531
- '</select>' +
532
- '</div>' +
533
- '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
534
- '<label class="block text-sm font-bold text-text-main mb-2">Theme</label>' +
535
- '<select class="w-full bg-slate-700/50 border border-slate-600 rounded-lg px-3 py-2 text-text-main" onchange="changeTheme(this.value)">' +
536
- '<option value="dark" selected>Dark</option>' +
537
- '<option value="light">Light</option>' +
538
- '</select>' +
539
- '</div>' +
540
- '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
541
- '<label class="block text-sm font-bold text-text-main mb-2">Items per Page</label>' +
542
- '<select class="w-full bg-slate-700/50 border border-slate-600 rounded-lg px-3 py-2 text-text-main" onchange="changeItemsPerPage(this.value)">' +
543
- '<option value="10">10</option>' +
544
- '<option value="25" selected>25</option>' +
545
- '<option value="50">50</option>' +
546
- '<option value="100">100</option>' +
547
- '</select>' +
548
- '</div>' +
549
- '<div class="flex justify-end space-x-3 mt-6">' +
550
- '<button onclick="closeModal(this)" class="px-4 py-2 bg-slate-700/50 hover:bg-slate-600/50 rounded-lg text-text-main transition-colors">Cancel</button>' +
551
- '<button onclick="saveSettings()" class="px-4 py-2 bg-primary text-black rounded-lg hover:bg-primary/80 transition-colors">Save</button>' +
552
- '</div>' +
553
- '</div>' +
554
- '</div>';
555
- document.body.appendChild(modal);
556
- }
557
-
558
- function closeModal(button) {
559
- const modal = button.closest('.fixed');
560
- if (modal) {
561
- modal.remove();
562
- }
563
- }
564
-
565
- function changeRefreshInterval(interval) {
566
- clearInterval(window.refreshInterval);
567
- window.refreshInterval = setInterval(loadData, parseInt(interval));
568
- }
569
-
570
- function changeTheme(theme) {
571
- if (theme === 'light') {
572
- document.documentElement.classList.remove('dark');
573
- } else {
574
- document.documentElement.classList.add('dark');
575
- }
576
- }
577
-
578
- function changeItemsPerPage(count) {
579
- // This would require updating the API calls, for now just store in localStorage
580
- localStorage.setItem('itemsPerPage', count);
581
- }
582
-
583
- function saveSettings() {
584
- // Close the modal and show success message
585
- const modal = document.querySelector('.fixed');
586
- if (modal) {
587
- // Could show a toast notification here
588
- modal.remove();
589
- }
590
- }
591
-
592
- function changeProject(path) {
593
- currentProjectPath = path;
594
- loadData();
595
- }
596
-
597
- // Initialize: load projects first, then data
598
- loadProjects().then(function() {
599
- loadData();
600
- window.refreshInterval = setInterval(loadData, 30000);
601
- });
602
- </script>
178
+ const html = `<!DOCTYPE html>
179
+ <html class="dark" lang="en"><head>
180
+ <meta charset="utf-8"/>
181
+ <meta content="width=device-width, initial-scale=1.0" name="viewport"/>
182
+ <title>Squish Memory Viewer - Playful Dashboard</title>
183
+ <link href="https://fonts.googleapis.com" rel="preconnect"/>
184
+ <link crossorigin="" href="https://fonts.gstatic.com" rel="preconnect"/>
185
+ <link href="https://fonts.googleapis.com/css2?family=Spline+Sans:wght@300;400;500;600;700;800&amp;display=swap" rel="stylesheet"/>
186
+ <link href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:wght,FILL@100..700,0..1&amp;display=swap" rel="stylesheet"/>
187
+ <script src="https://cdn.tailwindcss.com?plugins=forms,container-queries"></script>
188
+ <script id="tailwind-config">
189
+ tailwind.config = {
190
+ darkMode: "class",
191
+ theme: {
192
+ extend: {
193
+ colors: {
194
+ "primary": "#00ffbf",
195
+ "secondary": "#ff6392",
196
+ "accent": "#ffcd26",
197
+ "background-dark": "#0f172a",
198
+ "card-bg": "#1e293b",
199
+ "text-main": "#f8fafc",
200
+ "text-muted": "#94a3b8",
201
+ "alert-orange": "rgba(251, 146, 60, 0.1)"
202
+ },
203
+ fontFamily: {
204
+ "display": ["Spline Sans", "sans-serif"]
205
+ },
206
+ borderRadius: {
207
+ "pill": "2.5rem",
208
+ "blob": "30% 70% 70% 30% / 30% 30% 70% 70%"
209
+ }
210
+ },
211
+ },
212
+ }
213
+ </script>
214
+ <style type="text/tailwindcss">
215
+ @layer base {
216
+ body { @apply font-display text-text-main bg-background-dark antialiased; }
217
+ }
218
+ .squish-pill {
219
+ border-radius: 3rem;
220
+ }
221
+ .blob-alert {
222
+ border-radius: 40% 60% 70% 30% / 40% 50% 60% 50%;
223
+ }
224
+ .squishy-hover {
225
+ transition: transform 0.4s cubic-bezier(0.175, 0.885, 0.32, 1.275);
226
+ }
227
+ .squishy-hover:hover {
228
+ transform: scale(1.02) translateY(-4px);
229
+ }
230
+ .pulse-red {
231
+ animation: pulse 2s infinite;
232
+ }
233
+ @keyframes pulse {
234
+ 0% { transform: scale(0.95); box-shadow: 0 0 0 0 rgba(239, 68, 68, 0.7); }
235
+ 70% { transform: scale(1); box-shadow: 0 0 0 10px rgba(239, 68, 68, 0); }
236
+ 100% { transform: scale(0.95); box-shadow: 0 0 0 0 rgba(239, 68, 68, 0); }
237
+ }
238
+ </style>
239
+ </head>
240
+ <body class="min-h-screen selection:bg-primary/30 pb-20">
241
+ <header class="w-full px-6 py-8">
242
+ <div class="max-w-6xl mx-auto flex items-center justify-between">
243
+ <div class="flex items-center gap-4">
244
+ <div class="relative size-12 flex items-center justify-center">
245
+ <div class="absolute inset-0 bg-secondary/20 blur-xl rounded-full"></div>
246
+ <span class="material-symbols-outlined text-secondary text-5xl relative z-10">psychology</span>
247
+ </div>
248
+ <h1 class="text-3xl font-black tracking-tight flex items-center gap-2">
249
+ Squish <span class="text-primary italic">Memory Viewer</span>
250
+ </h1>
251
+ </div>
252
+ <div class="flex items-center gap-4">
253
+ <select id="project-select" onchange="changeProject(this.value)" class="bg-card-bg px-4 py-2 rounded-full border-2 border-slate-700/50 text-text-main text-sm font-medium focus:outline-none focus:border-primary">
254
+ <option value="">Loading projects...</option>
255
+ </select>
256
+ <div class="bg-card-bg px-4 py-2 rounded-full border-2 border-slate-700/50 flex items-center gap-2">
257
+ <div class="size-2 rounded-full bg-primary animate-pulse"></div>
258
+ <span class="text-xs font-bold uppercase tracking-widest text-text-muted">Local Server: Online</span>
259
+ </div>
260
+ </div>
261
+ </div>
262
+ </header>
263
+ <main class="max-w-6xl mx-auto px-6 space-y-12">
264
+ <div class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 gap-6">
265
+ <div class="bg-card-bg p-8 squish-pill border-2 border-slate-700/30 text-center squishy-hover shadow-xl">
266
+ <p class="text-4xl font-black mb-1" id="memories-count">-</p>
267
+ <p class="text-sm font-bold uppercase tracking-widest text-text-muted italic">Memories</p>
268
+ </div>
269
+ <div class="bg-card-bg p-8 squish-pill border-2 border-slate-700/30 text-center squishy-hover shadow-xl">
270
+ <p class="text-4xl font-black mb-1" id="observations-count">-</p>
271
+ <p class="text-sm font-bold uppercase tracking-widest text-text-muted italic">Observations</p>
272
+ </div>
273
+ <div class="bg-card-bg p-8 squish-pill border-2 border-slate-700/30 text-center squishy-hover shadow-xl">
274
+ <p class="text-4xl font-black mb-1" id="total-count">-</p>
275
+ <p class="text-sm font-bold uppercase tracking-widest text-text-muted italic">Total Items</p>
276
+ </div>
277
+ <div class="bg-card-bg p-8 squish-pill border-2 border-slate-700/30 flex flex-col items-center justify-center squishy-hover shadow-xl">
278
+ <div class="flex items-center gap-3">
279
+ <div class="size-4 bg-red-500 rounded-full pulse-red"></div>
280
+ <p class="text-2xl font-black text-red-400 italic">Error</p>
281
+ </div>
282
+ <p class="text-sm font-bold uppercase tracking-widest text-text-muted italic mt-1">Status</p>
283
+ </div>
284
+ </div>
285
+ <div class="relative py-6 px-10 bg-orange-500/10 border-2 border-orange-500/20 blob-alert flex items-center gap-6 overflow-hidden">
286
+ <div class="absolute top-0 left-0 w-full h-full bg-orange-500/5 -z-10"></div>
287
+ <span class="material-symbols-outlined text-orange-400 text-3xl">warning</span>
288
+ <div>
289
+ <h4 class="font-black text-orange-400 italic uppercase text-sm tracking-wider">Communication Breakdown</h4>
290
+ <p class="text-orange-200/80 font-medium">Failed to load data: Unknown error. Is the blob server running?</p>
291
+ </div>
292
+ </div>
293
+ <div class="grid grid-cols-1 lg:grid-cols-2 gap-12">
294
+ <section class="space-y-6">
295
+ <div class="flex items-center gap-4 border-b-4 border-primary/20 pb-4">
296
+ <div class="size-10 bg-primary/20 rounded-full flex items-center justify-center border-2 border-primary">
297
+ <span class="material-symbols-outlined text-primary font-bold">neurology</span>
298
+ </div>
299
+ <h2 class="text-2xl font-black italic text-primary uppercase">Recent Memories</h2>
300
+ </div>
301
+ <div class="space-y-4" id="memories">
302
+ <div class="bg-card-bg/50 p-6 rounded-3xl border-2 border-slate-700/20 flex flex-col items-center justify-center py-16 opacity-60">
303
+ <div class="size-12 border-4 border-primary border-t-transparent rounded-full animate-spin mb-4"></div>
304
+ <p class="font-black italic text-text-muted">Loading memories...</p>
305
+ </div>
306
+ </div>
307
+ </section>
308
+ <section class="space-y-6">
309
+ <div class="flex items-center gap-4 border-b-4 border-primary/20 pb-4">
310
+ <div class="size-10 bg-primary/20 rounded-full flex items-center justify-center border-2 border-primary">
311
+ <span class="material-symbols-outlined text-primary font-bold">visibility</span>
312
+ </div>
313
+ <h2 class="text-2xl font-black italic text-primary uppercase">Recent Observations</h2>
314
+ </div>
315
+ <div class="space-y-4" id="observations">
316
+ <div class="bg-card-bg/50 p-6 rounded-3xl border-2 border-slate-700/20 flex flex-col items-center justify-center py-16 opacity-60">
317
+ <div class="size-12 border-4 border-primary border-t-transparent rounded-full animate-spin mb-4"></div>
318
+ <p class="font-black italic text-text-muted">Loading observations...</p>
319
+ </div>
320
+ </div>
321
+ </section>
322
+ </div>
323
+ <div class="pt-12 flex justify-center">
324
+ <div class="bg-card-bg border-4 border-slate-700/50 p-2 rounded-full flex gap-2">
325
+ <button class="bg-primary text-black px-8 py-3 rounded-full font-black text-sm uppercase hover:scale-105 transition-transform flex items-center gap-2">
326
+ <span class="material-symbols-outlined text-sm">refresh</span>
327
+ Reconnect
328
+ </button>
329
+ <button class="text-text-main px-8 py-3 rounded-full font-black text-sm uppercase hover:bg-slate-700/50 transition-colors" onclick="openDocs()">
330
+ Docs
331
+ </button>
332
+ <button class="text-text-main px-8 py-3 rounded-full font-black text-sm uppercase hover:bg-slate-700/50 transition-colors" onclick="openSettings()">
333
+ Settings
334
+ </button>
335
+ </div>
336
+ </div>
337
+ </main>
338
+ <footer class="mt-20 px-6 opacity-30">
339
+ <div class="max-w-6xl mx-auto flex justify-between items-center py-8 border-t border-slate-700">
340
+ <p class="text-xs font-black uppercase">© 2026 Squish-Memory Dashboard</p>
341
+ <div class="flex gap-4">
342
+ <span class="material-symbols-outlined text-sm">database</span>
343
+ <span class="text-xs font-black uppercase italic">Local-First Engine v1.0</span>
344
+ </div>
345
+ </div>
346
+ </footer>
347
+ <script>
348
+ let currentProjectPath = null;
349
+
350
+ async function loadProjects() {
351
+ try {
352
+ const response = await fetch('/api/projects');
353
+ const data = await response.json();
354
+
355
+ if (data.status === 'ok' && data.data && data.data.length > 0) {
356
+ const select = document.getElementById('project-select');
357
+ if (select) {
358
+ select.innerHTML = data.data.map(function(p) {
359
+ return '<option value="' + escapeHtml(p.path) + '">' + escapeHtml(p.name || p.path) + '</option>';
360
+ }).join('');
361
+
362
+ // Try to select current directory
363
+ const cwd = window.location.pathname === '/' ? '' : window.location.pathname;
364
+ const defaultProject = data.data.find(function(p) { return p.path === cwd; }) || data.data[0];
365
+ if (defaultProject) {
366
+ currentProjectPath = defaultProject.path;
367
+ select.value = defaultProject.path;
368
+ }
369
+ }
370
+ } else {
371
+ // No projects yet, use current directory
372
+ currentProjectPath = window.location.pathname === '/' ? '' : window.location.pathname;
373
+ }
374
+ } catch (error) {
375
+ console.error('Failed to load projects:', error);
376
+ currentProjectPath = '';
377
+ }
378
+ }
379
+
380
+ async function loadData() {
381
+ try {
382
+ const url = currentProjectPath ? '/api/context?projectPath=' + encodeURIComponent(currentProjectPath) : '/api/context';
383
+ const response = await fetch(url);
384
+ const data = await response.json();
385
+
386
+ if (data.status === 'ok') {
387
+ document.getElementById('memories-count').textContent = data.memories ? data.memories.length : 0;
388
+ document.getElementById('observations-count').textContent = data.observations ? data.observations.length : 0;
389
+ document.getElementById('total-count').textContent = data.totalCount || 0;
390
+ updateStatus(data.memories && data.observations ? 'ok' : 'error');
391
+
392
+ renderMemories(data.memories || []);
393
+ renderObservations(data.observations || []);
394
+
395
+ // Update project info
396
+ if (data.project) {
397
+ const projectInfo = document.getElementById('project-info');
398
+ if (projectInfo) {
399
+ projectInfo.textContent = data.project.name || data.project.path || 'Unknown';
400
+ }
401
+ }
402
+
403
+ // Hide error alert if data loaded
404
+ const errorAlert = document.querySelector('.blob-alert');
405
+ if (errorAlert && (data.memories && data.memories.length > 0 || data.observations && data.observations.length > 0)) {
406
+ errorAlert.style.display = 'none';
407
+ }
408
+
409
+ // Show error alert if message present
410
+ if (data.message) {
411
+ const errorAlert = document.querySelector('.blob-alert');
412
+ if (errorAlert) {
413
+ errorAlert.querySelector('p').textContent = data.message;
414
+ errorAlert.style.display = 'flex';
415
+ }
416
+ }
417
+ } else {
418
+ throw new Error('API returned error status');
419
+ }
420
+ } catch (error) {
421
+ updateStatus('error');
422
+
423
+ // Show error alert
424
+ const errorAlert = document.querySelector('.blob-alert');
425
+ if (errorAlert) errorAlert.style.display = 'flex';
426
+ }
427
+ }
428
+
429
+ function renderMemories(memories) {
430
+ const container = document.getElementById('memories');
431
+ if (!memories || memories.length === 0) {
432
+ container.innerHTML = '<div class="bg-card-bg/50 p-6 rounded-3xl border-2 border-slate-700/20 flex flex-col items-center justify-center py-16 opacity-60"><p class="font-black italic text-text-muted">No memories found</p></div>';
433
+ return;
434
+ }
435
+
436
+ container.innerHTML = memories.map(function(memory) {
437
+ return '<div class="bg-card-bg p-6 rounded-3xl border-2 border-slate-700/20 squishy-hover">' +
438
+ '<div class="flex items-start justify-between mb-4">' +
439
+ '<span class="bg-primary text-black px-3 py-1 rounded-full text-xs font-bold uppercase">' + (memory.type || 'memory') + '</span>' +
440
+ '<span class="text-text-muted text-sm">' + formatTime(memory.createdAt) + '</span>' +
441
+ '</div>' +
442
+ '<div class="text-text-main mb-4">' + escapeHtml(memory.content || memory.text || '') + '</div>' +
443
+ '<div class="text-text-muted text-sm">' +
444
+ 'Tags: ' + (memory.tags ? memory.tags.join(', ') : 'none') +
445
+ '</div>' +
446
+ '</div>';
447
+ }).join('');
448
+ }
449
+
450
+ function renderObservations(observations) {
451
+ const container = document.getElementById('observations');
452
+ if (!observations || observations.length === 0) {
453
+ container.innerHTML = '<div class="bg-card-bg/50 p-6 rounded-3xl border-2 border-slate-700/20 flex flex-col items-center justify-center py-16 opacity-60"><p class="font-black italic text-text-muted">No observations found</p></div>';
454
+ return;
455
+ }
456
+
457
+ container.innerHTML = observations.map(function(obs) {
458
+ return '<div class="bg-card-bg p-6 rounded-3xl border-2 border-slate-700/20 squishy-hover">' +
459
+ '<div class="flex items-start justify-between mb-4">' +
460
+ '<span class="bg-secondary text-black px-3 py-1 rounded-full text-xs font-bold uppercase">' + (obs.type || 'observation') + '</span>' +
461
+ '<span class="text-text-muted text-sm">' + formatTime(obs.createdAt) + '</span>' +
462
+ '</div>' +
463
+ '<div class="text-text-main mb-4">' + escapeHtml(obs.summary || obs.content || '') + '</div>' +
464
+ '<div class="text-text-muted text-sm">' +
465
+ 'Action: ' + (obs.action || 'none') + ' | ' +
466
+ 'Target: ' + (obs.target || 'none') +
467
+ '</div>' +
468
+ '</div>';
469
+ }).join('');
470
+ }
471
+
472
+ function updateStatus(status) {
473
+ const statusCard = document.querySelector('.squishy-hover.flex-col');
474
+ if (!statusCard) return;
475
+
476
+ if (status === 'ok') {
477
+ statusCard.innerHTML = '<div class="flex items-center gap-3">' +
478
+ '<div class="size-4 bg-primary rounded-full animate-pulse"></div>' +
479
+ '<p class="text-2xl font-black text-primary italic">OK</p>' +
480
+ '</div>' +
481
+ '<p class="text-sm font-bold uppercase tracking-widest text-text-muted italic mt-1">Status</p>';
482
+ } else {
483
+ statusCard.innerHTML = '<div class="flex items-center gap-3">' +
484
+ '<div class="size-4 bg-red-500 rounded-full pulse-red"></div>' +
485
+ '<p class="text-2xl font-black text-red-400 italic">Error</p>' +
486
+ '</div>' +
487
+ '<p class="text-sm font-bold uppercase tracking-widest text-text-muted italic mt-1">Status</p>';
488
+ }
489
+ }
490
+
491
+ function formatTime(timestamp) {
492
+ if (!timestamp) return 'Unknown';
493
+ const date = new Date(timestamp);
494
+ return date.toLocaleString();
495
+ }
496
+
497
+ function escapeHtml(text) {
498
+ const div = document.createElement('div');
499
+ div.textContent = text;
500
+ return div.innerHTML;
501
+ }
502
+
503
+ function openDocs() {
504
+ // Create and show documentation modal
505
+ const modal = document.createElement('div');
506
+ modal.className = 'fixed inset-0 bg-black/50 flex items-center justify-center z-50';
507
+ modal.innerHTML = '<div class="bg-card-bg p-8 rounded-3xl border-2 border-slate-700/50 max-w-2xl w-full mx-4 max-h-[80vh] overflow-y-auto">' +
508
+ '<div class="flex justify-between items-center mb-6">' +
509
+ '<h2 class="text-2xl font-black text-primary italic">Documentation</h2>' +
510
+ '<button onclick="closeModal(this)" class="text-text-muted hover:text-text-main text-2xl">&times;</button>' +
511
+ '</div>' +
512
+ '<div class="space-y-4 text-text-main">' +
513
+ '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
514
+ '<h3 class="font-bold text-primary mb-2">🧠 Memory System</h3>' +
515
+ '<p class="text-sm">The Squish Memory Plugin captures and stores conversations, tool usage, and project insights across sessions.</p>' +
516
+ '</div>' +
517
+ '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
518
+ '<h3 class="font-bold text-primary mb-2">📊 API Endpoints</h3>' +
519
+ '<ul class="text-sm space-y-1">' +
520
+ '<li><code class="bg-slate-700/50 px-2 py-1 rounded">GET /api/health</code> - Service health status</li>' +
521
+ '<li><code class="bg-slate-700/50 px-2 py-1 rounded">GET /api/memories</code> - Recent memories</li>' +
522
+ '<li><code class="bg-slate-700/50 px-2 py-1 rounded">GET /api/observations</code> - Tool usage observations</li>' +
523
+ '<li><code class="bg-slate-700/50 px-2 py-1 rounded">GET /api/context</code> - Combined data</li>' +
524
+ '</ul>' +
525
+ '</div>' +
526
+ '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
527
+ '<h3 class="font-bold text-primary mb-2">⚙️ Configuration</h3>' +
528
+ '<p class="text-sm">Configure via environment variables: <code class="bg-slate-700/50 px-2 py-1 rounded">SQUISH_WEB_PORT</code>, <code class="bg-slate-700/50 px-2 py-1 rounded">DATABASE_URL</code>, etc.</p>' +
529
+ '</div>' +
530
+ '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
531
+ '<h3 class="font-bold text-primary mb-2">🚀 Getting Started</h3>' +
532
+ '<p class="text-sm">1. Install the plugin<br>2. Configure your database<br>3. Start the web UI<br>4. Access at http://localhost:37777</p>' +
533
+ '</div>' +
534
+ '</div>' +
535
+ '</div>';
536
+ document.body.appendChild(modal);
537
+ }
538
+
539
+ function openSettings() {
540
+ // Create and show settings modal
541
+ const modal = document.createElement('div');
542
+ modal.className = 'fixed inset-0 bg-black/50 flex items-center justify-center z-50';
543
+ modal.innerHTML = '<div class="bg-card-bg p-8 rounded-3xl border-2 border-slate-700/50 max-w-md w-full mx-4">' +
544
+ '<div class="flex justify-between items-center mb-6">' +
545
+ '<h2 class="text-2xl font-black text-primary italic">Settings</h2>' +
546
+ '<button onclick="closeModal(this)" class="text-text-muted hover:text-text-main text-2xl">&times;</button>' +
547
+ '</div>' +
548
+ '<div class="space-y-4">' +
549
+ '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
550
+ '<label class="block text-sm font-bold text-text-main mb-2">Refresh Interval</label>' +
551
+ '<select class="w-full bg-slate-700/50 border border-slate-600 rounded-lg px-3 py-2 text-text-main" onchange="changeRefreshInterval(this.value)">' +
552
+ '<option value="10000">10 seconds</option>' +
553
+ '<option value="30000" selected>30 seconds</option>' +
554
+ '<option value="60000">1 minute</option>' +
555
+ '<option value="300000">5 minutes</option>' +
556
+ '</select>' +
557
+ '</div>' +
558
+ '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
559
+ '<label class="block text-sm font-bold text-text-main mb-2">Theme</label>' +
560
+ '<select class="w-full bg-slate-700/50 border border-slate-600 rounded-lg px-3 py-2 text-text-main" onchange="changeTheme(this.value)">' +
561
+ '<option value="dark" selected>Dark</option>' +
562
+ '<option value="light">Light</option>' +
563
+ '</select>' +
564
+ '</div>' +
565
+ '<div class="bg-card-bg/50 p-4 rounded-xl border border-slate-700/30">' +
566
+ '<label class="block text-sm font-bold text-text-main mb-2">Items per Page</label>' +
567
+ '<select class="w-full bg-slate-700/50 border border-slate-600 rounded-lg px-3 py-2 text-text-main" onchange="changeItemsPerPage(this.value)">' +
568
+ '<option value="10">10</option>' +
569
+ '<option value="25" selected>25</option>' +
570
+ '<option value="50">50</option>' +
571
+ '<option value="100">100</option>' +
572
+ '</select>' +
573
+ '</div>' +
574
+ '<div class="flex justify-end space-x-3 mt-6">' +
575
+ '<button onclick="closeModal(this)" class="px-4 py-2 bg-slate-700/50 hover:bg-slate-600/50 rounded-lg text-text-main transition-colors">Cancel</button>' +
576
+ '<button onclick="saveSettings()" class="px-4 py-2 bg-primary text-black rounded-lg hover:bg-primary/80 transition-colors">Save</button>' +
577
+ '</div>' +
578
+ '</div>' +
579
+ '</div>';
580
+ document.body.appendChild(modal);
581
+ }
582
+
583
+ function closeModal(button) {
584
+ const modal = button.closest('.fixed');
585
+ if (modal) {
586
+ modal.remove();
587
+ }
588
+ }
589
+
590
+ function changeRefreshInterval(interval) {
591
+ clearInterval(window.refreshInterval);
592
+ window.refreshInterval = setInterval(loadData, parseInt(interval));
593
+ }
594
+
595
+ function changeTheme(theme) {
596
+ if (theme === 'light') {
597
+ document.documentElement.classList.remove('dark');
598
+ } else {
599
+ document.documentElement.classList.add('dark');
600
+ }
601
+ }
602
+
603
+ function changeItemsPerPage(count) {
604
+ // This would require updating the API calls, for now just store in localStorage
605
+ localStorage.setItem('itemsPerPage', count);
606
+ }
607
+
608
+ function saveSettings() {
609
+ // Close the modal and show success message
610
+ const modal = document.querySelector('.fixed');
611
+ if (modal) {
612
+ // Could show a toast notification here
613
+ modal.remove();
614
+ }
615
+ }
616
+
617
+ function changeProject(path) {
618
+ currentProjectPath = path;
619
+ loadData();
620
+ }
621
+
622
+ // Initialize: load projects first, then data
623
+ loadProjects().then(function() {
624
+ loadData();
625
+ window.refreshInterval = setInterval(loadData, 30000);
626
+ });
627
+ </script>
603
628
  </body></html>`;
604
629
  res.send(html);
605
630
  });
606
631
  // Start server
607
632
  export function startWebServer() {
608
- app.listen(PORT, () => {
609
- logger.info('Web UI available at http://localhost:' + PORT);
633
+ return new Promise((resolve, reject) => {
634
+ const server = app.listen(PORT, () => {
635
+ logger.info('Web UI available at http://localhost:' + PORT);
636
+ resolve(server);
637
+ });
638
+ server.on('error', reject);
610
639
  });
611
640
  }
612
641
  export default app;
613
- //# sourceMappingURL=web.js.map
642
+ //# sourceMappingURL=server.js.map