memento-mcp-server 0.1.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 (555) hide show
  1. package/.cursor/rules/cache-management.mdc +171 -0
  2. package/.cursor/rules/database-schema.mdc +344 -0
  3. package/.cursor/rules/deployment.mdc +596 -0
  4. package/.cursor/rules/error-logging.mdc +298 -0
  5. package/.cursor/rules/forgetting-algorithms.mdc +426 -0
  6. package/.cursor/rules/http-server.mdc +432 -0
  7. package/.cursor/rules/hybrid-search.mdc +424 -0
  8. package/.cursor/rules/implementation.mdc +369 -0
  9. package/.cursor/rules/lightweight-embedding.mdc +178 -0
  10. package/.cursor/rules/mcp-client-development.mdc +0 -0
  11. package/.cursor/rules/mcp-server-development.mdc +0 -0
  12. package/.cursor/rules/mcp-tools-architecture.mdc +205 -0
  13. package/.cursor/rules/memento-memory-rule.mdc +8 -0
  14. package/.cursor/rules/memento-project-overview.mdc +0 -0
  15. package/.cursor/rules/memory-algorithms.mdc +502 -0
  16. package/.cursor/rules/monitoring.mdc +622 -0
  17. package/.cursor/rules/performance-alerts.mdc +537 -0
  18. package/.cursor/rules/performance-monitoring.mdc +345 -0
  19. package/.cursor/rules/performance-optimization.mdc +563 -0
  20. package/.cursor/rules/project-structure.mdc +310 -0
  21. package/.cursor/rules/testing.mdc +473 -0
  22. package/.dockerignore +15 -0
  23. package/.eslintrc.json +34 -0
  24. package/.github/ISSUE_TEMPLATE/bug_report.md +51 -0
  25. package/.github/ISSUE_TEMPLATE/feature_request.md +45 -0
  26. package/.github/ISSUE_TEMPLATE/question.md +31 -0
  27. package/.github/PULL_REQUEST_TEMPLATE.md +97 -0
  28. package/AGENTS.md +28 -0
  29. package/CHANGELOG.md +416 -0
  30. package/CODE_OF_CONDUCT.md +62 -0
  31. package/CONTRIBUTING.md +165 -0
  32. package/DOCKER_SETUP_GUIDE.md +84 -0
  33. package/Dockerfile +90 -0
  34. package/INSTALL.en.md +303 -0
  35. package/INSTALL.md +303 -0
  36. package/README.en.md +330 -0
  37. package/README.md +427 -0
  38. package/backup/legacy-removal-20250930-201816/http-server-legacy.ts +1068 -0
  39. package/backup/legacy-removal-20250930-201816/rollback-to-legacy.ps1 +46 -0
  40. package/backup/legacy-removal-20250930-201816/rollback-to-legacy.sh +48 -0
  41. package/dist/algorithms/forgetting-algorithm.d.ts +90 -0
  42. package/dist/algorithms/forgetting-algorithm.d.ts.map +1 -0
  43. package/dist/algorithms/forgetting-algorithm.js +160 -0
  44. package/dist/algorithms/forgetting-algorithm.js.map +1 -0
  45. package/dist/algorithms/hybrid-search-engine.d.ts +108 -0
  46. package/dist/algorithms/hybrid-search-engine.d.ts.map +1 -0
  47. package/dist/algorithms/hybrid-search-engine.js +384 -0
  48. package/dist/algorithms/hybrid-search-engine.js.map +1 -0
  49. package/dist/algorithms/search-engine.d.ts +64 -0
  50. package/dist/algorithms/search-engine.d.ts.map +1 -0
  51. package/dist/algorithms/search-engine.js +304 -0
  52. package/dist/algorithms/search-engine.js.map +1 -0
  53. package/dist/algorithms/search-ranking.d.ts +145 -0
  54. package/dist/algorithms/search-ranking.d.ts.map +1 -0
  55. package/dist/algorithms/search-ranking.js +311 -0
  56. package/dist/algorithms/search-ranking.js.map +1 -0
  57. package/dist/algorithms/spaced-repetition.d.ts +83 -0
  58. package/dist/algorithms/spaced-repetition.d.ts.map +1 -0
  59. package/dist/algorithms/spaced-repetition.js +153 -0
  60. package/dist/algorithms/spaced-repetition.js.map +1 -0
  61. package/dist/algorithms/vector-search-engine.d.ts +96 -0
  62. package/dist/algorithms/vector-search-engine.d.ts.map +1 -0
  63. package/dist/algorithms/vector-search-engine.js +410 -0
  64. package/dist/algorithms/vector-search-engine.js.map +1 -0
  65. package/dist/client/index.d.ts +55 -0
  66. package/dist/client/index.d.ts.map +1 -0
  67. package/dist/client/index.js +179 -0
  68. package/dist/client/index.js.map +1 -0
  69. package/dist/config/index.d.ts +20 -0
  70. package/dist/config/index.d.ts.map +1 -0
  71. package/dist/config/index.js +87 -0
  72. package/dist/config/index.js.map +1 -0
  73. package/dist/database/init.d.ts +7 -0
  74. package/dist/database/init.d.ts.map +1 -0
  75. package/dist/database/init.js +117 -0
  76. package/dist/database/init.js.map +1 -0
  77. package/dist/database/migrate.d.ts +6 -0
  78. package/dist/database/migrate.d.ts.map +1 -0
  79. package/dist/database/migrate.js +84 -0
  80. package/dist/database/migrate.js.map +1 -0
  81. package/dist/database/schema.sql +159 -0
  82. package/dist/npm-client/context-injector.d.ts +106 -0
  83. package/dist/npm-client/context-injector.d.ts.map +1 -0
  84. package/dist/npm-client/context-injector.js +296 -0
  85. package/dist/npm-client/context-injector.js.map +1 -0
  86. package/dist/npm-client/index.d.ts +42 -0
  87. package/dist/npm-client/index.d.ts.map +1 -0
  88. package/dist/npm-client/index.js +43 -0
  89. package/dist/npm-client/index.js.map +1 -0
  90. package/dist/npm-client/memento-client.d.ts +114 -0
  91. package/dist/npm-client/memento-client.d.ts.map +1 -0
  92. package/dist/npm-client/memento-client.js +391 -0
  93. package/dist/npm-client/memento-client.js.map +1 -0
  94. package/dist/npm-client/memory-manager.d.ts +137 -0
  95. package/dist/npm-client/memory-manager.d.ts.map +1 -0
  96. package/dist/npm-client/memory-manager.js +341 -0
  97. package/dist/npm-client/memory-manager.js.map +1 -0
  98. package/dist/npm-client/types.d.ts +216 -0
  99. package/dist/npm-client/types.d.ts.map +1 -0
  100. package/dist/npm-client/types.js +44 -0
  101. package/dist/npm-client/types.js.map +1 -0
  102. package/dist/npm-client/utils.d.ts +91 -0
  103. package/dist/npm-client/utils.d.ts.map +1 -0
  104. package/dist/npm-client/utils.js +351 -0
  105. package/dist/npm-client/utils.js.map +1 -0
  106. package/dist/scripts/copy-assets.d.ts +3 -0
  107. package/dist/scripts/copy-assets.d.ts.map +1 -0
  108. package/dist/scripts/copy-assets.js +31 -0
  109. package/dist/scripts/copy-assets.js.map +1 -0
  110. package/dist/server/http-server.d.ts +28 -0
  111. package/dist/server/http-server.d.ts.map +1 -0
  112. package/dist/server/http-server.js +930 -0
  113. package/dist/server/http-server.js.map +1 -0
  114. package/dist/server/index-backup.d.ts +5 -0
  115. package/dist/server/index-backup.d.ts.map +1 -0
  116. package/dist/server/index-backup.js +793 -0
  117. package/dist/server/index-backup.js.map +1 -0
  118. package/dist/server/index-refactored.d.ts +6 -0
  119. package/dist/server/index-refactored.d.ts.map +1 -0
  120. package/dist/server/index-refactored.js +206 -0
  121. package/dist/server/index-refactored.js.map +1 -0
  122. package/dist/server/index.d.ts +12 -0
  123. package/dist/server/index.d.ts.map +1 -0
  124. package/dist/server/index.js +301 -0
  125. package/dist/server/index.js.map +1 -0
  126. package/dist/server/simple-mcp-server.d.ts +8 -0
  127. package/dist/server/simple-mcp-server.d.ts.map +1 -0
  128. package/dist/server/simple-mcp-server.js +152 -0
  129. package/dist/server/simple-mcp-server.js.map +1 -0
  130. package/dist/services/async-optimizer.d.ts +136 -0
  131. package/dist/services/async-optimizer.d.ts.map +1 -0
  132. package/dist/services/async-optimizer.js +406 -0
  133. package/dist/services/async-optimizer.js.map +1 -0
  134. package/dist/services/batch-scheduler.d.ts +156 -0
  135. package/dist/services/batch-scheduler.d.ts.map +1 -0
  136. package/dist/services/batch-scheduler.js +612 -0
  137. package/dist/services/batch-scheduler.js.map +1 -0
  138. package/dist/services/cache-service.d.ts +168 -0
  139. package/dist/services/cache-service.d.ts.map +1 -0
  140. package/dist/services/cache-service.js +360 -0
  141. package/dist/services/cache-service.js.map +1 -0
  142. package/dist/services/database-optimizer.d.ts +110 -0
  143. package/dist/services/database-optimizer.d.ts.map +1 -0
  144. package/dist/services/database-optimizer.js +369 -0
  145. package/dist/services/database-optimizer.js.map +1 -0
  146. package/dist/services/embedding-provider-factory.d.ts +62 -0
  147. package/dist/services/embedding-provider-factory.d.ts.map +1 -0
  148. package/dist/services/embedding-provider-factory.js +152 -0
  149. package/dist/services/embedding-provider-factory.js.map +1 -0
  150. package/dist/services/embedding-service.d.ts +95 -0
  151. package/dist/services/embedding-service.d.ts.map +1 -0
  152. package/dist/services/embedding-service.js +276 -0
  153. package/dist/services/embedding-service.js.map +1 -0
  154. package/dist/services/error-logging-service.d.ts +134 -0
  155. package/dist/services/error-logging-service.d.ts.map +1 -0
  156. package/dist/services/error-logging-service.js +271 -0
  157. package/dist/services/error-logging-service.js.map +1 -0
  158. package/dist/services/forgetting-policy-service.d.ts +94 -0
  159. package/dist/services/forgetting-policy-service.d.ts.map +1 -0
  160. package/dist/services/forgetting-policy-service.js +233 -0
  161. package/dist/services/forgetting-policy-service.js.map +1 -0
  162. package/dist/services/gemini-embedding-service.d.ts +82 -0
  163. package/dist/services/gemini-embedding-service.d.ts.map +1 -0
  164. package/dist/services/gemini-embedding-service.js +232 -0
  165. package/dist/services/gemini-embedding-service.js.map +1 -0
  166. package/dist/services/lightweight-embedding-service.d.ts +95 -0
  167. package/dist/services/lightweight-embedding-service.d.ts.map +1 -0
  168. package/dist/services/lightweight-embedding-service.js +263 -0
  169. package/dist/services/lightweight-embedding-service.js.map +1 -0
  170. package/dist/services/memory-embedding-service.d.ts +69 -0
  171. package/dist/services/memory-embedding-service.d.ts.map +1 -0
  172. package/dist/services/memory-embedding-service.js +223 -0
  173. package/dist/services/memory-embedding-service.js.map +1 -0
  174. package/dist/services/minilm-embedding-service.d.ts +92 -0
  175. package/dist/services/minilm-embedding-service.d.ts.map +1 -0
  176. package/dist/services/minilm-embedding-service.js +216 -0
  177. package/dist/services/minilm-embedding-service.js.map +1 -0
  178. package/dist/services/performance-alert-service.d.ts +142 -0
  179. package/dist/services/performance-alert-service.d.ts.map +1 -0
  180. package/dist/services/performance-alert-service.js +366 -0
  181. package/dist/services/performance-alert-service.js.map +1 -0
  182. package/dist/services/performance-monitor.d.ts +144 -0
  183. package/dist/services/performance-monitor.d.ts.map +1 -0
  184. package/dist/services/performance-monitor.js +416 -0
  185. package/dist/services/performance-monitor.js.map +1 -0
  186. package/dist/services/performance-monitoring-integration.d.ts +77 -0
  187. package/dist/services/performance-monitoring-integration.d.ts.map +1 -0
  188. package/dist/services/performance-monitoring-integration.js +177 -0
  189. package/dist/services/performance-monitoring-integration.js.map +1 -0
  190. package/dist/services/unified-embedding-service.d.ts +75 -0
  191. package/dist/services/unified-embedding-service.d.ts.map +1 -0
  192. package/dist/services/unified-embedding-service.js +211 -0
  193. package/dist/services/unified-embedding-service.js.map +1 -0
  194. package/dist/test/debug-http-v2.d.ts +5 -0
  195. package/dist/test/debug-http-v2.d.ts.map +1 -0
  196. package/dist/test/debug-http-v2.js +28 -0
  197. package/dist/test/debug-http-v2.js.map +1 -0
  198. package/dist/test/performance-benchmark.d.ts +57 -0
  199. package/dist/test/performance-benchmark.d.ts.map +1 -0
  200. package/dist/test/performance-benchmark.js +427 -0
  201. package/dist/test/performance-benchmark.js.map +1 -0
  202. package/dist/test/test-alerts-direct.d.ts +7 -0
  203. package/dist/test/test-alerts-direct.d.ts.map +1 -0
  204. package/dist/test/test-alerts-direct.js +101 -0
  205. package/dist/test/test-alerts-direct.js.map +1 -0
  206. package/dist/test/test-batch-scheduler.d.ts +2 -0
  207. package/dist/test/test-batch-scheduler.d.ts.map +1 -0
  208. package/dist/test/test-batch-scheduler.js +156 -0
  209. package/dist/test/test-batch-scheduler.js.map +1 -0
  210. package/dist/test/test-client.d.ts +5 -0
  211. package/dist/test/test-client.d.ts.map +1 -0
  212. package/dist/test/test-client.js +86 -0
  213. package/dist/test/test-client.js.map +1 -0
  214. package/dist/test/test-embedding.d.ts +6 -0
  215. package/dist/test/test-embedding.d.ts.map +1 -0
  216. package/dist/test/test-embedding.js +142 -0
  217. package/dist/test/test-embedding.js.map +1 -0
  218. package/dist/test/test-error-logging.d.ts +7 -0
  219. package/dist/test/test-error-logging.d.ts.map +1 -0
  220. package/dist/test/test-error-logging.js +105 -0
  221. package/dist/test/test-error-logging.js.map +1 -0
  222. package/dist/test/test-forgetting.d.ts +6 -0
  223. package/dist/test/test-forgetting.d.ts.map +1 -0
  224. package/dist/test/test-forgetting.js +154 -0
  225. package/dist/test/test-forgetting.js.map +1 -0
  226. package/dist/test/test-gemini-embedding.d.ts +7 -0
  227. package/dist/test/test-gemini-embedding.d.ts.map +1 -0
  228. package/dist/test/test-gemini-embedding.js +134 -0
  229. package/dist/test/test-gemini-embedding.js.map +1 -0
  230. package/dist/test/test-http-server-v2-simple.d.ts +6 -0
  231. package/dist/test/test-http-server-v2-simple.d.ts.map +1 -0
  232. package/dist/test/test-http-server-v2-simple.js +131 -0
  233. package/dist/test/test-http-server-v2-simple.js.map +1 -0
  234. package/dist/test/test-http-server-v2.d.ts +7 -0
  235. package/dist/test/test-http-server-v2.d.ts.map +1 -0
  236. package/dist/test/test-http-server-v2.js +529 -0
  237. package/dist/test/test-http-server-v2.js.map +1 -0
  238. package/dist/test/test-lightweight-embedding.d.ts +6 -0
  239. package/dist/test/test-lightweight-embedding.d.ts.map +1 -0
  240. package/dist/test/test-lightweight-embedding.js +189 -0
  241. package/dist/test/test-lightweight-embedding.js.map +1 -0
  242. package/dist/test/test-m1-completion.d.ts +7 -0
  243. package/dist/test/test-m1-completion.d.ts.map +1 -0
  244. package/dist/test/test-m1-completion.js +124 -0
  245. package/dist/test/test-m1-completion.js.map +1 -0
  246. package/dist/test/test-memory-injection-prompt.d.ts +2 -0
  247. package/dist/test/test-memory-injection-prompt.d.ts.map +1 -0
  248. package/dist/test/test-memory-injection-prompt.js +299 -0
  249. package/dist/test/test-memory-injection-prompt.js.map +1 -0
  250. package/dist/test/test-performance-alerts.d.ts +7 -0
  251. package/dist/test/test-performance-alerts.d.ts.map +1 -0
  252. package/dist/test/test-performance-alerts.js +109 -0
  253. package/dist/test/test-performance-alerts.js.map +1 -0
  254. package/dist/test/test-performance-monitor.d.ts +2 -0
  255. package/dist/test/test-performance-monitor.d.ts.map +1 -0
  256. package/dist/test/test-performance-monitor.js +182 -0
  257. package/dist/test/test-performance-monitor.js.map +1 -0
  258. package/dist/test/test-performance-monitoring.d.ts +6 -0
  259. package/dist/test/test-performance-monitoring.d.ts.map +1 -0
  260. package/dist/test/test-performance-monitoring.js +156 -0
  261. package/dist/test/test-performance-monitoring.js.map +1 -0
  262. package/dist/test/test-search.d.ts +5 -0
  263. package/dist/test/test-search.d.ts.map +1 -0
  264. package/dist/test/test-search.js +141 -0
  265. package/dist/test/test-search.js.map +1 -0
  266. package/dist/test/test-simple-alerts.d.ts +6 -0
  267. package/dist/test/test-simple-alerts.d.ts.map +1 -0
  268. package/dist/test/test-simple-alerts.js +106 -0
  269. package/dist/test/test-simple-alerts.js.map +1 -0
  270. package/dist/test/test-vector-search-engine.d.ts +2 -0
  271. package/dist/test/test-vector-search-engine.d.ts.map +1 -0
  272. package/dist/test/test-vector-search-engine.js +225 -0
  273. package/dist/test/test-vector-search-engine.js.map +1 -0
  274. package/dist/tools/base-tool.d.ts +64 -0
  275. package/dist/tools/base-tool.d.ts.map +1 -0
  276. package/dist/tools/base-tool.js +158 -0
  277. package/dist/tools/base-tool.js.map +1 -0
  278. package/dist/tools/cleanup-memory-tool.d.ts +10 -0
  279. package/dist/tools/cleanup-memory-tool.d.ts.map +1 -0
  280. package/dist/tools/cleanup-memory-tool.js +66 -0
  281. package/dist/tools/cleanup-memory-tool.js.map +1 -0
  282. package/dist/tools/database-optimize-tool.d.ts +10 -0
  283. package/dist/tools/database-optimize-tool.d.ts.map +1 -0
  284. package/dist/tools/database-optimize-tool.js +64 -0
  285. package/dist/tools/database-optimize-tool.js.map +1 -0
  286. package/dist/tools/error-stats.d.ts +93 -0
  287. package/dist/tools/error-stats.d.ts.map +1 -0
  288. package/dist/tools/error-stats.js +115 -0
  289. package/dist/tools/error-stats.js.map +1 -0
  290. package/dist/tools/forget-tool.d.ts +63 -0
  291. package/dist/tools/forget-tool.d.ts.map +1 -0
  292. package/dist/tools/forget-tool.js +340 -0
  293. package/dist/tools/forget-tool.js.map +1 -0
  294. package/dist/tools/forgetting-stats-tool.d.ts +10 -0
  295. package/dist/tools/forgetting-stats-tool.d.ts.map +1 -0
  296. package/dist/tools/forgetting-stats-tool.js +37 -0
  297. package/dist/tools/forgetting-stats-tool.js.map +1 -0
  298. package/dist/tools/index.d.ts +33 -0
  299. package/dist/tools/index.d.ts.map +1 -0
  300. package/dist/tools/index.js +55 -0
  301. package/dist/tools/index.js.map +1 -0
  302. package/dist/tools/memory-injection-prompt.d.ts +31 -0
  303. package/dist/tools/memory-injection-prompt.d.ts.map +1 -0
  304. package/dist/tools/memory-injection-prompt.js +203 -0
  305. package/dist/tools/memory-injection-prompt.js.map +1 -0
  306. package/dist/tools/performance-alerts.d.ts +127 -0
  307. package/dist/tools/performance-alerts.d.ts.map +1 -0
  308. package/dist/tools/performance-alerts.js +208 -0
  309. package/dist/tools/performance-alerts.js.map +1 -0
  310. package/dist/tools/performance-stats-tool.d.ts +10 -0
  311. package/dist/tools/performance-stats-tool.d.ts.map +1 -0
  312. package/dist/tools/performance-stats-tool.js +38 -0
  313. package/dist/tools/performance-stats-tool.js.map +1 -0
  314. package/dist/tools/pin-tool.d.ts +39 -0
  315. package/dist/tools/pin-tool.d.ts.map +1 -0
  316. package/dist/tools/pin-tool.js +211 -0
  317. package/dist/tools/pin-tool.js.map +1 -0
  318. package/dist/tools/recall-tool.d.ts +27 -0
  319. package/dist/tools/recall-tool.d.ts.map +1 -0
  320. package/dist/tools/recall-tool.js +335 -0
  321. package/dist/tools/recall-tool.js.map +1 -0
  322. package/dist/tools/remember-tool.d.ts +10 -0
  323. package/dist/tools/remember-tool.d.ts.map +1 -0
  324. package/dist/tools/remember-tool.js +101 -0
  325. package/dist/tools/remember-tool.js.map +1 -0
  326. package/dist/tools/resolve-error.d.ts +54 -0
  327. package/dist/tools/resolve-error.d.ts.map +1 -0
  328. package/dist/tools/resolve-error.js +63 -0
  329. package/dist/tools/resolve-error.js.map +1 -0
  330. package/dist/tools/tool-registry.d.ts +134 -0
  331. package/dist/tools/tool-registry.d.ts.map +1 -0
  332. package/dist/tools/tool-registry.js +337 -0
  333. package/dist/tools/tool-registry.js.map +1 -0
  334. package/dist/tools/types.d.ts +56 -0
  335. package/dist/tools/types.d.ts.map +1 -0
  336. package/dist/tools/types.js +23 -0
  337. package/dist/tools/types.js.map +1 -0
  338. package/dist/tools/unpin-tool.d.ts +43 -0
  339. package/dist/tools/unpin-tool.d.ts.map +1 -0
  340. package/dist/tools/unpin-tool.js +233 -0
  341. package/dist/tools/unpin-tool.js.map +1 -0
  342. package/dist/types/embedding.types.d.ts +65 -0
  343. package/dist/types/embedding.types.d.ts.map +1 -0
  344. package/dist/types/embedding.types.js +6 -0
  345. package/dist/types/embedding.types.js.map +1 -0
  346. package/dist/types/index.d.ts +103 -0
  347. package/dist/types/index.d.ts.map +1 -0
  348. package/dist/types/index.js +5 -0
  349. package/dist/types/index.js.map +1 -0
  350. package/dist/utils/database.d.ts +62 -0
  351. package/dist/utils/database.d.ts.map +1 -0
  352. package/dist/utils/database.js +399 -0
  353. package/dist/utils/database.js.map +1 -0
  354. package/dist/utils/stopwords.d.ts +18 -0
  355. package/dist/utils/stopwords.d.ts.map +1 -0
  356. package/dist/utils/stopwords.js +117 -0
  357. package/dist/utils/stopwords.js.map +1 -0
  358. package/docker-compose.dev.yml +50 -0
  359. package/docker-compose.prod.yml +77 -0
  360. package/docker-compose.yml +43 -0
  361. package/docs/en/Memento-Goals.md +217 -0
  362. package/docs/en/Memento-M1-DetailSpecs.md +130 -0
  363. package/docs/en/Memento-Milestones.md +135 -0
  364. package/docs/en/Search-Ranking-Memory-Decay-Formulas.md +177 -0
  365. package/docs/en/api-reference.md +658 -0
  366. package/docs/en/architecture.md +1302 -0
  367. package/docs/en/developer-guide.md +1005 -0
  368. package/docs/en/user-manual.md +595 -0
  369. package/docs/ko/Memento-Goals.md +217 -0
  370. package/docs/ko/Memento-M1-DetailSpecs.md +130 -0
  371. package/docs/ko/Memento-Milestones.md +134 -0
  372. package/docs/ko/Search-Ranking-Memory-Decay-Formulas.md +177 -0
  373. package/docs/ko/api-reference.md +658 -0
  374. package/docs/ko/architecture.md +1302 -0
  375. package/docs/ko/developer-guide.md +1006 -0
  376. package/docs/ko/embedding-api-reference.md +122 -0
  377. package/docs/ko/embedding-configuration.md +62 -0
  378. package/docs/ko/embedding-performance-benchmark.md +62 -0
  379. package/docs/ko/embedding-service-guide.md +314 -0
  380. package/docs/ko/user-manual.md +595 -0
  381. package/env.example +49 -0
  382. package/install.sh +191 -0
  383. package/mcp-http-client.js +218 -0
  384. package/mcp.json +0 -0
  385. package/package.json +125 -0
  386. package/packages/mcp-client/docs/API-REFERENCE.md +560 -0
  387. package/packages/mcp-client/docs/BEST-PRACTICES.md +564 -0
  388. package/packages/mcp-client/docs/MIGRATION-GUIDE.md +344 -0
  389. package/packages/mcp-client/docs/PERFORMANCE-GUIDE.md +476 -0
  390. package/packages/mcp-client/docs/TROUBLESHOOTING.md +564 -0
  391. package/packages/mcp-client/package-lock.json +907 -0
  392. package/packages/mcp-client/package.json +58 -0
  393. package/packages/mcp-client/performance-optimizer.js +428 -0
  394. package/packages/mcp-client/test-basic.js +65 -0
  395. package/packages/mcp-client/test-integration.js +366 -0
  396. package/scripts/auto-setup.js +234 -0
  397. package/scripts/backup-daily.bat +28 -0
  398. package/scripts/backup-embeddings.js +108 -0
  399. package/scripts/check-db-integrity.js +93 -0
  400. package/scripts/debug-embeddings.js +184 -0
  401. package/scripts/direct-sql-migration.sql +36 -0
  402. package/scripts/docker-migration.sh +105 -0
  403. package/scripts/fix-migration.js +93 -0
  404. package/scripts/fix-vector-dimensions.js +206 -0
  405. package/scripts/migrate-embedding-data.js +307 -0
  406. package/scripts/regenerate-embeddings.js +144 -0
  407. package/scripts/restore-legacy.ps1 +60 -0
  408. package/scripts/restore-legacy.sh +62 -0
  409. package/scripts/run-migration.js +122 -0
  410. package/scripts/safe-migration.js +150 -0
  411. package/scripts/simple-migrate.js +41 -0
  412. package/scripts/simple-update.js +123 -0
  413. package/scripts/start-container.sh +10 -0
  414. package/src/algorithms/forgetting-algorithm.spec.ts +538 -0
  415. package/src/algorithms/forgetting-algorithm.ts +243 -0
  416. package/src/algorithms/hybrid-search-engine.spec.ts +484 -0
  417. package/src/algorithms/hybrid-search-engine.ts +489 -0
  418. package/src/algorithms/search-engine.spec.ts +429 -0
  419. package/src/algorithms/search-engine.ts +392 -0
  420. package/src/algorithms/search-ranking.spec.ts +293 -0
  421. package/src/algorithms/search-ranking.ts +407 -0
  422. package/src/algorithms/spaced-repetition.spec.ts +510 -0
  423. package/src/algorithms/spaced-repetition.ts +238 -0
  424. package/src/algorithms/vector-search-engine.ts +505 -0
  425. package/src/client/index.spec.ts +427 -0
  426. package/src/client/index.ts +222 -0
  427. package/src/config/index.spec.ts +339 -0
  428. package/src/config/index.ts +106 -0
  429. package/src/database/init.spec.ts +239 -0
  430. package/src/database/init.ts +130 -0
  431. package/src/database/migrate.spec.ts +293 -0
  432. package/src/database/migrate.ts +94 -0
  433. package/src/database/migrations/001_add_embedding_metadata.sql +29 -0
  434. package/src/database/schema.sql +159 -0
  435. package/src/npm-client/context-injector.spec.ts +335 -0
  436. package/src/npm-client/context-injector.ts +412 -0
  437. package/src/npm-client/index.spec.ts +108 -0
  438. package/src/npm-client/index.ts +96 -0
  439. package/src/npm-client/memento-client.spec.ts +549 -0
  440. package/src/npm-client/memento-client.ts +501 -0
  441. package/src/npm-client/memory-manager.spec.ts +374 -0
  442. package/src/npm-client/memory-manager.ts +414 -0
  443. package/src/npm-client/types.spec.ts +427 -0
  444. package/src/npm-client/types.ts +296 -0
  445. package/src/npm-client/utils.spec.ts +355 -0
  446. package/src/npm-client/utils.ts +423 -0
  447. package/src/scripts/copy-assets.js +37 -0
  448. package/src/server/http-server.spec.ts +648 -0
  449. package/src/server/http-server.ts +1030 -0
  450. package/src/server/index-backup.ts +875 -0
  451. package/src/server/index-refactored.ts +237 -0
  452. package/src/server/index.spec.ts +281 -0
  453. package/src/server/index.ts +347 -0
  454. package/src/server/simple-mcp-server.spec.ts +207 -0
  455. package/src/server/simple-mcp-server.ts +173 -0
  456. package/src/services/async-optimizer.spec.ts +569 -0
  457. package/src/services/async-optimizer.ts +484 -0
  458. package/src/services/batch-scheduler.ts +759 -0
  459. package/src/services/cache-service.spec.ts +372 -0
  460. package/src/services/cache-service.ts +434 -0
  461. package/src/services/database-optimizer.spec.ts +344 -0
  462. package/src/services/database-optimizer.ts +450 -0
  463. package/src/services/embedding-provider-factory.ts +173 -0
  464. package/src/services/embedding-service.spec.ts +342 -0
  465. package/src/services/embedding-service.ts +333 -0
  466. package/src/services/error-logging-service.spec.ts +416 -0
  467. package/src/services/error-logging-service.ts +383 -0
  468. package/src/services/forgetting-policy-service.spec.ts +140 -0
  469. package/src/services/forgetting-policy-service.ts +334 -0
  470. package/src/services/gemini-embedding-service.spec.ts +463 -0
  471. package/src/services/gemini-embedding-service.ts +283 -0
  472. package/src/services/lightweight-embedding-service.spec.ts +458 -0
  473. package/src/services/lightweight-embedding-service.ts +324 -0
  474. package/src/services/memory-embedding-service.spec.ts +417 -0
  475. package/src/services/memory-embedding-service.ts +289 -0
  476. package/src/services/minilm-embedding-service.spec.ts +104 -0
  477. package/src/services/minilm-embedding-service.ts +262 -0
  478. package/src/services/performance-alert-service.spec.ts +517 -0
  479. package/src/services/performance-alert-service.ts +477 -0
  480. package/src/services/performance-monitor.spec.ts +401 -0
  481. package/src/services/performance-monitor.ts +532 -0
  482. package/src/services/performance-monitoring-integration.spec.ts +478 -0
  483. package/src/services/performance-monitoring-integration.ts.bak +276 -0
  484. package/src/services/unified-embedding-service.spec.ts +224 -0
  485. package/src/services/unified-embedding-service.ts +255 -0
  486. package/src/test/debug-http-v2.ts +30 -0
  487. package/src/test/embedding-integration-test.spec.ts +295 -0
  488. package/src/test/embedding-integration-test.ts +295 -0
  489. package/src/test/embedding-performance-benchmark.spec.ts +354 -0
  490. package/src/test/embedding-performance-benchmark.ts +312 -0
  491. package/src/test/performance-benchmark.ts +565 -0
  492. package/src/test/test-alerts-direct.ts +136 -0
  493. package/src/test/test-batch-scheduler-simple.spec.ts +122 -0
  494. package/src/test/test-batch-scheduler.spec.ts +453 -0
  495. package/src/test/test-batch-scheduler.ts +182 -0
  496. package/src/test/test-client.ts +97 -0
  497. package/src/test/test-embedding.ts +153 -0
  498. package/src/test/test-error-handling.spec.ts +575 -0
  499. package/src/test/test-error-logging.ts +117 -0
  500. package/src/test/test-forgetting.ts +162 -0
  501. package/src/test/test-gemini-embedding.ts +159 -0
  502. package/src/test/test-http-server-v2-simple.ts +147 -0
  503. package/src/test/test-http-server-v2.ts +586 -0
  504. package/src/test/test-hybrid-search-engine.spec.ts +521 -0
  505. package/src/test/test-integration-fixed.spec.ts +612 -0
  506. package/src/test/test-integration.spec.ts +463 -0
  507. package/src/test/test-lightweight-embedding.ts +208 -0
  508. package/src/test/test-m1-completion.spec.ts +614 -0
  509. package/src/test/test-m1-completion.ts +141 -0
  510. package/src/test/test-m1-integration.spec.ts +514 -0
  511. package/src/test/test-memory-injection-prompt.spec.ts +650 -0
  512. package/src/test/test-memory-injection-prompt.ts +391 -0
  513. package/src/test/test-performance-alerts.ts +125 -0
  514. package/src/test/test-performance-monitor-updates.spec.ts +490 -0
  515. package/src/test/test-performance-monitor.spec.ts +284 -0
  516. package/src/test/test-performance-monitor.ts +228 -0
  517. package/src/test/test-performance-monitoring.ts +171 -0
  518. package/src/test/test-search.ts +151 -0
  519. package/src/test/test-simple-alerts.ts +136 -0
  520. package/src/test/test-vector-search-engine.spec.ts +408 -0
  521. package/src/test/test-vector-search-engine.ts +303 -0
  522. package/src/tools/base-tool.ts +189 -0
  523. package/src/tools/cleanup-memory-tool.ts +77 -0
  524. package/src/tools/database-optimize-tool.ts +79 -0
  525. package/src/tools/error-stats.ts +119 -0
  526. package/src/tools/forget-tool.spec.ts +613 -0
  527. package/src/tools/forget-tool.ts +454 -0
  528. package/src/tools/forgetting-stats-tool.ts +47 -0
  529. package/src/tools/index.ts +71 -0
  530. package/src/tools/memory-injection-prompt.ts +257 -0
  531. package/src/tools/performance-alerts.ts +226 -0
  532. package/src/tools/performance-stats-tool.ts +48 -0
  533. package/src/tools/pin-tool.spec.ts +497 -0
  534. package/src/tools/pin-tool.ts +277 -0
  535. package/src/tools/recall-tool.spec.ts +475 -0
  536. package/src/tools/recall-tool.ts +389 -0
  537. package/src/tools/remember-tool.spec.ts +469 -0
  538. package/src/tools/remember-tool.ts +112 -0
  539. package/src/tools/resolve-error.ts +69 -0
  540. package/src/tools/tool-registry.ts +417 -0
  541. package/src/tools/types.ts +63 -0
  542. package/src/tools/unpin-tool.spec.ts +549 -0
  543. package/src/tools/unpin-tool.ts +306 -0
  544. package/src/types/embedding.types.ts +78 -0
  545. package/src/types/index.spec.ts +420 -0
  546. package/src/types/index.ts +117 -0
  547. package/src/utils/database.spec.ts +77 -0
  548. package/src/utils/database.ts +458 -0
  549. package/src/utils/stopwords.ts +128 -0
  550. package/start-docker-setup.bat +38 -0
  551. package/static/logo.png +0 -0
  552. package/static/memento_logo.svg +2 -0
  553. package/test-docker.js +103 -0
  554. package/tsconfig.json +46 -0
  555. package/vitest.config.ts +23 -0
