@kaelio/ktx 0.5.0 → 0.7.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 (875) hide show
  1. package/assets/python/{kaelio_ktx-0.5.0-py3-none-any.whl → kaelio_ktx-0.7.0-py3-none-any.whl} +0 -0
  2. package/assets/python/manifest.json +4 -4
  3. package/dist/.tsbuildinfo +1 -1
  4. package/dist/clack.d.ts +8 -0
  5. package/dist/clack.js +14 -0
  6. package/dist/connection.js +2 -9
  7. package/dist/connectors/bigquery/connector.d.ts +6 -1
  8. package/dist/connectors/bigquery/connector.js +38 -9
  9. package/dist/connectors/bigquery/dialect.d.ts +11 -9
  10. package/dist/connectors/bigquery/dialect.js +25 -45
  11. package/dist/connectors/clickhouse/connector.d.ts +5 -0
  12. package/dist/connectors/clickhouse/connector.js +36 -3
  13. package/dist/connectors/clickhouse/dialect.d.ts +11 -12
  14. package/dist/connectors/clickhouse/dialect.js +25 -100
  15. package/dist/connectors/mysql/connector.d.ts +7 -1
  16. package/dist/connectors/mysql/connector.js +67 -9
  17. package/dist/connectors/mysql/dialect.d.ts +11 -9
  18. package/dist/connectors/mysql/dialect.js +25 -43
  19. package/dist/connectors/postgres/connector.d.ts +6 -0
  20. package/dist/connectors/postgres/connector.js +67 -12
  21. package/dist/connectors/postgres/dialect.d.ts +11 -9
  22. package/dist/connectors/postgres/dialect.js +26 -35
  23. package/dist/connectors/snowflake/connector.d.ts +8 -3
  24. package/dist/connectors/snowflake/connector.js +39 -20
  25. package/dist/connectors/snowflake/dialect.d.ts +11 -9
  26. package/dist/connectors/snowflake/dialect.js +25 -24
  27. package/dist/connectors/sqlite/connector.d.ts +3 -1
  28. package/dist/connectors/sqlite/connector.js +23 -3
  29. package/dist/connectors/sqlite/dialect.d.ts +11 -9
  30. package/dist/connectors/sqlite/dialect.js +25 -29
  31. package/dist/connectors/sqlserver/connector.d.ts +6 -0
  32. package/dist/connectors/sqlserver/connector.js +56 -9
  33. package/dist/connectors/sqlserver/dialect.d.ts +11 -10
  34. package/dist/connectors/sqlserver/dialect.js +24 -40
  35. package/dist/context/connections/connection-type.d.ts +1 -1
  36. package/dist/context/connections/dialect-helpers.d.ts +9 -0
  37. package/dist/context/connections/dialect-helpers.js +67 -0
  38. package/dist/context/connections/dialects.d.ts +23 -5
  39. package/dist/context/connections/dialects.js +18 -56
  40. package/dist/context/connections/drivers.d.ts +23 -0
  41. package/dist/context/connections/drivers.js +171 -0
  42. package/dist/context/connections/local-query-executor.js +25 -7
  43. package/dist/context/connections/local-warehouse-descriptor.js +0 -2
  44. package/dist/context/connections/postgres-query-executor.js +1 -1
  45. package/dist/context/connections/sqlite-query-executor.js +1 -1
  46. package/dist/context/ingest/adapters/historic-sql/chunk-unified.js +1 -1
  47. package/dist/context/ingest/adapters/historic-sql/connection-dialect.js +11 -7
  48. package/dist/context/ingest/adapters/historic-sql/evidence-tool.d.ts +1 -1
  49. package/dist/context/ingest/adapters/historic-sql/evidence-tool.js +8 -5
  50. package/dist/context/ingest/adapters/historic-sql/evidence.d.ts +4 -4
  51. package/dist/context/ingest/adapters/historic-sql/evidence.js +2 -2
  52. package/dist/context/ingest/adapters/historic-sql/projection.js +5 -2
  53. package/dist/context/ingest/adapters/live-database/daemon-introspection.js +1 -1
  54. package/dist/context/ingest/adapters/live-database/stage.d.ts +2 -0
  55. package/dist/context/ingest/adapters/live-database/stage.js +9 -0
  56. package/dist/context/ingest/adapters/looker/mapping.d.ts +0 -3
  57. package/dist/context/ingest/adapters/looker/mapping.js +0 -3
  58. package/dist/context/ingest/adapters/looker/types.d.ts +1 -1
  59. package/dist/context/ingest/historic-sql-probes/bigquery-runner.d.ts +34 -0
  60. package/dist/context/ingest/historic-sql-probes/bigquery-runner.js +99 -0
  61. package/dist/context/ingest/historic-sql-probes/postgres-runner.d.ts +26 -0
  62. package/dist/context/ingest/historic-sql-probes/postgres-runner.js +76 -0
  63. package/dist/context/ingest/historic-sql-probes/snowflake-runner.d.ts +29 -0
  64. package/dist/context/ingest/historic-sql-probes/snowflake-runner.js +62 -0
  65. package/dist/context/ingest/historic-sql-probes.d.ts +46 -0
  66. package/dist/context/ingest/historic-sql-probes.js +62 -0
  67. package/dist/context/ingest/local-adapters.js +0 -1
  68. package/dist/context/ingest/local-ingest.js +1 -1
  69. package/dist/context/llm/claude-code-runtime.js +16 -1
  70. package/dist/context/mcp/context-tools.js +11 -48
  71. package/dist/context/mcp/local-project-ports.js +0 -3
  72. package/dist/context/project/config.d.ts +0 -8
  73. package/dist/context/project/driver-schemas.d.ts +0 -4
  74. package/dist/context/project/driver-schemas.js +0 -2
  75. package/dist/context/scan/constraint-discovery.d.ts +19 -0
  76. package/dist/context/scan/constraint-discovery.js +23 -0
  77. package/dist/context/scan/enabled-tables.d.ts +4 -5
  78. package/dist/context/scan/enabled-tables.js +4 -18
  79. package/dist/context/scan/entity-details.js +14 -44
  80. package/dist/context/scan/local-enrichment.js +13 -1
  81. package/dist/context/scan/local-scan.js +5 -4
  82. package/dist/context/scan/local-structural-artifacts.js +51 -0
  83. package/dist/context/scan/relationship-benchmarks.js +9 -6
  84. package/dist/context/scan/relationship-composite-candidates.d.ts +3 -2
  85. package/dist/context/scan/relationship-composite-candidates.js +21 -33
  86. package/dist/context/scan/relationship-discovery.d.ts +3 -2
  87. package/dist/context/scan/relationship-discovery.js +4 -4
  88. package/dist/context/scan/relationship-profiling.d.ts +2 -3
  89. package/dist/context/scan/relationship-profiling.js +25 -94
  90. package/dist/context/scan/relationship-validation.d.ts +3 -2
  91. package/dist/context/scan/relationship-validation.js +12 -22
  92. package/dist/context/scan/table-ref.d.ts +1 -2
  93. package/dist/context/scan/table-ref.js +3 -4
  94. package/dist/context/scan/types.d.ts +6 -2
  95. package/dist/context/scan/warehouse-catalog.js +31 -48
  96. package/dist/context/sl/local-query.js +0 -3
  97. package/dist/context/sl/local-sl.js +0 -13
  98. package/dist/context/sl/semantic-layer.service.js +1 -4
  99. package/dist/context/tools/context-candidate-write.tool.d.ts +2 -2
  100. package/dist/context-build-view.js +1 -1
  101. package/dist/database-tree-picker.js +14 -7
  102. package/dist/error-message.d.ts +1 -0
  103. package/dist/error-message.js +29 -0
  104. package/dist/ingest-depth.js +0 -1
  105. package/dist/ingest.js +2 -2
  106. package/dist/llm/embedding-health.js +2 -2
  107. package/dist/local-scan-connectors.js +13 -56
  108. package/dist/managed-local-embeddings.js +2 -1
  109. package/dist/managed-python-daemon.d.ts +5 -0
  110. package/dist/managed-python-daemon.js +29 -9
  111. package/dist/managed-python-http.js +2 -1
  112. package/dist/public-ingest.js +1 -6
  113. package/dist/runtime-requirements.js +2 -2
  114. package/dist/setup-agents.d.ts +1 -1
  115. package/dist/setup-agents.js +16 -74
  116. package/dist/setup-context.js +2 -1
  117. package/dist/setup-databases.d.ts +3 -13
  118. package/dist/setup-databases.js +141 -313
  119. package/dist/setup-embeddings.js +10 -2
  120. package/dist/setup-project.d.ts +0 -8
  121. package/dist/setup-project.js +3 -27
  122. package/dist/setup-runtime.js +2 -1
  123. package/dist/setup-sources.js +2 -1
  124. package/dist/setup.js +11 -18
  125. package/dist/skills/historic_sql_patterns/SKILL.md +1 -3
  126. package/dist/skills/historic_sql_table_digest/SKILL.md +0 -1
  127. package/dist/skills/sl/SKILL.md +2 -2
  128. package/dist/sql.js +0 -4
  129. package/dist/status-project.d.ts +3 -18
  130. package/dist/status-project.js +42 -216
  131. package/dist/telemetry/events.d.ts +1 -1
  132. package/dist/telemetry/index.js +8 -3
  133. package/dist/tree-picker-state.d.ts +2 -2
  134. package/dist/tree-picker-state.js +29 -13
  135. package/dist/tree-picker-tui.d.ts +3 -1
  136. package/dist/tree-picker-tui.js +20 -32
  137. package/package.json +6 -6
  138. package/dist/admin-reindex.test.d.ts +0 -1
  139. package/dist/admin-reindex.test.js +0 -119
  140. package/dist/admin.test.d.ts +0 -1
  141. package/dist/admin.test.js +0 -201
  142. package/dist/cli-program-telemetry.test.d.ts +0 -1
  143. package/dist/cli-program-telemetry.test.js +0 -89
  144. package/dist/cli-program.test.d.ts +0 -1
  145. package/dist/cli-program.test.js +0 -71
  146. package/dist/command-tree.test.d.ts +0 -1
  147. package/dist/command-tree.test.js +0 -126
  148. package/dist/commands/mcp-commands.test.d.ts +0 -1
  149. package/dist/commands/mcp-commands.test.js +0 -111
  150. package/dist/commands/sql-commands.test.d.ts +0 -1
  151. package/dist/commands/sql-commands.test.js +0 -68
  152. package/dist/connection.test.d.ts +0 -1
  153. package/dist/connection.test.js +0 -426
  154. package/dist/connectors/bigquery/connector.test.d.ts +0 -1
  155. package/dist/connectors/bigquery/connector.test.js +0 -363
  156. package/dist/connectors/bigquery/dialect.test.d.ts +0 -1
  157. package/dist/connectors/bigquery/dialect.test.js +0 -36
  158. package/dist/connectors/clickhouse/connector.test.d.ts +0 -1
  159. package/dist/connectors/clickhouse/connector.test.js +0 -342
  160. package/dist/connectors/clickhouse/dialect.test.d.ts +0 -1
  161. package/dist/connectors/clickhouse/dialect.test.js +0 -36
  162. package/dist/connectors/mysql/connector.test.d.ts +0 -1
  163. package/dist/connectors/mysql/connector.test.js +0 -365
  164. package/dist/connectors/mysql/dialect.test.d.ts +0 -1
  165. package/dist/connectors/mysql/dialect.test.js +0 -36
  166. package/dist/connectors/postgres/connector.test.d.ts +0 -1
  167. package/dist/connectors/postgres/connector.test.js +0 -391
  168. package/dist/connectors/postgres/dialect.test.d.ts +0 -1
  169. package/dist/connectors/postgres/dialect.test.js +0 -37
  170. package/dist/connectors/postgres/historic-sql-query-client.test.d.ts +0 -1
  171. package/dist/connectors/postgres/historic-sql-query-client.test.js +0 -45
  172. package/dist/connectors/snowflake/connector.test.d.ts +0 -1
  173. package/dist/connectors/snowflake/connector.test.js +0 -462
  174. package/dist/connectors/snowflake/dialect.test.d.ts +0 -1
  175. package/dist/connectors/snowflake/dialect.test.js +0 -34
  176. package/dist/connectors/snowflake/identifiers.test.d.ts +0 -1
  177. package/dist/connectors/snowflake/identifiers.test.js +0 -12
  178. package/dist/connectors/snowflake/sdk-logger.test.d.ts +0 -1
  179. package/dist/connectors/snowflake/sdk-logger.test.js +0 -47
  180. package/dist/connectors/sqlite/connector.test.d.ts +0 -1
  181. package/dist/connectors/sqlite/connector.test.js +0 -207
  182. package/dist/connectors/sqlite/dialect.test.d.ts +0 -1
  183. package/dist/connectors/sqlite/dialect.test.js +0 -23
  184. package/dist/connectors/sqlserver/connector.test.d.ts +0 -1
  185. package/dist/connectors/sqlserver/connector.test.js +0 -313
  186. package/dist/connectors/sqlserver/dialect.test.d.ts +0 -1
  187. package/dist/connectors/sqlserver/dialect.test.js +0 -36
  188. package/dist/context/connections/bigquery-identifiers.test.d.ts +0 -1
  189. package/dist/context/connections/bigquery-identifiers.test.js +0 -13
  190. package/dist/context/connections/dialects.test.d.ts +0 -1
  191. package/dist/context/connections/dialects.test.js +0 -24
  192. package/dist/context/connections/local-query-executor.test.d.ts +0 -1
  193. package/dist/context/connections/local-query-executor.test.js +0 -48
  194. package/dist/context/connections/local-warehouse-descriptor.test.d.ts +0 -1
  195. package/dist/context/connections/local-warehouse-descriptor.test.js +0 -53
  196. package/dist/context/connections/notion-config.test.d.ts +0 -1
  197. package/dist/context/connections/notion-config.test.js +0 -121
  198. package/dist/context/connections/postgres-query-executor.test.d.ts +0 -1
  199. package/dist/context/connections/postgres-query-executor.test.js +0 -91
  200. package/dist/context/connections/read-only-sql.test.d.ts +0 -1
  201. package/dist/context/connections/read-only-sql.test.js +0 -20
  202. package/dist/context/connections/sqlite-query-executor.test.d.ts +0 -1
  203. package/dist/context/connections/sqlite-query-executor.test.js +0 -113
  204. package/dist/context/core/config-reference.test.d.ts +0 -1
  205. package/dist/context/core/config-reference.test.js +0 -27
  206. package/dist/context/core/git.service.assert-worktree-clean.test.d.ts +0 -1
  207. package/dist/context/core/git.service.assert-worktree-clean.test.js +0 -62
  208. package/dist/context/core/git.service.delete-directories.test.d.ts +0 -1
  209. package/dist/context/core/git.service.delete-directories.test.js +0 -61
  210. package/dist/context/core/git.service.patch.test.d.ts +0 -1
  211. package/dist/context/core/git.service.patch.test.js +0 -40
  212. package/dist/context/core/git.service.reset-hard.test.d.ts +0 -1
  213. package/dist/context/core/git.service.reset-hard.test.js +0 -47
  214. package/dist/context/core/git.service.test.d.ts +0 -1
  215. package/dist/context/core/git.service.test.js +0 -357
  216. package/dist/context/core/session-worktree.service.test.d.ts +0 -1
  217. package/dist/context/core/session-worktree.service.test.js +0 -97
  218. package/dist/context/daemon/semantic-layer-compute.test.d.ts +0 -1
  219. package/dist/context/daemon/semantic-layer-compute.test.js +0 -305
  220. package/dist/context/index-sync/reindex.test.d.ts +0 -1
  221. package/dist/context/index-sync/reindex.test.js +0 -139
  222. package/dist/context/ingest/action-identity.test.d.ts +0 -1
  223. package/dist/context/ingest/action-identity.test.js +0 -19
  224. package/dist/context/ingest/adapters/dbt/chunk.test.d.ts +0 -1
  225. package/dist/context/ingest/adapters/dbt/chunk.test.js +0 -30
  226. package/dist/context/ingest/adapters/dbt/dbt.adapter.test.d.ts +0 -1
  227. package/dist/context/ingest/adapters/dbt/dbt.adapter.test.js +0 -43
  228. package/dist/context/ingest/adapters/dbt/fetch.test.d.ts +0 -1
  229. package/dist/context/ingest/adapters/dbt/fetch.test.js +0 -30
  230. package/dist/context/ingest/adapters/dbt/parse.test.d.ts +0 -1
  231. package/dist/context/ingest/adapters/dbt/parse.test.js +0 -7
  232. package/dist/context/ingest/adapters/dbt-descriptions/parse-schema.test.d.ts +0 -1
  233. package/dist/context/ingest/adapters/dbt-descriptions/parse-schema.test.js +0 -195
  234. package/dist/context/ingest/adapters/historic-sql/bigquery-query-history-reader.test.d.ts +0 -1
  235. package/dist/context/ingest/adapters/historic-sql/bigquery-query-history-reader.test.js +0 -121
  236. package/dist/context/ingest/adapters/historic-sql/buckets.test.d.ts +0 -1
  237. package/dist/context/ingest/adapters/historic-sql/buckets.test.js +0 -49
  238. package/dist/context/ingest/adapters/historic-sql/chunk-unified.test.d.ts +0 -1
  239. package/dist/context/ingest/adapters/historic-sql/chunk-unified.test.js +0 -160
  240. package/dist/context/ingest/adapters/historic-sql/detect.test.d.ts +0 -1
  241. package/dist/context/ingest/adapters/historic-sql/detect.test.js +0 -48
  242. package/dist/context/ingest/adapters/historic-sql/evidence-tool.test.d.ts +0 -1
  243. package/dist/context/ingest/adapters/historic-sql/evidence-tool.test.js +0 -67
  244. package/dist/context/ingest/adapters/historic-sql/evidence.test.d.ts +0 -1
  245. package/dist/context/ingest/adapters/historic-sql/evidence.test.js +0 -43
  246. package/dist/context/ingest/adapters/historic-sql/historic-sql.adapter.test.d.ts +0 -1
  247. package/dist/context/ingest/adapters/historic-sql/historic-sql.adapter.test.js +0 -98
  248. package/dist/context/ingest/adapters/historic-sql/local-ingest-acceptance.test.d.ts +0 -1
  249. package/dist/context/ingest/adapters/historic-sql/local-ingest-acceptance.test.js +0 -235
  250. package/dist/context/ingest/adapters/historic-sql/pattern-inputs.test.d.ts +0 -1
  251. package/dist/context/ingest/adapters/historic-sql/pattern-inputs.test.js +0 -68
  252. package/dist/context/ingest/adapters/historic-sql/postgres-pgss-reader.test.d.ts +0 -1
  253. package/dist/context/ingest/adapters/historic-sql/postgres-pgss-reader.test.js +0 -205
  254. package/dist/context/ingest/adapters/historic-sql/projection.test.d.ts +0 -1
  255. package/dist/context/ingest/adapters/historic-sql/projection.test.js +0 -392
  256. package/dist/context/ingest/adapters/historic-sql/redaction.test.d.ts +0 -1
  257. package/dist/context/ingest/adapters/historic-sql/redaction.test.js +0 -22
  258. package/dist/context/ingest/adapters/historic-sql/skill-schemas.test.d.ts +0 -1
  259. package/dist/context/ingest/adapters/historic-sql/skill-schemas.test.js +0 -62
  260. package/dist/context/ingest/adapters/historic-sql/snowflake-query-history-reader.test.d.ts +0 -1
  261. package/dist/context/ingest/adapters/historic-sql/snowflake-query-history-reader.test.js +0 -117
  262. package/dist/context/ingest/adapters/historic-sql/stage-unified.test.d.ts +0 -1
  263. package/dist/context/ingest/adapters/historic-sql/stage-unified.test.js +0 -405
  264. package/dist/context/ingest/adapters/historic-sql/types.test.d.ts +0 -1
  265. package/dist/context/ingest/adapters/historic-sql/types.test.js +0 -87
  266. package/dist/context/ingest/adapters/live-database/chunk.test.d.ts +0 -1
  267. package/dist/context/ingest/adapters/live-database/chunk.test.js +0 -95
  268. package/dist/context/ingest/adapters/live-database/daemon-introspection.test.d.ts +0 -1
  269. package/dist/context/ingest/adapters/live-database/daemon-introspection.test.js +0 -241
  270. package/dist/context/ingest/adapters/live-database/live-database.adapter.test.d.ts +0 -1
  271. package/dist/context/ingest/adapters/live-database/live-database.adapter.test.js +0 -105
  272. package/dist/context/ingest/adapters/live-database/manifest.test.d.ts +0 -1
  273. package/dist/context/ingest/adapters/live-database/manifest.test.js +0 -291
  274. package/dist/context/ingest/adapters/live-database/stage.test.d.ts +0 -1
  275. package/dist/context/ingest/adapters/live-database/stage.test.js +0 -133
  276. package/dist/context/ingest/adapters/looker/chunk.test.d.ts +0 -1
  277. package/dist/context/ingest/adapters/looker/chunk.test.js +0 -142
  278. package/dist/context/ingest/adapters/looker/client-boundary.test.d.ts +0 -1
  279. package/dist/context/ingest/adapters/looker/client-boundary.test.js +0 -12
  280. package/dist/context/ingest/adapters/looker/client.test.d.ts +0 -1
  281. package/dist/context/ingest/adapters/looker/client.test.js +0 -407
  282. package/dist/context/ingest/adapters/looker/daemon-table-identifier-parser.test.d.ts +0 -1
  283. package/dist/context/ingest/adapters/looker/daemon-table-identifier-parser.test.js +0 -40
  284. package/dist/context/ingest/adapters/looker/detect.test.d.ts +0 -1
  285. package/dist/context/ingest/adapters/looker/detect.test.js +0 -39
  286. package/dist/context/ingest/adapters/looker/evidence-documents.test.d.ts +0 -1
  287. package/dist/context/ingest/adapters/looker/evidence-documents.test.js +0 -178
  288. package/dist/context/ingest/adapters/looker/factory.test.d.ts +0 -1
  289. package/dist/context/ingest/adapters/looker/factory.test.js +0 -55
  290. package/dist/context/ingest/adapters/looker/fetch-report.test.d.ts +0 -1
  291. package/dist/context/ingest/adapters/looker/fetch-report.test.js +0 -71
  292. package/dist/context/ingest/adapters/looker/fetch.test.d.ts +0 -1
  293. package/dist/context/ingest/adapters/looker/fetch.test.js +0 -592
  294. package/dist/context/ingest/adapters/looker/local-runtime-store.test.d.ts +0 -1
  295. package/dist/context/ingest/adapters/looker/local-runtime-store.test.js +0 -106
  296. package/dist/context/ingest/adapters/looker/looker.adapter.test.d.ts +0 -1
  297. package/dist/context/ingest/adapters/looker/looker.adapter.test.js +0 -99
  298. package/dist/context/ingest/adapters/looker/mapping.test.d.ts +0 -1
  299. package/dist/context/ingest/adapters/looker/mapping.test.js +0 -334
  300. package/dist/context/ingest/adapters/looker/reconcile.test.d.ts +0 -1
  301. package/dist/context/ingest/adapters/looker/reconcile.test.js +0 -12
  302. package/dist/context/ingest/adapters/looker/scope.test.d.ts +0 -1
  303. package/dist/context/ingest/adapters/looker/scope.test.js +0 -84
  304. package/dist/context/ingest/adapters/looker/target-connections.test.d.ts +0 -1
  305. package/dist/context/ingest/adapters/looker/target-connections.test.js +0 -71
  306. package/dist/context/ingest/adapters/looker/tools/looker-query-to-sl.tool.test.d.ts +0 -1
  307. package/dist/context/ingest/adapters/looker/tools/looker-query-to-sl.tool.test.js +0 -211
  308. package/dist/context/ingest/adapters/looker/types.test.d.ts +0 -1
  309. package/dist/context/ingest/adapters/looker/types.test.js +0 -261
  310. package/dist/context/ingest/adapters/lookml/chunk.test.d.ts +0 -1
  311. package/dist/context/ingest/adapters/lookml/chunk.test.js +0 -213
  312. package/dist/context/ingest/adapters/lookml/detect.test.d.ts +0 -1
  313. package/dist/context/ingest/adapters/lookml/detect.test.js +0 -37
  314. package/dist/context/ingest/adapters/lookml/fetch-report.test.d.ts +0 -1
  315. package/dist/context/ingest/adapters/lookml/fetch-report.test.js +0 -82
  316. package/dist/context/ingest/adapters/lookml/fetch.test.d.ts +0 -1
  317. package/dist/context/ingest/adapters/lookml/fetch.test.js +0 -121
  318. package/dist/context/ingest/adapters/lookml/graph.test.d.ts +0 -1
  319. package/dist/context/ingest/adapters/lookml/graph.test.js +0 -105
  320. package/dist/context/ingest/adapters/lookml/lookml.adapter.test.d.ts +0 -1
  321. package/dist/context/ingest/adapters/lookml/lookml.adapter.test.js +0 -49
  322. package/dist/context/ingest/adapters/lookml/parse.test.d.ts +0 -1
  323. package/dist/context/ingest/adapters/lookml/parse.test.js +0 -118
  324. package/dist/context/ingest/adapters/lookml/pull-config.test.d.ts +0 -1
  325. package/dist/context/ingest/adapters/lookml/pull-config.test.js +0 -128
  326. package/dist/context/ingest/adapters/metabase/card-references.test.d.ts +0 -1
  327. package/dist/context/ingest/adapters/metabase/card-references.test.js +0 -36
  328. package/dist/context/ingest/adapters/metabase/chunk.test.d.ts +0 -1
  329. package/dist/context/ingest/adapters/metabase/chunk.test.js +0 -299
  330. package/dist/context/ingest/adapters/metabase/client-boundary.test.d.ts +0 -1
  331. package/dist/context/ingest/adapters/metabase/client-boundary.test.js +0 -38
  332. package/dist/context/ingest/adapters/metabase/client-port.test.d.ts +0 -1
  333. package/dist/context/ingest/adapters/metabase/client-port.test.js +0 -86
  334. package/dist/context/ingest/adapters/metabase/client.test.d.ts +0 -1
  335. package/dist/context/ingest/adapters/metabase/client.test.js +0 -377
  336. package/dist/context/ingest/adapters/metabase/detect.test.d.ts +0 -1
  337. package/dist/context/ingest/adapters/metabase/detect.test.js +0 -42
  338. package/dist/context/ingest/adapters/metabase/fanout-planner.test.d.ts +0 -1
  339. package/dist/context/ingest/adapters/metabase/fanout-planner.test.js +0 -44
  340. package/dist/context/ingest/adapters/metabase/fetch-scope.test.d.ts +0 -1
  341. package/dist/context/ingest/adapters/metabase/fetch-scope.test.js +0 -124
  342. package/dist/context/ingest/adapters/metabase/fetch.test.d.ts +0 -1
  343. package/dist/context/ingest/adapters/metabase/fetch.test.js +0 -557
  344. package/dist/context/ingest/adapters/metabase/local-metabase.adapter.test.d.ts +0 -1
  345. package/dist/context/ingest/adapters/metabase/local-metabase.adapter.test.js +0 -56
  346. package/dist/context/ingest/adapters/metabase/local-source-state-store.test.d.ts +0 -1
  347. package/dist/context/ingest/adapters/metabase/local-source-state-store.test.js +0 -99
  348. package/dist/context/ingest/adapters/metabase/mapping.test.d.ts +0 -1
  349. package/dist/context/ingest/adapters/metabase/mapping.test.js +0 -215
  350. package/dist/context/ingest/adapters/metabase/metabase.adapter.test.d.ts +0 -1
  351. package/dist/context/ingest/adapters/metabase/metabase.adapter.test.js +0 -129
  352. package/dist/context/ingest/adapters/metabase/serialize-card.test.d.ts +0 -1
  353. package/dist/context/ingest/adapters/metabase/serialize-card.test.js +0 -205
  354. package/dist/context/ingest/adapters/metabase/types.test.d.ts +0 -1
  355. package/dist/context/ingest/adapters/metabase/types.test.js +0 -75
  356. package/dist/context/ingest/adapters/metricflow/chunk.test.d.ts +0 -1
  357. package/dist/context/ingest/adapters/metricflow/chunk.test.js +0 -114
  358. package/dist/context/ingest/adapters/metricflow/deep-parse.test.d.ts +0 -1
  359. package/dist/context/ingest/adapters/metricflow/deep-parse.test.js +0 -1139
  360. package/dist/context/ingest/adapters/metricflow/detect.test.d.ts +0 -1
  361. package/dist/context/ingest/adapters/metricflow/detect.test.js +0 -43
  362. package/dist/context/ingest/adapters/metricflow/fetch.test.d.ts +0 -1
  363. package/dist/context/ingest/adapters/metricflow/fetch.test.js +0 -97
  364. package/dist/context/ingest/adapters/metricflow/graph.test.d.ts +0 -1
  365. package/dist/context/ingest/adapters/metricflow/graph.test.js +0 -245
  366. package/dist/context/ingest/adapters/metricflow/import-semantic-models.test.d.ts +0 -1
  367. package/dist/context/ingest/adapters/metricflow/import-semantic-models.test.js +0 -318
  368. package/dist/context/ingest/adapters/metricflow/metricflow.adapter.test.d.ts +0 -1
  369. package/dist/context/ingest/adapters/metricflow/metricflow.adapter.test.js +0 -212
  370. package/dist/context/ingest/adapters/metricflow/parse.test.d.ts +0 -1
  371. package/dist/context/ingest/adapters/metricflow/parse.test.js +0 -171
  372. package/dist/context/ingest/adapters/metricflow/pull-config.test.d.ts +0 -1
  373. package/dist/context/ingest/adapters/metricflow/pull-config.test.js +0 -57
  374. package/dist/context/ingest/adapters/metricflow/semantic-models.test.d.ts +0 -1
  375. package/dist/context/ingest/adapters/metricflow/semantic-models.test.js +0 -204
  376. package/dist/context/ingest/adapters/notion/cluster.test.d.ts +0 -1
  377. package/dist/context/ingest/adapters/notion/cluster.test.js +0 -123
  378. package/dist/context/ingest/adapters/notion/fetch.test.d.ts +0 -1
  379. package/dist/context/ingest/adapters/notion/fetch.test.js +0 -358
  380. package/dist/context/ingest/adapters/notion/local-state-store.test.d.ts +0 -1
  381. package/dist/context/ingest/adapters/notion/local-state-store.test.js +0 -29
  382. package/dist/context/ingest/adapters/notion/normalize.test.d.ts +0 -1
  383. package/dist/context/ingest/adapters/notion/normalize.test.js +0 -64
  384. package/dist/context/ingest/adapters/notion/notion-client.test.d.ts +0 -1
  385. package/dist/context/ingest/adapters/notion/notion-client.test.js +0 -49
  386. package/dist/context/ingest/adapters/notion/notion.adapter.test.d.ts +0 -1
  387. package/dist/context/ingest/adapters/notion/notion.adapter.test.js +0 -315
  388. package/dist/context/ingest/artifact-gates.test.d.ts +0 -1
  389. package/dist/context/ingest/artifact-gates.test.js +0 -167
  390. package/dist/context/ingest/canonical-pins.test.d.ts +0 -1
  391. package/dist/context/ingest/canonical-pins.test.js +0 -66
  392. package/dist/context/ingest/clustering/kmeans.test.d.ts +0 -1
  393. package/dist/context/ingest/clustering/kmeans.test.js +0 -61
  394. package/dist/context/ingest/context-candidates/candidate-dedup.service.test.d.ts +0 -1
  395. package/dist/context/ingest/context-candidates/candidate-dedup.service.test.js +0 -216
  396. package/dist/context/ingest/context-candidates/context-candidate-carryforward.service.test.d.ts +0 -1
  397. package/dist/context/ingest/context-candidates/context-candidate-carryforward.service.test.js +0 -161
  398. package/dist/context/ingest/context-candidates/curator-pagination.service.test.d.ts +0 -1
  399. package/dist/context/ingest/context-candidates/curator-pagination.service.test.js +0 -168
  400. package/dist/context/ingest/context-candidates/embedding-text.test.d.ts +0 -1
  401. package/dist/context/ingest/context-candidates/embedding-text.test.js +0 -10
  402. package/dist/context/ingest/context-candidates/store.test.d.ts +0 -1
  403. package/dist/context/ingest/context-candidates/store.test.js +0 -67
  404. package/dist/context/ingest/context-evidence/context-evidence-index.service.test.d.ts +0 -1
  405. package/dist/context/ingest/context-evidence/context-evidence-index.service.test.js +0 -374
  406. package/dist/context/ingest/context-evidence/sqlite-context-evidence-store.test.d.ts +0 -1
  407. package/dist/context/ingest/context-evidence/sqlite-context-evidence-store.test.js +0 -416
  408. package/dist/context/ingest/context-evidence/store.test.d.ts +0 -1
  409. package/dist/context/ingest/context-evidence/store.test.js +0 -55
  410. package/dist/context/ingest/dbt-shared/project-vars.test.d.ts +0 -1
  411. package/dist/context/ingest/dbt-shared/project-vars.test.js +0 -90
  412. package/dist/context/ingest/dbt-shared/schema-files.test.d.ts +0 -1
  413. package/dist/context/ingest/dbt-shared/schema-files.test.js +0 -35
  414. package/dist/context/ingest/diff-set.service.test.d.ts +0 -1
  415. package/dist/context/ingest/diff-set.service.test.js +0 -132
  416. package/dist/context/ingest/final-gate-repair.test.d.ts +0 -1
  417. package/dist/context/ingest/final-gate-repair.test.js +0 -109
  418. package/dist/context/ingest/finalization-scope.test.d.ts +0 -1
  419. package/dist/context/ingest/finalization-scope.test.js +0 -114
  420. package/dist/context/ingest/ingest-bundle.runner.isolated-diff.test.d.ts +0 -1
  421. package/dist/context/ingest/ingest-bundle.runner.isolated-diff.test.js +0 -1928
  422. package/dist/context/ingest/ingest-bundle.runner.test.d.ts +0 -1
  423. package/dist/context/ingest/ingest-bundle.runner.test.js +0 -1899
  424. package/dist/context/ingest/ingest-prompts.test.d.ts +0 -1
  425. package/dist/context/ingest/ingest-prompts.test.js +0 -32
  426. package/dist/context/ingest/ingest-runtime-assets.test.d.ts +0 -1
  427. package/dist/context/ingest/ingest-runtime-assets.test.js +0 -89
  428. package/dist/context/ingest/ingest-trace.test.d.ts +0 -1
  429. package/dist/context/ingest/ingest-trace.test.js +0 -76
  430. package/dist/context/ingest/isolated-diff/git-patch.test.d.ts +0 -1
  431. package/dist/context/ingest/isolated-diff/git-patch.test.js +0 -76
  432. package/dist/context/ingest/isolated-diff/patch-integrator.test.d.ts +0 -1
  433. package/dist/context/ingest/isolated-diff/patch-integrator.test.js +0 -369
  434. package/dist/context/ingest/isolated-diff/textual-conflict-resolver.test.d.ts +0 -1
  435. package/dist/context/ingest/isolated-diff/textual-conflict-resolver.test.js +0 -101
  436. package/dist/context/ingest/isolated-diff/work-unit-executor.test.d.ts +0 -1
  437. package/dist/context/ingest/isolated-diff/work-unit-executor.test.js +0 -137
  438. package/dist/context/ingest/local-adapters.test.d.ts +0 -1
  439. package/dist/context/ingest/local-adapters.test.js +0 -612
  440. package/dist/context/ingest/local-bundle-ingest.test.d.ts +0 -1
  441. package/dist/context/ingest/local-bundle-ingest.test.js +0 -794
  442. package/dist/context/ingest/local-bundle-runtime.test.d.ts +0 -1
  443. package/dist/context/ingest/local-bundle-runtime.test.js +0 -240
  444. package/dist/context/ingest/local-embedding-provider.integration.test.d.ts +0 -1
  445. package/dist/context/ingest/local-embedding-provider.integration.test.js +0 -139
  446. package/dist/context/ingest/local-mapping-reconcile.test.d.ts +0 -1
  447. package/dist/context/ingest/local-mapping-reconcile.test.js +0 -61
  448. package/dist/context/ingest/local-metabase-ingest.test.d.ts +0 -1
  449. package/dist/context/ingest/local-metabase-ingest.test.js +0 -227
  450. package/dist/context/ingest/local-stage-ingest.test.d.ts +0 -1
  451. package/dist/context/ingest/local-stage-ingest.test.js +0 -581
  452. package/dist/context/ingest/memory-flow/acceptance-fixtures.d.ts +0 -6
  453. package/dist/context/ingest/memory-flow/acceptance-fixtures.js +0 -155
  454. package/dist/context/ingest/memory-flow/acceptance.test.d.ts +0 -1
  455. package/dist/context/ingest/memory-flow/acceptance.test.js +0 -43
  456. package/dist/context/ingest/memory-flow/events.test.d.ts +0 -1
  457. package/dist/context/ingest/memory-flow/events.test.js +0 -319
  458. package/dist/context/ingest/memory-flow/interaction.test.d.ts +0 -1
  459. package/dist/context/ingest/memory-flow/interaction.test.js +0 -264
  460. package/dist/context/ingest/memory-flow/interactive-render.test.d.ts +0 -1
  461. package/dist/context/ingest/memory-flow/interactive-render.test.js +0 -160
  462. package/dist/context/ingest/memory-flow/live-buffer.test.d.ts +0 -1
  463. package/dist/context/ingest/memory-flow/live-buffer.test.js +0 -77
  464. package/dist/context/ingest/memory-flow/render.test.d.ts +0 -1
  465. package/dist/context/ingest/memory-flow/render.test.js +0 -105
  466. package/dist/context/ingest/memory-flow/schema.test.d.ts +0 -1
  467. package/dist/context/ingest/memory-flow/schema.test.js +0 -147
  468. package/dist/context/ingest/memory-flow/summary.test.d.ts +0 -1
  469. package/dist/context/ingest/memory-flow/summary.test.js +0 -130
  470. package/dist/context/ingest/memory-flow/view-model.test.d.ts +0 -1
  471. package/dist/context/ingest/memory-flow/view-model.test.js +0 -397
  472. package/dist/context/ingest/memory-flow/visuals.test.d.ts +0 -1
  473. package/dist/context/ingest/memory-flow/visuals.test.js +0 -49
  474. package/dist/context/ingest/page-triage/page-triage.service.test.d.ts +0 -1
  475. package/dist/context/ingest/page-triage/page-triage.service.test.js +0 -311
  476. package/dist/context/ingest/raw-sources-paths.test.d.ts +0 -1
  477. package/dist/context/ingest/raw-sources-paths.test.js +0 -18
  478. package/dist/context/ingest/repo-fetch.test.d.ts +0 -1
  479. package/dist/context/ingest/repo-fetch.test.js +0 -168
  480. package/dist/context/ingest/report-snapshot.test.d.ts +0 -1
  481. package/dist/context/ingest/report-snapshot.test.js +0 -329
  482. package/dist/context/ingest/semantic-layer-target-policy.test.d.ts +0 -1
  483. package/dist/context/ingest/semantic-layer-target-policy.test.js +0 -25
  484. package/dist/context/ingest/source-adapter-registry.test.d.ts +0 -1
  485. package/dist/context/ingest/source-adapter-registry.test.js +0 -35
  486. package/dist/context/ingest/sqlite-bundle-ingest-store.test.d.ts +0 -1
  487. package/dist/context/ingest/sqlite-bundle-ingest-store.test.js +0 -517
  488. package/dist/context/ingest/sqlite-local-ingest-store.test.d.ts +0 -1
  489. package/dist/context/ingest/sqlite-local-ingest-store.test.js +0 -143
  490. package/dist/context/ingest/stages/build-reconcile-context.context-candidates.test.d.ts +0 -1
  491. package/dist/context/ingest/stages/build-reconcile-context.context-candidates.test.js +0 -102
  492. package/dist/context/ingest/stages/build-reconcile-context.test.d.ts +0 -1
  493. package/dist/context/ingest/stages/build-reconcile-context.test.js +0 -141
  494. package/dist/context/ingest/stages/build-wu-context.test.d.ts +0 -1
  495. package/dist/context/ingest/stages/build-wu-context.test.js +0 -196
  496. package/dist/context/ingest/stages/stage-1-stage-raw-files.test.d.ts +0 -1
  497. package/dist/context/ingest/stages/stage-1-stage-raw-files.test.js +0 -54
  498. package/dist/context/ingest/stages/stage-3-work-units.test.d.ts +0 -1
  499. package/dist/context/ingest/stages/stage-3-work-units.test.js +0 -175
  500. package/dist/context/ingest/stages/stage-4-reconciliation.test.d.ts +0 -1
  501. package/dist/context/ingest/stages/stage-4-reconciliation.test.js +0 -144
  502. package/dist/context/ingest/stages/validate-wu-sources.test.d.ts +0 -1
  503. package/dist/context/ingest/stages/validate-wu-sources.test.js +0 -27
  504. package/dist/context/ingest/tools/emit-reconciliation-records.tool.test.d.ts +0 -1
  505. package/dist/context/ingest/tools/emit-reconciliation-records.tool.test.js +0 -237
  506. package/dist/context/ingest/tools/eviction-list.tool.test.d.ts +0 -1
  507. package/dist/context/ingest/tools/eviction-list.tool.test.js +0 -44
  508. package/dist/context/ingest/tools/read-raw-file.tool.test.d.ts +0 -1
  509. package/dist/context/ingest/tools/read-raw-file.tool.test.js +0 -45
  510. package/dist/context/ingest/tools/read-raw-span.tool.test.d.ts +0 -1
  511. package/dist/context/ingest/tools/read-raw-span.tool.test.js +0 -34
  512. package/dist/context/ingest/tools/stage-diff.tool.test.d.ts +0 -1
  513. package/dist/context/ingest/tools/stage-diff.tool.test.js +0 -112
  514. package/dist/context/ingest/tools/stage-list.tool.test.d.ts +0 -1
  515. package/dist/context/ingest/tools/stage-list.tool.test.js +0 -58
  516. package/dist/context/ingest/tools/tool-transcript-summary.test.d.ts +0 -1
  517. package/dist/context/ingest/tools/tool-transcript-summary.test.js +0 -141
  518. package/dist/context/ingest/tools/warehouse-verification/discover-data.tool.test.d.ts +0 -1
  519. package/dist/context/ingest/tools/warehouse-verification/discover-data.tool.test.js +0 -107
  520. package/dist/context/ingest/tools/warehouse-verification/entity-details.tool.test.d.ts +0 -1
  521. package/dist/context/ingest/tools/warehouse-verification/entity-details.tool.test.js +0 -146
  522. package/dist/context/ingest/tools/warehouse-verification/sql-execution.tool.test.d.ts +0 -1
  523. package/dist/context/ingest/tools/warehouse-verification/sql-execution.tool.test.js +0 -50
  524. package/dist/context/ingest/wiki-body-refs.test.d.ts +0 -1
  525. package/dist/context/ingest/wiki-body-refs.test.js +0 -138
  526. package/dist/context/ingest/wiki-sl-ref-repair.test.d.ts +0 -1
  527. package/dist/context/ingest/wiki-sl-ref-repair.test.js +0 -81
  528. package/dist/context/llm/ai-sdk-runtime.test.d.ts +0 -1
  529. package/dist/context/llm/ai-sdk-runtime.test.js +0 -308
  530. package/dist/context/llm/claude-code-env.test.d.ts +0 -1
  531. package/dist/context/llm/claude-code-env.test.js +0 -17
  532. package/dist/context/llm/claude-code-models.test.d.ts +0 -1
  533. package/dist/context/llm/claude-code-models.test.js +0 -15
  534. package/dist/context/llm/claude-code-runtime.test.d.ts +0 -1
  535. package/dist/context/llm/claude-code-runtime.test.js +0 -434
  536. package/dist/context/llm/debug-request-recorder.test.d.ts +0 -1
  537. package/dist/context/llm/debug-request-recorder.test.js +0 -112
  538. package/dist/context/llm/embedding-port.test.d.ts +0 -1
  539. package/dist/context/llm/embedding-port.test.js +0 -34
  540. package/dist/context/llm/local-config.test.d.ts +0 -1
  541. package/dist/context/llm/local-config.test.js +0 -164
  542. package/dist/context/llm/runtime-local-config.test.d.ts +0 -1
  543. package/dist/context/llm/runtime-local-config.test.js +0 -17
  544. package/dist/context/llm/runtime-tools.test.d.ts +0 -1
  545. package/dist/context/llm/runtime-tools.test.js +0 -36
  546. package/dist/context/mcp/local-project-ports.test.d.ts +0 -1
  547. package/dist/context/mcp/local-project-ports.test.js +0 -689
  548. package/dist/context/mcp/server.test.d.ts +0 -1
  549. package/dist/context/mcp/server.test.js +0 -902
  550. package/dist/context/memory/local-memory.test.d.ts +0 -1
  551. package/dist/context/memory/local-memory.test.js +0 -173
  552. package/dist/context/memory/memory-agent.service.ingest.test.d.ts +0 -1
  553. package/dist/context/memory/memory-agent.service.ingest.test.js +0 -355
  554. package/dist/context/memory/memory-agent.service.test.d.ts +0 -1
  555. package/dist/context/memory/memory-agent.service.test.js +0 -413
  556. package/dist/context/memory/memory-runs.test.d.ts +0 -1
  557. package/dist/context/memory/memory-runs.test.js +0 -158
  558. package/dist/context/memory/memory-runtime-assets.test.d.ts +0 -1
  559. package/dist/context/memory/memory-runtime-assets.test.js +0 -162
  560. package/dist/context/project/config.test.d.ts +0 -1
  561. package/dist/context/project/config.test.js +0 -467
  562. package/dist/context/project/driver-schemas.test.d.ts +0 -1
  563. package/dist/context/project/driver-schemas.test.js +0 -125
  564. package/dist/context/project/local-git-file-store.test.d.ts +0 -1
  565. package/dist/context/project/local-git-file-store.test.js +0 -71
  566. package/dist/context/project/mappings-yaml-schema.test.d.ts +0 -1
  567. package/dist/context/project/mappings-yaml-schema.test.js +0 -79
  568. package/dist/context/project/project.test.d.ts +0 -1
  569. package/dist/context/project/project.test.js +0 -55
  570. package/dist/context/project/setup-config.test.d.ts +0 -1
  571. package/dist/context/project/setup-config.test.js +0 -38
  572. package/dist/context/prompts/prompt.service.test.d.ts +0 -1
  573. package/dist/context/prompts/prompt.service.test.js +0 -43
  574. package/dist/context/scan/credentials.test.d.ts +0 -1
  575. package/dist/context/scan/credentials.test.js +0 -162
  576. package/dist/context/scan/data-dictionary.test.d.ts +0 -1
  577. package/dist/context/scan/data-dictionary.test.js +0 -92
  578. package/dist/context/scan/description-generation.test.d.ts +0 -1
  579. package/dist/context/scan/description-generation.test.js +0 -693
  580. package/dist/context/scan/embedding-text.test.d.ts +0 -1
  581. package/dist/context/scan/embedding-text.test.js +0 -36
  582. package/dist/context/scan/enrichment-state.test.d.ts +0 -1
  583. package/dist/context/scan/enrichment-state.test.js +0 -147
  584. package/dist/context/scan/enrichment-summary.test.d.ts +0 -1
  585. package/dist/context/scan/enrichment-summary.test.js +0 -34
  586. package/dist/context/scan/enrichment-types.test.d.ts +0 -1
  587. package/dist/context/scan/enrichment-types.test.js +0 -141
  588. package/dist/context/scan/entity-details.test.d.ts +0 -1
  589. package/dist/context/scan/entity-details.test.js +0 -234
  590. package/dist/context/scan/local-enrichment-artifacts.test.d.ts +0 -1
  591. package/dist/context/scan/local-enrichment-artifacts.test.js +0 -771
  592. package/dist/context/scan/local-enrichment.test.d.ts +0 -1
  593. package/dist/context/scan/local-enrichment.test.js +0 -765
  594. package/dist/context/scan/local-scan.test.d.ts +0 -1
  595. package/dist/context/scan/local-scan.test.js +0 -1663
  596. package/dist/context/scan/local-structural-artifacts.test.d.ts +0 -1
  597. package/dist/context/scan/local-structural-artifacts.test.js +0 -144
  598. package/dist/context/scan/relationship-benchmark-report.test.d.ts +0 -1
  599. package/dist/context/scan/relationship-benchmark-report.test.js +0 -389
  600. package/dist/context/scan/relationship-benchmarks.test.d.ts +0 -1
  601. package/dist/context/scan/relationship-benchmarks.test.js +0 -1072
  602. package/dist/context/scan/relationship-budget.test.d.ts +0 -1
  603. package/dist/context/scan/relationship-budget.test.js +0 -71
  604. package/dist/context/scan/relationship-candidates.test.d.ts +0 -1
  605. package/dist/context/scan/relationship-candidates.test.js +0 -747
  606. package/dist/context/scan/relationship-composite-candidates.test.d.ts +0 -1
  607. package/dist/context/scan/relationship-composite-candidates.test.js +0 -69
  608. package/dist/context/scan/relationship-diagnostics.test.d.ts +0 -1
  609. package/dist/context/scan/relationship-diagnostics.test.js +0 -333
  610. package/dist/context/scan/relationship-discovery.test.d.ts +0 -1
  611. package/dist/context/scan/relationship-discovery.test.js +0 -618
  612. package/dist/context/scan/relationship-formal-metadata.test.d.ts +0 -1
  613. package/dist/context/scan/relationship-formal-metadata.test.js +0 -125
  614. package/dist/context/scan/relationship-graph-resolver.test.d.ts +0 -1
  615. package/dist/context/scan/relationship-graph-resolver.test.js +0 -604
  616. package/dist/context/scan/relationship-llm-proposal.test.d.ts +0 -1
  617. package/dist/context/scan/relationship-llm-proposal.test.js +0 -197
  618. package/dist/context/scan/relationship-locality.test.d.ts +0 -1
  619. package/dist/context/scan/relationship-locality.test.js +0 -128
  620. package/dist/context/scan/relationship-name-similarity.test.d.ts +0 -1
  621. package/dist/context/scan/relationship-name-similarity.test.js +0 -68
  622. package/dist/context/scan/relationship-profiling.test.d.ts +0 -1
  623. package/dist/context/scan/relationship-profiling.test.js +0 -392
  624. package/dist/context/scan/relationship-scoring.test.d.ts +0 -1
  625. package/dist/context/scan/relationship-scoring.test.js +0 -86
  626. package/dist/context/scan/relationship-validation.test.d.ts +0 -1
  627. package/dist/context/scan/relationship-validation.test.js +0 -455
  628. package/dist/context/scan/table-ref.test.d.ts +0 -1
  629. package/dist/context/scan/table-ref.test.js +0 -53
  630. package/dist/context/scan/type-normalization.test.d.ts +0 -1
  631. package/dist/context/scan/type-normalization.test.js +0 -21
  632. package/dist/context/scan/types.test.d.ts +0 -1
  633. package/dist/context/scan/types.test.js +0 -206
  634. package/dist/context/scan/warehouse-catalog.test.d.ts +0 -1
  635. package/dist/context/scan/warehouse-catalog.test.js +0 -158
  636. package/dist/context/search/backend-conformance.test-utils.d.ts +0 -39
  637. package/dist/context/search/backend-conformance.test-utils.js +0 -88
  638. package/dist/context/search/backend-conformance.test-utils.test.d.ts +0 -1
  639. package/dist/context/search/backend-conformance.test-utils.test.js +0 -408
  640. package/dist/context/search/discover.test.d.ts +0 -1
  641. package/dist/context/search/discover.test.js +0 -197
  642. package/dist/context/search/hybrid-search-core.test.d.ts +0 -1
  643. package/dist/context/search/hybrid-search-core.test.js +0 -113
  644. package/dist/context/search/pglite-owner-process.test.d.ts +0 -1
  645. package/dist/context/search/pglite-owner-process.test.js +0 -273
  646. package/dist/context/search/pglite-runtime-boundary.test.d.ts +0 -1
  647. package/dist/context/search/pglite-runtime-boundary.test.js +0 -40
  648. package/dist/context/search/pglite-spike.test.d.ts +0 -1
  649. package/dist/context/search/pglite-spike.test.js +0 -249
  650. package/dist/context/search/query.test.d.ts +0 -1
  651. package/dist/context/search/query.test.js +0 -23
  652. package/dist/context/search/rrf.test.d.ts +0 -1
  653. package/dist/context/search/rrf.test.js +0 -47
  654. package/dist/context/skills/skills-registry.service.test.d.ts +0 -1
  655. package/dist/context/skills/skills-registry.service.test.js +0 -161
  656. package/dist/context/sl/dictionary-search.test.d.ts +0 -1
  657. package/dist/context/sl/dictionary-search.test.js +0 -204
  658. package/dist/context/sl/local-query.test.d.ts +0 -1
  659. package/dist/context/sl/local-query.test.js +0 -283
  660. package/dist/context/sl/local-sl.test.d.ts +0 -1
  661. package/dist/context/sl/local-sl.test.js +0 -334
  662. package/dist/context/sl/pglite-sl-search-prototype.test.d.ts +0 -1
  663. package/dist/context/sl/pglite-sl-search-prototype.test.js +0 -240
  664. package/dist/context/sl/schemas.contract.test.d.ts +0 -1
  665. package/dist/context/sl/schemas.contract.test.js +0 -62
  666. package/dist/context/sl/semantic-layer.service.test.d.ts +0 -1
  667. package/dist/context/sl/semantic-layer.service.test.js +0 -1107
  668. package/dist/context/sl/sl-dictionary-profile.test.d.ts +0 -1
  669. package/dist/context/sl/sl-dictionary-profile.test.js +0 -88
  670. package/dist/context/sl/sl-search.service.test.d.ts +0 -1
  671. package/dist/context/sl/sl-search.service.test.js +0 -256
  672. package/dist/context/sl/sqlite-sl-sources-index.test.d.ts +0 -1
  673. package/dist/context/sl/sqlite-sl-sources-index.test.js +0 -175
  674. package/dist/context/sl/tools/connection-id-schema.test.d.ts +0 -1
  675. package/dist/context/sl/tools/connection-id-schema.test.js +0 -14
  676. package/dist/context/sl/tools/sl-discover.tool.test.d.ts +0 -1
  677. package/dist/context/sl/tools/sl-discover.tool.test.js +0 -72
  678. package/dist/context/sl/tools/sl-edit-source.tool.test.d.ts +0 -1
  679. package/dist/context/sl/tools/sl-edit-source.tool.test.js +0 -184
  680. package/dist/context/sl/tools/sl-read-source.tool.session.test.d.ts +0 -1
  681. package/dist/context/sl/tools/sl-read-source.tool.session.test.js +0 -55
  682. package/dist/context/sl/tools/sl-rollback.tool.test.d.ts +0 -1
  683. package/dist/context/sl/tools/sl-rollback.tool.test.js +0 -57
  684. package/dist/context/sl/tools/sl-validate.tool.test.d.ts +0 -1
  685. package/dist/context/sl/tools/sl-validate.tool.test.js +0 -54
  686. package/dist/context/sl/tools/sl-warehouse-validation.test.d.ts +0 -1
  687. package/dist/context/sl/tools/sl-warehouse-validation.test.js +0 -136
  688. package/dist/context/sl/tools/sl-write-source.tool.test.d.ts +0 -1
  689. package/dist/context/sl/tools/sl-write-source.tool.test.js +0 -307
  690. package/dist/context/sql-analysis/http-sql-analysis-port.test.d.ts +0 -1
  691. package/dist/context/sql-analysis/http-sql-analysis-port.test.js +0 -147
  692. package/dist/context/test/make-local-git-repo.d.ts +0 -10
  693. package/dist/context/test/make-local-git-repo.js +0 -34
  694. package/dist/context/tools/context-evidence-tools.test.d.ts +0 -1
  695. package/dist/context/tools/context-evidence-tools.test.js +0 -486
  696. package/dist/context/tools/touched-sl-sources.test.d.ts +0 -1
  697. package/dist/context/tools/touched-sl-sources.test.js +0 -31
  698. package/dist/context/wiki/knowledge-wiki.service.test.d.ts +0 -1
  699. package/dist/context/wiki/knowledge-wiki.service.test.js +0 -205
  700. package/dist/context/wiki/local-knowledge.test.d.ts +0 -1
  701. package/dist/context/wiki/local-knowledge.test.js +0 -270
  702. package/dist/context/wiki/sqlite-knowledge-index.test.d.ts +0 -1
  703. package/dist/context/wiki/sqlite-knowledge-index.test.js +0 -129
  704. package/dist/context/wiki/tools/wiki-list-tags.tool.test.d.ts +0 -1
  705. package/dist/context/wiki/tools/wiki-list-tags.tool.test.js +0 -35
  706. package/dist/context/wiki/tools/wiki-read.tool.test.d.ts +0 -1
  707. package/dist/context/wiki/tools/wiki-read.tool.test.js +0 -66
  708. package/dist/context/wiki/tools/wiki-remove.tool.test.d.ts +0 -1
  709. package/dist/context/wiki/tools/wiki-remove.tool.test.js +0 -95
  710. package/dist/context/wiki/tools/wiki-search.tool.test.d.ts +0 -1
  711. package/dist/context/wiki/tools/wiki-search.tool.test.js +0 -35
  712. package/dist/context/wiki/tools/wiki-write.tool.test.d.ts +0 -1
  713. package/dist/context/wiki/tools/wiki-write.tool.test.js +0 -264
  714. package/dist/context/wiki/wiki-ref-validation.test.d.ts +0 -1
  715. package/dist/context/wiki/wiki-ref-validation.test.js +0 -64
  716. package/dist/context-build-view.test.d.ts +0 -1
  717. package/dist/context-build-view.test.js +0 -942
  718. package/dist/database-tree-picker.test.d.ts +0 -1
  719. package/dist/database-tree-picker.test.js +0 -188
  720. package/dist/demo-assets.test.d.ts +0 -1
  721. package/dist/demo-assets.test.js +0 -121
  722. package/dist/demo-metrics.test.d.ts +0 -1
  723. package/dist/demo-metrics.test.js +0 -108
  724. package/dist/doctor.test.d.ts +0 -1
  725. package/dist/doctor.test.js +0 -596
  726. package/dist/embedding-resolution.test.d.ts +0 -1
  727. package/dist/embedding-resolution.test.js +0 -132
  728. package/dist/example-smoke.test.d.ts +0 -1
  729. package/dist/example-smoke.test.js +0 -83
  730. package/dist/index.test.d.ts +0 -1
  731. package/dist/index.test.js +0 -1300
  732. package/dist/ingest-query-executor.test.d.ts +0 -1
  733. package/dist/ingest-query-executor.test.js +0 -71
  734. package/dist/ingest-report-file.test.d.ts +0 -1
  735. package/dist/ingest-report-file.test.js +0 -63
  736. package/dist/ingest-viz.test.d.ts +0 -1
  737. package/dist/ingest-viz.test.js +0 -691
  738. package/dist/ingest.test-utils.d.ts +0 -126
  739. package/dist/ingest.test-utils.js +0 -629
  740. package/dist/ingest.test.d.ts +0 -1
  741. package/dist/ingest.test.js +0 -1568
  742. package/dist/io/logger.test.d.ts +0 -1
  743. package/dist/io/logger.test.js +0 -55
  744. package/dist/io/mode.test.d.ts +0 -1
  745. package/dist/io/mode.test.js +0 -48
  746. package/dist/io/print-list.test.d.ts +0 -1
  747. package/dist/io/print-list.test.js +0 -277
  748. package/dist/knowledge.test.d.ts +0 -1
  749. package/dist/knowledge.test.js +0 -198
  750. package/dist/llm/embedding-health.test.d.ts +0 -1
  751. package/dist/llm/embedding-health.test.js +0 -72
  752. package/dist/llm/embedding-provider.test.d.ts +0 -1
  753. package/dist/llm/embedding-provider.test.js +0 -84
  754. package/dist/llm/message-builder.test.d.ts +0 -1
  755. package/dist/llm/message-builder.test.js +0 -127
  756. package/dist/llm/model-health.test.d.ts +0 -1
  757. package/dist/llm/model-health.test.js +0 -55
  758. package/dist/llm/model-provider.test.d.ts +0 -1
  759. package/dist/llm/model-provider.test.js +0 -246
  760. package/dist/llm/repair.test.d.ts +0 -1
  761. package/dist/llm/repair.test.js +0 -78
  762. package/dist/local-adapters.test.d.ts +0 -1
  763. package/dist/local-adapters.test.js +0 -166
  764. package/dist/local-scan-connectors.test.d.ts +0 -1
  765. package/dist/local-scan-connectors.test.js +0 -92
  766. package/dist/managed-local-embeddings.test.d.ts +0 -1
  767. package/dist/managed-local-embeddings.test.js +0 -229
  768. package/dist/managed-mcp-daemon.test.d.ts +0 -1
  769. package/dist/managed-mcp-daemon.test.js +0 -187
  770. package/dist/managed-python-command.test.d.ts +0 -1
  771. package/dist/managed-python-command.test.js +0 -262
  772. package/dist/managed-python-daemon.test.d.ts +0 -1
  773. package/dist/managed-python-daemon.test.js +0 -360
  774. package/dist/managed-python-http.test.d.ts +0 -1
  775. package/dist/managed-python-http.test.js +0 -177
  776. package/dist/managed-python-runtime.test.d.ts +0 -1
  777. package/dist/managed-python-runtime.test.js +0 -426
  778. package/dist/mcp-http-server.test.d.ts +0 -1
  779. package/dist/mcp-http-server.test.js +0 -209
  780. package/dist/mcp-server-factory.test.d.ts +0 -1
  781. package/dist/mcp-server-factory.test.js +0 -142
  782. package/dist/memory-flow-interactive.test.d.ts +0 -1
  783. package/dist/memory-flow-interactive.test.js +0 -109
  784. package/dist/memory-flow-tui.test.d.ts +0 -1
  785. package/dist/memory-flow-tui.test.js +0 -247
  786. package/dist/next-steps.test.d.ts +0 -1
  787. package/dist/next-steps.test.js +0 -77
  788. package/dist/notion-page-picker.test.d.ts +0 -1
  789. package/dist/notion-page-picker.test.js +0 -244
  790. package/dist/print-command-tree.test.d.ts +0 -1
  791. package/dist/print-command-tree.test.js +0 -37
  792. package/dist/project-dir.test.d.ts +0 -1
  793. package/dist/project-dir.test.js +0 -124
  794. package/dist/project-resolver.test.d.ts +0 -1
  795. package/dist/project-resolver.test.js +0 -49
  796. package/dist/prompt-navigation.test.d.ts +0 -1
  797. package/dist/prompt-navigation.test.js +0 -33
  798. package/dist/proxy-env.test.d.ts +0 -1
  799. package/dist/proxy-env.test.js +0 -17
  800. package/dist/public-ingest-copy.test.d.ts +0 -1
  801. package/dist/public-ingest-copy.test.js +0 -24
  802. package/dist/public-ingest.test.d.ts +0 -1
  803. package/dist/public-ingest.test.js +0 -891
  804. package/dist/runtime-requirements.test.d.ts +0 -1
  805. package/dist/runtime-requirements.test.js +0 -73
  806. package/dist/runtime.test.d.ts +0 -1
  807. package/dist/runtime.test.js +0 -381
  808. package/dist/scan.test.d.ts +0 -1
  809. package/dist/scan.test.js +0 -1123
  810. package/dist/setup-agents.test.d.ts +0 -1
  811. package/dist/setup-agents.test.js +0 -1028
  812. package/dist/setup-context.test.d.ts +0 -1
  813. package/dist/setup-context.test.js +0 -491
  814. package/dist/setup-databases.test.d.ts +0 -1
  815. package/dist/setup-databases.test.js +0 -2101
  816. package/dist/setup-demo-tour.test.d.ts +0 -1
  817. package/dist/setup-demo-tour.test.js +0 -221
  818. package/dist/setup-embeddings.test.d.ts +0 -1
  819. package/dist/setup-embeddings.test.js +0 -436
  820. package/dist/setup-interrupt.test.d.ts +0 -1
  821. package/dist/setup-interrupt.test.js +0 -77
  822. package/dist/setup-models.test.d.ts +0 -1
  823. package/dist/setup-models.test.js +0 -885
  824. package/dist/setup-project.test.d.ts +0 -1
  825. package/dist/setup-project.test.js +0 -209
  826. package/dist/setup-prompts.test.d.ts +0 -1
  827. package/dist/setup-prompts.test.js +0 -208
  828. package/dist/setup-ready-menu.test.d.ts +0 -1
  829. package/dist/setup-ready-menu.test.js +0 -44
  830. package/dist/setup-runtime.test.d.ts +0 -1
  831. package/dist/setup-runtime.test.js +0 -111
  832. package/dist/setup-secrets.test.d.ts +0 -1
  833. package/dist/setup-secrets.test.js +0 -30
  834. package/dist/setup-sources-notion.test.d.ts +0 -1
  835. package/dist/setup-sources-notion.test.js +0 -109
  836. package/dist/setup-sources.test.d.ts +0 -1
  837. package/dist/setup-sources.test.js +0 -1303
  838. package/dist/setup.test.d.ts +0 -1
  839. package/dist/setup.test.js +0 -1825
  840. package/dist/sl.test.d.ts +0 -1
  841. package/dist/sl.test.js +0 -567
  842. package/dist/source-mapping.test.d.ts +0 -1
  843. package/dist/source-mapping.test.js +0 -65
  844. package/dist/sql.test.d.ts +0 -1
  845. package/dist/sql.test.js +0 -253
  846. package/dist/standalone-smoke.test.d.ts +0 -1
  847. package/dist/standalone-smoke.test.js +0 -250
  848. package/dist/status-project.test.d.ts +0 -1
  849. package/dist/status-project.test.js +0 -502
  850. package/dist/telemetry/command-hook.test.d.ts +0 -1
  851. package/dist/telemetry/command-hook.test.js +0 -31
  852. package/dist/telemetry/demo-detect.test.d.ts +0 -1
  853. package/dist/telemetry/demo-detect.test.js +0 -22
  854. package/dist/telemetry/emitter.test.d.ts +0 -1
  855. package/dist/telemetry/emitter.test.js +0 -103
  856. package/dist/telemetry/events.snapshot.test.d.ts +0 -1
  857. package/dist/telemetry/events.snapshot.test.js +0 -135
  858. package/dist/telemetry/events.test.d.ts +0 -1
  859. package/dist/telemetry/events.test.js +0 -136
  860. package/dist/telemetry/identity.test.d.ts +0 -1
  861. package/dist/telemetry/identity.test.js +0 -148
  862. package/dist/telemetry/project-snapshot.test.d.ts +0 -1
  863. package/dist/telemetry/project-snapshot.test.js +0 -71
  864. package/dist/telemetry/schema-writer.test.d.ts +0 -1
  865. package/dist/telemetry/schema-writer.test.js +0 -23
  866. package/dist/telemetry/scrubber.test.d.ts +0 -1
  867. package/dist/telemetry/scrubber.test.js +0 -21
  868. package/dist/text-ingest.test.d.ts +0 -1
  869. package/dist/text-ingest.test.js +0 -247
  870. package/dist/tree-picker-state.test.d.ts +0 -1
  871. package/dist/tree-picker-state.test.js +0 -303
  872. package/dist/tree-picker-tui.test.d.ts +0 -1
  873. package/dist/tree-picker-tui.test.js +0 -248
  874. package/dist/viz-fallback.test.d.ts +0 -1
  875. package/dist/viz-fallback.test.js +0 -77
