@jagilber-org/index-server 1.22.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 (372) hide show
  1. package/CHANGELOG.md +1354 -0
  2. package/CODE_OF_CONDUCT.md +49 -0
  3. package/CONTRIBUTING.md +99 -0
  4. package/LICENSE +21 -0
  5. package/README.md +228 -0
  6. package/SECURITY.md +50 -0
  7. package/dist/config/configUtils.d.ts +11 -0
  8. package/dist/config/configUtils.js +87 -0
  9. package/dist/config/dashboardConfig.d.ts +46 -0
  10. package/dist/config/dashboardConfig.js +67 -0
  11. package/dist/config/defaultValues.d.ts +63 -0
  12. package/dist/config/defaultValues.js +72 -0
  13. package/dist/config/dirConstants.d.ts +18 -0
  14. package/dist/config/dirConstants.js +29 -0
  15. package/dist/config/featureConfig.d.ts +61 -0
  16. package/dist/config/featureConfig.js +121 -0
  17. package/dist/config/runtimeConfig.d.ts +151 -0
  18. package/dist/config/runtimeConfig.js +380 -0
  19. package/dist/config/serverConfig.d.ts +90 -0
  20. package/dist/config/serverConfig.js +167 -0
  21. package/dist/dashboard/analytics/AnalyticsEngine.d.ts +142 -0
  22. package/dist/dashboard/analytics/AnalyticsEngine.js +373 -0
  23. package/dist/dashboard/analytics/BusinessIntelligence.d.ts +187 -0
  24. package/dist/dashboard/analytics/BusinessIntelligence.js +594 -0
  25. package/dist/dashboard/client/admin.html +2175 -0
  26. package/dist/dashboard/client/chunks/mermaid-layout-elk.esm.min/chunk-SP2CHFBE.mjs +1 -0
  27. package/dist/dashboard/client/chunks/mermaid-layout-elk.esm.min/render-T6MDALS3.mjs +27 -0
  28. package/dist/dashboard/client/css/admin.css +1587 -0
  29. package/dist/dashboard/client/js/admin.auth.js +179 -0
  30. package/dist/dashboard/client/js/admin.boot.js +359 -0
  31. package/dist/dashboard/client/js/admin.config.js +196 -0
  32. package/dist/dashboard/client/js/admin.embeddings.js +426 -0
  33. package/dist/dashboard/client/js/admin.graph.js +615 -0
  34. package/dist/dashboard/client/js/admin.instances.js +120 -0
  35. package/dist/dashboard/client/js/admin.instructions.js +579 -0
  36. package/dist/dashboard/client/js/admin.logs.js +113 -0
  37. package/dist/dashboard/client/js/admin.maintenance.js +393 -0
  38. package/dist/dashboard/client/js/admin.messaging.js +636 -0
  39. package/dist/dashboard/client/js/admin.monitor.js +184 -0
  40. package/dist/dashboard/client/js/admin.overview.js +221 -0
  41. package/dist/dashboard/client/js/admin.performance.js +61 -0
  42. package/dist/dashboard/client/js/admin.sessions.js +292 -0
  43. package/dist/dashboard/client/js/admin.sqlite.js +373 -0
  44. package/dist/dashboard/client/js/admin.utils.js +49 -0
  45. package/dist/dashboard/client/js/chart.umd.js +14 -0
  46. package/dist/dashboard/client/js/elk.bundled.js +6696 -0
  47. package/dist/dashboard/client/js/marked.umd.js +74 -0
  48. package/dist/dashboard/client/js/mermaid.min.js +3022 -0
  49. package/dist/dashboard/client/mermaid-layout-elk.esm.min.mjs +1 -0
  50. package/dist/dashboard/export/DataExporter.d.ts +169 -0
  51. package/dist/dashboard/export/DataExporter.js +737 -0
  52. package/dist/dashboard/export/exporters/csvExporter.d.ts +11 -0
  53. package/dist/dashboard/export/exporters/csvExporter.js +47 -0
  54. package/dist/dashboard/export/exporters/exportTypes.d.ts +89 -0
  55. package/dist/dashboard/export/exporters/exportTypes.js +5 -0
  56. package/dist/dashboard/export/exporters/jsonExporter.d.ts +7 -0
  57. package/dist/dashboard/export/exporters/jsonExporter.js +23 -0
  58. package/dist/dashboard/export/exporters/xmlExporter.d.ts +17 -0
  59. package/dist/dashboard/export/exporters/xmlExporter.js +176 -0
  60. package/dist/dashboard/integration/APIIntegration.d.ts +41 -0
  61. package/dist/dashboard/integration/APIIntegration.js +95 -0
  62. package/dist/dashboard/security/SecurityMonitor.d.ts +167 -0
  63. package/dist/dashboard/security/SecurityMonitor.js +560 -0
  64. package/dist/dashboard/server/AdminPanel.d.ts +195 -0
  65. package/dist/dashboard/server/AdminPanel.js +861 -0
  66. package/dist/dashboard/server/AdminPanelConfig.d.ts +42 -0
  67. package/dist/dashboard/server/AdminPanelConfig.js +80 -0
  68. package/dist/dashboard/server/AdminPanelState.d.ts +47 -0
  69. package/dist/dashboard/server/AdminPanelState.js +215 -0
  70. package/dist/dashboard/server/ApiRoutes.d.ts +17 -0
  71. package/dist/dashboard/server/ApiRoutes.js +184 -0
  72. package/dist/dashboard/server/DashboardServer.d.ts +49 -0
  73. package/dist/dashboard/server/DashboardServer.js +160 -0
  74. package/dist/dashboard/server/FileMetricsStorage.d.ts +49 -0
  75. package/dist/dashboard/server/FileMetricsStorage.js +196 -0
  76. package/dist/dashboard/server/HttpTransport.d.ts +23 -0
  77. package/dist/dashboard/server/HttpTransport.js +116 -0
  78. package/dist/dashboard/server/InstanceManager.d.ts +53 -0
  79. package/dist/dashboard/server/InstanceManager.js +295 -0
  80. package/dist/dashboard/server/KnowledgeStore.d.ts +35 -0
  81. package/dist/dashboard/server/KnowledgeStore.js +105 -0
  82. package/dist/dashboard/server/LeaderElection.d.ts +81 -0
  83. package/dist/dashboard/server/LeaderElection.js +268 -0
  84. package/dist/dashboard/server/MetricsCollector.d.ts +200 -0
  85. package/dist/dashboard/server/MetricsCollector.js +810 -0
  86. package/dist/dashboard/server/SessionPersistenceManager.d.ts +88 -0
  87. package/dist/dashboard/server/SessionPersistenceManager.js +458 -0
  88. package/dist/dashboard/server/ThinClient.d.ts +64 -0
  89. package/dist/dashboard/server/ThinClient.js +237 -0
  90. package/dist/dashboard/server/WebSocketManager.d.ts +161 -0
  91. package/dist/dashboard/server/WebSocketManager.js +448 -0
  92. package/dist/dashboard/server/httpLifecycle.d.ts +17 -0
  93. package/dist/dashboard/server/httpLifecycle.js +35 -0
  94. package/dist/dashboard/server/legacyDashboardHtml.d.ts +9 -0
  95. package/dist/dashboard/server/legacyDashboardHtml.js +618 -0
  96. package/dist/dashboard/server/legacyDashboardStyles.d.ts +5 -0
  97. package/dist/dashboard/server/legacyDashboardStyles.js +490 -0
  98. package/dist/dashboard/server/metricsAggregation.d.ts +252 -0
  99. package/dist/dashboard/server/metricsAggregation.js +210 -0
  100. package/dist/dashboard/server/metricsSerializer.d.ts +25 -0
  101. package/dist/dashboard/server/metricsSerializer.js +195 -0
  102. package/dist/dashboard/server/middleware/ensureLoadedMiddleware.d.ts +25 -0
  103. package/dist/dashboard/server/middleware/ensureLoadedMiddleware.js +24 -0
  104. package/dist/dashboard/server/routes/admin.routes.d.ts +16 -0
  105. package/dist/dashboard/server/routes/admin.routes.js +574 -0
  106. package/dist/dashboard/server/routes/adminAuth.d.ts +4 -0
  107. package/dist/dashboard/server/routes/adminAuth.js +46 -0
  108. package/dist/dashboard/server/routes/alerts.routes.d.ts +7 -0
  109. package/dist/dashboard/server/routes/alerts.routes.js +91 -0
  110. package/dist/dashboard/server/routes/api.feedback.routes.d.ts +73 -0
  111. package/dist/dashboard/server/routes/api.feedback.routes.js +171 -0
  112. package/dist/dashboard/server/routes/api.instructions.routes.d.ts +101 -0
  113. package/dist/dashboard/server/routes/api.instructions.routes.js +213 -0
  114. package/dist/dashboard/server/routes/api.usage.routes.d.ts +57 -0
  115. package/dist/dashboard/server/routes/api.usage.routes.js +374 -0
  116. package/dist/dashboard/server/routes/embeddings.routes.d.ts +6 -0
  117. package/dist/dashboard/server/routes/embeddings.routes.js +246 -0
  118. package/dist/dashboard/server/routes/graph.routes.d.ts +6 -0
  119. package/dist/dashboard/server/routes/graph.routes.js +279 -0
  120. package/dist/dashboard/server/routes/index.d.ts +39 -0
  121. package/dist/dashboard/server/routes/index.js +229 -0
  122. package/dist/dashboard/server/routes/instances.routes.d.ts +6 -0
  123. package/dist/dashboard/server/routes/instances.routes.js +35 -0
  124. package/dist/dashboard/server/routes/instructions.routes.d.ts +8 -0
  125. package/dist/dashboard/server/routes/instructions.routes.js +268 -0
  126. package/dist/dashboard/server/routes/knowledge.routes.d.ts +6 -0
  127. package/dist/dashboard/server/routes/knowledge.routes.js +80 -0
  128. package/dist/dashboard/server/routes/logs.routes.d.ts +6 -0
  129. package/dist/dashboard/server/routes/logs.routes.js +166 -0
  130. package/dist/dashboard/server/routes/messaging.routes.d.ts +16 -0
  131. package/dist/dashboard/server/routes/messaging.routes.js +307 -0
  132. package/dist/dashboard/server/routes/metrics.routes.d.ts +10 -0
  133. package/dist/dashboard/server/routes/metrics.routes.js +335 -0
  134. package/dist/dashboard/server/routes/scripts.routes.d.ts +9 -0
  135. package/dist/dashboard/server/routes/scripts.routes.js +84 -0
  136. package/dist/dashboard/server/routes/sqlite.routes.d.ts +9 -0
  137. package/dist/dashboard/server/routes/sqlite.routes.js +570 -0
  138. package/dist/dashboard/server/routes/status.routes.d.ts +7 -0
  139. package/dist/dashboard/server/routes/status.routes.js +179 -0
  140. package/dist/dashboard/server/routes/synthetic.routes.d.ts +7 -0
  141. package/dist/dashboard/server/routes/synthetic.routes.js +197 -0
  142. package/dist/dashboard/server/routes/tools.routes.d.ts +6 -0
  143. package/dist/dashboard/server/routes/tools.routes.js +47 -0
  144. package/dist/dashboard/server/routes/usage.routes.d.ts +6 -0
  145. package/dist/dashboard/server/routes/usage.routes.js +26 -0
  146. package/dist/dashboard/server/wsInit.d.ts +16 -0
  147. package/dist/dashboard/server/wsInit.js +35 -0
  148. package/dist/externalClientLib.d.ts +1 -0
  149. package/dist/externalClientLib.js +2 -0
  150. package/dist/minimal/index.d.ts +1 -0
  151. package/dist/minimal/index.js +140 -0
  152. package/dist/models/SessionPersistence.d.ts +115 -0
  153. package/dist/models/SessionPersistence.js +66 -0
  154. package/dist/models/instruction.d.ts +46 -0
  155. package/dist/models/instruction.js +2 -0
  156. package/dist/perf/benchmark.d.ts +1 -0
  157. package/dist/perf/benchmark.js +50 -0
  158. package/dist/portableClientWrapper.d.ts +1 -0
  159. package/dist/portableClientWrapper.js +2 -0
  160. package/dist/schemas/index.d.ts +132 -0
  161. package/dist/schemas/index.js +372 -0
  162. package/dist/scripts/runPerformanceBaseline.d.ts +1 -0
  163. package/dist/scripts/runPerformanceBaseline.js +17 -0
  164. package/dist/server/backgroundServicesStartup.d.ts +3 -0
  165. package/dist/server/backgroundServicesStartup.js +51 -0
  166. package/dist/server/handshakeManager.d.ts +25 -0
  167. package/dist/server/handshakeManager.js +470 -0
  168. package/dist/server/index-server.d.ts +38 -0
  169. package/dist/server/index-server.js +620 -0
  170. package/dist/server/multiInstanceStartup.d.ts +6 -0
  171. package/dist/server/multiInstanceStartup.js +132 -0
  172. package/dist/server/registry.d.ts +44 -0
  173. package/dist/server/registry.js +236 -0
  174. package/dist/server/sdkServer.d.ts +8 -0
  175. package/dist/server/sdkServer.js +299 -0
  176. package/dist/server/shutdownGuard.d.ts +41 -0
  177. package/dist/server/shutdownGuard.js +52 -0
  178. package/dist/server/startupDiagnostics.d.ts +2 -0
  179. package/dist/server/startupDiagnostics.js +33 -0
  180. package/dist/server/thin-client.d.ts +22 -0
  181. package/dist/server/thin-client.js +111 -0
  182. package/dist/server/transport.d.ts +41 -0
  183. package/dist/server/transport.js +312 -0
  184. package/dist/server/transportFactory.d.ts +21 -0
  185. package/dist/server/transportFactory.js +429 -0
  186. package/dist/services/atomicFs.d.ts +22 -0
  187. package/dist/services/atomicFs.js +103 -0
  188. package/dist/services/auditLog.d.ts +38 -0
  189. package/dist/services/auditLog.js +142 -0
  190. package/dist/services/autoBackup.d.ts +14 -0
  191. package/dist/services/autoBackup.js +171 -0
  192. package/dist/services/autoSplit.d.ts +32 -0
  193. package/dist/services/autoSplit.js +113 -0
  194. package/dist/services/backupZip.d.ts +25 -0
  195. package/dist/services/backupZip.js +112 -0
  196. package/dist/services/bootstrapGating.d.ts +123 -0
  197. package/dist/services/bootstrapGating.js +221 -0
  198. package/dist/services/canonical.d.ts +23 -0
  199. package/dist/services/canonical.js +65 -0
  200. package/dist/services/categoryRules.d.ts +7 -0
  201. package/dist/services/categoryRules.js +37 -0
  202. package/dist/services/classificationService.d.ts +42 -0
  203. package/dist/services/classificationService.js +168 -0
  204. package/dist/services/embeddingService.d.ts +62 -0
  205. package/dist/services/embeddingService.js +264 -0
  206. package/dist/services/errors.d.ts +22 -0
  207. package/dist/services/errors.js +31 -0
  208. package/dist/services/featureFlags.d.ts +25 -0
  209. package/dist/services/featureFlags.js +89 -0
  210. package/dist/services/features.d.ts +13 -0
  211. package/dist/services/features.js +35 -0
  212. package/dist/services/handlers/instructions.add.d.ts +1 -0
  213. package/dist/services/handlers/instructions.add.js +510 -0
  214. package/dist/services/handlers/instructions.groom.d.ts +1 -0
  215. package/dist/services/handlers/instructions.groom.js +575 -0
  216. package/dist/services/handlers/instructions.import.d.ts +1 -0
  217. package/dist/services/handlers/instructions.import.js +205 -0
  218. package/dist/services/handlers/instructions.patch.d.ts +1 -0
  219. package/dist/services/handlers/instructions.patch.js +121 -0
  220. package/dist/services/handlers/instructions.query.d.ts +159 -0
  221. package/dist/services/handlers/instructions.query.js +469 -0
  222. package/dist/services/handlers/instructions.reload.d.ts +1 -0
  223. package/dist/services/handlers/instructions.reload.js +13 -0
  224. package/dist/services/handlers/instructions.remove.d.ts +1 -0
  225. package/dist/services/handlers/instructions.remove.js +122 -0
  226. package/dist/services/handlers/instructions.shared.d.ts +32 -0
  227. package/dist/services/handlers/instructions.shared.js +91 -0
  228. package/dist/services/handlers.activation.d.ts +1 -0
  229. package/dist/services/handlers.activation.js +203 -0
  230. package/dist/services/handlers.bootstrap.d.ts +1 -0
  231. package/dist/services/handlers.bootstrap.js +38 -0
  232. package/dist/services/handlers.dashboardConfig.d.ts +34 -0
  233. package/dist/services/handlers.dashboardConfig.js +110 -0
  234. package/dist/services/handlers.diagnostics.d.ts +1 -0
  235. package/dist/services/handlers.diagnostics.js +64 -0
  236. package/dist/services/handlers.feedback.d.ts +15 -0
  237. package/dist/services/handlers.feedback.js +389 -0
  238. package/dist/services/handlers.gates.d.ts +1 -0
  239. package/dist/services/handlers.gates.js +47 -0
  240. package/dist/services/handlers.graph.d.ts +53 -0
  241. package/dist/services/handlers.graph.js +231 -0
  242. package/dist/services/handlers.help.d.ts +1 -0
  243. package/dist/services/handlers.help.js +119 -0
  244. package/dist/services/handlers.instructionSchema.d.ts +1 -0
  245. package/dist/services/handlers.instructionSchema.js +227 -0
  246. package/dist/services/handlers.instructions.d.ts +8 -0
  247. package/dist/services/handlers.instructions.js +14 -0
  248. package/dist/services/handlers.instructionsDiagnostics.d.ts +1 -0
  249. package/dist/services/handlers.instructionsDiagnostics.js +14 -0
  250. package/dist/services/handlers.integrity.d.ts +1 -0
  251. package/dist/services/handlers.integrity.js +35 -0
  252. package/dist/services/handlers.manifest.d.ts +1 -0
  253. package/dist/services/handlers.manifest.js +24 -0
  254. package/dist/services/handlers.messaging.d.ts +12 -0
  255. package/dist/services/handlers.messaging.js +203 -0
  256. package/dist/services/handlers.metrics.d.ts +1 -0
  257. package/dist/services/handlers.metrics.js +43 -0
  258. package/dist/services/handlers.promote.d.ts +1 -0
  259. package/dist/services/handlers.promote.js +326 -0
  260. package/dist/services/handlers.prompt.d.ts +1 -0
  261. package/dist/services/handlers.prompt.js +7 -0
  262. package/dist/services/handlers.search.d.ts +69 -0
  263. package/dist/services/handlers.search.js +669 -0
  264. package/dist/services/handlers.testPrimitive.d.ts +1 -0
  265. package/dist/services/handlers.testPrimitive.js +5 -0
  266. package/dist/services/handlers.trace.d.ts +1 -0
  267. package/dist/services/handlers.trace.js +35 -0
  268. package/dist/services/handlers.usage.d.ts +1 -0
  269. package/dist/services/handlers.usage.js +11 -0
  270. package/dist/services/hotScore.d.ts +137 -0
  271. package/dist/services/hotScore.js +244 -0
  272. package/dist/services/indexContext.d.ts +117 -0
  273. package/dist/services/indexContext.js +989 -0
  274. package/dist/services/indexLoader.d.ts +44 -0
  275. package/dist/services/indexLoader.js +920 -0
  276. package/dist/services/indexRepository.d.ts +32 -0
  277. package/dist/services/indexRepository.js +71 -0
  278. package/dist/services/indexingService.d.ts +1 -0
  279. package/dist/services/indexingService.js +2 -0
  280. package/dist/services/instructions.dispatcher.d.ts +1 -0
  281. package/dist/services/instructions.dispatcher.js +231 -0
  282. package/dist/services/logPrefix.d.ts +1 -0
  283. package/dist/services/logPrefix.js +30 -0
  284. package/dist/services/logger.d.ts +52 -0
  285. package/dist/services/logger.js +268 -0
  286. package/dist/services/manifestManager.d.ts +82 -0
  287. package/dist/services/manifestManager.js +200 -0
  288. package/dist/services/messaging/agentMailbox.d.ts +60 -0
  289. package/dist/services/messaging/agentMailbox.js +353 -0
  290. package/dist/services/messaging/messagingPersistence.d.ts +20 -0
  291. package/dist/services/messaging/messagingPersistence.js +111 -0
  292. package/dist/services/messaging/messagingTypes.d.ts +150 -0
  293. package/dist/services/messaging/messagingTypes.js +66 -0
  294. package/dist/services/ownershipService.d.ts +1 -0
  295. package/dist/services/ownershipService.js +36 -0
  296. package/dist/services/performanceBaseline.d.ts +19 -0
  297. package/dist/services/performanceBaseline.js +210 -0
  298. package/dist/services/preflight.d.ts +12 -0
  299. package/dist/services/preflight.js +79 -0
  300. package/dist/services/promptReviewService.d.ts +44 -0
  301. package/dist/services/promptReviewService.js +101 -0
  302. package/dist/services/responseEnvelope.d.ts +6 -0
  303. package/dist/services/responseEnvelope.js +25 -0
  304. package/dist/services/seedBootstrap.d.ts +34 -0
  305. package/dist/services/seedBootstrap.js +259 -0
  306. package/dist/services/storage/factory.d.ts +17 -0
  307. package/dist/services/storage/factory.js +35 -0
  308. package/dist/services/storage/hashUtils.d.ts +11 -0
  309. package/dist/services/storage/hashUtils.js +35 -0
  310. package/dist/services/storage/index.d.ts +12 -0
  311. package/dist/services/storage/index.js +18 -0
  312. package/dist/services/storage/jsonFileStore.d.ts +32 -0
  313. package/dist/services/storage/jsonFileStore.js +241 -0
  314. package/dist/services/storage/migrationEngine.d.ts +35 -0
  315. package/dist/services/storage/migrationEngine.js +93 -0
  316. package/dist/services/storage/sqliteMessageStore.d.ts +53 -0
  317. package/dist/services/storage/sqliteMessageStore.js +146 -0
  318. package/dist/services/storage/sqliteSchema.d.ts +12 -0
  319. package/dist/services/storage/sqliteSchema.js +123 -0
  320. package/dist/services/storage/sqliteStore.d.ts +42 -0
  321. package/dist/services/storage/sqliteStore.js +361 -0
  322. package/dist/services/storage/sqliteUsageStore.d.ts +35 -0
  323. package/dist/services/storage/sqliteUsageStore.js +94 -0
  324. package/dist/services/storage/types.d.ts +171 -0
  325. package/dist/services/storage/types.js +12 -0
  326. package/dist/services/toolHandlers.d.ts +23 -0
  327. package/dist/services/toolHandlers.js +50 -0
  328. package/dist/services/toolRegistry.d.ts +20 -0
  329. package/dist/services/toolRegistry.js +490 -0
  330. package/dist/services/toolRegistry.zod.d.ts +10 -0
  331. package/dist/services/toolRegistry.zod.js +325 -0
  332. package/dist/services/tracing.d.ts +26 -0
  333. package/dist/services/tracing.js +260 -0
  334. package/dist/services/usageBuckets.d.ts +161 -0
  335. package/dist/services/usageBuckets.js +364 -0
  336. package/dist/services/validationService.d.ts +38 -0
  337. package/dist/services/validationService.js +125 -0
  338. package/dist/utils/BufferRing.d.ts +203 -0
  339. package/dist/utils/BufferRing.js +551 -0
  340. package/dist/utils/BufferRingExamples.d.ts +55 -0
  341. package/dist/utils/BufferRingExamples.js +188 -0
  342. package/dist/utils/envUtils.d.ts +42 -0
  343. package/dist/utils/envUtils.js +80 -0
  344. package/dist/utils/memoryMonitor.d.ts +83 -0
  345. package/dist/utils/memoryMonitor.js +275 -0
  346. package/dist/versioning/schemaVersion.d.ts +6 -0
  347. package/dist/versioning/schemaVersion.js +94 -0
  348. package/package.json +139 -0
  349. package/schemas/README.md +13 -0
  350. package/schemas/feedback-entry.schema.json +27 -0
  351. package/schemas/graph-export-v2.schema.json +60 -0
  352. package/schemas/index-server.code-schema.json +40670 -0
  353. package/schemas/instruction.schema.json +262 -0
  354. package/schemas/json-schema/SessionPersistence-persisted-admin-session.schema.json +54 -0
  355. package/schemas/json-schema/SessionPersistence-persisted-session-history-entry.schema.json +51 -0
  356. package/schemas/json-schema/SessionPersistence-persisted-web-socket-connection.schema.json +54 -0
  357. package/schemas/json-schema/SessionPersistence-session-persistence-config.schema.json +110 -0
  358. package/schemas/json-schema/SessionPersistence-session-persistence-data.schema.json +229 -0
  359. package/schemas/json-schema/SessionPersistence-session-persistence-manifest.schema.json +109 -0
  360. package/schemas/json-schema/SessionPersistence-session-persistence-metadata.schema.json +55 -0
  361. package/schemas/json-schema/instruction-audience-scope.schema.json +14 -0
  362. package/schemas/json-schema/instruction-content-type.schema.json +17 -0
  363. package/schemas/json-schema/instruction-instruction-entry.schema.json +210 -0
  364. package/schemas/json-schema/instruction-requirement-level.schema.json +16 -0
  365. package/schemas/manifest.json +78 -0
  366. package/schemas/manifest.schema.json +33 -0
  367. package/schemas/usage-batch.schema.json +16 -0
  368. package/schemas/usage-buckets.schema.json +30 -0
  369. package/schemas/usage-event.schema.json +17 -0
  370. package/scripts/copy-dashboard-assets.mjs +170 -0
  371. package/scripts/dist/README.md +15 -0
  372. package/scripts/setup-hooks.cjs +28 -0