@@ -0,0 +1,1068 @@
1
+ /**
2
+ * HTTP/WebSocket 기반 MCP μ„œλ²„
3
+ * μ½˜μ†” λ‘œκ·Έμ™€ MCP ν”„λ‘œν† μ½œ 좩돌 문제λ₯Ό ν•΄κ²°
4
+ */
5
+
6
+ import express from 'express';
7
+ import { WebSocketServer } from 'ws';
8
+ import cors from 'cors';
9
+ import { createServer } from 'http';
10
+ import { initializeDatabase, closeDatabase } from '../database/init.js';
11
+ import { mementoConfig, validateConfig } from '../config/index.js';
12
+ import { DatabaseUtils } from '../utils/database.js';
13
+ import { SearchEngine } from '../algorithms/search-engine.js';
14
+ import { HybridSearchEngine } from '../algorithms/hybrid-search-engine.js';
15
+ import { MemoryEmbeddingService } from '../services/memory-embedding-service.js';
16
+ import type { MemoryType, PrivacyScope } from '../types/index.js';
17
+ import Database from 'better-sqlite3';
18
+ import { z } from 'zod';
19
+
20
+ // MCP Tools μŠ€ν‚€λ§ˆ μ •μ˜
21
+ const RememberSchema = z.object({
22
+ content: z.string().min(1, 'Content cannot be empty'),
23
+ type: z.enum(['working', 'episodic', 'semantic', 'procedural']).default('episodic'),
24
+ tags: z.array(z.string()).optional(),
25
+ importance: z.number().min(0).max(1).default(0.5),
26
+ source: z.string().optional(),
27
+ privacy_scope: z.enum(['private', 'team', 'public']).default('private')
28
+ });
29
+
30
+ const RecallSchema = z.object({
31
+ query: z.string().min(1, 'Query cannot be empty'),
32
+ filters: z.object({
33
+ type: z.array(z.enum(['working', 'episodic', 'semantic', 'procedural'])).optional(),
34
+ tags: z.array(z.string()).optional(),
35
+ privacy_scope: z.array(z.enum(['private', 'team', 'public'])).optional(),
36
+ time_from: z.string().optional(),
37
+ time_to: z.string().optional(),
38
+ pinned: z.boolean().optional()
39
+ }).optional(),
40
+ limit: z.number().min(1).max(50).default(10)
41
+ });
42
+
43
+ const ForgetSchema = z.object({
44
+ id: z.string().min(1, 'Memory ID cannot be empty'),
45
+ hard: z.boolean().default(false)
46
+ });
47
+
48
+ const PinSchema = z.object({
49
+ id: z.string().min(1, 'Memory ID cannot be empty')
50
+ });
51
+
52
+ const UnpinSchema = z.object({
53
+ id: z.string().min(1, 'Memory ID cannot be empty')
54
+ });
55
+
56
+ const ContextInjectionSchema = z.object({
57
+ query: z.string().min(1, 'Query cannot be empty'),
58
+ context: z.string().min(1, 'Context cannot be empty'),
59
+ limit: z.number().min(1).max(50).default(5)
60
+ });
61
+
62
+ // μ „μ—­ λ³€μˆ˜
63
+ let db: Database.Database | null = null;
64
+ let searchEngine: SearchEngine;
65
+ let hybridSearchEngine: HybridSearchEngine;
66
+ let embeddingService: MemoryEmbeddingService;
67
+
68
+ type TestDependencies = {
69
+ database: Database.Database;
70
+ searchEngine?: SearchEngine;
71
+ hybridSearchEngine?: HybridSearchEngine;
72
+ embeddingService?: MemoryEmbeddingService;
73
+ };
74
+
75
+ function setTestDependencies({
76
+ database,
77
+ searchEngine: search,
78
+ hybridSearchEngine: hybrid,
79
+ embeddingService: embedding
80
+ }: TestDependencies): void {
81
+ db = database;
82
+ searchEngine = search ?? new SearchEngine();
83
+ hybridSearchEngine = hybrid ?? new HybridSearchEngine();
84
+ embeddingService = embedding ?? new MemoryEmbeddingService();
85
+ }
86
+
87
+ // Tool ν•Έλ“€λŸ¬λ“€ (κΈ°μ‘΄κ³Ό 동일)
88
+ async function handleRemember(params: z.infer<typeof RememberSchema>) {
89
+ const { content, type, tags, importance, source, privacy_scope } = params;
90
+
91
+ const id = `mem_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
92
+
93
+ if (!db) {
94
+ throw new Error('λ°μ΄ν„°λ² μ΄μŠ€κ°€ μ΄ˆκΈ°ν™”λ˜μ§€ μ•Šμ•˜μŠ΅λ‹ˆλ‹€');
95
+ }
96
+
97
+ try {
98
+ const result = await DatabaseUtils.runTransaction(db!, async () => {
99
+ await DatabaseUtils.run(db!, `
100
+ INSERT INTO memory_item (id, type, content, importance, privacy_scope, tags, source, created_at)
101
+ VALUES (?, ?, ?, ?, ?, ?, ?, CURRENT_TIMESTAMP)
102
+ `, [id, type, content, importance, privacy_scope,
103
+ tags ? JSON.stringify(tags) : null, source]);
104
+
105
+ return { id, type, content, importance, privacy_scope, tags, source };
106
+ });
107
+
108
+ // μž„λ² λ”© 생성 (비동기)
109
+ if (embeddingService.isAvailable()) {
110
+ embeddingService.createAndStoreEmbedding(db, id, content, type)
111
+ .then(result => {
112
+ if (result) {
113
+ console.log(`βœ… μž„λ² λ”© 생성 μ™„λ£Œ: ${id} (${result.embedding.length}차원)`);
114
+ }
115
+ })
116
+ .catch(error => {
117
+ console.warn(`⚠️ μž„λ² λ”© 생성 μ‹€νŒ¨ (${id}):`, error.message);
118
+ });
119
+ }
120
+
121
+ return {
122
+ memory_id: id,
123
+ message: `기얡이 μ €μž₯λ˜μ—ˆμŠ΅λ‹ˆλ‹€: ${id}`,
124
+ embedding_created: embeddingService.isAvailable()
125
+ };
126
+ } catch (error) {
127
+ if ((error as any).code === 'SQLITE_BUSY') {
128
+ console.log('πŸ”§ λ°μ΄ν„°λ² μ΄μŠ€ 락 감지, WAL 체크포인트 μ‹œλ„...');
129
+ try {
130
+ await DatabaseUtils.checkpointWAL(db);
131
+ console.log('βœ… WAL 체크포인트 μ™„λ£Œ, μž¬μ‹œλ„ κ°€λŠ₯');
132
+ } catch (checkpointError) {
133
+ console.error('❌ WAL 체크포인트 μ‹€νŒ¨:', checkpointError);
134
+ }
135
+ }
136
+ throw error;
137
+ }
138
+ }
139
+
140
+ async function handleRecall(params: z.infer<typeof RecallSchema>) {
141
+ const { query, filters, limit } = params;
142
+
143
+ if (!db) {
144
+ throw new Error('λ°μ΄ν„°λ² μ΄μŠ€κ°€ μ΄ˆκΈ°ν™”λ˜μ§€ μ•Šμ•˜μŠ΅λ‹ˆλ‹€');
145
+ }
146
+
147
+ try {
148
+ console.log('πŸ” HTTP μ„œλ²„μ—μ„œ ν•˜μ΄λΈŒλ¦¬λ“œ 검색 μ—”μ§„ 호좜 μ‹œμž‘');
149
+ const results = await hybridSearchEngine.search(db, {
150
+ query,
151
+ filters: filters || {},
152
+ limit
153
+ });
154
+ console.log('πŸ” HTTP μ„œλ²„μ—μ„œ ν•˜μ΄λΈŒλ¦¬λ“œ 검색 μ—”μ§„ 호좜 μ™„λ£Œ, κ²°κ³Ό 개수:', results.items.length);
155
+
156
+ return {
157
+ items: results,
158
+ search_type: 'hybrid',
159
+ vector_search_available: true
160
+ };
161
+ } catch (error) {
162
+ console.error('❌ ν•˜μ΄λΈŒλ¦¬λ“œ 검색 μ—”μ§„ 호좜 μ‹€νŒ¨:', error);
163
+ throw error;
164
+ }
165
+ }
166
+
167
+ async function handleForget(params: z.infer<typeof ForgetSchema>) {
168
+ const { id, hard } = params;
169
+
170
+ if (!db) {
171
+ throw new Error('λ°μ΄ν„°λ² μ΄μŠ€κ°€ μ΄ˆκΈ°ν™”λ˜μ§€ μ•Šμ•˜μŠ΅λ‹ˆλ‹€');
172
+ }
173
+
174
+ try {
175
+ const result = await DatabaseUtils.runTransaction(db!, async () => {
176
+ if (hard) {
177
+ const deleteResult = await DatabaseUtils.run(db!, 'DELETE FROM memory_item WHERE id = ?', [id]);
178
+
179
+ if (deleteResult.changes === 0) {
180
+ throw new Error(`Memory with ID ${id} not found`);
181
+ }
182
+
183
+ return { type: 'hard', changes: deleteResult.changes };
184
+ } else {
185
+ const updateResult = await DatabaseUtils.run(db!, 'UPDATE memory_item SET pinned = FALSE WHERE id = ?', [id]);
186
+
187
+ if (updateResult.changes === 0) {
188
+ throw new Error(`Memory with ID ${id} not found`);
189
+ }
190
+
191
+ return { type: 'soft', changes: updateResult.changes };
192
+ }
193
+ });
194
+
195
+ if (hard && embeddingService.isAvailable()) {
196
+ try {
197
+ await embeddingService.deleteEmbedding(db, id);
198
+ } catch (embeddingError) {
199
+ console.warn(`⚠️ μž„λ² λ”© μ‚­μ œ μ‹€νŒ¨ (${id}):`, embeddingError);
200
+ }
201
+ }
202
+
203
+ return {
204
+ memory_id: id,
205
+ message: hard ? `기얡이 μ™„μ „νžˆ μ‚­μ œλ˜μ—ˆμŠ΅λ‹ˆλ‹€: ${id}` : `기얡이 μ‚­μ œ λŒ€μƒμœΌλ‘œ ν‘œμ‹œλ˜μ—ˆμŠ΅λ‹ˆλ‹€: ${id}`
206
+ };
207
+ } catch (error) {
208
+ if ((error as any).code === 'SQLITE_BUSY') {
209
+ console.log('πŸ”§ λ°μ΄ν„°λ² μ΄μŠ€ 락 감지, WAL 체크포인트 μ‹œλ„...');
210
+ try {
211
+ await DatabaseUtils.checkpointWAL(db);
212
+ console.log('βœ… WAL 체크포인트 μ™„λ£Œ, μž¬μ‹œλ„ κ°€λŠ₯');
213
+ } catch (checkpointError) {
214
+ console.error('❌ WAL 체크포인트 μ‹€νŒ¨:', checkpointError);
215
+ }
216
+ }
217
+ throw error;
218
+ }
219
+ }
220
+
221
+ async function handlePin(params: z.infer<typeof PinSchema>) {
222
+ const { id } = params;
223
+
224
+ if (!db) {
225
+ throw new Error('λ°μ΄ν„°λ² μ΄μŠ€κ°€ μ΄ˆκΈ°ν™”λ˜μ§€ μ•Šμ•˜μŠ΅λ‹ˆλ‹€');
226
+ }
227
+
228
+ try {
229
+ await DatabaseUtils.runTransaction(db!, async () => {
230
+ const result = await DatabaseUtils.run(db!, 'UPDATE memory_item SET pinned = TRUE WHERE id = ?', [id]);
231
+
232
+ if (result.changes === 0) {
233
+ throw new Error(`Memory with ID ${id} not found`);
234
+ }
235
+
236
+ return result;
237
+ });
238
+
239
+ return {
240
+ memory_id: id,
241
+ message: `기얡이 κ³ μ •λ˜μ—ˆμŠ΅λ‹ˆλ‹€: ${id}`
242
+ };
243
+ } catch (error) {
244
+ if ((error as any).code === 'SQLITE_BUSY') {
245
+ console.log('πŸ”§ λ°μ΄ν„°λ² μ΄μŠ€ 락 감지, WAL 체크포인트 μ‹œλ„...');
246
+ try {
247
+ await DatabaseUtils.checkpointWAL(db);
248
+ console.log('βœ… WAL 체크포인트 μ™„λ£Œ, μž¬μ‹œλ„ κ°€λŠ₯');
249
+ } catch (checkpointError) {
250
+ console.error('❌ WAL 체크포인트 μ‹€νŒ¨:', checkpointError);
251
+ }
252
+ }
253
+ throw error;
254
+ }
255
+ }
256
+
257
+ async function handleUnpin(params: z.infer<typeof UnpinSchema>) {
258
+ const { id } = params;
259
+
260
+ if (!db) {
261
+ throw new Error('λ°μ΄ν„°λ² μ΄μŠ€κ°€ μ΄ˆκΈ°ν™”λ˜μ§€ μ•Šμ•˜μŠ΅λ‹ˆλ‹€');
262
+ }
263
+
264
+ try {
265
+ await DatabaseUtils.runTransaction(db!, async () => {
266
+ const result = await DatabaseUtils.run(db!, 'UPDATE memory_item SET pinned = FALSE WHERE id = ?', [id]);
267
+
268
+ if (result.changes === 0) {
269
+ throw new Error(`Memory with ID ${id} not found`);
270
+ }
271
+
272
+ return result;
273
+ });
274
+
275
+ return {
276
+ memory_id: id,
277
+ message: `κΈ°μ–΅ 고정이 ν•΄μ œλ˜μ—ˆμŠ΅λ‹ˆλ‹€: ${id}`
278
+ };
279
+ } catch (error) {
280
+ if ((error as any).code === 'SQLITE_BUSY') {
281
+ console.log('πŸ”§ λ°μ΄ν„°λ² μ΄μŠ€ 락 감지, WAL 체크포인트 μ‹œλ„...');
282
+ try {
283
+ await DatabaseUtils.checkpointWAL(db);
284
+ console.log('βœ… WAL 체크포인트 μ™„λ£Œ, μž¬μ‹œλ„ κ°€λŠ₯');
285
+ } catch (checkpointError) {
286
+ console.error('❌ WAL 체크포인트 μ‹€νŒ¨:', checkpointError);
287
+ }
288
+ }
289
+ throw error;
290
+ }
291
+ }
292
+
293
+ async function handleContextInjection(params: z.infer<typeof ContextInjectionSchema>) {
294
+ const { query, context, limit } = params;
295
+
296
+ if (!db) {
297
+ throw new Error('λ°μ΄ν„°λ² μ΄μŠ€κ°€ μ΄ˆκΈ°ν™”λ˜μ§€ μ•Šμ•˜μŠ΅λ‹ˆλ‹€');
298
+ }
299
+
300
+ try {
301
+ // 쿼리둜 κ΄€λ ¨ κΈ°μ–΅ 검색
302
+ const searchResult = await hybridSearchEngine.search(db, {
303
+ query,
304
+ limit: limit * 2, // 더 λ§Žμ€ 후보λ₯Ό κ°€μ Έμ™€μ„œ μ»¨ν…μŠ€νŠΈμ™€ λ§€μΉ­
305
+ vectorWeight: 0.6,
306
+ textWeight: 0.4
307
+ });
308
+
309
+ // μ»¨ν…μŠ€νŠΈμ™€ κ΄€λ ¨μ„± 높은 κΈ°μ–΅λ“€ 필터링
310
+ const relevantMemories = searchResult.items
311
+ .filter(memory => {
312
+ // κ°„λ‹¨ν•œ ν‚€μ›Œλ“œ 맀칭으둜 κ΄€λ ¨μ„± νŒλ‹¨
313
+ const contextKeywords = context.toLowerCase().split(/\s+/);
314
+ const memoryContent = memory.content.toLowerCase();
315
+ return contextKeywords.some(keyword => memoryContent.includes(keyword));
316
+ })
317
+ .slice(0, limit);
318
+
319
+ return {
320
+ memories: relevantMemories,
321
+ total_count: relevantMemories.length,
322
+ context: context,
323
+ query: query
324
+ };
325
+ } catch (error) {
326
+ if ((error as any).code === 'SQLITE_BUSY') {
327
+ console.log('πŸ”§ λ°μ΄ν„°λ² μ΄μŠ€ 락 감지, WAL 체크포인트 μ‹œλ„...');
328
+ try {
329
+ await DatabaseUtils.checkpointWAL(db);
330
+ console.log('βœ… WAL 체크포인트 μ™„λ£Œ, μž¬μ‹œλ„ κ°€λŠ₯');
331
+ } catch (checkpointError) {
332
+ console.error('❌ WAL 체크포인트 μ‹€νŒ¨:', checkpointError);
333
+ }
334
+ }
335
+ throw error;
336
+ }
337
+ }
338
+
339
+ // Express μ•± 생성
340
+ const app = express();
341
+ const server = createServer(app);
342
+
343
+ // 미듀웨어 μ„€μ •
344
+ app.use(cors());
345
+ app.use(express.json());
346
+
347
+ // MCP Tools λͺ©λ‘
348
+ const tools = [
349
+ {
350
+ name: 'remember',
351
+ description: '기얡을 μ €μž₯ν•©λ‹ˆλ‹€',
352
+ inputSchema: {
353
+ type: 'object',
354
+ properties: {
355
+ content: { type: 'string', description: 'μ €μž₯ν•  κΈ°μ–΅ λ‚΄μš©' },
356
+ type: {
357
+ type: 'string',
358
+ enum: ['working', 'episodic', 'semantic', 'procedural'],
359
+ default: 'episodic',
360
+ description: 'κΈ°μ–΅ νƒ€μž…'
361
+ },
362
+ tags: { type: 'array', items: { type: 'string' }, description: 'νƒœκ·Έ λͺ©λ‘' },
363
+ importance: { type: 'number', minimum: 0, maximum: 1, default: 0.5, description: 'μ€‘μš”λ„ (0-1)' },
364
+ source: { type: 'string', description: '좜처' },
365
+ privacy_scope: {
366
+ type: 'string',
367
+ enum: ['private', 'team', 'public'],
368
+ default: 'private',
369
+ description: 'ν”„λΌμ΄λ²„μ‹œ λ²”μœ„'
370
+ }
371
+ },
372
+ required: ['content']
373
+ }
374
+ },
375
+ {
376
+ name: 'recall',
377
+ description: '기얡을 κ²€μƒ‰ν•©λ‹ˆλ‹€',
378
+ inputSchema: {
379
+ type: 'object',
380
+ properties: {
381
+ query: { type: 'string', description: '검색 쿼리' },
382
+ filters: {
383
+ type: 'object',
384
+ properties: {
385
+ type: { type: 'array', items: { type: 'string' }, description: 'κΈ°μ–΅ νƒ€μž… ν•„ν„°' },
386
+ tags: { type: 'array', items: { type: 'string' }, description: 'νƒœκ·Έ ν•„ν„°' },
387
+ privacy_scope: { type: 'array', items: { type: 'string' }, description: 'ν”„λΌμ΄λ²„μ‹œ λ²”μœ„ ν•„ν„°' },
388
+ time_from: { type: 'string', description: 'μ‹œμž‘ μ‹œκ°„' },
389
+ time_to: { type: 'string', description: 'μ’…λ£Œ μ‹œκ°„' },
390
+ pinned: { type: 'boolean', description: 'κ³ μ •λœ κΈ°μ–΅λ§Œ' }
391
+ }
392
+ },
393
+ limit: { type: 'number', minimum: 1, maximum: 50, default: 10, description: 'κ²°κ³Ό 개수 μ œν•œ' }
394
+ },
395
+ required: ['query']
396
+ }
397
+ },
398
+ {
399
+ name: 'forget',
400
+ description: '기얡을 μ‚­μ œν•©λ‹ˆλ‹€',
401
+ inputSchema: {
402
+ type: 'object',
403
+ properties: {
404
+ id: { type: 'string', description: 'μ‚­μ œν•  κΈ°μ–΅ ID' },
405
+ hard: { type: 'boolean', default: false, description: 'μ™„μ „ μ‚­μ œ μ—¬λΆ€' }
406
+ },
407
+ required: ['id']
408
+ }
409
+ },
410
+ {
411
+ name: 'pin',
412
+ description: '기얡을 κ³ μ •ν•©λ‹ˆλ‹€',
413
+ inputSchema: {
414
+ type: 'object',
415
+ properties: {
416
+ id: { type: 'string', description: 'κ³ μ •ν•  κΈ°μ–΅ ID' }
417
+ },
418
+ required: ['id']
419
+ }
420
+ },
421
+ {
422
+ name: 'unpin',
423
+ description: 'κΈ°μ–΅ 고정을 ν•΄μ œν•©λ‹ˆλ‹€',
424
+ inputSchema: {
425
+ type: 'object',
426
+ properties: {
427
+ id: { type: 'string', description: 'κ³ μ • ν•΄μ œν•  κΈ°μ–΅ ID' }
428
+ },
429
+ required: ['id']
430
+ }
431
+ }
432
+ ];
433
+
434
+ // API μ—”λ“œν¬μΈνŠΈ
435
+ app.get('/health', (req, res) => {
436
+ res.json({
437
+ status: 'healthy',
438
+ server: mementoConfig.serverName,
439
+ version: mementoConfig.serverVersion,
440
+ database: db ? 'connected' : 'disconnected'
441
+ });
442
+ });
443
+
444
+ app.get('/tools', (req, res) => {
445
+ res.json({ tools });
446
+ });
447
+
448
+ // κ΄€λ¦¬μž API μ—”λ“œν¬μΈνŠΈλ“€
449
+ app.post('/admin/memory/cleanup', async (req, res) => {
450
+ try {
451
+ // λ©”λͺ¨λ¦¬ 정리 둜직 (κΈ°μ‘΄ CleanupMemoryTool 둜직)
452
+ res.json({ message: 'λ©”λͺ¨λ¦¬ 정리 μ™„λ£Œ' });
453
+ } catch (error) {
454
+ res.status(500).json({ error: 'λ©”λͺ¨λ¦¬ 정리 μ‹€νŒ¨' });
455
+ }
456
+ });
457
+
458
+ app.get('/admin/stats/forgetting', async (req, res) => {
459
+ try {
460
+ // 망각 톡계 둜직 (κΈ°μ‘΄ ForgettingStatsTool 둜직)
461
+ res.json({ message: '망각 톡계 쑰회 μ™„λ£Œ' });
462
+ } catch (error) {
463
+ res.status(500).json({ error: '망각 톡계 쑰회 μ‹€νŒ¨' });
464
+ }
465
+ });
466
+
467
+ app.get('/admin/stats/performance', async (req, res) => {
468
+ try {
469
+ // μ„±λŠ₯ 톡계 둜직 (κΈ°μ‘΄ PerformanceStatsTool 둜직)
470
+ res.json({ message: 'μ„±λŠ₯ 톡계 쑰회 μ™„λ£Œ' });
471
+ } catch (error) {
472
+ res.status(500).json({ error: 'μ„±λŠ₯ 톡계 쑰회 μ‹€νŒ¨' });
473
+ }
474
+ });
475
+
476
+ app.post('/admin/database/optimize', async (req, res) => {
477
+ try {
478
+ // λ°μ΄ν„°λ² μ΄μŠ€ μ΅œμ ν™” 둜직 (κΈ°μ‘΄ DatabaseOptimizeTool 둜직)
479
+ res.json({ message: 'λ°μ΄ν„°λ² μ΄μŠ€ μ΅œμ ν™” μ™„λ£Œ' });
480
+ } catch (error) {
481
+ res.status(500).json({ error: 'λ°μ΄ν„°λ² μ΄μŠ€ μ΅œμ ν™” μ‹€νŒ¨' });
482
+ }
483
+ });
484
+
485
+ app.get('/admin/stats/errors', async (req, res) => {
486
+ try {
487
+ // μ—λŸ¬ 톡계 둜직 (κΈ°μ‘΄ errorStatsTool 둜직)
488
+ res.json({ message: 'μ—λŸ¬ 톡계 쑰회 μ™„λ£Œ' });
489
+ } catch (error) {
490
+ res.status(500).json({ error: 'μ—λŸ¬ 톡계 쑰회 μ‹€νŒ¨' });
491
+ }
492
+ });
493
+
494
+ app.post('/admin/errors/resolve', async (req, res) => {
495
+ try {
496
+ const { errorId, resolvedBy, reason } = req.body;
497
+ // μ—λŸ¬ ν•΄κ²° 둜직 (κΈ°μ‘΄ resolveErrorTool 둜직)
498
+ res.json({ message: 'μ—λŸ¬ ν•΄κ²° μ™„λ£Œ' });
499
+ } catch (error) {
500
+ res.status(500).json({ error: 'μ—λŸ¬ ν•΄κ²° μ‹€νŒ¨' });
501
+ }
502
+ });
503
+
504
+ app.get('/admin/alerts/performance', async (req, res) => {
505
+ try {
506
+ // μ„±λŠ₯ μ•Œλ¦Ό 둜직 (κΈ°μ‘΄ performanceAlertsTool 둜직)
507
+ res.json({ message: 'μ„±λŠ₯ μ•Œλ¦Ό 쑰회 μ™„λ£Œ' });
508
+ } catch (error) {
509
+ res.status(500).json({ error: 'μ„±λŠ₯ μ•Œλ¦Ό 쑰회 μ‹€νŒ¨' });
510
+ }
511
+ });
512
+
513
+ app.post('/tools/:name', async (req, res) => {
514
+ const { name } = req.params;
515
+ const params = req.body;
516
+
517
+ try {
518
+ let result;
519
+
520
+ switch (name) {
521
+ case 'remember':
522
+ result = await handleRemember(RememberSchema.parse(params));
523
+ break;
524
+ case 'recall':
525
+ result = await handleRecall(RecallSchema.parse(params));
526
+ break;
527
+ case 'forget':
528
+ result = await handleForget(ForgetSchema.parse(params));
529
+ break;
530
+ case 'pin':
531
+ result = await handlePin(PinSchema.parse(params));
532
+ break;
533
+ case 'unpin':
534
+ result = await handleUnpin(UnpinSchema.parse(params));
535
+ break;
536
+ case 'context-injection':
537
+ result = await handleContextInjection(ContextInjectionSchema.parse(params));
538
+ break;
539
+ default:
540
+ return res.status(404).json({ error: `Unknown tool: ${name}` });
541
+ }
542
+
543
+ return res.json({ result });
544
+ } catch (error) {
545
+ if (error instanceof z.ZodError) {
546
+ return res.status(400).json({
547
+ error: 'Invalid parameters',
548
+ details: error.errors.map(e => e.message).join(', ')
549
+ });
550
+ }
551
+
552
+ console.error(`❌ Tool ${name} μ‹€ν–‰ μ‹€νŒ¨:`, error);
553
+ return res.status(500).json({
554
+ error: 'Internal server error',
555
+ message: error instanceof Error ? error.message : 'Unknown error'
556
+ });
557
+ }
558
+ });
559
+
560
+ // MCP SSE μ—”λ“œν¬μΈνŠΈ - MCP SDK ν˜Έν™˜ κ΅¬ν˜„
561
+ // Store transports by session ID
562
+ const transports: { [sessionId: string]: any } = {};
563
+
564
+ // SSE endpoint for establishing the stream
565
+ app.get('/mcp', async (req, res) => {
566
+ console.log('πŸ”— MCP SSE ν΄λΌμ΄μ–ΈνŠΈ μ—°κ²° μš”μ²­');
567
+
568
+ try {
569
+ // SSE 헀더 μ„€μ •
570
+ res.writeHead(200, {
571
+ 'Content-Type': 'text/event-stream',
572
+ 'Cache-Control': 'no-cache, no-transform',
573
+ 'Connection': 'keep-alive',
574
+ 'Access-Control-Allow-Origin': '*',
575
+ 'Access-Control-Allow-Headers': 'Cache-Control, Content-Type, Authorization',
576
+ 'Access-Control-Allow-Methods': 'GET, POST, OPTIONS',
577
+ 'X-Accel-Buffering': 'no' // nginx 버퍼링 λΉ„ν™œμ„±ν™”
578
+ });
579
+
580
+ // Generate session ID
581
+ const sessionId = `session_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
582
+
583
+ // Send the endpoint event with session ID
584
+ const endpointUrl = `/messages?sessionId=${sessionId}`;
585
+ res.write(`event: endpoint\ndata: ${endpointUrl}\n\n`);
586
+
587
+ // MCP μ„œλ²„ μ€€λΉ„ μ™„λ£Œ μ•Œλ¦Ό (ν΄λΌμ΄μ–ΈνŠΈκ°€ initializeλ₯Ό 보내야 함)
588
+ res.write(`data: {"type": "ready"}\n\n`);
589
+
590
+ // μ¦‰μ‹œ 응닡 ν”ŒλŸ¬μ‹œ (Expressμ—μ„œλŠ” μžλ™μœΌλ‘œ 처리됨)
591
+
592
+ // Keep-alive ping 전솑
593
+ const keepAliveInterval = setInterval(() => {
594
+ if (res.writableEnded) {
595
+ clearInterval(keepAliveInterval);
596
+ return;
597
+ }
598
+ try {
599
+ res.write(`data: {"type": "ping"}\n\n`);
600
+ } catch (error) {
601
+ clearInterval(keepAliveInterval);
602
+ }
603
+ }, 30000); // 30μ΄ˆλ§ˆλ‹€ ping
604
+
605
+ // Store the transport info
606
+ transports[sessionId] = {
607
+ res: res,
608
+ sessionId: sessionId,
609
+ keepAliveInterval: keepAliveInterval
610
+ };
611
+
612
+ // μ—°κ²° μ’…λ£Œ 처리
613
+ req.on('close', () => {
614
+ console.log(`πŸ”Œ MCP SSE ν΄λΌμ΄μ–ΈνŠΈ μ—°κ²° 정상 μ’…λ£Œλ¨ (session: ${sessionId})`);
615
+ clearInterval(keepAliveInterval);
616
+ delete transports[sessionId];
617
+ });
618
+
619
+ req.on('error', (error) => {
620
+ // ECONNRESET은 정상적인 μ—°κ²° μ’…λ£Œμ΄λ―€λ‘œ μ—λŸ¬λ‘œ μ²˜λ¦¬ν•˜μ§€ μ•ŠμŒ
621
+ if ((error as any).code === 'ECONNRESET') {
622
+ console.log(`πŸ”Œ MCP SSE ν΄λΌμ΄μ–ΈνŠΈ μ—°κ²° 정상 μ’…λ£Œλ¨ (session: ${sessionId})`);
623
+ } else {
624
+ console.error(`❌ MCP SSE μ—°κ²° μ—λŸ¬ (session: ${sessionId}):`, error);
625
+ }
626
+ clearInterval(keepAliveInterval);
627
+ delete transports[sessionId];
628
+ });
629
+
630
+ console.log(`βœ… MCP SSE 슀트림 μ„€μ • μ™„λ£Œ (session: ${sessionId})`);
631
+
632
+ } catch (error) {
633
+ console.error('❌ SSE 슀트림 μ„€μ • μ‹€νŒ¨:', error);
634
+ if (!res.headersSent) {
635
+ res.status(500).send('Error establishing SSE stream');
636
+ }
637
+ }
638
+ });
639
+
640
+ // Messages endpoint for receiving client JSON-RPC requests
641
+ app.post('/messages', express.json(), async (req, res) => {
642
+ console.log('πŸ“¨ MCP λ©”μ‹œμ§€ μˆ˜μ‹ :', req.body.method);
643
+
644
+ // Extract session ID from URL query parameter
645
+ const sessionId = req.query.sessionId as string;
646
+ if (!sessionId) {
647
+ console.error('❌ No session ID provided in request URL');
648
+ res.status(400).send('Missing sessionId parameter');
649
+ return;
650
+ }
651
+
652
+ const transport = transports[sessionId];
653
+ if (!transport) {
654
+ console.error(`❌ No active transport found for session ID: ${sessionId}`);
655
+ res.status(404).send('Session not found');
656
+ return;
657
+ }
658
+
659
+ let message = req.body;
660
+ let result;
661
+
662
+ console.log(`πŸ” MCP λ©”μ‹œμ§€ 처리 쀑: ${message.method}`, JSON.stringify(message, null, 2));
663
+
664
+ try {
665
+
666
+ if (message.method === 'initialize') {
667
+ console.log('πŸš€ MCP initialize μš”μ²­ 처리 쀑...');
668
+ result = {
669
+ jsonrpc: '2.0',
670
+ id: message.id,
671
+ result: {
672
+ protocolVersion: '2024-11-05',
673
+ capabilities: {
674
+ tools: {}
675
+ },
676
+ serverInfo: {
677
+ name: 'memento-memory',
678
+ version: '0.1.0'
679
+ }
680
+ }
681
+ };
682
+ console.log('βœ… MCP initialize 응닡 생성 μ™„λ£Œ:', JSON.stringify(result, null, 2));
683
+ } else if (message.method === 'notifications/initialized') {
684
+ console.log('πŸ”” MCP initialized μ•Œλ¦Ό μˆ˜μ‹ ');
685
+ result = {
686
+ jsonrpc: '2.0',
687
+ id: message.id,
688
+ result: {}
689
+ };
690
+ } else if (message.method === 'tools/list') {
691
+ console.log('πŸ“‹ MCP tools/list μš”μ²­ 처리 쀑...');
692
+
693
+ try {
694
+ // κ°„λ‹¨ν•œ 도ꡬ λͺ©λ‘μœΌλ‘œ ν…ŒμŠ€νŠΈ
695
+ const simpleTools = [
696
+ {
697
+ name: 'remember',
698
+ description: '기얡을 μ €μž₯ν•©λ‹ˆλ‹€',
699
+ inputSchema: {
700
+ type: 'object',
701
+ properties: {
702
+ content: { type: 'string', description: 'μ €μž₯ν•  κΈ°μ–΅ λ‚΄μš©' }
703
+ },
704
+ required: ['content']
705
+ }
706
+ },
707
+ {
708
+ name: 'recall',
709
+ description: '기얡을 κ²€μƒ‰ν•©λ‹ˆλ‹€',
710
+ inputSchema: {
711
+ type: 'object',
712
+ properties: {
713
+ query: { type: 'string', description: '검색 쿼리' }
714
+ },
715
+ required: ['query']
716
+ }
717
+ }
718
+ ];
719
+
720
+ console.log('πŸ” κ°„λ‹¨ν•œ 도ꡬ λͺ©λ‘ μ‚¬μš©, 길이:', simpleTools.length);
721
+
722
+ result = {
723
+ jsonrpc: '2.0',
724
+ id: message.id,
725
+ result: { tools: simpleTools }
726
+ };
727
+
728
+ console.log('βœ… MCP tools/list 응닡 생성 μ™„λ£Œ, tools 개수:', simpleTools.length);
729
+ console.log('πŸ” 응닡 크기:', JSON.stringify(result).length, 'bytes');
730
+
731
+ // SSE 응닡 μ¦‰μ‹œ 전솑
732
+ console.log('πŸ“€ SSE 응닡 μ¦‰μ‹œ 전솑 쀑...');
733
+ if (transport && transport.res && !transport.res.writableEnded) {
734
+ const sseData = `data: ${JSON.stringify(result)}\n\n`;
735
+ transport.res.write(sseData);
736
+ console.log('βœ… SSE 응닡 μ¦‰μ‹œ 전솑 μ™„λ£Œ, 크기:', sseData.length, 'bytes');
737
+ } else {
738
+ console.error('❌ SSE transportκ°€ μœ νš¨ν•˜μ§€ μ•ŠμŒ');
739
+ }
740
+
741
+ // HTTP 응닡 전솑
742
+ res.json({ status: 'ok' });
743
+ return;
744
+
745
+ } catch (toolsError) {
746
+ console.error('❌ tools/list 처리 쀑 였λ₯˜:', toolsError);
747
+ const errorResult = {
748
+ jsonrpc: '2.0',
749
+ id: message.id,
750
+ error: {
751
+ code: -32603,
752
+ message: 'Internal error',
753
+ data: toolsError instanceof Error ? toolsError.message : String(toolsError)
754
+ }
755
+ };
756
+
757
+ if (transport && transport.res && !transport.res.writableEnded) {
758
+ transport.res.write(`data: ${JSON.stringify(errorResult)}\n\n`);
759
+ }
760
+ res.json({ status: 'error' });
761
+ return;
762
+ }
763
+ } else if (message.method === 'tools/call') {
764
+ const { name, arguments: args } = message.params;
765
+
766
+ switch (name) {
767
+ case 'remember':
768
+ result = {
769
+ jsonrpc: '2.0',
770
+ id: message.id,
771
+ result: { content: [{ type: 'text', text: JSON.stringify(await handleRemember(RememberSchema.parse(args))) }] }
772
+ };
773
+ break;
774
+ case 'recall':
775
+ result = {
776
+ jsonrpc: '2.0',
777
+ id: message.id,
778
+ result: { content: [{ type: 'text', text: JSON.stringify(await handleRecall(RecallSchema.parse(args))) }] }
779
+ };
780
+ break;
781
+ case 'forget':
782
+ result = {
783
+ jsonrpc: '2.0',
784
+ id: message.id,
785
+ result: { content: [{ type: 'text', text: JSON.stringify(await handleForget(ForgetSchema.parse(args))) }] }
786
+ };
787
+ break;
788
+ case 'pin':
789
+ result = {
790
+ jsonrpc: '2.0',
791
+ id: message.id,
792
+ result: { content: [{ type: 'text', text: JSON.stringify(await handlePin(PinSchema.parse(args))) }] }
793
+ };
794
+ break;
795
+ case 'unpin':
796
+ result = {
797
+ jsonrpc: '2.0',
798
+ id: message.id,
799
+ result: { content: [{ type: 'text', text: JSON.stringify(await handleUnpin(UnpinSchema.parse(args))) }] }
800
+ };
801
+ break;
802
+ default:
803
+ throw new Error(`Unknown tool: ${name}`);
804
+ }
805
+ } else {
806
+ result = {
807
+ jsonrpc: '2.0',
808
+ id: message.id,
809
+ error: {
810
+ code: -32601,
811
+ message: 'Method not found'
812
+ }
813
+ };
814
+ }
815
+
816
+ // Send response via SSE
817
+ console.log('πŸ“€ SSE 응닡 전솑 쀑:', JSON.stringify(result).substring(0, 200) + '...');
818
+ try {
819
+ // transport 객체 μœ νš¨μ„± 확인
820
+ if (!transport || !transport.res || transport.res.writableEnded) {
821
+ console.error('❌ SSE transportκ°€ μœ νš¨ν•˜μ§€ μ•ŠμŒ');
822
+ res.status(500).json({ error: 'SSE transport invalid' });
823
+ return;
824
+ }
825
+
826
+ // SSE 응닡 전솑
827
+ const sseData = `data: ${JSON.stringify(result)}\n\n`;
828
+ transport.res.write(sseData);
829
+ console.log('βœ… SSE 응닡 전솑 μ™„λ£Œ, 크기:', sseData.length, 'bytes');
830
+ } catch (sseError) {
831
+ console.error('❌ SSE 응닡 전솑 μ‹€νŒ¨:', sseError);
832
+ // SSE 전솑 μ‹€νŒ¨ μ‹œμ—λ„ HTTP 응닡은 정상 처리
833
+ }
834
+
835
+ // Send HTTP response
836
+ res.json({ status: 'ok' });
837
+
838
+ } catch (error) {
839
+ console.error('❌ MCP λ©”μ‹œμ§€ 처리 μ‹€νŒ¨:', error);
840
+ const errorResponse = {
841
+ jsonrpc: '2.0',
842
+ id: message?.id || null,
843
+ error: {
844
+ code: -32603,
845
+ message: 'Internal error',
846
+ data: error instanceof Error ? error.message : 'Unknown error'
847
+ }
848
+ };
849
+
850
+ // Send error via SSE
851
+ try {
852
+ if (transport && transport.res && !transport.res.writableEnded) {
853
+ const errorSseData = `data: ${JSON.stringify(errorResponse)}\n\n`;
854
+ transport.res.write(errorSseData);
855
+ console.log('βœ… SSE μ—λŸ¬ 응닡 전솑 μ™„λ£Œ');
856
+ } else {
857
+ console.error('❌ SSE transportκ°€ μœ νš¨ν•˜μ§€ μ•Šμ•„ μ—λŸ¬ 응닡 전솑 μ‹€νŒ¨');
858
+ }
859
+ } catch (errorSseError) {
860
+ console.error('❌ SSE μ—λŸ¬ 응닡 전솑 μ‹€νŒ¨:', errorSseError);
861
+ }
862
+
863
+ // Send HTTP response
864
+ res.json({ status: 'error' });
865
+ }
866
+ });
867
+
868
+ // WebSocket μ„œλ²„ μ„€μ •
869
+ const wss = new WebSocketServer({ server });
870
+
871
+ wss.on('connection', (ws) => {
872
+ console.log('πŸ”— WebSocket ν΄λΌμ΄μ–ΈνŠΈ 연결됨');
873
+
874
+ ws.on('message', async (data) => {
875
+ let message: any;
876
+ try {
877
+ message = JSON.parse(data.toString());
878
+
879
+ if (message.method === 'tools/list') {
880
+ ws.send(JSON.stringify({
881
+ jsonrpc: '2.0',
882
+ id: message.id,
883
+ result: { tools }
884
+ }));
885
+ } else if (message.method === 'tools/call') {
886
+ const { name, arguments: args } = message.params;
887
+ let result;
888
+
889
+ switch (name) {
890
+ case 'remember':
891
+ result = await handleRemember(RememberSchema.parse(args));
892
+ break;
893
+ case 'recall':
894
+ result = await handleRecall(RecallSchema.parse(args));
895
+ break;
896
+ case 'forget':
897
+ result = await handleForget(ForgetSchema.parse(args));
898
+ break;
899
+ case 'pin':
900
+ result = await handlePin(PinSchema.parse(args));
901
+ break;
902
+ case 'unpin':
903
+ result = await handleUnpin(UnpinSchema.parse(args));
904
+ break;
905
+ default:
906
+ throw new Error(`Unknown tool: ${name}`);
907
+ }
908
+
909
+ ws.send(JSON.stringify({
910
+ jsonrpc: '2.0',
911
+ id: message.id,
912
+ result: { content: [{ type: 'text', text: JSON.stringify(result) }] }
913
+ }));
914
+ }
915
+ } catch (error) {
916
+ console.error('❌ WebSocket λ©”μ‹œμ§€ 처리 μ‹€νŒ¨:', error);
917
+ ws.send(JSON.stringify({
918
+ jsonrpc: '2.0',
919
+ id: message?.id || null,
920
+ error: {
921
+ code: -32603,
922
+ message: 'Internal error',
923
+ data: error instanceof Error ? error.message : 'Unknown error'
924
+ }
925
+ }));
926
+ }
927
+ });
928
+
929
+ ws.on('close', () => {
930
+ console.log('πŸ”Œ WebSocket ν΄λΌμ΄μ–ΈνŠΈ μ—°κ²° ν•΄μ œλ¨');
931
+ });
932
+ });
933
+
934
+ // μ„œλ²„ μ΄ˆκΈ°ν™”
935
+ async function initializeServer() {
936
+ try {
937
+ console.log('πŸš€ HTTP/WebSocket MCP μ„œλ²„ μ‹œμž‘ 쀑...');
938
+
939
+ // μ„€μ • 검증
940
+ validateConfig();
941
+
942
+ // λ°μ΄ν„°λ² μ΄μŠ€ μ΄ˆκΈ°ν™”
943
+ db = await initializeDatabase();
944
+
945
+ // 검색 μ—”μ§„ μ΄ˆκΈ°ν™”
946
+ searchEngine = new SearchEngine();
947
+ hybridSearchEngine = new HybridSearchEngine();
948
+ embeddingService = new MemoryEmbeddingService();
949
+
950
+ // μž„λ² λ”© ν”„λ‘œλ°”μ΄λ” 정보 ν‘œμ‹œ
951
+ console.log(`πŸ”§ μž„λ² λ”© ν”„λ‘œλ°”μ΄λ”: ${mementoConfig.embeddingProvider.toUpperCase()}`);
952
+ if (mementoConfig.embeddingProvider === 'openai' && mementoConfig.openaiApiKey) {
953
+ console.log(` πŸ“ λͺ¨λΈ: ${mementoConfig.openaiModel} (${mementoConfig.embeddingDimensions}차원)`);
954
+ } else if (mementoConfig.embeddingProvider === 'gemini' && mementoConfig.geminiApiKey) {
955
+ console.log(` πŸ“ λͺ¨λΈ: ${mementoConfig.geminiModel} (${mementoConfig.embeddingDimensions}차원)`);
956
+ } else if (mementoConfig.embeddingProvider === 'lightweight') {
957
+ console.log(` πŸ“ λͺ¨λΈ: lightweight-hybrid (512차원)`);
958
+ }
959
+
960
+ console.log('βœ… μ„œλ²„ μ΄ˆκΈ°ν™” μ™„λ£Œ');
961
+ console.log(`πŸ“Š μ„œλ²„: ${mementoConfig.serverName} v${mementoConfig.serverVersion}`);
962
+ console.log(`πŸ—„οΈ λ°μ΄ν„°λ² μ΄μŠ€: ${mementoConfig.dbPath}`);
963
+
964
+ } catch (error) {
965
+ console.error('❌ μ„œλ²„ μ΄ˆκΈ°ν™” μ‹€νŒ¨:', error);
966
+ process.exit(1);
967
+ }
968
+ }
969
+
970
+ // 정리 ν•¨μˆ˜
971
+ let isCleaningUp = false;
972
+ async function cleanup() {
973
+ if (isCleaningUp) {
974
+ return; // 이미 정리 쀑이면 쀑볡 μ‹€ν–‰ λ°©μ§€
975
+ }
976
+
977
+ isCleaningUp = true;
978
+
979
+ try {
980
+ if (db) {
981
+ closeDatabase(db);
982
+ db = null;
983
+ }
984
+ console.log('πŸ‘‹ HTTP/WebSocket MCP μ„œλ²„ μ’…λ£Œ');
985
+ } catch (error) {
986
+ console.error('❌ 정리 쀑 였λ₯˜:', error);
987
+ }
988
+ }
989
+
990
+ // ν”„λ‘œμ„ΈμŠ€ μ’…λ£Œ μ‹œ 정리 (ν•œ 번만 등둝)
991
+ let cleanupRegistered = false;
992
+ function registerCleanupHandlers() {
993
+ if (cleanupRegistered) {
994
+ return;
995
+ }
996
+
997
+ cleanupRegistered = true;
998
+
999
+ process.on('SIGINT', async () => {
1000
+ await cleanup();
1001
+ process.exit(0);
1002
+ });
1003
+
1004
+ process.on('SIGTERM', async () => {
1005
+ await cleanup();
1006
+ process.exit(0);
1007
+ });
1008
+
1009
+ process.on('uncaughtException', async (error) => {
1010
+ console.error('❌ μ˜ˆμƒμΉ˜ λͺ»ν•œ 였λ₯˜:', error);
1011
+ await cleanup();
1012
+ process.exit(1);
1013
+ });
1014
+ }
1015
+
1016
+ // μ„œλ²„ μ‹œμž‘
1017
+ const PORT = process.env.PORT || 9001;
1018
+
1019
+ async function startServer() {
1020
+ await initializeServer();
1021
+
1022
+ // 정리 ν•Έλ“€λŸ¬ 등둝
1023
+ registerCleanupHandlers();
1024
+
1025
+ // Express app을 μ‚¬μš©ν•˜μ—¬ λͺ¨λ“  μΈν„°νŽ˜μ΄μŠ€μ— 바인딩
1026
+ app.listen(Number(PORT), '0.0.0.0', () => {
1027
+ console.log(`🌐 HTTP μ„œλ²„: http://0.0.0.0:${PORT}`);
1028
+ console.log(`πŸ”Œ WebSocket μ„œλ²„: ws://0.0.0.0:${PORT}`);
1029
+ console.log(`πŸ“‹ API λ¬Έμ„œ: http://0.0.0.0:${PORT}/tools`);
1030
+ console.log(`❀️ ν—¬μŠ€ 체크: http://0.0.0.0:${PORT}/health`);
1031
+ });
1032
+
1033
+ // μΆ”κ°€: λͺ¨λ“  μΈν„°νŽ˜μ΄μŠ€μ— 바인딩 확인
1034
+ server.on('listening', () => {
1035
+ const address = server.address();
1036
+ if (address && typeof address === 'object') {
1037
+ console.log(`πŸ”— μ„œλ²„κ°€ ${address.address}:${address.port}에 바인딩됨`);
1038
+ }
1039
+ });
1040
+ }
1041
+
1042
+ // μ„œλ²„ μ‹œμž‘
1043
+ if (process.argv[1] && process.argv[1].endsWith('http-server.js')) {
1044
+ startServer().catch(error => {
1045
+ console.error('❌ μ„œλ²„ μ‹œμž‘ μ‹€νŒ¨:', error);
1046
+ process.exit(1);
1047
+ });
1048
+ }
1049
+
1050
+ export const __test: {
1051
+ setTestDependencies: (deps: TestDependencies) => void;
1052
+ getApp: () => express.Application;
1053
+ getServer: () => any;
1054
+ getDatabase: () => Database.Database | null;
1055
+ getSearchEngine: () => SearchEngine | undefined;
1056
+ getHybridSearchEngine: () => HybridSearchEngine | undefined;
1057
+ getEmbeddingService: () => MemoryEmbeddingService | undefined;
1058
+ } = {
1059
+ setTestDependencies,
1060
+ getApp: () => app,
1061
+ getServer: () => server,
1062
+ getDatabase: () => db,
1063
+ getSearchEngine: () => searchEngine,
1064
+ getHybridSearchEngine: () => hybridSearchEngine,
1065
+ getEmbeddingService: () => embeddingService
1066
+ };
1067
+
1068
+ export { startServer, cleanup };