@@ -1,1107 +0,0 @@
1
- import { beforeEach, describe, expect, it, vi } from 'vitest';
2
- import { ColumnNameCollisionError, composeOverlay, ConflictingExcludeAndOverrideError, enrichColumnsFromManifest, findDanglingSegmentRefs, projectManifestEntry, SemanticLayerService, toResolvedWire, UnknownColumnOverrideError, } from './semantic-layer.service.js';
3
- import { resolvedSourceSchema, sourceDefinitionSchema, sourceOverlaySchema } from './schemas.js';
4
- const pythonPort = {
5
- validateSources: vi.fn(),
6
- generateSources: vi.fn(),
7
- query: vi.fn(),
8
- };
9
- function connectionCatalog(connectionType = 'SNOWFLAKE') {
10
- return {
11
- listEnabledConnections: vi.fn().mockResolvedValue([]),
12
- getConnectionById: vi.fn().mockResolvedValue({ id: 'conn-1', name: 'conn-1', connectionType }),
13
- executeQuery: vi.fn(),
14
- };
15
- }
16
- const baseTable = {
17
- name: 'fct_labs',
18
- grain: ['lab_order_id'],
19
- table: 'analytics.fct_labs',
20
- columns: [
21
- { name: 'lab_order_id', type: 'string' },
22
- { name: 'admin_user_id', type: 'string' },
23
- { name: 'lab_type', type: 'string' },
24
- ],
25
- joins: [],
26
- measures: [],
27
- };
28
- describe('listConnectionIdsWithNames', () => {
29
- it('discovers local KTX connection ids from semantic-layer directories', async () => {
30
- const configService = {
31
- listFiles: vi.fn().mockResolvedValue({
32
- files: [
33
- 'semantic-layer/warehouse/_schema/public.yaml',
34
- 'semantic-layer/dbt-main/orders.yaml',
35
- 'semantic-layer/.gitkeep',
36
- ],
37
- }),
38
- };
39
- const catalog = connectionCatalog();
40
- catalog.listEnabledConnections.mockImplementation(async (ids) => ids.map((id) => ({ id, name: id, connectionType: id === 'warehouse' ? 'postgres' : 'dbt' })));
41
- const service = new SemanticLayerService(configService, catalog, pythonPort);
42
- await expect(service.listConnectionIdsWithNames()).resolves.toEqual([
43
- { id: 'dbt-main', name: 'dbt-main', connectionType: 'dbt' },
44
- { id: 'warehouse', name: 'warehouse', connectionType: 'postgres' },
45
- ]);
46
- expect(catalog.listEnabledConnections).toHaveBeenCalledWith(['dbt-main', 'warehouse']);
47
- });
48
- });
49
- describe('loadSource', () => {
50
- it('warns and returns null when an existing source file has invalid YAML', async () => {
51
- const logger = { log: vi.fn(), warn: vi.fn(), error: vi.fn() };
52
- const configService = {
53
- readFile: vi.fn().mockResolvedValue({ content: 'name: [' }),
54
- };
55
- const service = new SemanticLayerService(configService, connectionCatalog(), pythonPort, logger);
56
- await expect(service.loadSource('warehouse', 'orders')).resolves.toBeNull();
57
- expect(configService.readFile).toHaveBeenCalledWith('semantic-layer/warehouse/orders.yaml');
58
- expect(logger.warn).toHaveBeenCalledWith(expect.stringContaining('[loadSource] warehouse/orders.yaml: YAML parse failed:'));
59
- });
60
- });
61
- describe('composeOverlay', () => {
62
- it('carries top-level segments from overlay into the composed source', () => {
63
- const overlay = {
64
- name: 'fct_labs',
65
- segments: [{ name: 'byol', expr: "lab_type = 'byol'", description: 'BYOL cohort' }],
66
- };
67
- const composed = composeOverlay(baseTable, overlay);
68
- expect(composed.segments).toHaveLength(1);
69
- expect(composed.segments?.[0].name).toBe('byol');
70
- expect(composed.segments?.[0].expr).toBe("lab_type = 'byol'");
71
- });
72
- it('preserves measure-level segments references', () => {
73
- const overlay = {
74
- name: 'fct_labs',
75
- segments: [{ name: 'byol', expr: "lab_type = 'byol'" }],
76
- measures: [
77
- {
78
- name: 'byol_subscriber_count',
79
- expr: 'count(distinct admin_user_id)',
80
- segments: ['byol'],
81
- description: 'BYOL subscribers',
82
- },
83
- ],
84
- };
85
- const composed = composeOverlay(baseTable, overlay);
86
- expect(composed.measures).toHaveLength(1);
87
- expect(composed.measures[0].segments).toEqual(['byol']);
88
- });
89
- it('leaves base segments unchanged when overlay does not specify segments', () => {
90
- const baseWithSegments = {
91
- ...baseTable,
92
- segments: [{ name: 'pre_existing', expr: 'is_paid = true' }],
93
- };
94
- const overlay = { name: 'fct_labs', descriptions: { user: 'no segments here' } };
95
- const composed = composeOverlay(baseWithSegments, overlay);
96
- expect(composed.segments).toEqual([{ name: 'pre_existing', expr: 'is_paid = true' }]);
97
- });
98
- it('replaces base segments when overlay provides its own (even an empty array)', () => {
99
- const baseWithSegments = {
100
- ...baseTable,
101
- segments: [{ name: 'pre_existing', expr: 'is_paid = true' }],
102
- };
103
- const overlay = { name: 'fct_labs', segments: [] };
104
- const composed = composeOverlay(baseWithSegments, overlay);
105
- expect(composed.segments).toEqual([]);
106
- });
107
- it('throws on unknown top-level overlay keys with a pointed error', () => {
108
- const overlay = { name: 'fct_labs', frobnicate: true };
109
- expect(() => composeOverlay(baseTable, overlay)).toThrow(/overlay for 'fct_labs' has unhandled keys \[frobnicate\]/);
110
- });
111
- it('lists every unknown key in the error message, not just the first', () => {
112
- const overlay = { name: 'fct_labs', foo: 1, bar: 2 };
113
- expect(() => composeOverlay(baseTable, overlay)).toThrow(/foo, bar/);
114
- });
115
- it('still handles existing known keys without regression', () => {
116
- const overlay = {
117
- name: 'fct_labs',
118
- descriptions: { user: 'patient lab orders' },
119
- exclude_columns: ['admin_user_id'],
120
- columns: [{ name: 'is_byol', type: 'boolean', expr: "lab_type = 'byol'" }],
121
- measures: [{ name: 'count_all', expr: 'count(*)' }],
122
- };
123
- const composed = composeOverlay(baseTable, overlay);
124
- expect(composed.columns.find((c) => c.name === 'admin_user_id')).toBeUndefined();
125
- expect(composed.columns.find((c) => c.name === 'is_byol')).toBeDefined();
126
- expect(composed.measures).toHaveLength(1);
127
- });
128
- it('applies column_overrides to same-named manifest columns', () => {
129
- const overlay = {
130
- name: 'fct_labs',
131
- column_overrides: [
132
- { name: 'lab_order_id', descriptions: { user: 'Primary key' } },
133
- { name: 'admin_user_id', descriptions: { user: 'FK to admin_users' } },
134
- ],
135
- };
136
- const composed = composeOverlay(baseTable, overlay);
137
- // No duplicate columns appended — same-named overlay entries merged onto the base.
138
- expect(composed.columns).toHaveLength(3);
139
- const labOrder = composed.columns.find((c) => c.name === 'lab_order_id');
140
- expect(labOrder?.type).toBe('string');
141
- expect(labOrder?.descriptions).toEqual({ user: 'Primary key' });
142
- const adminUser = composed.columns.find((c) => c.name === 'admin_user_id');
143
- expect(adminUser?.type).toBe('string');
144
- expect(adminUser?.descriptions).toEqual({ user: 'FK to admin_users' });
145
- });
146
- it('appends computed columns alongside column overrides', () => {
147
- const overlay = {
148
- name: 'fct_labs',
149
- column_overrides: [
150
- { name: 'lab_order_id', descriptions: { user: 'PK doc' } },
151
- ],
152
- columns: [
153
- { name: 'is_byol', type: 'boolean', expr: "lab_type = 'byol'" },
154
- ],
155
- };
156
- const composed = composeOverlay(baseTable, overlay);
157
- expect(composed.columns).toHaveLength(4);
158
- expect(composed.columns.find((c) => c.name === 'is_byol')?.expr).toBe("lab_type = 'byol'");
159
- expect(composed.columns.find((c) => c.name === 'lab_order_id')?.type).toBe('string');
160
- });
161
- it('rejects column_overrides that target unknown manifest columns', () => {
162
- expect(() => composeOverlay(baseTable, {
163
- name: 'fct_labs',
164
- column_overrides: [{ name: 'missing', descriptions: { user: 'Nope' } }],
165
- })).toThrow(UnknownColumnOverrideError);
166
- });
167
- it('rejects computed columns whose names collide with manifest columns', () => {
168
- expect(() => composeOverlay(baseTable, {
169
- name: 'fct_labs',
170
- columns: [{ name: 'lab_order_id', type: 'string', expr: 'lab_order_id' }],
171
- })).toThrow(ColumnNameCollisionError);
172
- });
173
- it('rejects exclude/override conflicts before applying exclusions', () => {
174
- expect(() => composeOverlay(baseTable, {
175
- name: 'fct_labs',
176
- exclude_columns: ['lab_order_id'],
177
- column_overrides: [{ name: 'lab_order_id', descriptions: { user: 'Hidden PK' } }],
178
- })).toThrow(ConflictingExcludeAndOverrideError);
179
- });
180
- it('merges overlay descriptions (plural) with base descriptions keyed by source', () => {
181
- const baseWithDescriptions = {
182
- ...baseTable,
183
- descriptions: { db: 'scan-derived description', ai: 'AI description' },
184
- };
185
- const overlay = {
186
- name: 'fct_labs',
187
- descriptions: { dbt: 'dbt description', ai: 'AI description (overridden)' },
188
- };
189
- const composed = composeOverlay(baseWithDescriptions, overlay);
190
- expect(composed.descriptions).toEqual({
191
- db: 'scan-derived description',
192
- ai: 'AI description (overridden)',
193
- dbt: 'dbt description',
194
- });
195
- });
196
- it('replaces manifest usage only when an overlay explicitly provides usage', () => {
197
- const baseWithUsage = {
198
- ...baseTable,
199
- usage: {
200
- narrative: 'Orders are commonly queried by lifecycle status.',
201
- frequencyTier: 'high',
202
- commonFilters: ['status'],
203
- commonJoins: [{ table: 'public.customers', on: ['customer_id'] }],
204
- },
205
- };
206
- expect(composeOverlay(baseWithUsage, { name: 'fct_labs', measures: [] }).usage).toEqual(baseWithUsage.usage);
207
- const composed = composeOverlay(baseWithUsage, {
208
- name: 'fct_labs',
209
- usage: {
210
- narrative: 'Overlay-curated usage note.',
211
- frequencyTier: 'mid',
212
- commonFilters: ['created_at'],
213
- commonGroupBys: ['created_at'],
214
- commonJoins: [],
215
- },
216
- });
217
- expect(composed.usage).toEqual({
218
- narrative: 'Overlay-curated usage note.',
219
- frequencyTier: 'mid',
220
- commonFilters: ['created_at'],
221
- commonGroupBys: ['created_at'],
222
- commonJoins: [],
223
- });
224
- });
225
- });
226
- describe('enrichColumnsFromManifest', () => {
227
- const manifest = {
228
- name: 'CONSIGNMENTS',
229
- table: 'ANALYTICS.MARTS.CONSIGNMENTS',
230
- grain: ['CONSIGNED_ITEM_ID'],
231
- columns: [
232
- {
233
- name: 'CONSIGNED_ITEM_ID',
234
- type: 'string',
235
- descriptions: { ai: 'Unique identifier for the consigned item record.' },
236
- },
237
- {
238
- name: 'CASH_ADV_AMOUNT',
239
- type: 'number',
240
- descriptions: { ai: 'Amount of cash advance disbursed to consigners.' },
241
- },
242
- {
243
- name: 'CONSIGNMENT_CREATED_AT',
244
- type: 'time',
245
- role: 'time',
246
- descriptions: { ai: 'Timestamp when the consignment was created.' },
247
- },
248
- ],
249
- joins: [],
250
- measures: [],
251
- };
252
- it('fills blank type and descriptions on source columns from the manifest', () => {
253
- const source = {
254
- name: 'aav_consignments',
255
- sql: 'SELECT CONSIGNED_ITEM_ID, CASH_ADV_AMOUNT FROM MARTS.CONSIGNMENTS WHERE ...',
256
- inherits_columns_from: 'CONSIGNMENTS',
257
- grain: ['CONSIGNED_ITEM_ID'],
258
- columns: [
259
- { name: 'CONSIGNED_ITEM_ID', type: '' },
260
- { name: 'CASH_ADV_AMOUNT', type: '' },
261
- ],
262
- joins: [],
263
- measures: [],
264
- };
265
- const enriched = enrichColumnsFromManifest(source, manifest);
266
- expect(enriched.columns[0]).toEqual({
267
- name: 'CONSIGNED_ITEM_ID',
268
- type: 'string',
269
- descriptions: { ai: 'Unique identifier for the consigned item record.' },
270
- });
271
- expect(enriched.columns[1]).toEqual({
272
- name: 'CASH_ADV_AMOUNT',
273
- type: 'number',
274
- descriptions: { ai: 'Amount of cash advance disbursed to consigners.' },
275
- });
276
- });
277
- it('preserves a local description if the source already declared one', () => {
278
- const source = {
279
- name: 'aav_consignments',
280
- sql: 'SELECT CONSIGNED_ITEM_ID FROM ...',
281
- inherits_columns_from: 'CONSIGNMENTS',
282
- grain: ['CONSIGNED_ITEM_ID'],
283
- columns: [
284
- {
285
- name: 'CONSIGNED_ITEM_ID',
286
- type: 'string',
287
- descriptions: { ai: 'AAV-specific note: always non-null in this filtered view.' },
288
- },
289
- ],
290
- joins: [],
291
- measures: [],
292
- };
293
- const enriched = enrichColumnsFromManifest(source, manifest);
294
- expect(enriched.columns[0].descriptions).toEqual({
295
- ai: 'AAV-specific note: always non-null in this filtered view.',
296
- });
297
- });
298
- it('passes through columns absent from the manifest unchanged', () => {
299
- const source = {
300
- name: 'aav_consignments',
301
- sql: 'SELECT ALT_VALUE_COMBINED, my_derived FROM ...',
302
- inherits_columns_from: 'CONSIGNMENTS',
303
- grain: ['CONSIGNED_ITEM_ID'],
304
- columns: [{ name: 'my_derived', type: 'number', expr: 'CASH_ADV_AMOUNT * 2' }],
305
- joins: [],
306
- measures: [],
307
- };
308
- const enriched = enrichColumnsFromManifest(source, manifest);
309
- expect(enriched.columns[0]).toEqual({
310
- name: 'my_derived',
311
- type: 'number',
312
- expr: 'CASH_ADV_AMOUNT * 2',
313
- });
314
- });
315
- it('copies role from the manifest when the source omits it', () => {
316
- const source = {
317
- name: 'aav_consignments',
318
- sql: 'SELECT CONSIGNMENT_CREATED_AT FROM ...',
319
- inherits_columns_from: 'CONSIGNMENTS',
320
- grain: ['CONSIGNED_ITEM_ID'],
321
- columns: [{ name: 'CONSIGNMENT_CREATED_AT', type: '' }],
322
- joins: [],
323
- measures: [],
324
- };
325
- const enriched = enrichColumnsFromManifest(source, manifest);
326
- expect(enriched.columns[0].role).toBe('time');
327
- expect(enriched.columns[0].type).toBe('time');
328
- });
329
- it('returns the source unchanged when manifestEntry is null/undefined', () => {
330
- const source = {
331
- name: 'aav_consignments',
332
- sql: 'SELECT FOO FROM ...',
333
- grain: ['FOO'],
334
- columns: [{ name: 'FOO', type: '' }],
335
- joins: [],
336
- measures: [],
337
- };
338
- const enriched = enrichColumnsFromManifest(source, null);
339
- expect(enriched).toEqual(source);
340
- });
341
- });
342
- describe('sourceDefinitionSchema', () => {
343
- it('preserves dbt structural metadata fields used by manifest-backed SL readers', () => {
344
- const result = sourceDefinitionSchema.safeParse({
345
- name: 'orders',
346
- descriptions: { dbt: 'Order facts from dbt.' },
347
- table: 'public.orders',
348
- grain: ['id'],
349
- columns: [
350
- {
351
- name: 'status',
352
- type: 'string',
353
- descriptions: { dbt: 'Order lifecycle status.' },
354
- constraints: { dbt: { not_null: true, unique: true } },
355
- enum_values: { dbt: ['placed', 'shipped'] },
356
- tests: {
357
- dbt: [{ name: 'accepted_values', package: 'dbt' }],
358
- dbt_by_package: { dbt: ['accepted_values'] },
359
- },
360
- },
361
- ],
362
- joins: [],
363
- measures: [],
364
- tags: { dbt: ['mart', 'finance'] },
365
- freshness: { dbt: { loaded_at_field: 'updated_at', raw: { warn_after: { count: 12, period: 'hour' } } } },
366
- default_time_dimension: { dbt: 'updated_at' },
367
- });
368
- expect(result.success).toBe(true);
369
- if (!result.success) {
370
- return;
371
- }
372
- expect(result.data.descriptions).toEqual({ dbt: 'Order facts from dbt.' });
373
- expect(result.data.columns[0]).toMatchObject({
374
- descriptions: { dbt: 'Order lifecycle status.' },
375
- constraints: { dbt: { not_null: true, unique: true } },
376
- enum_values: { dbt: ['placed', 'shipped'] },
377
- tests: {
378
- dbt: [{ name: 'accepted_values', package: 'dbt' }],
379
- dbt_by_package: { dbt: ['accepted_values'] },
380
- },
381
- });
382
- expect(result.data.tags).toEqual({ dbt: ['mart', 'finance'] });
383
- expect(result.data.freshness).toEqual({
384
- dbt: { loaded_at_field: 'updated_at', raw: { warn_after: { count: 12, period: 'hour' } } },
385
- });
386
- });
387
- it('accepts historic SQL usage on standalone sources', () => {
388
- const result = sourceDefinitionSchema.safeParse({
389
- name: 'orders',
390
- table: 'public.orders',
391
- grain: ['id'],
392
- columns: [{ name: 'id', type: 'string' }],
393
- joins: [],
394
- measures: [],
395
- usage: {
396
- narrative: 'Orders are queried for fulfillment and revenue analysis.',
397
- frequencyTier: 'high',
398
- commonFilters: ['status', 'created_at'],
399
- commonJoins: [{ table: 'public.customers', on: ['customer_id'] }],
400
- externalOwner: 'analytics',
401
- },
402
- });
403
- expect(result.success).toBe(true);
404
- if (!result.success) {
405
- return;
406
- }
407
- expect(result.data.usage).toMatchObject({
408
- narrative: 'Orders are queried for fulfillment and revenue analysis.',
409
- frequencyTier: 'high',
410
- commonFilters: ['status', 'created_at'],
411
- commonJoins: [{ table: 'public.customers', on: ['customer_id'] }],
412
- externalOwner: 'analytics',
413
- });
414
- });
415
- it("rejects qualified grain names (e.g. 'activity.account_id')", () => {
416
- const result = sourceDefinitionSchema.safeParse({
417
- name: 'activity',
418
- table: 'public.activity',
419
- grain: ['activity.account_id'],
420
- columns: [{ name: 'account_id', type: 'number' }],
421
- joins: [],
422
- measures: [],
423
- });
424
- expect(result.success).toBe(false);
425
- if (result.success)
426
- return;
427
- expect(result.error.issues.some((i) => i.path.join('.').startsWith('grain'))).toBe(true);
428
- });
429
- it('rejects qualified column names', () => {
430
- const result = sourceDefinitionSchema.safeParse({
431
- name: 'activity',
432
- table: 'public.activity',
433
- grain: ['account_id'],
434
- columns: [{ name: 'activity.account_id', type: 'number' }],
435
- joins: [],
436
- measures: [],
437
- });
438
- expect(result.success).toBe(false);
439
- if (result.success)
440
- return;
441
- expect(result.error.issues.some((i) => i.path.join('.').startsWith('columns'))).toBe(true);
442
- });
443
- });
444
- describe('sourceOverlaySchema', () => {
445
- it('accepts column_overrides and keeps columns computed-only', () => {
446
- const result = sourceOverlaySchema.safeParse({
447
- name: 'orders',
448
- column_overrides: [{ name: 'status', descriptions: { user: 'Lifecycle status' } }],
449
- columns: [{ name: 'is_paid', type: 'boolean', expr: "status = 'paid'" }],
450
- });
451
- expect(result.success).toBe(true);
452
- });
453
- it('rejects typeless overlay columns and singular description on overrides', () => {
454
- const result = sourceOverlaySchema.safeParse({
455
- name: 'orders',
456
- column_overrides: [{ name: 'status', description: 'Lifecycle status' }],
457
- columns: [{ name: 'status', descriptions: { user: 'Lifecycle status' } }],
458
- });
459
- expect(result.success).toBe(false);
460
- if (!result.success) {
461
- const paths = result.error.issues.map((issue) => issue.path.join('.'));
462
- expect(paths).toContain('column_overrides.0');
463
- expect(paths).toContain('columns.0.type');
464
- expect(paths).toContain('columns.0.expr');
465
- }
466
- });
467
- });
468
- describe('toResolvedWire', () => {
469
- it('strips TS-only authoring and provenance fields before the Python boundary', () => {
470
- const wire = toResolvedWire({
471
- name: 'orders',
472
- table: 'public.orders',
473
- inherits_columns_from: 'orders',
474
- grain: ['id'],
475
- columns: [{ name: 'id', type: 'string' }],
476
- joins: [{ to: 'customers', on: 'orders.customer_id = customers.id', relationship: 'many_to_one', source: 'formal' }],
477
- measures: [],
478
- usage: {
479
- narrative: 'Frequently queried orders.',
480
- frequencyTier: 'high',
481
- commonFilters: ['status'],
482
- commonJoins: [],
483
- },
484
- });
485
- expect(wire).toEqual({
486
- name: 'orders',
487
- table: 'public.orders',
488
- grain: ['id'],
489
- columns: [{ name: 'id', type: 'string' }],
490
- joins: [{ to: 'customers', on: 'orders.customer_id = customers.id', relationship: 'many_to_one' }],
491
- measures: [],
492
- });
493
- expect(resolvedSourceSchema.parse(wire)).toEqual(wire);
494
- });
495
- });
496
- describe('projectManifestEntry', () => {
497
- it('projects manifest usage onto the semantic-layer source', () => {
498
- const source = projectManifestEntry('orders', {
499
- table: 'public.orders',
500
- usage: {
501
- narrative: 'Orders are frequently filtered by status.',
502
- frequencyTier: 'high',
503
- commonFilters: ['status'],
504
- commonJoins: [{ table: 'public.customers', on: ['customer_id'] }],
505
- },
506
- columns: [
507
- { name: 'id', type: 'string', pk: true },
508
- { name: 'status', type: 'string' },
509
- ],
510
- });
511
- expect(source.usage).toEqual({
512
- narrative: 'Orders are frequently filtered by status.',
513
- frequencyTier: 'high',
514
- commonFilters: ['status'],
515
- commonJoins: [{ table: 'public.customers', on: ['customer_id'] }],
516
- });
517
- });
518
- });
519
- describe('findManifestEntryByTableRef', () => {
520
- let configService;
521
- let service;
522
- beforeEach(() => {
523
- configService = {
524
- listFiles: vi.fn().mockResolvedValue({
525
- files: ['semantic-layer/conn-1/_schema/marts.yaml'],
526
- }),
527
- readFile: vi.fn().mockResolvedValue({
528
- content: [
529
- 'tables:',
530
- ' CONSIGNMENTS:',
531
- ' table: ANALYTICS.MARTS.CONSIGNMENTS',
532
- ' columns:',
533
- ' - { name: CONSIGNED_ITEM_ID, type: string, pk: true }',
534
- ].join('\n'),
535
- }),
536
- };
537
- service = new SemanticLayerService(configService, connectionCatalog(), pythonPort);
538
- });
539
- it('finds by exact bare manifest key', async () => {
540
- const entry = await service.findManifestEntryByTableRef('conn-1', 'CONSIGNMENTS');
541
- expect(entry?.name).toBe('CONSIGNMENTS');
542
- });
543
- it('finds by fully-qualified table path', async () => {
544
- const entry = await service.findManifestEntryByTableRef('conn-1', 'ANALYTICS.MARTS.CONSIGNMENTS');
545
- expect(entry?.name).toBe('CONSIGNMENTS');
546
- });
547
- it('finds by schema-qualified suffix', async () => {
548
- const entry = await service.findManifestEntryByTableRef('conn-1', 'MARTS.CONSIGNMENTS');
549
- expect(entry?.name).toBe('CONSIGNMENTS');
550
- });
551
- it('matches case-insensitively on table path', async () => {
552
- const entry = await service.findManifestEntryByTableRef('conn-1', 'analytics.marts.consignments');
553
- expect(entry?.name).toBe('CONSIGNMENTS');
554
- });
555
- it('returns null when nothing matches', async () => {
556
- const entry = await service.findManifestEntryByTableRef('conn-1', 'NOT_A_TABLE');
557
- expect(entry).toBeNull();
558
- });
559
- });
560
- describe('loadAllSources — standalone enrichment via inherits_columns_from', () => {
561
- let configService;
562
- let service;
563
- beforeEach(() => {
564
- configService = {
565
- listFiles: vi.fn(),
566
- readFile: vi.fn(),
567
- };
568
- service = new SemanticLayerService(configService, connectionCatalog(), pythonPort);
569
- });
570
- it('preserves dbt metadata when projecting manifest-backed sources', async () => {
571
- const schemaPath = 'semantic-layer/conn-1/_schema/marts.yaml';
572
- configService.listFiles.mockImplementation((dir) => {
573
- if (dir === 'semantic-layer/conn-1' || dir === 'semantic-layer/conn-1/_schema') {
574
- return Promise.resolve({ files: [schemaPath] });
575
- }
576
- return Promise.resolve({ files: [] });
577
- });
578
- configService.readFile.mockResolvedValue({
579
- content: [
580
- 'tables:',
581
- ' orders:',
582
- ' table: public.orders',
583
- ' tags: { dbt: [mart] }',
584
- ' freshness:',
585
- ' dbt:',
586
- ' loaded_at_field: updated_at',
587
- ' columns:',
588
- ' - name: status',
589
- ' type: string',
590
- ' constraints: { dbt: { not_null: true } }',
591
- ' enum_values: { dbt: [placed, shipped] }',
592
- ' tests:',
593
- ' dbt:',
594
- ' - { name: accepted_values, package: dbt }',
595
- ].join('\n'),
596
- });
597
- const { sources, loadErrors } = await service.loadAllSources('conn-1');
598
- expect(loadErrors).toEqual([]);
599
- expect(sources[0]).toMatchObject({
600
- name: 'orders',
601
- tags: { dbt: ['mart'] },
602
- freshness: { dbt: { loaded_at_field: 'updated_at' } },
603
- columns: [
604
- {
605
- name: 'status',
606
- constraints: { dbt: { not_null: true } },
607
- enum_values: { dbt: ['placed', 'shipped'] },
608
- tests: { dbt: [{ name: 'accepted_values', package: 'dbt' }] },
609
- },
610
- ],
611
- });
612
- });
613
- it('fills blank columns on a standalone source from the manifest entry it points at', async () => {
614
- const schemaPath = 'semantic-layer/conn-1/_schema/marts.yaml';
615
- const standalonePath = 'semantic-layer/conn-1/aav_consignments.yaml';
616
- configService.listFiles.mockImplementation((dir) => {
617
- if (dir === 'semantic-layer/conn-1') {
618
- return Promise.resolve({ files: [schemaPath, standalonePath] });
619
- }
620
- if (dir === 'semantic-layer/conn-1/_schema') {
621
- return Promise.resolve({ files: [schemaPath] });
622
- }
623
- return Promise.resolve({ files: [] });
624
- });
625
- configService.readFile.mockImplementation((path) => {
626
- if (path === schemaPath) {
627
- return Promise.resolve({
628
- content: [
629
- 'tables:',
630
- ' CONSIGNMENTS:',
631
- ' table: ANALYTICS.MARTS.CONSIGNMENTS',
632
- ' columns:',
633
- ' - name: CONSIGNED_ITEM_ID',
634
- ' type: string',
635
- ' descriptions: { ai: "Unique consigned-item id." }',
636
- ' - name: CASH_ADV_AMOUNT',
637
- ' type: number',
638
- ' descriptions: { ai: "Cash advance amount." }',
639
- ].join('\n'),
640
- });
641
- }
642
- if (path === standalonePath) {
643
- return Promise.resolve({
644
- content: [
645
- 'name: aav_consignments',
646
- 'sql: |',
647
- ' SELECT CONSIGNED_ITEM_ID, CASH_ADV_AMOUNT FROM ANALYTICS.MARTS.CONSIGNMENTS WHERE x',
648
- 'inherits_columns_from: CONSIGNMENTS',
649
- 'grain: [CONSIGNED_ITEM_ID]',
650
- 'columns:',
651
- ' - { name: CONSIGNED_ITEM_ID }',
652
- ' - { name: CASH_ADV_AMOUNT }',
653
- ].join('\n'),
654
- });
655
- }
656
- return Promise.reject(new Error(`Unexpected readFile: ${path}`));
657
- });
658
- const { sources, loadErrors } = await service.loadAllSources('conn-1');
659
- expect(loadErrors).toEqual([]);
660
- const aav = sources.find((s) => s.name === 'aav_consignments');
661
- expect(aav).toBeDefined();
662
- expect(aav?.columns).toEqual([
663
- { name: 'CONSIGNED_ITEM_ID', type: 'string', descriptions: { ai: 'Unique consigned-item id.' } },
664
- { name: 'CASH_ADV_AMOUNT', type: 'number', descriptions: { ai: 'Cash advance amount.' } },
665
- ]);
666
- });
667
- it('accepts a fully-qualified path in inherits_columns_from', async () => {
668
- const schemaPath = 'semantic-layer/conn-1/_schema/marts.yaml';
669
- const standalonePath = 'semantic-layer/conn-1/aav_consignments.yaml';
670
- configService.listFiles.mockImplementation((dir) => {
671
- if (dir === 'semantic-layer/conn-1') {
672
- return Promise.resolve({ files: [schemaPath, standalonePath] });
673
- }
674
- if (dir === 'semantic-layer/conn-1/_schema') {
675
- return Promise.resolve({ files: [schemaPath] });
676
- }
677
- return Promise.resolve({ files: [] });
678
- });
679
- configService.readFile.mockImplementation((path) => {
680
- if (path === schemaPath) {
681
- return Promise.resolve({
682
- content: [
683
- 'tables:',
684
- ' CONSIGNMENTS:',
685
- ' table: ANALYTICS.MARTS.CONSIGNMENTS',
686
- ' columns:',
687
- ' - { name: CONSIGNED_ITEM_ID, type: string }',
688
- ].join('\n'),
689
- });
690
- }
691
- return Promise.resolve({
692
- content: [
693
- 'name: aav_consignments',
694
- 'sql: SELECT 1',
695
- 'inherits_columns_from: ANALYTICS.MARTS.CONSIGNMENTS',
696
- 'grain: [CONSIGNED_ITEM_ID]',
697
- 'columns:',
698
- ' - { name: CONSIGNED_ITEM_ID }',
699
- ].join('\n'),
700
- });
701
- });
702
- const { sources, loadErrors } = await service.loadAllSources('conn-1');
703
- expect(loadErrors).toEqual([]);
704
- const aav = sources.find((s) => s.name === 'aav_consignments');
705
- expect(aav?.columns[0].type).toBe('string');
706
- });
707
- it('passes the source through unchanged if inherits_columns_from misses', async () => {
708
- const standalonePath = 'semantic-layer/conn-1/aav_consignments.yaml';
709
- configService.listFiles.mockImplementation((dir) => {
710
- if (dir === 'semantic-layer/conn-1') {
711
- return Promise.resolve({ files: [standalonePath] });
712
- }
713
- return Promise.resolve({ files: [] });
714
- });
715
- configService.readFile.mockResolvedValue({
716
- content: [
717
- 'name: aav_consignments',
718
- 'sql: SELECT 1',
719
- 'inherits_columns_from: NO_SUCH_TABLE',
720
- 'grain: [FOO]',
721
- 'columns:',
722
- ' - { name: FOO, type: string }',
723
- ].join('\n'),
724
- });
725
- const { sources, loadErrors } = await service.loadAllSources('conn-1');
726
- expect(loadErrors).toEqual([]);
727
- const aav = sources.find((s) => s.name === 'aav_consignments');
728
- expect(aav?.columns).toEqual([{ name: 'FOO', type: 'string' }]);
729
- });
730
- it('loads standalone source and column description maps', async () => {
731
- const standalonePath = 'semantic-layer/conn-1/orders.yaml';
732
- configService.listFiles.mockResolvedValue({ files: [standalonePath] });
733
- configService.readFile.mockResolvedValue({
734
- content: [
735
- 'name: orders',
736
- 'descriptions:',
737
- ' user: Finance orders used for invoice reconciliation.',
738
- 'table: public.orders',
739
- 'grain: [id]',
740
- 'columns:',
741
- ' - name: id',
742
- ' type: string',
743
- ' descriptions:',
744
- ' user: Stable order identifier.',
745
- ].join('\n'),
746
- });
747
- const { sources, loadErrors } = await service.loadAllSources('conn-1');
748
- expect(loadErrors).toEqual([]);
749
- expect(sources[0]).toMatchObject({
750
- name: 'orders',
751
- descriptions: { user: 'Finance orders used for invoice reconciliation.' },
752
- columns: [{ name: 'id', type: 'string', descriptions: { user: 'Stable order identifier.' } }],
753
- });
754
- });
755
- it('reports file-attributed errors for legacy overlay column patches', async () => {
756
- const schemaPath = 'semantic-layer/conn-1/_schema/marts.yaml';
757
- const overlayPath = 'semantic-layer/conn-1/orders.yaml';
758
- configService.listFiles.mockResolvedValue({ files: [schemaPath, overlayPath] });
759
- configService.readFile.mockImplementation((path) => {
760
- if (path === schemaPath) {
761
- return Promise.resolve({
762
- content: [
763
- 'tables:',
764
- ' orders:',
765
- ' table: public.orders',
766
- ' columns:',
767
- ' - { name: id, type: string, pk: true }',
768
- ].join('\n'),
769
- });
770
- }
771
- return Promise.resolve({
772
- content: ['name: orders', 'columns:', ' - name: id', ' descriptions: { user: "Stable id." }'].join('\n'),
773
- });
774
- });
775
- const { loadErrors } = await service.loadAllSources('conn-1');
776
- expect(loadErrors.join('\n')).toContain(overlayPath);
777
- expect(loadErrors.join('\n')).toContain("move it to 'column_overrides:'");
778
- });
779
- it('reports and logs directory listing failures instead of treating them as empty sources', async () => {
780
- const logger = { log: vi.fn(), warn: vi.fn(), error: vi.fn() };
781
- configService.listFiles.mockRejectedValue(new Error('permission denied'));
782
- service = new SemanticLayerService(configService, connectionCatalog(), pythonPort, logger);
783
- const { sources, loadErrors } = await service.loadAllSources('conn-1');
784
- expect(sources).toEqual([]);
785
- expect(loadErrors).toEqual([
786
- 'Failed to list semantic-layer files under semantic-layer/conn-1: permission denied',
787
- ]);
788
- expect(logger.warn).toHaveBeenCalledWith('Failed to list semantic-layer files under semantic-layer/conn-1: permission denied');
789
- });
790
- });
791
- describe('validateWithProposedSource', () => {
792
- let configService;
793
- let service;
794
- beforeEach(() => {
795
- pythonPort.validateSources.mockReset();
796
- configService = {
797
- listFiles: vi.fn().mockResolvedValue({
798
- files: [],
799
- }),
800
- readFile: vi.fn(),
801
- };
802
- service = new SemanticLayerService(configService, connectionCatalog('BIGQUERY'), pythonPort);
803
- });
804
- it('uses the connection warehouse dialect, not hardcoded postgres', async () => {
805
- pythonPort.validateSources.mockResolvedValue({
806
- data: { errors: [], warnings: [] },
807
- });
808
- await service.validateWithProposedSource('conn-1', {
809
- name: 'std',
810
- table: 'analytics.std',
811
- grain: ['id'],
812
- columns: [{ name: 'id', type: 'number' }],
813
- joins: [],
814
- measures: [],
815
- });
816
- expect(pythonPort.validateSources).toHaveBeenCalledWith(expect.objectContaining({
817
- dialect: 'bigquery',
818
- }));
819
- });
820
- it('composes a bare overlay with its manifest base before validating', async () => {
821
- const schemaPath = 'semantic-layer/conn-1/_schema/core.yaml';
822
- const listFilesImpl = (dir) => {
823
- if (dir === 'semantic-layer/conn-1') {
824
- return Promise.resolve({ files: [schemaPath, 'semantic-layer/conn-1/fct_orders.yaml'] });
825
- }
826
- if (dir === 'semantic-layer/conn-1/_schema') {
827
- return Promise.resolve({ files: [schemaPath] });
828
- }
829
- return Promise.resolve({ files: [] });
830
- };
831
- const readFileImpl = (path) => {
832
- if (path === schemaPath) {
833
- return Promise.resolve({
834
- content: [
835
- 'tables:',
836
- ' fct_orders:',
837
- ' table: analytics.fct_orders',
838
- ' columns:',
839
- ' - { name: id, type: string, pk: true }',
840
- ' - { name: amount, type: number }',
841
- ].join('\n'),
842
- });
843
- }
844
- if (path === 'semantic-layer/conn-1/fct_orders.yaml') {
845
- return Promise.resolve({ content: 'name: fct_orders\nmeasures: []\n' });
846
- }
847
- return Promise.reject(new Error(`Unexpected readFile: ${path}`));
848
- };
849
- configService.listFiles.mockImplementation(listFilesImpl);
850
- configService.readFile.mockImplementation(readFileImpl);
851
- pythonPort.validateSources.mockResolvedValue({
852
- data: { errors: [], warnings: [] },
853
- });
854
- const overlay = {
855
- name: 'fct_orders',
856
- grain: ['id'],
857
- columns: [],
858
- joins: [],
859
- measures: [{ name: 'total_amount', expr: 'sum(amount)' }],
860
- };
861
- await service.validateWithProposedSource('conn-1', overlay);
862
- expect(pythonPort.validateSources).toHaveBeenCalledTimes(1);
863
- const sources = (pythonPort.validateSources.mock.calls[0][0]?.sources ?? []);
864
- const composed = sources.find((s) => s.name === 'fct_orders');
865
- expect(composed).toBeDefined();
866
- expect(composed?.table).toBe('analytics.fct_orders');
867
- expect(composed?.measures).toEqual([{ name: 'total_amount', expr: 'sum(amount)' }]);
868
- });
869
- it('returns a pointed error when a bare overlay has no manifest base', async () => {
870
- configService.listFiles.mockResolvedValue({ files: [] });
871
- const overlay = {
872
- name: 'orphan',
873
- grain: [],
874
- columns: [],
875
- joins: [],
876
- measures: [],
877
- };
878
- const result = await service.validateWithProposedSource('conn-1', overlay);
879
- expect(result.errors[0]).toMatch(/Overlay 'orphan' has no matching manifest entry/);
880
- expect(pythonPort.validateSources).not.toHaveBeenCalled();
881
- });
882
- it('rejects table-backed sources whose declared columns are absent from a matching physical manifest', async () => {
883
- const schemaPath = 'semantic-layer/postgres-warehouse/_schema/orbit_analytics.yaml';
884
- configService.listFiles.mockImplementation((dir) => {
885
- if (dir === 'semantic-layer/dbt-main') {
886
- return Promise.resolve({ files: [] });
887
- }
888
- if (dir === 'semantic-layer') {
889
- return Promise.resolve({ files: [schemaPath] });
890
- }
891
- if (dir === 'semantic-layer/dbt-main/_schema' || dir === 'semantic-layer/postgres-warehouse/_schema') {
892
- return Promise.resolve({ files: dir.endsWith('postgres-warehouse/_schema') ? [schemaPath] : [] });
893
- }
894
- return Promise.resolve({ files: [] });
895
- });
896
- configService.readFile.mockImplementation((path) => {
897
- if (path === schemaPath) {
898
- return Promise.resolve({
899
- content: [
900
- 'tables:',
901
- ' int_procurement_qualifying_actions:',
902
- ' table: orbit_analytics.int_procurement_qualifying_actions',
903
- ' columns:',
904
- ' - { name: action_id, type: string }',
905
- ' - { name: account_id, type: string }',
906
- ' - { name: user_id, type: string }',
907
- ' - { name: action_date, type: time }',
908
- ' - { name: action_type, type: string }',
909
- ].join('\n'),
910
- });
911
- }
912
- return Promise.reject(new Error(`Unexpected readFile: ${path}`));
913
- });
914
- pythonPort.validateSources.mockResolvedValue({
915
- data: { errors: [], warnings: [] },
916
- });
917
- const result = await service.validateWithProposedSource('dbt-main', {
918
- name: 'int_procurement_qualifying_actions',
919
- table: 'orbit_analytics.int_procurement_qualifying_actions',
920
- grain: ['purchase_request_id'],
921
- columns: [
922
- { name: 'purchase_request_id', type: 'string' },
923
- { name: 'account_id', type: 'string' },
924
- { name: 'requester_user_id', type: 'string' },
925
- { name: 'action_week', type: 'time' },
926
- ],
927
- joins: [],
928
- measures: [{ name: 'qualifying_action_count', expr: 'count(purchase_request_id)' }],
929
- });
930
- expect(result.errors.join('\n')).toMatch(/declared column\(s\) absent from physical table/);
931
- expect(result.errors.join('\n')).toMatch(/purchase_request_id/);
932
- expect(result.errors.join('\n')).toMatch(/requester_user_id/);
933
- expect(result.errors.join('\n')).toMatch(/action_week/);
934
- expect(result.errors.join('\n')).toMatch(/measure "qualifying_action_count" references unknown column\(s\)/);
935
- });
936
- it('keeps valid table-backed sources clean when a physical manifest matches', async () => {
937
- const schemaPath = 'semantic-layer/postgres-warehouse/_schema/orbit_analytics.yaml';
938
- configService.listFiles.mockImplementation((dir) => {
939
- if (dir === 'semantic-layer/dbt-main') {
940
- return Promise.resolve({ files: [] });
941
- }
942
- if (dir === 'semantic-layer') {
943
- return Promise.resolve({ files: [schemaPath] });
944
- }
945
- if (dir === 'semantic-layer/dbt-main/_schema' || dir === 'semantic-layer/postgres-warehouse/_schema') {
946
- return Promise.resolve({ files: dir.endsWith('postgres-warehouse/_schema') ? [schemaPath] : [] });
947
- }
948
- return Promise.resolve({ files: [] });
949
- });
950
- configService.readFile.mockResolvedValue({
951
- content: [
952
- 'tables:',
953
- ' mart_revenue_daily:',
954
- ' table: orbit_analytics.mart_revenue_daily',
955
- ' columns:',
956
- ' - { name: revenue_date, type: time }',
957
- ' - { name: gross_revenue_cents, type: number }',
958
- ' - { name: credits_cents, type: number }',
959
- ' - { name: refunds_cents, type: number }',
960
- ' - { name: net_revenue_cents, type: number }',
961
- ].join('\n'),
962
- });
963
- pythonPort.validateSources.mockResolvedValue({
964
- data: { errors: [], warnings: [] },
965
- });
966
- const result = await service.validateWithProposedSource('dbt-main', {
967
- name: 'mart_revenue_daily',
968
- table: 'orbit_analytics.mart_revenue_daily',
969
- grain: ['revenue_date'],
970
- columns: [
971
- { name: 'revenue_date', type: 'time' },
972
- { name: 'gross_revenue_cents', type: 'number' },
973
- { name: 'credits_cents', type: 'number' },
974
- { name: 'refunds_cents', type: 'number' },
975
- { name: 'net_revenue_cents', type: 'number' },
976
- ],
977
- joins: [],
978
- measures: [{ name: 'net_revenue', expr: 'sum(net_revenue_cents)' }],
979
- });
980
- expect(result.errors).toEqual([]);
981
- });
982
- it('allows SQL syntax tokens and cast types in physical expression validation', async () => {
983
- const schemaPath = 'semantic-layer/postgres-warehouse/_schema/orbit_analytics.yaml';
984
- configService.listFiles.mockImplementation((dir) => {
985
- if (dir === 'semantic-layer/dbt-main') {
986
- return Promise.resolve({ files: [] });
987
- }
988
- if (dir === 'semantic-layer') {
989
- return Promise.resolve({ files: [schemaPath] });
990
- }
991
- if (dir === 'semantic-layer/dbt-main/_schema' || dir === 'semantic-layer/postgres-warehouse/_schema') {
992
- return Promise.resolve({ files: dir.endsWith('postgres-warehouse/_schema') ? [schemaPath] : [] });
993
- }
994
- return Promise.resolve({ files: [] });
995
- });
996
- configService.readFile.mockResolvedValue({
997
- content: [
998
- 'tables:',
999
- ' mart_revenue_daily:',
1000
- ' table: orbit_analytics.mart_revenue_daily',
1001
- ' columns:',
1002
- ' - { name: order_id, type: string }',
1003
- ' - { name: revenue_date, type: time }',
1004
- ' - { name: amount, type: number }',
1005
- ' - { name: status, type: string }',
1006
- ' - { name: created_at, type: time }',
1007
- ].join('\n'),
1008
- });
1009
- pythonPort.validateSources.mockResolvedValue({
1010
- data: { errors: [], warnings: [] },
1011
- });
1012
- const result = await service.validateWithProposedSource('dbt-main', {
1013
- name: 'mart_revenue_daily',
1014
- table: 'orbit_analytics.mart_revenue_daily',
1015
- grain: ['order_id'],
1016
- columns: [
1017
- { name: 'order_id', type: 'string' },
1018
- { name: 'revenue_date', type: 'time' },
1019
- { name: 'amount', type: 'number' },
1020
- { name: 'status', type: 'string' },
1021
- { name: 'created_at', type: 'time' },
1022
- { name: 'status_text', type: 'string', expr: 'status::text' },
1023
- ],
1024
- segments: [{ name: 'current_or_paid', expr: "created_at <= current_date OR status = 'paid'" }],
1025
- joins: [],
1026
- measures: [
1027
- { name: 'paid_amount', expr: "sum(amount) FILTER (WHERE status = 'paid')" },
1028
- { name: 'cast_amount_count', expr: 'count(cast(amount as text))' },
1029
- ],
1030
- });
1031
- expect(result.errors).toEqual([]);
1032
- });
1033
- it('rejects join keys that are absent from matched physical sources', async () => {
1034
- const schemaPath = 'semantic-layer/postgres-warehouse/_schema/orbit_analytics.yaml';
1035
- configService.listFiles.mockImplementation((dir) => {
1036
- if (dir === 'semantic-layer/dbt-main') {
1037
- return Promise.resolve({ files: [] });
1038
- }
1039
- if (dir === 'semantic-layer') {
1040
- return Promise.resolve({ files: [schemaPath] });
1041
- }
1042
- if (dir === 'semantic-layer/dbt-main/_schema' || dir === 'semantic-layer/postgres-warehouse/_schema') {
1043
- return Promise.resolve({ files: dir.endsWith('postgres-warehouse/_schema') ? [schemaPath] : [] });
1044
- }
1045
- return Promise.resolve({ files: [] });
1046
- });
1047
- configService.readFile.mockResolvedValue({
1048
- content: [
1049
- 'tables:',
1050
- ' activity:',
1051
- ' table: orbit_analytics.activity',
1052
- ' columns:',
1053
- ' - { name: account_id, type: string }',
1054
- ' accounts:',
1055
- ' table: orbit_analytics.accounts',
1056
- ' columns:',
1057
- ' - { name: account_id, type: string }',
1058
- ].join('\n'),
1059
- });
1060
- pythonPort.validateSources.mockResolvedValue({
1061
- data: { errors: [], warnings: [] },
1062
- });
1063
- const result = await service.validateWithProposedSource('dbt-main', {
1064
- name: 'activity',
1065
- table: 'orbit_analytics.activity',
1066
- grain: ['account_id'],
1067
- columns: [{ name: 'account_id', type: 'string' }],
1068
- joins: [{ to: 'accounts', on: 'activity.account_name = accounts.account_uuid', relationship: 'many_to_one' }],
1069
- measures: [],
1070
- });
1071
- expect(result.errors.join('\n')).toMatch(/local column "account_name"/);
1072
- expect(result.errors.join('\n')).toMatch(/target column "account_uuid"/);
1073
- });
1074
- });
1075
- describe('findDanglingSegmentRefs', () => {
1076
- it('returns empty when every measure segment resolves', () => {
1077
- const source = {
1078
- segments: [{ name: 'byol' }, { name: 'paid' }],
1079
- measures: [
1080
- { name: 'byol_count', segments: ['byol'] },
1081
- { name: 'paid_count', segments: ['paid', 'byol'] },
1082
- ],
1083
- };
1084
- expect(findDanglingSegmentRefs(source)).toEqual([]);
1085
- });
1086
- it('flags measures whose segment reference does not exist on the source', () => {
1087
- const source = {
1088
- segments: [{ name: 'byol' }],
1089
- measures: [{ name: 'broken', segments: ['byol', 'missing'] }],
1090
- };
1091
- const refs = findDanglingSegmentRefs(source);
1092
- expect(refs).toHaveLength(1);
1093
- expect(refs[0]).toMatch(/measure 'broken' references unknown segment 'missing'/);
1094
- });
1095
- it('flags when a source has zero segments but measures reference one', () => {
1096
- const source = {
1097
- measures: [{ name: 'broken', segments: ['byol'] }],
1098
- };
1099
- const refs = findDanglingSegmentRefs(source);
1100
- expect(refs).toHaveLength(1);
1101
- expect(refs[0]).toMatch(/unknown segment 'byol'/);
1102
- });
1103
- it('is a no-op for sources with no measures or no segment references', () => {
1104
- expect(findDanglingSegmentRefs({ measures: [{ name: 'simple', expr: 'count(*)' }] })).toEqual([]);
1105
- expect(findDanglingSegmentRefs({})).toEqual([]);
1106
- });
1107
- });