@@ -0,0 +1,196 @@
1
+ /* eslint-disable */
2
+ // Configuration panel – dynamic flags grouped by category with search, auto-refresh, and doc links
3
+ (function(){
4
+ var _configRefreshTimer = null;
5
+ var _collapsedCategories = {};
6
+
7
+ function escapeHtml(s) { return String(s).replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;').replace(/"/g,'&quot;'); }
8
+
9
+ function buildFlagRow(f, featureFlags) {
10
+ var isBool = f.type === 'boolean';
11
+ var currentVal = (f.enabled !== undefined ? (f.enabled ? 'on' : 'off') : (f.value !== undefined ? f.value : '')) || '';
12
+ var control = isBool
13
+ ? '<select data-flag="' + escapeHtml(f.name.toLowerCase()) + '" class="form-input cfg-flag-select">'
14
+ + '<option value="1"' + ((featureFlags[f.name.toLowerCase()] ?? f.enabled) ? ' selected' : '') + '>On</option>'
15
+ + '<option value="0"' + (!(featureFlags[f.name.toLowerCase()] ?? f.enabled) ? ' selected' : '') + '>Off</option>'
16
+ + '</select>'
17
+ : '<span class="cfg-flag-ro">' + escapeHtml(currentVal) + '</span>';
18
+ var stabClass = 'cfg-stab-' + (f.stability || 'stable');
19
+ var docHref = f.docAnchor ? ('https://github.com/jagilber-org/index-server/blob/main/docs/configuration.md#' + encodeURIComponent(f.docAnchor)) : '';
20
+ var docIcon = docHref ? ' <a class="cfg-doc-link" href="' + docHref + '" target="_blank" rel="noopener" title="Documentation for ' + escapeHtml(f.name) + '">📖</a>' : '';
21
+ return '<tr class="cfg-flag-row" data-flagname="' + escapeHtml(f.name.toLowerCase()) + '">'
22
+ + '<td class="cfg-flag-name">' + escapeHtml(f.name) + docIcon + '</td>'
23
+ + '<td>' + control + '</td>'
24
+ + '<td class="cfg-flag-default">' + escapeHtml(f.default || '') + '</td>'
25
+ + '<td class="' + stabClass + '">' + escapeHtml(f.stability || '') + '</td>'
26
+ + '<td class="cfg-flag-desc">' + escapeHtml(f.description || '') + '</td>'
27
+ + '</tr>';
28
+ }
29
+
30
+ function groupByCategory(flags) {
31
+ var groups = {};
32
+ var order = [];
33
+ flags.forEach(function(f) {
34
+ if (!groups[f.category]) { groups[f.category] = []; order.push(f.category); }
35
+ groups[f.category].push(f);
36
+ });
37
+ return { groups: groups, order: order };
38
+ }
39
+
40
+ function buildFlagsHtml(allFlags, featureFlags) {
41
+ if (!allFlags.length) return '<div class="cfg-no-flags">No feature flags detected</div>';
42
+ var catData = groupByCategory(allFlags);
43
+ var html = '<div class="cfg-flag-filter"><input type="text" id="cfg-flag-search" class="form-input" placeholder="Filter flags..." /></div>';
44
+ catData.order.forEach(function(cat) {
45
+ var collapsed = _collapsedCategories[cat];
46
+ var flags = catData.groups[cat];
47
+ html += '<div class="cfg-category-group" data-category="' + escapeHtml(cat) + '">';
48
+ html += '<div class="cfg-category-header" onclick="toggleConfigCategory(\'' + escapeHtml(cat) + '\')">';
49
+ html += '<span class="cfg-category-chevron">' + (collapsed ? '▶' : '▼') + '</span> ';
50
+ html += '<span class="cfg-category-name">' + escapeHtml(cat.charAt(0).toUpperCase() + cat.slice(1)) + '</span>';
51
+ html += ' <span class="cfg-category-count">(' + flags.length + ')</span>';
52
+ html += '</div>';
53
+ html += '<table class="cfg-flag-table"' + (collapsed ? ' style="display:none"' : '') + '>';
54
+ html += '<thead><tr><th>Flag</th><th>Value</th><th>Default</th><th>Stability</th><th>Description</th></tr></thead>';
55
+ html += '<tbody>';
56
+ flags.forEach(function(f) { html += buildFlagRow(f, featureFlags); });
57
+ html += '</tbody></table>';
58
+ html += '</div>';
59
+ });
60
+ return html;
61
+ }
62
+
63
+ async function loadConfiguration() {
64
+ try {
65
+ var res = await adminAuth.adminFetch('/api/admin/config');
66
+ var data = await res.json();
67
+ if (!data.success) throw new Error('Failed to load config');
68
+ var cfg = data.config;
69
+ var featureFlags = data.featureFlags || {};
70
+ var allFlags = Array.isArray(data.allFlags) ? data.allFlags : [];
71
+ if (!allFlags.length) {
72
+ try {
73
+ var fres = await adminAuth.adminFetch('/api/admin/flags');
74
+ var fdata = await fres.json();
75
+ if (fdata.success && Array.isArray(fdata.allFlags)) allFlags = fdata.allFlags;
76
+ } catch(e) { /* ignore */ }
77
+ }
78
+ var refreshedAt = data.timestamp ? new Date(data.timestamp).toLocaleTimeString() : new Date().toLocaleTimeString();
79
+
80
+ var html = '<div class="cfg-panel">'
81
+ + '<form onsubmit="return updateConfiguration(event)" class="cfg-server-form">'
82
+ + '<div class="cfg-server-grid">'
83
+ + '<div class="form-group"><label class="form-label">Max Connections</label>'
84
+ + '<input class="form-input" type="number" id="cfg-maxConnections" value="' + cfg.serverSettings.maxConnections + '" /></div>'
85
+ + '<div class="form-group"><label class="form-label">Request Timeout (ms)</label>'
86
+ + '<input class="form-input" type="number" id="cfg-requestTimeout" value="' + cfg.serverSettings.requestTimeout + '" /></div>'
87
+ + '<div class="form-group"><label class="form-label">Verbose Logging</label>'
88
+ + '<select class="form-input" id="cfg-verbose"><option value="1"' + (cfg.serverSettings.enableVerboseLogging ? ' selected' : '') + '>Enabled</option>'
89
+ + '<option value="0"' + (!cfg.serverSettings.enableVerboseLogging ? ' selected' : '') + '>Disabled</option></select></div>'
90
+ + '<div class="form-group"><label class="form-label">Enable Mutation</label>'
91
+ + '<select class="form-input" id="cfg-mutation"><option value="1"' + (cfg.serverSettings.enableMutation ? ' selected' : '') + '>Enabled</option>'
92
+ + '<option value="0"' + (!cfg.serverSettings.enableMutation ? ' selected' : '') + '>Disabled</option></select></div>'
93
+ + '<div class="form-group"><label class="form-label">Rate Limit Window (ms)</label>'
94
+ + '<input class="form-input" type="number" id="cfg-windowMs" value="' + cfg.serverSettings.rateLimit.windowMs + '" /></div>'
95
+ + '<div class="form-group"><label class="form-label">Rate Limit Max Requests</label>'
96
+ + '<input class="form-input" type="number" id="cfg-maxRequests" value="' + cfg.serverSettings.rateLimit.maxRequests + '" /></div>'
97
+ + '</div>'
98
+ + '<div class="cfg-save-row"><button class="action-btn" type="submit">💾 Save Config</button></div>'
99
+ + '</form>'
100
+ + '<div class="cfg-flags-section">'
101
+ + '<div class="cfg-flags-header">'
102
+ + '<h3 class="cfg-flags-title">Feature Flags</h3>'
103
+ + '<span class="cfg-refreshed">Last refreshed: ' + escapeHtml(refreshedAt) + '</span>'
104
+ + '</div>'
105
+ + '<div class="cfg-flags-note">All recognized flags grouped by category. Edit boolean flags inline – changes persist to file. Non-boolean flags are read-only.</div>'
106
+ + buildFlagsHtml(allFlags, featureFlags)
107
+ + '</div></div>';
108
+
109
+ var target = document.getElementById('config-form');
110
+ if (target) {
111
+ target.innerHTML = html;
112
+ target.classList.remove('loading');
113
+ var searchInput = document.getElementById('cfg-flag-search');
114
+ if (searchInput) searchInput.addEventListener('input', filterConfigFlags);
115
+ }
116
+ } catch (e) {
117
+ var target = document.getElementById('config-form');
118
+ if (target) target.innerHTML = '<div class="error">Failed to load configuration</div>';
119
+ }
120
+ }
121
+
122
+ function filterConfigFlags() {
123
+ var term = (document.getElementById('cfg-flag-search') || {}).value;
124
+ if (term === undefined) return;
125
+ term = term.toLowerCase();
126
+ var rows = document.querySelectorAll('.cfg-flag-row');
127
+ rows.forEach(function(row) {
128
+ var name = row.getAttribute('data-flagname') || '';
129
+ var desc = (row.querySelector('.cfg-flag-desc') || {}).textContent || '';
130
+ row.style.display = (name.indexOf(term) !== -1 || desc.toLowerCase().indexOf(term) !== -1) ? '' : 'none';
131
+ });
132
+ // Show all category groups when filtering
133
+ if (term) {
134
+ document.querySelectorAll('.cfg-flag-table').forEach(function(t) { t.style.display = ''; });
135
+ }
136
+ }
137
+
138
+ function toggleConfigCategory(cat) {
139
+ _collapsedCategories[cat] = !_collapsedCategories[cat];
140
+ var group = document.querySelector('.cfg-category-group[data-category="' + cat + '"]');
141
+ if (!group) return;
142
+ var table = group.querySelector('.cfg-flag-table');
143
+ var chevron = group.querySelector('.cfg-category-chevron');
144
+ if (table) table.style.display = _collapsedCategories[cat] ? 'none' : '';
145
+ if (chevron) chevron.textContent = _collapsedCategories[cat] ? '▶' : '▼';
146
+ }
147
+
148
+ function startConfigAutoRefresh() {
149
+ stopConfigAutoRefresh();
150
+ _configRefreshTimer = setInterval(function() {
151
+ // Only auto-refresh if config section is visible
152
+ var section = document.getElementById('config-section');
153
+ if (section && !section.classList.contains('hidden')) loadConfiguration();
154
+ }, 15000);
155
+ }
156
+
157
+ function stopConfigAutoRefresh() {
158
+ if (_configRefreshTimer) { clearInterval(_configRefreshTimer); _configRefreshTimer = null; }
159
+ }
160
+
161
+ async function updateConfiguration(ev) {
162
+ ev.preventDefault();
163
+ var flagSelects = document.querySelectorAll('[data-flag]');
164
+ var featureFlags = {};
165
+ flagSelects.forEach(function(sel) {
166
+ var name = sel.getAttribute('data-flag');
167
+ if (name) featureFlags[name] = sel.value === '1';
168
+ });
169
+ var updates = {
170
+ serverSettings: {
171
+ maxConnections: parseInt(document.getElementById('cfg-maxConnections').value),
172
+ requestTimeout: parseInt(document.getElementById('cfg-requestTimeout').value),
173
+ enableVerboseLogging: document.getElementById('cfg-verbose').value === '1',
174
+ enableMutation: document.getElementById('cfg-mutation').value === '1',
175
+ rateLimit: {
176
+ windowMs: parseInt(document.getElementById('cfg-windowMs').value),
177
+ maxRequests: parseInt(document.getElementById('cfg-maxRequests').value)
178
+ }
179
+ },
180
+ featureFlags: featureFlags
181
+ };
182
+ try {
183
+ var res = await adminAuth.adminFetch('/api/admin/config', { method:'POST', headers:{'Content-Type':'application/json'}, body: JSON.stringify(updates)});
184
+ var data = await res.json();
185
+ if (data.success) { if (typeof showSuccess === 'function') showSuccess('Configuration updated'); loadConfiguration(); }
186
+ else { if (typeof showError === 'function') showError(data.error || 'Update failed'); }
187
+ } catch (e) { if (typeof showError === 'function') showError('Update failed'); }
188
+ return false;
189
+ }
190
+
191
+ window.loadConfiguration = loadConfiguration;
192
+ window.updateConfiguration = updateConfiguration;
193
+ window.toggleConfigCategory = toggleConfigCategory;
194
+ window.startConfigAutoRefresh = startConfigAutoRefresh;
195
+ window.stopConfigAutoRefresh = stopConfigAutoRefresh;
196
+ })();
@@ -0,0 +1,426 @@
1
+ /* global adminAuth */
2
+ /**
3
+ * admin.embeddings.js — Embeddings Visualization Panel
4
+ *
5
+ * Interactive 2D scatter plot of PCA-projected instruction embeddings.
6
+ * Features: category coloring, norm coloring, HTML tooltips, grid lines,
7
+ * devicePixelRatio, click-to-inspect, search filter, zoom/pan.
8
+ */
9
+ (function () {
10
+ 'use strict';
11
+
12
+ const CAT_COLORS = {
13
+ 'AI/ML': '#e74c3c', 'Azure': '#3498db', 'Service Fabric': '#e67e22',
14
+ 'PowerShell': '#2ecc71', 'Agent': '#9b59b6', 'MCP': '#1abc9c',
15
+ 'VS Code': '#f39c12', 'Git/Repo': '#34495e', 'Testing': '#16a085',
16
+ 'Debugging': '#c0392b', 'Containers': '#8e44ad', 'Security': '#d35400',
17
+ 'Runbooks/Guides': '#27ae60', 'Other': '#95a5a6',
18
+ };
19
+
20
+ let embData = null;
21
+ let canvas = null;
22
+ let ctx = null;
23
+ let searchTerm = '';
24
+ let highlightCat = null;
25
+ let colorByNorm = false;
26
+
27
+ // Data bounds (computed once on load)
28
+ let minX = 0, maxX = 0, minY = 0, maxY = 0, rangeX = 1, rangeY = 1;
29
+ let minNorm = 1, maxNorm = 1;
30
+
31
+ // Pan/zoom state
32
+ let panX = 0, panY = 0, zoom = 1;
33
+ let dragging = false, lastMouse = null;
34
+ let selectedIdx = -1;
35
+
36
+ function init() {
37
+ canvas = document.getElementById('embeddings-canvas');
38
+ if (!canvas) return;
39
+ ctx = canvas.getContext('2d');
40
+ resize();
41
+ window.addEventListener('resize', resize);
42
+ canvas.addEventListener('mousemove', onMouseMove);
43
+ canvas.addEventListener('click', onClick);
44
+ canvas.addEventListener('wheel', onWheel, { passive: false });
45
+ canvas.addEventListener('mousedown', onMouseDown);
46
+ canvas.addEventListener('mouseup', onMouseUp);
47
+ canvas.addEventListener('mouseleave', onMouseLeave);
48
+
49
+ var searchInput = document.getElementById('emb-search');
50
+ if (searchInput) {
51
+ searchInput.addEventListener('input', function () {
52
+ searchTerm = this.value.toLowerCase();
53
+ draw();
54
+ });
55
+ }
56
+ }
57
+
58
+ function resize() {
59
+ if (!canvas) return;
60
+ var dpr = window.devicePixelRatio || 1;
61
+ var w = canvas.parentElement.clientWidth;
62
+ var h = canvas.parentElement.clientHeight;
63
+ canvas.width = w * dpr;
64
+ canvas.height = h * dpr;
65
+ canvas.style.width = w + 'px';
66
+ canvas.style.height = h + 'px';
67
+ ctx.setTransform(dpr, 0, 0, dpr, 0, 0);
68
+ draw();
69
+ }
70
+
71
+ // ── Data Loading ──────────────────────────────────────────────────────
72
+
73
+ window.loadEmbeddings = async function loadEmbeddings() {
74
+ if (!canvas) init();
75
+ var statusEl = document.getElementById('emb-status');
76
+ if (statusEl) statusEl.textContent = 'Loading embeddings\u2026';
77
+ try {
78
+ var res = await adminAuth.adminFetch('/api/embeddings/projection');
79
+ if (!res.ok) {
80
+ var err = await res.json().catch(function () { return {}; });
81
+ if (statusEl) statusEl.textContent = 'Error: ' + (err.error || res.statusText);
82
+ return;
83
+ }
84
+ embData = await res.json();
85
+ computeBounds();
86
+ buildLegend();
87
+ buildStats();
88
+ buildSimilarPairs();
89
+ panX = 0; panY = 0; zoom = 1;
90
+ draw();
91
+ if (statusEl) statusEl.textContent = embData.count + ' embeddings loaded (' + embData.dimensions + 'D \u2192 2D)';
92
+ } catch (e) {
93
+ if (statusEl) statusEl.textContent = 'Failed: ' + e.message;
94
+ }
95
+ };
96
+
97
+ function computeBounds() {
98
+ if (!embData || !embData.points.length) return;
99
+ var pts = embData.points;
100
+ minX = maxX = pts[0].x;
101
+ minY = maxY = pts[0].y;
102
+ minNorm = maxNorm = pts[0].norm || 1;
103
+ for (var i = 1; i < pts.length; i++) {
104
+ if (pts[i].x < minX) minX = pts[i].x;
105
+ if (pts[i].x > maxX) maxX = pts[i].x;
106
+ if (pts[i].y < minY) minY = pts[i].y;
107
+ if (pts[i].y > maxY) maxY = pts[i].y;
108
+ var n = pts[i].norm || 1;
109
+ if (n < minNorm) minNorm = n;
110
+ if (n > maxNorm) maxNorm = n;
111
+ }
112
+ rangeX = maxX - minX || 1;
113
+ rangeY = maxY - minY || 1;
114
+ }
115
+
116
+ // ── Coordinate transforms ────────────────────────────────────────────
117
+
118
+ function toScreen(px, py) {
119
+ var w = canvas.width / (window.devicePixelRatio || 1);
120
+ var h = canvas.height / (window.devicePixelRatio || 1);
121
+ var pad = 40;
122
+ var sx = pad + ((px - minX) / rangeX) * (w - 2 * pad);
123
+ var sy = pad + ((py - minY) / rangeY) * (h - 2 * pad);
124
+ return [(sx - w / 2) * zoom + w / 2 + panX, (sy - h / 2) * zoom + h / 2 + panY];
125
+ }
126
+
127
+ // ── Category & UI builders ────────────────────────────────────────────
128
+
129
+ function catColor(cat) {
130
+ return CAT_COLORS[cat] || '#95a5a6';
131
+ }
132
+
133
+ function normColor(n) {
134
+ var t = (n - minNorm) / (maxNorm - minNorm || 1);
135
+ return 'rgb(' + Math.round(30 + t * 225) + ',' + Math.round(180 - t * 100) + ',' + Math.round(255 - t * 200) + ')';
136
+ }
137
+
138
+ function buildLegend() {
139
+ var el = document.getElementById('emb-legend');
140
+ if (!el || !embData) return;
141
+ // Count per category
142
+ var counts = {};
143
+ for (var i = 0; i < embData.points.length; i++) {
144
+ var c = embData.points[i].category || 'Other';
145
+ counts[c] = (counts[c] || 0) + 1;
146
+ }
147
+ var cats = Object.keys(counts).sort();
148
+ if (!cats.length) { el.innerHTML = '<span class="emb-status-text">No category data</span>'; return; }
149
+ el.innerHTML = cats.map(function (c) {
150
+ return '<div class="emb-cat-item" data-cat="' + c + '">' +
151
+ '<div class="emb-cat-dot" style="background:' + catColor(c) + '"></div>' +
152
+ c + ' (' + counts[c] + ')</div>';
153
+ }).join('');
154
+ el.querySelectorAll('.emb-cat-item').forEach(function (item) {
155
+ item.addEventListener('click', function () {
156
+ var cat = item.getAttribute('data-cat');
157
+ highlightCat = highlightCat === cat ? null : cat;
158
+ el.querySelectorAll('.emb-cat-item').forEach(function (e) { e.style.background = ''; });
159
+ if (highlightCat) item.style.background = 'var(--admin-surface-alt)';
160
+ draw();
161
+ });
162
+ });
163
+ }
164
+
165
+ function buildStats() {
166
+ var el = document.getElementById('emb-stats');
167
+ if (!el || !embData) return;
168
+ var s = embData.stats || {};
169
+ var hash = embData.indexHash || '';
170
+ var shortHash = hash.length > 16 ? hash.substring(0, 16) + '\u2026' : (hash || '?');
171
+ el.innerHTML =
172
+ stat('Instructions', embData.count) +
173
+ stat('Dimensions', embData.dimensions) +
174
+ stat('Model', embData.model || '?') +
175
+ stat('index Hash', '<span title="' + hash + '">' + shortHash + '</span>') +
176
+ stat('Avg Cosine Sim', s.avgCosineSim) +
177
+ stat('Min / Max Sim', (s.minCosineSim || '?') + ' / ' + (s.maxCosineSim || '?')) +
178
+ stat('Avg Norm', s.avgNorm);
179
+ }
180
+ function stat(label, value) {
181
+ return '<div class="emb-stat"><span class="label">' + label + ':</span> <span class="value">' + value + '</span></div>';
182
+ }
183
+
184
+ function buildSimilarPairs() {
185
+ var el = document.getElementById('emb-similar');
186
+ if (!el || !embData || !embData.similarPairs) return;
187
+ if (!embData.similarPairs.length) { el.innerHTML = '<em class="emb-status-text">No similar pairs</em>'; return; }
188
+ el.innerHTML = embData.similarPairs.slice(0, 10).map(function (p) {
189
+ var aShort = p.a.length > 30 ? p.a.substring(0, 30) : p.a;
190
+ var bShort = p.b.length > 30 ? p.b.substring(0, 30) : p.b;
191
+ return '<div class="emb-pair"><span class="sim">' + p.similarity + '</span> ' + aShort + ' \u2194 ' + bShort + '</div>';
192
+ }).join('');
193
+ }
194
+
195
+ // ── Drawing ───────────────────────────────────────────────────────────
196
+
197
+ function draw() {
198
+ if (!ctx || !canvas) return;
199
+ var dpr = window.devicePixelRatio || 1;
200
+ var w = canvas.width / dpr;
201
+ var h = canvas.height / dpr;
202
+ ctx.clearRect(0, 0, w, h);
203
+
204
+ if (!embData || !embData.points.length) {
205
+ ctx.fillStyle = '#8e959e';
206
+ ctx.font = '14px Inter, system-ui, sans-serif';
207
+ ctx.textAlign = 'center';
208
+ ctx.fillText('Click "Load" to visualize embeddings', w / 2, h / 2);
209
+ ctx.textAlign = 'start';
210
+ return;
211
+ }
212
+
213
+ // Grid
214
+ ctx.strokeStyle = '#21262d';
215
+ ctx.lineWidth = 0.5;
216
+ for (var gi = 0; gi <= 10; gi++) {
217
+ var gx = (gi / 10) * rangeX + minX;
218
+ var gy = (gi / 10) * rangeY + minY;
219
+ var sx1 = toScreen(gx, minY)[0];
220
+ var sy1 = toScreen(minX, gy)[1];
221
+ ctx.beginPath(); ctx.moveTo(sx1, 0); ctx.lineTo(sx1, h); ctx.stroke();
222
+ ctx.beginPath(); ctx.moveTo(0, sy1); ctx.lineTo(w, sy1); ctx.stroke();
223
+ }
224
+
225
+ // Points
226
+ var search = searchTerm;
227
+ var pts = embData.points;
228
+ for (var i = 0; i < pts.length; i++) {
229
+ var pt = pts[i];
230
+ var scr = toScreen(pt.x, pt.y);
231
+ var sx = scr[0], sy = scr[1];
232
+ if (sx < -20 || sx > w + 20 || sy < -20 || sy > h + 20) continue;
233
+
234
+ var matchSearch = !search || pt.id.toLowerCase().indexOf(search) !== -1;
235
+ var matchCat = !highlightCat || pt.category === highlightCat;
236
+ var highlighted = matchSearch && matchCat;
237
+
238
+ var color = colorByNorm ? normColor(pt.norm || 1) : catColor(pt.category);
239
+ ctx.globalAlpha = highlighted ? 0.9 : 0.12;
240
+ ctx.fillStyle = color;
241
+ ctx.beginPath();
242
+ var r = highlighted ? (4 * Math.sqrt(zoom)) : (2.5 * Math.sqrt(zoom));
243
+ r = Math.max(r, 1.5);
244
+ if (i === selectedIdx) r *= 1.5;
245
+ ctx.arc(sx, sy, r, 0, Math.PI * 2);
246
+ ctx.fill();
247
+
248
+ // Label on search match when zoomed
249
+ if (highlighted && search && zoom > 1.5) {
250
+ ctx.globalAlpha = 0.8;
251
+ ctx.fillStyle = '#c9d1d9';
252
+ ctx.font = '10px Inter, system-ui, sans-serif';
253
+ ctx.fillText(pt.id.substring(0, 25), sx + r + 3, sy + 3);
254
+ }
255
+ }
256
+ ctx.globalAlpha = 1;
257
+
258
+ // Axis labels
259
+ ctx.fillStyle = '#484f58';
260
+ ctx.font = '11px Inter, system-ui, sans-serif';
261
+ ctx.fillText('PC1 \u2192', w - 50, h - 8);
262
+ ctx.save();
263
+ ctx.translate(12, 50);
264
+ ctx.rotate(-Math.PI / 2);
265
+ ctx.fillText('\u2190 PC2', 0, 0);
266
+ ctx.restore();
267
+ }
268
+
269
+ // ── Interaction ───────────────────────────────────────────────────────
270
+
271
+ function hitTest(mx, my) {
272
+ if (!embData) return -1;
273
+ var closest = -1, closestDist = 15;
274
+ for (var i = 0; i < embData.points.length; i++) {
275
+ var scr = toScreen(embData.points[i].x, embData.points[i].y);
276
+ var d = Math.hypot(scr[0] - mx, scr[1] - my);
277
+ if (d < closestDist) { closest = i; closestDist = d; }
278
+ }
279
+ return closest;
280
+ }
281
+
282
+ function onMouseMove(e) {
283
+ var mx = e.offsetX, my = e.offsetY;
284
+ if (dragging && lastMouse) {
285
+ panX += mx - lastMouse.x;
286
+ panY += my - lastMouse.y;
287
+ lastMouse = { x: mx, y: my };
288
+ draw();
289
+ return;
290
+ }
291
+ // HTML tooltip
292
+ var tooltip = document.getElementById('emb-tooltip');
293
+ var idx = hitTest(mx, my);
294
+ if (idx >= 0 && embData) {
295
+ var pt = embData.points[idx];
296
+ canvas.style.cursor = 'pointer';
297
+ if (tooltip) {
298
+ tooltip.style.display = 'block';
299
+ tooltip.style.left = (mx + 16) + 'px';
300
+ tooltip.style.top = (my - 10) + 'px';
301
+ tooltip.innerHTML =
302
+ '<div class="tt-id">' + pt.id + '</div>' +
303
+ '<div class="tt-cat">Category: ' + (pt.category || 'Other') + '</div>' +
304
+ '<div class="tt-norm">Norm: ' + (pt.norm ? pt.norm.toFixed(4) : '?') + '</div>' +
305
+ '<div>PC1: ' + pt.x.toFixed(4) + ', PC2: ' + pt.y.toFixed(4) + '</div>';
306
+ }
307
+ } else {
308
+ canvas.style.cursor = 'default';
309
+ if (tooltip) tooltip.style.display = 'none';
310
+ }
311
+ }
312
+
313
+ function onClick(e) {
314
+ var mx = e.offsetX, my = e.offsetY;
315
+ selectedIdx = hitTest(mx, my);
316
+ var detailEl = document.getElementById('emb-detail');
317
+ if (detailEl && selectedIdx >= 0 && embData) {
318
+ var pt = embData.points[selectedIdx];
319
+ detailEl.innerHTML =
320
+ '<b>' + pt.id + '</b>' +
321
+ (pt.title ? '<br>' + pt.title : '') +
322
+ (pt.category ? '<br>Category: ' + pt.category : '') +
323
+ '<br>Norm: ' + (pt.norm ? pt.norm.toFixed(4) : '?') +
324
+ '<br>Position: (' + pt.x.toFixed(4) + ', ' + pt.y.toFixed(4) + ')';
325
+ } else if (detailEl) {
326
+ detailEl.innerHTML = '<em style="color:var(--admin-text-dim)">Click a point to inspect</em>';
327
+ }
328
+ draw();
329
+ }
330
+
331
+ function onWheel(e) {
332
+ e.preventDefault();
333
+ var factor = e.deltaY > 0 ? 0.9 : 1.1;
334
+ var mx = e.offsetX, my = e.offsetY;
335
+ panX = mx - factor * (mx - panX);
336
+ panY = my - factor * (my - panY);
337
+ zoom *= factor;
338
+ draw();
339
+ }
340
+
341
+ function onMouseDown(e) {
342
+ dragging = true;
343
+ lastMouse = { x: e.offsetX, y: e.offsetY };
344
+ canvas.style.cursor = 'grabbing';
345
+ }
346
+
347
+ function onMouseUp() {
348
+ dragging = false;
349
+ lastMouse = null;
350
+ canvas.style.cursor = 'default';
351
+ }
352
+
353
+ function onMouseLeave() {
354
+ var tooltip = document.getElementById('emb-tooltip');
355
+ if (tooltip) tooltip.style.display = 'none';
356
+ dragging = false;
357
+ lastMouse = null;
358
+ }
359
+
360
+ // ── Public API ────────────────────────────────────────────────────────
361
+
362
+ window.resetEmbeddingsView = function () {
363
+ panX = 0; panY = 0; zoom = 1;
364
+ draw();
365
+ };
366
+
367
+ window.toggleNormColor = function () {
368
+ colorByNorm = !colorByNorm;
369
+ var btn = document.getElementById('emb-norm-btn');
370
+ if (btn) {
371
+ if (colorByNorm) btn.classList.add('active');
372
+ else btn.classList.remove('active');
373
+ }
374
+ draw();
375
+ };
376
+
377
+ // Hook into showSection for lazy-load
378
+ var origShowSection = window.showSection;
379
+ window.showSection = function (name) {
380
+ if (origShowSection) origShowSection(name);
381
+ if (name === 'embeddings' && !embData) {
382
+ setTimeout(init, 50);
383
+ }
384
+ };
385
+
386
+ window.computeEmbeddings = async function computeEmbeddings() {
387
+ if (!canvas) init();
388
+ var statusEl = document.getElementById('emb-status');
389
+ if (statusEl) statusEl.textContent = 'Computing embeddings (model download on first run)…';
390
+ try {
391
+ var res = await adminAuth.adminFetch('/api/embeddings/compute', { method: 'POST', headers: { 'Content-Type': 'application/json' } });
392
+ if (!res.ok) {
393
+ var err = await res.json().catch(function () { return {}; });
394
+ if (statusEl) statusEl.textContent = 'Error: ' + (err.error || res.statusText) + (err.hint ? ' — ' + err.hint : '');
395
+ return;
396
+ }
397
+ var result = await res.json();
398
+ if (statusEl) statusEl.textContent = 'Computed ' + result.count + ' embeddings (' + result.model + ', ' + result.elapsedMs + 'ms). Loading visualization…';
399
+ // Auto-load the projection after compute
400
+ await window.loadEmbeddings();
401
+ } catch (e) {
402
+ if (statusEl) statusEl.textContent = 'Compute failed: ' + e.message;
403
+ }
404
+ };
405
+
406
+ // ── Event delegation for CSP-safe buttons ────────────────────────────
407
+ var embSection = document.getElementById('embeddings-section');
408
+ if (embSection) {
409
+ embSection.addEventListener('click', function (e) {
410
+ var btn = e.target.closest('[data-emb-action]');
411
+ if (!btn) return;
412
+ var action = btn.getAttribute('data-emb-action');
413
+ if (action === 'compute') window.computeEmbeddings();
414
+ else if (action === 'load') window.loadEmbeddings();
415
+ else if (action === 'reset') window.resetEmbeddingsView();
416
+ else if (action === 'norm') window.toggleNormColor();
417
+ });
418
+ }
419
+
420
+ document.addEventListener('DOMContentLoaded', function () {
421
+ var sec = document.getElementById('embeddings-section');
422
+ if (sec && !sec.classList.contains('hidden')) {
423
+ init();
424
+ }
425
+ });
426
+ })();