@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,1928 +0,0 @@
1
- import { mkdir, mkdtemp, readFile, readdir, rm, writeFile } from 'node:fs/promises';
2
- import { tmpdir } from 'node:os';
3
- import { join } from 'node:path';
4
- import { describe, expect, it, vi } from 'vitest';
5
- import { GitService } from '../../context/core/git.service.js';
6
- import { SessionWorktreeService } from '../../context/core/session-worktree.service.js';
7
- import { LocalGitFileStore } from '../project/local-git-file-store.js';
8
- import { addTouchedSlSource } from '../../context/tools/touched-sl-sources.js';
9
- import { IngestBundleRunner } from './ingest-bundle.runner.js';
10
- async function makeRealGitRuntime() {
11
- const homeDir = await mkdtemp(join(tmpdir(), 'ktx-isolated-runner-'));
12
- const configDir = join(homeDir, 'config');
13
- const git = new GitService({
14
- storage: { configDir, homeDir },
15
- git: {
16
- userName: 'System User',
17
- userEmail: 'system@example.com',
18
- bootstrapMessage: 'init',
19
- bootstrapAuthor: 'system',
20
- bootstrapAuthorEmail: 'system@example.com',
21
- },
22
- });
23
- await git.onModuleInit();
24
- const configService = new LocalGitFileStore({ rootDir: configDir, git });
25
- const sessionWorktreeService = new SessionWorktreeService({
26
- coreConfig: {
27
- storage: { configDir, homeDir },
28
- git: {
29
- userName: 'System User',
30
- userEmail: 'system@example.com',
31
- bootstrapMessage: 'init',
32
- bootstrapAuthor: 'system',
33
- bootstrapAuthorEmail: 'system@example.com',
34
- },
35
- },
36
- gitService: git,
37
- configService,
38
- });
39
- return { homeDir, configDir, git, configService, sessionWorktreeService };
40
- }
41
- function rootOfConfig(configService, fallback) {
42
- const rootDir = configService.rootDir;
43
- return typeof rootDir === 'string' ? rootDir : fallback;
44
- }
45
- async function loadSourcesFromRoot(root) {
46
- const raw = await readFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'utf-8').catch(() => '');
47
- const hasCents = raw.includes('total_contract_arr_cents');
48
- const hasDollars = raw.includes('total_contract_arr');
49
- return {
50
- sources: hasCents || hasDollars
51
- ? [
52
- {
53
- name: 'mart_account_segments',
54
- grain: ['account_id'],
55
- columns: [{ name: 'account_id', type: 'string' }],
56
- joins: [],
57
- measures: [{ name: hasCents ? 'total_contract_arr_cents' : 'total_contract_arr', expr: 'sum(contract_arr)' }],
58
- table: 'analytics.mart_account_segments',
59
- },
60
- ]
61
- : [],
62
- loadErrors: [],
63
- };
64
- }
65
- async function listGlobalWikiPageKeys(root) {
66
- const dir = join(root, 'wiki/global');
67
- const entries = await readdir(dir).catch(() => []);
68
- return entries
69
- .filter((entry) => entry.endsWith('.md'))
70
- .map((entry) => entry.slice(0, -'.md'.length))
71
- .sort();
72
- }
73
- function frontmatterList(yaml, key) {
74
- const pattern = new RegExp(`(?:^|\\n)${key}:\\n((?: - .+\\n?)*)`);
75
- return (pattern
76
- .exec(yaml)?.[1]
77
- ?.split('\n')
78
- .map((line) => line.trim().replace(/^- /, ''))
79
- .filter(Boolean) ?? []);
80
- }
81
- function legacyFallbackSettingKey() {
82
- return ['sharedWorktree', 'SourceKeys'].join('');
83
- }
84
- function legacySharedTraceEvent() {
85
- return ['shared', 'worktree', 'path', 'enabled'].join('_');
86
- }
87
- function makeWikiService(root) {
88
- return {
89
- listPageKeys: vi.fn(async (scope) => (scope === 'GLOBAL' ? listGlobalWikiPageKeys(root) : [])),
90
- readPage: vi.fn(async (_scope, _scopeId, key) => {
91
- const path = join(root, 'wiki/global', `${key}.md`);
92
- const raw = await readFile(path, 'utf-8').catch(() => null);
93
- if (!raw) {
94
- return null;
95
- }
96
- const [, yaml = '', content = ''] = /^---\n([\s\S]*?)\n---\n?([\s\S]*)$/.exec(raw) ?? [];
97
- return {
98
- pageKey: key,
99
- frontmatter: {
100
- summary: key,
101
- usage_mode: 'auto',
102
- refs: frontmatterList(yaml, 'refs'),
103
- sl_refs: frontmatterList(yaml, 'sl_refs'),
104
- },
105
- content: content.trim(),
106
- };
107
- }),
108
- writePage: vi.fn(async (_scope, _scopeId, key, frontmatter, content) => {
109
- await mkdir(join(root, 'wiki/global'), { recursive: true });
110
- const refs = (frontmatter.refs ?? []).map((ref) => ` - ${ref}`).join('\n');
111
- const slRefs = (frontmatter.sl_refs ?? []).map((ref) => ` - ${ref}`).join('\n');
112
- await writeFile(join(root, 'wiki/global', `${key}.md`), [
113
- '---',
114
- `summary: ${frontmatter.summary ?? key}`,
115
- `usage_mode: ${frontmatter.usage_mode ?? 'auto'}`,
116
- 'refs:',
117
- refs,
118
- 'sl_refs:',
119
- slRefs,
120
- '---',
121
- '',
122
- content,
123
- '',
124
- ].join('\n'));
125
- }),
126
- syncFromCommit: vi.fn(),
127
- };
128
- }
129
- function makeDeps(runtime, sourceKey = 'metabase', settings = {}) {
130
- const adapter = {
131
- source: sourceKey,
132
- skillNames: [],
133
- detect: vi.fn().mockResolvedValue(true),
134
- chunk: vi.fn().mockResolvedValue({
135
- workUnits: [
136
- { unitKey: 'card-wiki', rawFiles: ['cards/wiki.json'], peerFileIndex: [], dependencyPaths: [] },
137
- { unitKey: 'card-source', rawFiles: ['cards/source.json'], peerFileIndex: [], dependencyPaths: [] },
138
- ],
139
- }),
140
- };
141
- const wikiService = makeWikiService(runtime.configDir);
142
- const semanticLayerService = {
143
- loadAllSources: vi.fn(async () => loadSourcesFromRoot(runtime.configDir)),
144
- listFilesForConnection: vi.fn().mockResolvedValue(['mart_account_segments.yaml']),
145
- };
146
- semanticLayerService.forWorktree = vi.fn((workdir) => ({
147
- ...semanticLayerService,
148
- loadAllSources: vi.fn(async () => loadSourcesFromRoot(workdir)),
149
- listFilesForConnection: vi.fn().mockResolvedValue(['mart_account_segments.yaml']),
150
- }));
151
- const deps = {
152
- runs: { create: vi.fn().mockResolvedValue({ id: 'run-1' }), markCompleted: vi.fn(), markFailed: vi.fn() },
153
- provenance: {
154
- insertMany: vi.fn(),
155
- findLatestHashesForCompletedSyncs: vi.fn().mockResolvedValue(new Map()),
156
- findLatestArtifactsForRawPaths: vi.fn().mockResolvedValue(new Map()),
157
- },
158
- reports: { create: vi.fn().mockResolvedValue({ id: 'report-1' }), findByJobId: vi.fn().mockResolvedValue(null), markSuperseded: vi.fn() },
159
- canonicalPins: { listPins: vi.fn().mockResolvedValue([]) },
160
- registry: { get: vi.fn().mockReturnValue(adapter), register: vi.fn(), has: vi.fn(), list: vi.fn() },
161
- diffSetService: {
162
- compute: vi.fn().mockResolvedValue({ added: ['cards/wiki.json', 'cards/source.json'], modified: [], deleted: [], unchanged: [] }),
163
- },
164
- sessionWorktreeService: runtime.sessionWorktreeService,
165
- agentRunner: { runLoop: vi.fn() },
166
- gitService: runtime.git,
167
- lockingService: { withLock: vi.fn(async (_key, fn) => fn()) },
168
- storage: {
169
- homeDir: join(runtime.configDir, '.ktx'),
170
- systemGitAuthor: { name: 'KTX Test', email: 'system@ktx.local' },
171
- resolveUploadDir: (id) => join(runtime.homeDir, 'upload', id),
172
- resolvePullDir: (id) => join(runtime.homeDir, 'pull', id),
173
- resolveTranscriptDir: (id) => join(runtime.configDir, '.ktx/ingest-transcripts', id),
174
- resolveTracePath: (id) => join(runtime.configDir, '.ktx/ingest-traces', id, 'trace.jsonl'),
175
- },
176
- settings: {
177
- memoryIngestionModel: 'test',
178
- probeRowCount: 1,
179
- ingestTraceLevel: 'trace',
180
- ...settings,
181
- },
182
- skillsRegistry: {
183
- listSkills: vi.fn().mockResolvedValue([]),
184
- getSkill: vi.fn().mockResolvedValue(null),
185
- buildSkillsPrompt: vi.fn().mockReturnValue(''),
186
- stripFrontmatter: vi.fn((body) => body),
187
- },
188
- promptService: { loadPrompt: vi.fn().mockResolvedValue('base') },
189
- wikiService: { ...wikiService, forWorktree: vi.fn((workdir) => makeWikiService(workdir)) },
190
- knowledgeIndex: { listPagesForUser: vi.fn().mockResolvedValue([]) },
191
- knowledgeSlRefs: { syncFromWiki: vi.fn() },
192
- semanticLayerService,
193
- slSearchService: { indexSources: vi.fn() },
194
- slSourcesRepository: {},
195
- slValidator: { validateSingleSource: vi.fn().mockResolvedValue({ errors: [], warnings: [] }) },
196
- connections: { listEnabledConnections: vi.fn().mockResolvedValue([]), getConnectionById: vi.fn() },
197
- toolsetFactory: { createIngestWuToolset: vi.fn(() => ({ toRuntimeTools: vi.fn(() => ({})) })) },
198
- commitMessages: { enqueueForExternalCommit: vi.fn() },
199
- embedding: { maxBatchSize: 64, computeEmbedding: vi.fn(), computeEmbeddingsBulk: vi.fn() },
200
- };
201
- return { deps, adapter };
202
- }
203
- async function mockStageRawFiles(runner, runtime, hashes, sourceKey = 'metabase') {
204
- runner.resolveStagedDir = vi.fn().mockResolvedValue(join(runtime.homeDir, 'stage'));
205
- runner.stageRawFilesStage1 = vi.fn(async ({ worktreeRoot }) => {
206
- const rawDir = join(worktreeRoot, 'raw-sources/warehouse', sourceKey, 's');
207
- await mkdir(rawDir, { recursive: true });
208
- for (const [rawPath] of hashes) {
209
- await mkdir(join(rawDir, rawPath.split('/').slice(0, -1).join('/')), { recursive: true });
210
- await writeFile(join(rawDir, rawPath), '{}');
211
- }
212
- return { currentHashes: new Map(hashes), rawDirInWorktree: `raw-sources/warehouse/${sourceKey}/s` };
213
- });
214
- }
215
- describe('IngestBundleRunner isolated diff path', () => {
216
- it('routes an unlisted direct-writing source through isolated diffs by default', async () => {
217
- const runtime = await makeRealGitRuntime();
218
- try {
219
- const sourceKey = 'custom-direct-source';
220
- const { deps, adapter } = makeDeps(runtime, sourceKey);
221
- adapter.chunk.mockResolvedValue({
222
- workUnits: [
223
- {
224
- unitKey: 'custom-wiki',
225
- rawFiles: ['custom/page.json'],
226
- peerFileIndex: [],
227
- dependencyPaths: [],
228
- },
229
- ],
230
- });
231
- let currentSession = null;
232
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
233
- currentSession = toolSession;
234
- return { toRuntimeTools: vi.fn(() => ({})) };
235
- });
236
- deps.agentRunner.runLoop = vi.fn(async (params) => {
237
- if (params.telemetryTags.operationName !== 'ingest-bundle-wu') {
238
- return { stopReason: 'natural' };
239
- }
240
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
241
- await mkdir(join(root, 'wiki/global'), { recursive: true });
242
- await writeFile(join(root, 'wiki/global/custom-isolated.md'), '---\nsummary: Custom isolated write\nusage_mode: auto\n---\n\nCustom isolated write.\n', 'utf-8');
243
- currentSession.actions.push({
244
- target: 'wiki',
245
- type: 'created',
246
- key: 'custom-isolated',
247
- detail: 'Custom isolated write',
248
- rawPaths: ['custom/page.json'],
249
- });
250
- await currentSession.gitService.commitFiles(['wiki/global/custom-isolated.md'], 'custom wiki', 'KTX Test', 'system@ktx.local');
251
- return { stopReason: 'natural' };
252
- });
253
- const runner = new IngestBundleRunner(deps);
254
- await mockStageRawFiles(runner, runtime, [['custom/page.json', 'h1']], sourceKey);
255
- await expect(runner.run({
256
- jobId: 'job-custom-default',
257
- connectionId: 'warehouse',
258
- sourceKey,
259
- trigger: 'upload',
260
- bundleRef: { kind: 'upload', uploadId: 'upload' },
261
- })).resolves.toMatchObject({
262
- jobId: 'job-custom-default',
263
- failedWorkUnits: [],
264
- workUnitCount: 1,
265
- });
266
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-custom-default/trace.jsonl'), 'utf-8');
267
- expect(trace).toContain('isolated_diff_enabled');
268
- expect(trace).toContain('work_unit_child_created');
269
- expect(trace).not.toContain(legacySharedTraceEvent());
270
- const reportCreate = vi.mocked(deps.reports.create).mock.calls.at(-1)?.[0];
271
- const reportBody = reportCreate?.body;
272
- expect(reportBody?.isolatedDiff).toMatchObject({
273
- enabled: true,
274
- acceptedPatches: 1,
275
- });
276
- }
277
- finally {
278
- await rm(runtime.homeDir, { recursive: true, force: true });
279
- }
280
- });
281
- it('does not support shared-worktree fallback settings', async () => {
282
- const runtime = await makeRealGitRuntime();
283
- try {
284
- const sourceKey = 'legacy-source';
285
- const staleSettings = {
286
- [legacyFallbackSettingKey()]: ['legacy-source'],
287
- };
288
- const { deps, adapter } = makeDeps(runtime, sourceKey, staleSettings);
289
- adapter.chunk.mockResolvedValue({
290
- workUnits: [
291
- {
292
- unitKey: 'legacy-wiki',
293
- rawFiles: ['legacy/page.json'],
294
- peerFileIndex: [],
295
- dependencyPaths: [],
296
- },
297
- ],
298
- });
299
- let currentSession = null;
300
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
301
- currentSession = toolSession;
302
- return { toRuntimeTools: vi.fn(() => ({})) };
303
- });
304
- deps.agentRunner.runLoop = vi.fn(async (params) => {
305
- if (params.telemetryTags.operationName !== 'ingest-bundle-wu') {
306
- return { stopReason: 'natural' };
307
- }
308
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
309
- await mkdir(join(root, 'wiki/global'), { recursive: true });
310
- await writeFile(join(root, 'wiki/global/legacy-isolated.md'), '---\nsummary: Legacy isolated write\nusage_mode: auto\n---\n\nLegacy isolated write.\n', 'utf-8');
311
- currentSession.actions.push({
312
- target: 'wiki',
313
- type: 'created',
314
- key: 'legacy-isolated',
315
- detail: 'Legacy isolated write',
316
- rawPaths: ['legacy/page.json'],
317
- });
318
- await currentSession.gitService.commitFiles(['wiki/global/legacy-isolated.md'], 'legacy isolated wiki', 'KTX Test', 'system@ktx.local');
319
- return { stopReason: 'natural' };
320
- });
321
- const runner = new IngestBundleRunner(deps);
322
- await mockStageRawFiles(runner, runtime, [['legacy/page.json', 'h1']], sourceKey);
323
- await expect(runner.run({
324
- jobId: 'job-legacy-isolated',
325
- connectionId: 'warehouse',
326
- sourceKey,
327
- trigger: 'upload',
328
- bundleRef: { kind: 'upload', uploadId: 'upload' },
329
- })).resolves.toMatchObject({
330
- jobId: 'job-legacy-isolated',
331
- failedWorkUnits: [],
332
- workUnitCount: 1,
333
- });
334
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-legacy-isolated/trace.jsonl'), 'utf-8');
335
- expect(trace).toContain('isolated_diff_enabled');
336
- expect(trace).toContain('work_unit_child_created');
337
- expect(trace).not.toContain(legacySharedTraceEvent());
338
- const reportCreate = vi.mocked(deps.reports.create).mock.calls.at(-1)?.[0];
339
- const reportBody = reportCreate?.body;
340
- expect(reportBody?.isolatedDiff).toMatchObject({
341
- enabled: true,
342
- acceptedPatches: 1,
343
- });
344
- }
345
- finally {
346
- await rm(runtime.homeDir, { recursive: true, force: true });
347
- }
348
- });
349
- it('does not integrate failed isolated WorkUnit patches', async () => {
350
- const runtime = await makeRealGitRuntime();
351
- try {
352
- const { deps, adapter } = makeDeps(runtime, 'fake');
353
- adapter.chunk.mockResolvedValue({
354
- workUnits: [
355
- { unitKey: 'wu-good', rawFiles: ['good.raw'], peerFileIndex: [], dependencyPaths: [] },
356
- { unitKey: 'wu-bad', rawFiles: ['bad.raw'], peerFileIndex: [], dependencyPaths: [] },
357
- ],
358
- });
359
- deps.diffSetService.compute = vi.fn().mockResolvedValue({
360
- added: ['good.raw', 'bad.raw'],
361
- modified: [],
362
- deleted: [],
363
- unchanged: [],
364
- });
365
- deps.slValidator.validateSingleSource = vi.fn(async (_validationDeps, _connectionId, sourceName) => ({
366
- errors: sourceName === 'bad' ? [{ message: 'bad source rejected' }] : [],
367
- warnings: [],
368
- }));
369
- let currentSession = null;
370
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
371
- currentSession = toolSession;
372
- return { toRuntimeTools: vi.fn(() => ({})) };
373
- });
374
- deps.agentRunner.runLoop = vi.fn(async (params) => {
375
- if (params.telemetryTags.operationName !== 'ingest-bundle-wu') {
376
- return { stopReason: 'natural' };
377
- }
378
- const unitKey = params.telemetryTags.unitKey;
379
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
380
- await mkdir(join(root, 'semantic-layer/warehouse'), { recursive: true });
381
- if (unitKey === 'wu-good') {
382
- await writeFile(join(root, 'semantic-layer/warehouse/good.yaml'), 'name: good\n', 'utf-8');
383
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'good');
384
- currentSession.actions.push({
385
- target: 'sl',
386
- type: 'created',
387
- key: 'good',
388
- detail: 'good source',
389
- targetConnectionId: 'warehouse',
390
- rawPaths: ['good.raw'],
391
- });
392
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/good.yaml'], 'test: add good source', 'KTX Test', 'system@ktx.local');
393
- }
394
- if (unitKey === 'wu-bad') {
395
- await writeFile(join(root, 'semantic-layer/warehouse/bad.yaml'), 'name: bad\n', 'utf-8');
396
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'bad');
397
- currentSession.actions.push({
398
- target: 'sl',
399
- type: 'created',
400
- key: 'bad',
401
- detail: 'bad source',
402
- targetConnectionId: 'warehouse',
403
- rawPaths: ['bad.raw'],
404
- });
405
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/bad.yaml'], 'test: add bad source', 'KTX Test', 'system@ktx.local');
406
- }
407
- return { stopReason: 'natural' };
408
- });
409
- const runner = new IngestBundleRunner(deps);
410
- await mockStageRawFiles(runner, runtime, [
411
- ['good.raw', 'good-hash'],
412
- ['bad.raw', 'bad-hash'],
413
- ], 'fake');
414
- const result = await runner.run({
415
- jobId: 'job-failed-wu-isolated',
416
- connectionId: 'warehouse',
417
- sourceKey: 'fake',
418
- trigger: 'upload',
419
- bundleRef: { kind: 'upload', uploadId: 'upload' },
420
- });
421
- expect(result.failedWorkUnits).toEqual(['wu-bad']);
422
- await expect(readFile(join(runtime.configDir, 'semantic-layer/warehouse/good.yaml'), 'utf-8')).resolves.toContain('good');
423
- await expect(readFile(join(runtime.configDir, 'semantic-layer/warehouse/bad.yaml'), 'utf-8')).rejects.toThrow();
424
- const reportCreate = vi.mocked(deps.reports.create).mock.calls.at(-1)?.[0];
425
- const reportBody = reportCreate?.body;
426
- expect(reportBody.failedWorkUnits).toEqual(['wu-bad']);
427
- expect(reportBody.isolatedDiff).toMatchObject({ enabled: true, acceptedPatches: 1 });
428
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-failed-wu-isolated/trace.jsonl'), 'utf-8');
429
- expect(trace).toContain('work_unit_failed_before_patch');
430
- expect(trace).toContain('patch_accepted');
431
- expect(trace).not.toContain(legacySharedTraceEvent());
432
- }
433
- finally {
434
- await rm(runtime.homeDir, { recursive: true, force: true });
435
- }
436
- });
437
- it.each(['notion', 'lookml', 'looker', 'dbt', 'metricflow'])('routes %s direct writes through isolated child worktrees', async (sourceKey) => {
438
- const runtime = await makeRealGitRuntime();
439
- try {
440
- const { deps, adapter } = makeDeps(runtime, sourceKey);
441
- adapter.chunk.mockResolvedValue({
442
- workUnits: [
443
- {
444
- unitKey: `${sourceKey}-wiki`,
445
- rawFiles: [`${sourceKey}/page.json`],
446
- peerFileIndex: [],
447
- dependencyPaths: [],
448
- },
449
- ],
450
- });
451
- let currentSession = null;
452
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
453
- currentSession = toolSession;
454
- return { toRuntimeTools: vi.fn(() => ({})) };
455
- });
456
- deps.agentRunner.runLoop = vi.fn(async (params) => {
457
- if (params.telemetryTags.operationName !== 'ingest-bundle-wu') {
458
- return { stopReason: 'natural' };
459
- }
460
- expect(params.telemetryTags).toMatchObject({
461
- operationName: 'ingest-bundle-wu',
462
- source: sourceKey,
463
- unitKey: `${sourceKey}-wiki`,
464
- });
465
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
466
- await mkdir(join(root, 'wiki/global'), { recursive: true });
467
- await writeFile(join(root, 'wiki/global', `${sourceKey}-isolated.md`), `---\nsummary: ${sourceKey} isolated write\nusage_mode: auto\n---\n\nIsolated ${sourceKey} write.\n`, 'utf-8');
468
- currentSession.actions.push({
469
- target: 'wiki',
470
- type: 'created',
471
- key: `${sourceKey}-isolated`,
472
- detail: `${sourceKey} isolated write`,
473
- rawPaths: [`${sourceKey}/page.json`],
474
- });
475
- await currentSession.gitService.commitFiles([`wiki/global/${sourceKey}-isolated.md`], `${sourceKey} wiki`, 'KTX Test', 'system@ktx.local');
476
- return { stopReason: 'natural' };
477
- });
478
- const runner = new IngestBundleRunner(deps);
479
- await mockStageRawFiles(runner, runtime, [[`${sourceKey}/page.json`, 'h1']], sourceKey);
480
- await expect(runner.run({
481
- jobId: `job-${sourceKey}`,
482
- connectionId: 'warehouse',
483
- sourceKey,
484
- trigger: 'upload',
485
- bundleRef: { kind: 'upload', uploadId: 'upload' },
486
- })).resolves.toMatchObject({
487
- jobId: `job-${sourceKey}`,
488
- failedWorkUnits: [],
489
- workUnitCount: 1,
490
- });
491
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces', `job-${sourceKey}`, 'trace.jsonl'), 'utf-8');
492
- expect(trace).toContain('isolated_diff_enabled');
493
- expect(trace).toContain('work_unit_child_created');
494
- expect(trace).toContain('work_unit_patch_collected');
495
- expect(trace).toContain('patch_apply_started');
496
- expect(trace).not.toContain(legacySharedTraceEvent());
497
- const reportCreate = vi.mocked(deps.reports.create).mock.calls.at(-1)?.[0];
498
- const reportBody = reportCreate?.body;
499
- expect(reportBody?.isolatedDiff).toMatchObject({
500
- enabled: true,
501
- acceptedPatches: 1,
502
- });
503
- }
504
- finally {
505
- await rm(runtime.homeDir, { recursive: true, force: true });
506
- }
507
- });
508
- it('rejects the Metabase stale-measure wiki body regression before squash', async () => {
509
- const runtime = await makeRealGitRuntime();
510
- try {
511
- const { deps, adapter } = makeDeps(runtime);
512
- adapter.project = vi.fn(async ({ workdir }) => {
513
- await mkdir(join(workdir, 'semantic-layer/warehouse'), { recursive: true });
514
- await writeFile(join(workdir, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr_cents\n expr: sum(contract_arr)\n');
515
- return {
516
- warnings: [],
517
- errors: [],
518
- touchedSources: [{ connectionId: 'warehouse', sourceName: 'mart_account_segments' }],
519
- changedWikiPageKeys: [],
520
- };
521
- });
522
- let currentSession = null;
523
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
524
- currentSession = toolSession;
525
- return { toRuntimeTools: vi.fn(() => ({})) };
526
- });
527
- deps.agentRunner.runLoop = vi.fn(async (params) => {
528
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
529
- if (params.telemetryTags.unitKey === 'card-wiki') {
530
- await mkdir(join(root, 'wiki/global'), { recursive: true });
531
- await writeFile(join(root, 'wiki/global/account-segments.md'), '---\nsummary: Account segments\nusage_mode: auto\nsl_refs:\n - mart_account_segments\n---\n\nARR is `mart_account_segments.total_contract_arr_cents`.\n');
532
- currentSession.actions.push({ target: 'wiki', type: 'created', key: 'account-segments', detail: 'Account segments' });
533
- await currentSession.gitService.commitFiles(['wiki/global/account-segments.md'], 'wu wiki', 'KTX Test', 'system@ktx.local');
534
- }
535
- if (params.telemetryTags.unitKey === 'card-source') {
536
- await writeFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr\n expr: sum(contract_arr)\n');
537
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'mart_account_segments');
538
- currentSession.actions.push({
539
- target: 'sl',
540
- type: 'updated',
541
- key: 'mart_account_segments',
542
- detail: 'Dollar measure',
543
- targetConnectionId: 'warehouse',
544
- });
545
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml'], 'wu source', 'KTX Test', 'system@ktx.local');
546
- }
547
- return { stopReason: 'natural' };
548
- });
549
- const runner = new IngestBundleRunner(deps);
550
- await mockStageRawFiles(runner, runtime, [
551
- ['cards/wiki.json', 'h1'],
552
- ['cards/source.json', 'h2'],
553
- ]);
554
- await expect(runner.run({ jobId: 'job-1', connectionId: 'warehouse', sourceKey: 'metabase', trigger: 'upload', bundleRef: { kind: 'upload', uploadId: 'upload' } })).rejects.toThrow(/total_contract_arr_cents/);
555
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-1/trace.jsonl'), 'utf-8');
556
- expect(trace).toContain('input_snapshot');
557
- expect(trace).toContain('isolated_diff_enabled');
558
- expect(trace).toContain('work_unit_child_created');
559
- expect(trace).toContain('work_unit_patch_collected');
560
- expect(trace).toContain('patch_apply_started');
561
- expect(trace).toContain('final_artifact_gates_failed');
562
- expect(trace).toContain('ingest_failed');
563
- }
564
- finally {
565
- await rm(runtime.homeDir, { recursive: true, force: true });
566
- }
567
- });
568
- it('rejects unchanged wiki body refs made stale by isolated semantic-layer changes', async () => {
569
- const runtime = await makeRealGitRuntime();
570
- try {
571
- await mkdir(join(runtime.configDir, 'semantic-layer/warehouse'), { recursive: true });
572
- await mkdir(join(runtime.configDir, 'wiki/global'), { recursive: true });
573
- await writeFile(join(runtime.configDir, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr_cents\n expr: sum(contract_arr)\n');
574
- await writeFile(join(runtime.configDir, 'wiki/global/account-segments.md'), '---\nsummary: Account segments\nusage_mode: auto\n---\n\nExisting ARR uses `mart_account_segments.total_contract_arr_cents`.\n');
575
- await runtime.git.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml', 'wiki/global/account-segments.md'], 'seed existing wiki body ref', 'KTX Test', 'system@ktx.local');
576
- const preRunHead = await runtime.git.revParseHead();
577
- const { deps, adapter } = makeDeps(runtime);
578
- adapter.chunk.mockResolvedValue({
579
- workUnits: [{ unitKey: 'source-only', rawFiles: ['cards/source.json'], peerFileIndex: [], dependencyPaths: [] }],
580
- });
581
- let currentSession = null;
582
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
583
- currentSession = toolSession;
584
- return { toRuntimeTools: vi.fn(() => ({})) };
585
- });
586
- deps.agentRunner.runLoop = vi.fn(async () => {
587
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
588
- await writeFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr\n expr: sum(contract_arr)\n');
589
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'mart_account_segments');
590
- currentSession.actions.push({
591
- target: 'sl',
592
- type: 'updated',
593
- key: 'mart_account_segments',
594
- detail: 'Rename ARR measure',
595
- targetConnectionId: 'warehouse',
596
- rawPaths: ['cards/source.json'],
597
- });
598
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml'], 'wu source rename', 'KTX Test', 'system@ktx.local');
599
- return { stopReason: 'natural' };
600
- });
601
- const runner = new IngestBundleRunner(deps);
602
- await mockStageRawFiles(runner, runtime, [['cards/source.json', 'h1']]);
603
- await expect(runner.run({
604
- jobId: 'job-existing-body-stale',
605
- connectionId: 'warehouse',
606
- sourceKey: 'metabase',
607
- trigger: 'upload',
608
- bundleRef: { kind: 'upload', uploadId: 'upload' },
609
- })).rejects.toThrow(/total_contract_arr_cents/);
610
- expect(await runtime.git.revParseHead()).toBe(preRunHead);
611
- const events = (await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-existing-body-stale/trace.jsonl'), 'utf-8'))
612
- .trim()
613
- .split('\n')
614
- .map((line) => JSON.parse(line));
615
- expect(events.map((event) => event.event)).toEqual(expect.arrayContaining([
616
- 'final_artifact_gates_started',
617
- 'final_artifact_gates_failed',
618
- 'ingest_failed',
619
- 'failure_report_created',
620
- ]));
621
- expect(events.map((event) => event.event)).not.toContain('squash_finished');
622
- const gateFailure = events.find((event) => event.event === 'final_artifact_gates_failed');
623
- expect(gateFailure).toMatchObject({
624
- data: {
625
- wikiReferenceGateScope: {
626
- global: true,
627
- reasons: expect.arrayContaining(['semantic_layer_changed']),
628
- pageKeysValidated: expect.arrayContaining(['account-segments']),
629
- },
630
- actionOrigins: expect.arrayContaining([
631
- expect.objectContaining({
632
- source: 'work_unit_action',
633
- unitKey: 'source-only',
634
- unitRawFiles: ['cards/source.json'],
635
- action: expect.objectContaining({
636
- target: 'sl',
637
- type: 'updated',
638
- key: 'mart_account_segments',
639
- rawPaths: ['cards/source.json'],
640
- targetConnectionId: 'warehouse',
641
- }),
642
- }),
643
- ]),
644
- },
645
- error: { message: expect.stringContaining('total_contract_arr_cents') },
646
- });
647
- const failureReport = deps.reports.create.mock.calls
648
- .map((call) => call[0])
649
- .find((report) => report.body.status === 'failed');
650
- expect(failureReport.body.failure).toMatchObject({
651
- phase: 'final_gates',
652
- message: expect.stringContaining('total_contract_arr_cents'),
653
- details: expect.objectContaining({
654
- wikiReferenceGateScope: expect.objectContaining({
655
- global: true,
656
- reasons: expect.arrayContaining(['semantic_layer_changed']),
657
- pageKeysValidated: expect.arrayContaining(['account-segments']),
658
- }),
659
- touchedSlSources: expect.arrayContaining([
660
- expect.objectContaining({ connectionId: 'warehouse', sourceName: 'mart_account_segments' }),
661
- ]),
662
- actionOrigins: expect.arrayContaining([
663
- expect.objectContaining({
664
- source: 'work_unit_action',
665
- unitKey: 'source-only',
666
- action: expect.objectContaining({
667
- target: 'sl',
668
- type: 'updated',
669
- key: 'mart_account_segments',
670
- rawPaths: ['cards/source.json'],
671
- targetConnectionId: 'warehouse',
672
- }),
673
- }),
674
- ]),
675
- }),
676
- });
677
- expect(failureReport.body.workUnits).toEqual(expect.arrayContaining([
678
- expect.objectContaining({
679
- unitKey: 'source-only',
680
- actions: expect.arrayContaining([
681
- expect.objectContaining({
682
- target: 'sl',
683
- type: 'updated',
684
- key: 'mart_account_segments',
685
- rawPaths: ['cards/source.json'],
686
- }),
687
- ]),
688
- }),
689
- ]));
690
- }
691
- finally {
692
- await rm(runtime.homeDir, { recursive: true, force: true });
693
- }
694
- });
695
- it('accepts two isolated work units that edit different wiki pages', async () => {
696
- const runtime = await makeRealGitRuntime();
697
- try {
698
- const { deps, adapter } = makeDeps(runtime);
699
- adapter.chunk.mockResolvedValue({
700
- workUnits: [
701
- { unitKey: 'page-a', rawFiles: ['pages/a.json'], peerFileIndex: [], dependencyPaths: [] },
702
- { unitKey: 'page-b', rawFiles: ['pages/b.json'], peerFileIndex: [], dependencyPaths: [] },
703
- ],
704
- });
705
- let currentSession = null;
706
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
707
- currentSession = toolSession;
708
- return { toRuntimeTools: vi.fn(() => ({})) };
709
- });
710
- deps.agentRunner.runLoop = vi.fn(async (params) => {
711
- const unitKey = params.telemetryTags.unitKey;
712
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
713
- await mkdir(join(root, 'wiki/global'), { recursive: true });
714
- await writeFile(join(root, `wiki/global/${unitKey}.md`), `---\nsummary: ${unitKey}\nusage_mode: auto\n---\n\n${unitKey}\n`);
715
- currentSession.actions.push({ target: 'wiki', type: 'created', key: unitKey, detail: unitKey });
716
- await currentSession.gitService.commitFiles([`wiki/global/${unitKey}.md`], `wu ${unitKey}`, 'KTX Test', 'system@ktx.local');
717
- return { stopReason: 'natural' };
718
- });
719
- const runner = new IngestBundleRunner(deps);
720
- await mockStageRawFiles(runner, runtime, [
721
- ['pages/a.json', 'h1'],
722
- ['pages/b.json', 'h2'],
723
- ]);
724
- const result = await runner.run({ jobId: 'job-clean', connectionId: 'warehouse', sourceKey: 'metabase', trigger: 'upload', bundleRef: { kind: 'upload', uploadId: 'upload' } });
725
- expect(result.failedWorkUnits).toEqual([]);
726
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-clean/trace.jsonl'), 'utf-8');
727
- expect(trace.match(/patch_accepted/g)).toHaveLength(2);
728
- expect(trace).toContain('ingest_finished');
729
- }
730
- finally {
731
- await rm(runtime.homeDir, { recursive: true, force: true });
732
- }
733
- });
734
- it('classifies same-source patch application failure as a textual conflict', async () => {
735
- const runtime = await makeRealGitRuntime();
736
- try {
737
- const { deps, adapter } = makeDeps(runtime);
738
- adapter.chunk.mockResolvedValue({
739
- workUnits: [
740
- { unitKey: 'orders-a', rawFiles: ['orders/a.json'], peerFileIndex: [], dependencyPaths: [] },
741
- { unitKey: 'orders-b', rawFiles: ['orders/b.json'], peerFileIndex: [], dependencyPaths: [] },
742
- ],
743
- });
744
- let currentSession = null;
745
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
746
- currentSession = toolSession;
747
- return { toRuntimeTools: vi.fn(() => ({})) };
748
- });
749
- deps.agentRunner.runLoop = vi.fn(async (params) => {
750
- if (params.telemetryTags.operationName === 'ingest-isolated-diff-textual-resolver') {
751
- return { stopReason: 'natural' };
752
- }
753
- const suffix = params.telemetryTags.unitKey === 'orders-a' ? 'a' : 'b';
754
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
755
- await mkdir(join(root, 'semantic-layer/warehouse'), { recursive: true });
756
- await writeFile(join(root, 'semantic-layer/warehouse/orders.yaml'), `name: orders\ngrain: [id]\ncolumns: [{name: id, type: string}]\njoins: []\nmeasures:\n - name: order_count_${suffix}\n expr: count(*)\n`);
757
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'orders');
758
- currentSession.actions.push({ target: 'sl', type: 'updated', key: 'orders', detail: suffix, targetConnectionId: 'warehouse' });
759
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/orders.yaml'], `wu ${suffix}`, 'KTX Test', 'system@ktx.local');
760
- return { stopReason: 'natural' };
761
- });
762
- const runner = new IngestBundleRunner(deps);
763
- await mockStageRawFiles(runner, runtime, [
764
- ['orders/a.json', 'h1'],
765
- ['orders/b.json', 'h2'],
766
- ]);
767
- await expect(runner.run({ jobId: 'job-text-conflict', connectionId: 'warehouse', sourceKey: 'metabase', trigger: 'upload', bundleRef: { kind: 'upload', uploadId: 'upload' } })).rejects.toThrow(/isolated diff textual conflict/);
768
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-text-conflict/trace.jsonl'), 'utf-8');
769
- expect(trace).toContain('patch_textual_conflict');
770
- expect(trace).toContain('textual_conflict_resolver_failed');
771
- }
772
- finally {
773
- await rm(runtime.homeDir, { recursive: true, force: true });
774
- }
775
- });
776
- it('makes deterministic projection visible to child worktrees before WorkUnit synthesis', async () => {
777
- const runtime = await makeRealGitRuntime();
778
- try {
779
- const { deps, adapter } = makeDeps(runtime);
780
- adapter.chunk.mockResolvedValue({
781
- workUnits: [{ unitKey: 'wiki-projected', rawFiles: ['projected/wiki.json'], peerFileIndex: [], dependencyPaths: [] }],
782
- });
783
- adapter.project = vi.fn(async ({ workdir }) => {
784
- await mkdir(join(workdir, 'semantic-layer/warehouse'), { recursive: true });
785
- await writeFile(join(workdir, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr\n expr: sum(contract_arr)\n');
786
- return {
787
- warnings: [],
788
- errors: [],
789
- touchedSources: [{ connectionId: 'warehouse', sourceName: 'mart_account_segments' }],
790
- changedWikiPageKeys: [],
791
- };
792
- });
793
- let currentSession = null;
794
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
795
- currentSession = toolSession;
796
- return { toRuntimeTools: vi.fn(() => ({})) };
797
- });
798
- deps.agentRunner.runLoop = vi.fn(async () => {
799
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
800
- await expect(readFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'utf-8')).resolves.toContain('total_contract_arr');
801
- await mkdir(join(root, 'wiki/global'), { recursive: true });
802
- await writeFile(join(root, 'wiki/global/projected-orders.md'), '---\nsummary: Projected orders\nusage_mode: auto\nsl_refs:\n - mart_account_segments\n---\n\nARR `mart_account_segments.total_contract_arr`.\n');
803
- currentSession.actions.push({ target: 'wiki', type: 'created', key: 'projected-orders', detail: 'Projected orders' });
804
- await currentSession.gitService.commitFiles(['wiki/global/projected-orders.md'], 'wu projected wiki', 'KTX Test', 'system@ktx.local');
805
- return { stopReason: 'natural' };
806
- });
807
- const runner = new IngestBundleRunner(deps);
808
- await mockStageRawFiles(runner, runtime, [['projected/wiki.json', 'h1']]);
809
- const result = await runner.run({ jobId: 'job-projection', connectionId: 'warehouse', sourceKey: 'metabase', trigger: 'upload', bundleRef: { kind: 'upload', uploadId: 'upload' } });
810
- expect(result.failedWorkUnits).toEqual([]);
811
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-projection/trace.jsonl'), 'utf-8');
812
- expect(trace).toContain('deterministic_projection_finished');
813
- expect(trace).toContain('deterministic_projection_committed');
814
- }
815
- finally {
816
- await rm(runtime.homeDir, { recursive: true, force: true });
817
- }
818
- });
819
- it('rejects Notion-style changed wiki pages with invalid sl_refs', async () => {
820
- const runtime = await makeRealGitRuntime();
821
- try {
822
- const { deps, adapter } = makeDeps(runtime);
823
- adapter.chunk.mockResolvedValue({
824
- workUnits: [{ unitKey: 'notion-page', rawFiles: ['pages/notion.json'], peerFileIndex: [], dependencyPaths: [] }],
825
- });
826
- let currentSession = null;
827
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
828
- currentSession = toolSession;
829
- return { toRuntimeTools: vi.fn(() => ({})) };
830
- });
831
- deps.agentRunner.runLoop = vi.fn(async (params) => {
832
- if (params.telemetryTags.operationName === 'ingest-isolated-diff-gate-repair') {
833
- return { stopReason: 'natural' };
834
- }
835
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
836
- await mkdir(join(root, 'wiki/global'), { recursive: true });
837
- await writeFile(join(root, 'wiki/global/notion-page.md'), '---\nsummary: Notion page\nusage_mode: auto\nsl_refs:\n - missing_source\n---\n\nBody\n');
838
- currentSession.actions.push({ target: 'wiki', type: 'created', key: 'notion-page', detail: 'Notion page' });
839
- await currentSession.gitService.commitFiles(['wiki/global/notion-page.md'], 'wu notion', 'KTX Test', 'system@ktx.local');
840
- return { stopReason: 'natural' };
841
- });
842
- const runner = new IngestBundleRunner(deps);
843
- await mockStageRawFiles(runner, runtime, [['pages/notion.json', 'h1']]);
844
- await expect(runner.run({ jobId: 'job-invalid-slrefs', connectionId: 'warehouse', sourceKey: 'metabase', trigger: 'upload', bundleRef: { kind: 'upload', uploadId: 'upload' } })).rejects.toThrow(/gate repair completed without editing an allowed path/);
845
- }
846
- finally {
847
- await rm(runtime.homeDir, { recursive: true, force: true });
848
- }
849
- });
850
- it('runs final artifact gates after reconciliation mutates the integration tree', async () => {
851
- const runtime = await makeRealGitRuntime();
852
- try {
853
- const { deps, adapter } = makeDeps(runtime);
854
- adapter.chunk.mockResolvedValue({
855
- workUnits: [{ unitKey: 'card-source', rawFiles: ['cards/source.json'], peerFileIndex: [], dependencyPaths: [] }],
856
- });
857
- let currentSession = null;
858
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
859
- currentSession = toolSession;
860
- return { toRuntimeTools: vi.fn(() => ({})) };
861
- });
862
- deps.agentRunner.runLoop = vi.fn(async (params) => {
863
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
864
- if (params.telemetryTags.operationName === 'ingest-bundle-wu') {
865
- await mkdir(join(root, 'semantic-layer/warehouse'), { recursive: true });
866
- await writeFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr\n expr: sum(contract_arr)\n');
867
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'mart_account_segments');
868
- currentSession.actions.push({
869
- target: 'sl',
870
- type: 'created',
871
- key: 'mart_account_segments',
872
- detail: 'Source with renamed ARR measure',
873
- targetConnectionId: 'warehouse',
874
- rawPaths: ['cards/source.json'],
875
- });
876
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml'], 'wu source', 'KTX Test', 'system@ktx.local');
877
- }
878
- else {
879
- await mkdir(join(root, 'wiki/global'), { recursive: true });
880
- await writeFile(join(root, 'wiki/global/account-segments.md'), '---\nsummary: Account segments\nusage_mode: auto\nsl_refs:\n - mart_account_segments\n---\n\nReconcile wrote stale ARR `mart_account_segments.total_contract_arr_cents`.\n');
881
- currentSession.actions.push({
882
- target: 'wiki',
883
- type: 'created',
884
- key: 'account-segments',
885
- detail: 'Stale reconcile wiki page',
886
- rawPaths: ['cards/source.json'],
887
- });
888
- await currentSession.gitService.commitFiles(['wiki/global/account-segments.md'], 'reconcile wiki', 'KTX Test', 'system@ktx.local');
889
- }
890
- return { stopReason: 'natural' };
891
- });
892
- const runner = new IngestBundleRunner(deps);
893
- await mockStageRawFiles(runner, runtime, [['cards/source.json', 'h1']]);
894
- await expect(runner.run({
895
- jobId: 'job-reconcile-stale',
896
- connectionId: 'warehouse',
897
- sourceKey: 'metabase',
898
- trigger: 'upload',
899
- bundleRef: { kind: 'upload', uploadId: 'upload' },
900
- })).rejects.toThrow(/total_contract_arr_cents/);
901
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-reconcile-stale/trace.jsonl'), 'utf-8');
902
- expect(trace).toContain('reconciliation_finished');
903
- expect(trace).toContain('final_artifact_gates_failed');
904
- expect(trace).toContain('ingest_failed');
905
- expect(await runtime.git.revParseHead()).not.toContain('reconcile wiki');
906
- }
907
- finally {
908
- await rm(runtime.homeDir, { recursive: true, force: true });
909
- }
910
- });
911
- it('stores a failure report and postmortem trace for final gate failures', async () => {
912
- const runtime = await makeRealGitRuntime();
913
- try {
914
- const { deps, adapter } = makeDeps(runtime);
915
- const createdReports = [];
916
- deps.reports.create = vi.fn(async (args) => {
917
- createdReports.push(args);
918
- return { id: `report-${createdReports.length}` };
919
- });
920
- adapter.chunk.mockResolvedValue({
921
- workUnits: [
922
- { unitKey: 'card-wiki', rawFiles: ['cards/wiki.json'], peerFileIndex: [], dependencyPaths: [] },
923
- { unitKey: 'card-source', rawFiles: ['cards/source.json'], peerFileIndex: [], dependencyPaths: [] },
924
- ],
925
- });
926
- let currentSession = null;
927
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
928
- currentSession = toolSession;
929
- return { toRuntimeTools: vi.fn(() => ({})) };
930
- });
931
- deps.agentRunner.runLoop = vi.fn(async (params) => {
932
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
933
- if (params.telemetryTags.unitKey === 'card-wiki') {
934
- await mkdir(join(root, 'wiki/global'), { recursive: true });
935
- await writeFile(join(root, 'wiki/global/account-segments.md'), '---\nsummary: Account segments\nusage_mode: auto\n---\n\nARR is `mart_account_segments.total_contract_arr_cents`.\n');
936
- currentSession.actions.push({
937
- target: 'wiki',
938
- type: 'created',
939
- key: 'account-segments',
940
- detail: 'Account segments',
941
- rawPaths: ['cards/wiki.json'],
942
- });
943
- await currentSession.gitService.commitFiles(['wiki/global/account-segments.md'], 'wu wiki', 'KTX Test', 'system@ktx.local');
944
- }
945
- if (params.telemetryTags.unitKey === 'card-source') {
946
- await mkdir(join(root, 'semantic-layer/warehouse'), { recursive: true });
947
- await writeFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr\n expr: sum(contract_arr)\n');
948
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'mart_account_segments');
949
- currentSession.actions.push({
950
- target: 'sl',
951
- type: 'created',
952
- key: 'mart_account_segments',
953
- detail: 'Dollar measure',
954
- targetConnectionId: 'warehouse',
955
- rawPaths: ['cards/source.json'],
956
- });
957
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml'], 'wu source', 'KTX Test', 'system@ktx.local');
958
- }
959
- return { stopReason: 'natural' };
960
- });
961
- const runner = new IngestBundleRunner(deps);
962
- await mockStageRawFiles(runner, runtime, [
963
- ['cards/wiki.json', 'h1'],
964
- ['cards/source.json', 'h2'],
965
- ]);
966
- await expect(runner.run({
967
- jobId: 'job-trace-failure',
968
- connectionId: 'warehouse',
969
- sourceKey: 'metabase',
970
- trigger: 'upload',
971
- bundleRef: { kind: 'upload', uploadId: 'upload' },
972
- })).rejects.toThrow(/total_contract_arr_cents/);
973
- const failureReport = createdReports.find((report) => report.body.status === 'failed');
974
- expect(failureReport.body.tracePath).toContain('job-trace-failure/trace.jsonl');
975
- expect(failureReport.body.failure).toMatchObject({ phase: 'final_gates' });
976
- const events = (await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-trace-failure/trace.jsonl'), 'utf-8'))
977
- .trim()
978
- .split('\n')
979
- .map((line) => JSON.parse(line));
980
- expect(events.map((event) => event.event)).toEqual(expect.arrayContaining([
981
- 'ingest_started',
982
- 'input_snapshot',
983
- 'work_units_planned',
984
- 'isolated_diff_enabled',
985
- 'work_unit_child_created',
986
- 'work_unit_patch_collected',
987
- 'patch_apply_started',
988
- 'patch_accepted',
989
- 'reconciliation_finished',
990
- 'final_artifact_gates_failed',
991
- 'ingest_failed',
992
- 'failure_report_created',
993
- ]));
994
- const failed = events.find((event) => event.event === 'ingest_failed');
995
- expect(failed).toMatchObject({
996
- runId: 'run-1',
997
- syncId: expect.any(String),
998
- data: { phase: 'final_gates', tracePath: expect.stringContaining('trace.jsonl') },
999
- error: { message: expect.stringContaining('total_contract_arr_cents') },
1000
- });
1001
- }
1002
- finally {
1003
- await rm(runtime.homeDir, { recursive: true, force: true });
1004
- }
1005
- });
1006
- it('rejects invalid provenance raw paths before squash reaches main', async () => {
1007
- const runtime = await makeRealGitRuntime();
1008
- try {
1009
- const { deps, adapter } = makeDeps(runtime);
1010
- const createdReports = [];
1011
- deps.reports.create = vi.fn(async (args) => {
1012
- createdReports.push(args);
1013
- return { id: `report-${createdReports.length}` };
1014
- });
1015
- adapter.chunk.mockResolvedValue({
1016
- workUnits: [
1017
- {
1018
- unitKey: 'card-valid-artifacts',
1019
- rawFiles: ['cards/source.json'],
1020
- peerFileIndex: [],
1021
- dependencyPaths: [],
1022
- },
1023
- ],
1024
- });
1025
- let currentSession = null;
1026
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1027
- currentSession = toolSession;
1028
- return { toRuntimeTools: vi.fn(() => ({})) };
1029
- });
1030
- deps.agentRunner.runLoop = vi.fn(async () => {
1031
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1032
- await mkdir(join(root, 'semantic-layer/warehouse'), { recursive: true });
1033
- await mkdir(join(root, 'wiki/global'), { recursive: true });
1034
- await writeFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr\n expr: sum(contract_arr)\n');
1035
- await writeFile(join(root, 'wiki/global/account-segments.md'), '---\nsummary: Account segments\nusage_mode: auto\nsl_refs:\n - mart_account_segments\n---\n\nARR is `mart_account_segments.total_contract_arr`.\n');
1036
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'mart_account_segments');
1037
- currentSession.actions.push({
1038
- target: 'sl',
1039
- type: 'created',
1040
- key: 'mart_account_segments',
1041
- detail: 'Valid source',
1042
- targetConnectionId: 'warehouse',
1043
- rawPaths: ['cards/source.json'],
1044
- });
1045
- currentSession.actions.push({
1046
- target: 'wiki',
1047
- type: 'created',
1048
- key: 'account-segments',
1049
- detail: 'Valid wiki with invalid provenance raw path',
1050
- rawPaths: ['cards/missing.json'],
1051
- });
1052
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml', 'wiki/global/account-segments.md'], 'valid artifacts with invalid provenance', 'KTX Test', 'system@ktx.local');
1053
- return { stopReason: 'natural' };
1054
- });
1055
- const runner = new IngestBundleRunner(deps);
1056
- await mockStageRawFiles(runner, runtime, [['cards/source.json', 'h1']]);
1057
- const preRunHead = await runtime.git.revParseHead();
1058
- await expect(runner.run({
1059
- jobId: 'job-invalid-provenance',
1060
- connectionId: 'warehouse',
1061
- sourceKey: 'metabase',
1062
- trigger: 'upload',
1063
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1064
- })).rejects.toThrow(/provenance row references raw path outside this snapshot: cards\/missing\.json/);
1065
- expect(await runtime.git.revParseHead()).toBe(preRunHead);
1066
- expect(deps.provenance.insertMany).not.toHaveBeenCalled();
1067
- const failureReport = createdReports.find((report) => report.body.status === 'failed');
1068
- expect(failureReport.body.tracePath).toContain('job-invalid-provenance/trace.jsonl');
1069
- expect(failureReport.body.failure).toMatchObject({
1070
- phase: 'provenance_validation',
1071
- message: expect.stringContaining('cards/missing.json'),
1072
- });
1073
- expect(failureReport.body.failure.details).toMatchObject({
1074
- invalidRawPaths: ['cards/missing.json'],
1075
- currentRawPaths: ['cards/source.json'],
1076
- invalidRows: expect.arrayContaining([
1077
- expect.objectContaining({
1078
- row: expect.objectContaining({
1079
- rawPath: 'cards/missing.json',
1080
- artifactKind: 'wiki',
1081
- artifactKey: 'account-segments',
1082
- actionType: 'wiki_written',
1083
- }),
1084
- origin: expect.objectContaining({
1085
- source: 'work_unit_action',
1086
- unitKey: 'card-valid-artifacts',
1087
- actionIndex: 1,
1088
- unitRawFiles: ['cards/source.json'],
1089
- action: expect.objectContaining({
1090
- target: 'wiki',
1091
- type: 'created',
1092
- key: 'account-segments',
1093
- rawPaths: ['cards/missing.json'],
1094
- }),
1095
- }),
1096
- }),
1097
- ]),
1098
- });
1099
- expect(failureReport.body.provenanceRows).toEqual(expect.arrayContaining([
1100
- expect.objectContaining({ rawPath: 'cards/source.json', artifactKind: 'sl', artifactKey: 'mart_account_segments' }),
1101
- expect.objectContaining({ rawPath: 'cards/missing.json', artifactKind: 'wiki', artifactKey: 'account-segments' }),
1102
- ]));
1103
- expect(failureReport.body.workUnits).toEqual(expect.arrayContaining([
1104
- expect.objectContaining({
1105
- unitKey: 'card-valid-artifacts',
1106
- rawFiles: ['cards/source.json'],
1107
- actions: expect.arrayContaining([
1108
- expect.objectContaining({
1109
- target: 'wiki',
1110
- key: 'account-segments',
1111
- rawPaths: ['cards/missing.json'],
1112
- }),
1113
- ]),
1114
- }),
1115
- ]));
1116
- const events = (await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-invalid-provenance/trace.jsonl'), 'utf-8'))
1117
- .trim()
1118
- .split('\n')
1119
- .map((line) => JSON.parse(line));
1120
- expect(events.map((event) => event.event)).toEqual(expect.arrayContaining([
1121
- 'final_artifact_gates_finished',
1122
- 'provenance_rows_validation_failed',
1123
- 'ingest_failed',
1124
- 'failure_report_created',
1125
- ]));
1126
- expect(events.map((event) => event.event)).not.toContain('squash_finished');
1127
- const validationFailure = events.find((event) => event.event === 'provenance_rows_validation_failed');
1128
- expect(validationFailure).toMatchObject({
1129
- phase: 'provenance',
1130
- data: {
1131
- invalidRawPaths: ['cards/missing.json'],
1132
- currentRawPaths: ['cards/source.json'],
1133
- invalidRows: expect.arrayContaining([
1134
- expect.objectContaining({
1135
- row: expect.objectContaining({ rawPath: 'cards/missing.json' }),
1136
- origin: expect.objectContaining({
1137
- source: 'work_unit_action',
1138
- unitKey: 'card-valid-artifacts',
1139
- actionIndex: 1,
1140
- }),
1141
- }),
1142
- ]),
1143
- },
1144
- });
1145
- }
1146
- finally {
1147
- await rm(runtime.homeDir, { recursive: true, force: true });
1148
- }
1149
- });
1150
- it('rejects slDisallowed patches that touch semantic-layer files', async () => {
1151
- const runtime = await makeRealGitRuntime();
1152
- try {
1153
- const { deps, adapter } = makeDeps(runtime);
1154
- adapter.chunk.mockResolvedValue({
1155
- workUnits: [
1156
- {
1157
- unitKey: 'lookml-mismatch',
1158
- rawFiles: ['views/orders.lkml'],
1159
- peerFileIndex: [],
1160
- dependencyPaths: [],
1161
- slDisallowed: true,
1162
- slDisallowedReason: 'lookml_connection_mismatch',
1163
- },
1164
- ],
1165
- });
1166
- let currentSession = null;
1167
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1168
- currentSession = toolSession;
1169
- return { toRuntimeTools: vi.fn(() => ({})) };
1170
- });
1171
- deps.agentRunner.runLoop = vi.fn(async () => {
1172
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1173
- await mkdir(join(root, 'semantic-layer/warehouse'), { recursive: true });
1174
- await writeFile(join(root, 'semantic-layer/warehouse/orders.yaml'), 'name: orders\ngrain: [id]\ncolumns: [{name: id, type: string}]\njoins: []\nmeasures: []\n');
1175
- currentSession.actions.push({ target: 'sl', type: 'created', key: 'orders', detail: 'forbidden', targetConnectionId: 'warehouse' });
1176
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/orders.yaml'], 'forbidden sl', 'KTX Test', 'system@ktx.local');
1177
- return { stopReason: 'natural' };
1178
- });
1179
- const runner = new IngestBundleRunner(deps);
1180
- await mockStageRawFiles(runner, runtime, [['views/orders.lkml', 'h1']]);
1181
- await expect(runner.run({ jobId: 'job-sl-disallowed', connectionId: 'warehouse', sourceKey: 'metabase', trigger: 'upload', bundleRef: { kind: 'upload', uploadId: 'upload' } })).rejects.toThrow(/isolated diff textual conflict/);
1182
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-sl-disallowed/trace.jsonl'), 'utf-8');
1183
- expect(trace).toContain('patch_policy_rejected');
1184
- expect(trace).toContain('slDisallowed WorkUnit lookml-mismatch touched semantic-layer/warehouse/orders.yaml');
1185
- }
1186
- finally {
1187
- await rm(runtime.homeDir, { recursive: true, force: true });
1188
- }
1189
- });
1190
- it('rejects final wiki refs broken by another accepted WorkUnit before squash', async () => {
1191
- const runtime = await makeRealGitRuntime();
1192
- try {
1193
- await mkdir(join(runtime.configDir, 'wiki/global'), { recursive: true });
1194
- await writeFile(join(runtime.configDir, 'wiki/global/source-page.md'), '---\nsummary: Source page\nusage_mode: auto\n---\n\nSource page\n');
1195
- await runtime.git.commitFiles(['wiki/global/source-page.md'], 'seed source page', 'KTX Test', 'system@ktx.local');
1196
- const preRunHead = await runtime.git.revParseHead();
1197
- const { deps, adapter } = makeDeps(runtime);
1198
- adapter.chunk.mockResolvedValue({
1199
- workUnits: [
1200
- { unitKey: 'page-ref', rawFiles: ['pages/ref.json'], peerFileIndex: [], dependencyPaths: [] },
1201
- { unitKey: 'page-delete', rawFiles: ['pages/delete.json'], peerFileIndex: [], dependencyPaths: [] },
1202
- ],
1203
- });
1204
- let currentSession = null;
1205
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1206
- currentSession = toolSession;
1207
- return { toRuntimeTools: vi.fn(() => ({})) };
1208
- });
1209
- deps.agentRunner.runLoop = vi.fn(async (params) => {
1210
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1211
- if (params.telemetryTags.unitKey === 'page-ref') {
1212
- await mkdir(join(root, 'wiki/global'), { recursive: true });
1213
- await writeFile(join(root, 'wiki/global/account-segments.md'), '---\nsummary: Account segments\nusage_mode: auto\nrefs:\n - source-page\n---\n\nSee [[source-page]].\n');
1214
- currentSession.actions.push({
1215
- target: 'wiki',
1216
- type: 'created',
1217
- key: 'account-segments',
1218
- detail: 'Page with wiki ref',
1219
- rawPaths: ['pages/ref.json'],
1220
- });
1221
- await currentSession.gitService.commitFiles(['wiki/global/account-segments.md'], 'wu page ref', 'KTX Test', 'system@ktx.local');
1222
- }
1223
- if (params.telemetryTags.unitKey === 'page-delete') {
1224
- await rm(join(root, 'wiki/global/source-page.md'), { force: true });
1225
- currentSession.actions.push({
1226
- target: 'wiki',
1227
- type: 'removed',
1228
- key: 'source-page',
1229
- detail: 'Delete referenced page',
1230
- rawPaths: ['pages/delete.json'],
1231
- });
1232
- await currentSession.gitService.commitFiles(['wiki/global/source-page.md'], 'wu delete source page', 'KTX Test', 'system@ktx.local');
1233
- }
1234
- return { stopReason: 'natural' };
1235
- });
1236
- const runner = new IngestBundleRunner(deps);
1237
- await mockStageRawFiles(runner, runtime, [
1238
- ['pages/ref.json', 'h1'],
1239
- ['pages/delete.json', 'h2'],
1240
- ]);
1241
- await expect(runner.run({
1242
- jobId: 'job-wiki-ref-conflict',
1243
- connectionId: 'warehouse',
1244
- sourceKey: 'metabase',
1245
- trigger: 'upload',
1246
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1247
- })).rejects.toThrow(/wiki references target missing page\(s\): account-segments -> source-page/);
1248
- expect(await runtime.git.revParseHead()).toBe(preRunHead);
1249
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-wiki-ref-conflict/trace.jsonl'), 'utf-8');
1250
- expect(trace).toContain('final_artifact_gates_failed');
1251
- expect(trace).toContain('account-segments -> source-page');
1252
- expect(trace).toContain('ingest_failed');
1253
- expect(trace).toContain('failure_report_created');
1254
- expect(trace).not.toContain('squash_finished');
1255
- const failureReport = deps.reports.create.mock.calls
1256
- .map((call) => call[0])
1257
- .find((report) => report.body.status === 'failed');
1258
- expect(failureReport.body.failure).toMatchObject({
1259
- phase: 'final_gates',
1260
- message: expect.stringContaining('account-segments -> source-page'),
1261
- details: expect.objectContaining({
1262
- changedWikiPageKeys: expect.arrayContaining(['account-segments']),
1263
- workUnitPatchTouchedPaths: expect.arrayContaining([
1264
- 'wiki/global/account-segments.md',
1265
- 'wiki/global/source-page.md',
1266
- ]),
1267
- }),
1268
- });
1269
- }
1270
- finally {
1271
- await rm(runtime.homeDir, { recursive: true, force: true });
1272
- }
1273
- });
1274
- it('rejects unchanged inbound wiki refs broken by an isolated wiki deletion', async () => {
1275
- const runtime = await makeRealGitRuntime();
1276
- try {
1277
- await mkdir(join(runtime.configDir, 'wiki/global'), { recursive: true });
1278
- await writeFile(join(runtime.configDir, 'wiki/global/source-page.md'), '---\nsummary: Source page\nusage_mode: auto\n---\n\nSource page\n');
1279
- await writeFile(join(runtime.configDir, 'wiki/global/account-segments.md'), '---\nsummary: Account segments\nusage_mode: auto\nrefs:\n - source-page\n---\n\nSee [[source-page]].\n');
1280
- await runtime.git.commitFiles(['wiki/global/source-page.md', 'wiki/global/account-segments.md'], 'seed inbound wiki refs', 'KTX Test', 'system@ktx.local');
1281
- const preRunHead = await runtime.git.revParseHead();
1282
- const { deps, adapter } = makeDeps(runtime);
1283
- adapter.chunk.mockResolvedValue({
1284
- workUnits: [{ unitKey: 'delete-target-page', rawFiles: ['pages/delete.json'], peerFileIndex: [], dependencyPaths: [] }],
1285
- });
1286
- let currentSession = null;
1287
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1288
- currentSession = toolSession;
1289
- return { toRuntimeTools: vi.fn(() => ({})) };
1290
- });
1291
- deps.agentRunner.runLoop = vi.fn(async (params) => {
1292
- if (params.telemetryTags.unitKey !== 'delete-target-page') {
1293
- return { stopReason: 'natural' };
1294
- }
1295
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1296
- await rm(join(root, 'wiki/global/source-page.md'), { force: true });
1297
- currentSession.actions.push({
1298
- target: 'wiki',
1299
- type: 'removed',
1300
- key: 'source-page',
1301
- detail: 'Delete referenced page',
1302
- rawPaths: ['pages/delete.json'],
1303
- });
1304
- await currentSession.gitService.commitFiles(['wiki/global/source-page.md'], 'wu delete target page', 'KTX Test', 'system@ktx.local');
1305
- return { stopReason: 'natural' };
1306
- });
1307
- const runner = new IngestBundleRunner(deps);
1308
- await mockStageRawFiles(runner, runtime, [['pages/delete.json', 'h1']]);
1309
- await expect(runner.run({
1310
- jobId: 'job-existing-wiki-ref-stale',
1311
- connectionId: 'warehouse',
1312
- sourceKey: 'metabase',
1313
- trigger: 'upload',
1314
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1315
- })).rejects.toThrow(/wiki references target missing page\(s\): account-segments -> source-page/);
1316
- expect(await runtime.git.revParseHead()).toBe(preRunHead);
1317
- const events = (await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-existing-wiki-ref-stale/trace.jsonl'), 'utf-8'))
1318
- .trim()
1319
- .split('\n')
1320
- .map((line) => JSON.parse(line));
1321
- expect(events.map((event) => event.event)).toEqual(expect.arrayContaining([
1322
- 'final_artifact_gates_started',
1323
- 'final_artifact_gates_failed',
1324
- 'ingest_failed',
1325
- 'failure_report_created',
1326
- ]));
1327
- expect(events.map((event) => event.event)).not.toContain('squash_finished');
1328
- const gateFailure = events.find((event) => event.event === 'final_artifact_gates_failed');
1329
- expect(gateFailure).toMatchObject({
1330
- data: {
1331
- wikiReferenceGateScope: {
1332
- global: true,
1333
- reasons: expect.arrayContaining(['wiki_page_removed']),
1334
- removedWikiPageKeys: expect.arrayContaining(['source-page']),
1335
- pageKeysValidated: expect.arrayContaining(['account-segments']),
1336
- },
1337
- actionOrigins: expect.arrayContaining([
1338
- expect.objectContaining({
1339
- source: 'work_unit_action',
1340
- unitKey: 'delete-target-page',
1341
- unitRawFiles: ['pages/delete.json'],
1342
- action: expect.objectContaining({
1343
- target: 'wiki',
1344
- type: 'removed',
1345
- key: 'source-page',
1346
- rawPaths: ['pages/delete.json'],
1347
- }),
1348
- }),
1349
- ]),
1350
- },
1351
- error: { message: expect.stringContaining('account-segments -> source-page') },
1352
- });
1353
- const failureReport = deps.reports.create.mock.calls
1354
- .map((call) => call[0])
1355
- .find((report) => report.body.status === 'failed');
1356
- expect(failureReport.body.failure).toMatchObject({
1357
- phase: 'final_gates',
1358
- message: expect.stringContaining('account-segments -> source-page'),
1359
- details: expect.objectContaining({
1360
- wikiReferenceGateScope: expect.objectContaining({
1361
- global: true,
1362
- reasons: expect.arrayContaining(['wiki_page_removed']),
1363
- removedWikiPageKeys: expect.arrayContaining(['source-page']),
1364
- pageKeysValidated: expect.arrayContaining(['account-segments']),
1365
- }),
1366
- changedWikiPageKeys: expect.arrayContaining(['source-page']),
1367
- actionOrigins: expect.arrayContaining([
1368
- expect.objectContaining({
1369
- source: 'work_unit_action',
1370
- unitKey: 'delete-target-page',
1371
- action: expect.objectContaining({
1372
- target: 'wiki',
1373
- type: 'removed',
1374
- key: 'source-page',
1375
- rawPaths: ['pages/delete.json'],
1376
- }),
1377
- }),
1378
- ]),
1379
- }),
1380
- });
1381
- expect(failureReport.body.workUnits).toEqual(expect.arrayContaining([
1382
- expect.objectContaining({
1383
- unitKey: 'delete-target-page',
1384
- actions: expect.arrayContaining([
1385
- expect.objectContaining({
1386
- target: 'wiki',
1387
- type: 'removed',
1388
- key: 'source-page',
1389
- rawPaths: ['pages/delete.json'],
1390
- }),
1391
- ]),
1392
- }),
1393
- ]));
1394
- }
1395
- finally {
1396
- await rm(runtime.homeDir, { recursive: true, force: true });
1397
- }
1398
- });
1399
- it('rejects WorkUnit patches that touch unauthorized semantic-layer target connections', async () => {
1400
- const runtime = await makeRealGitRuntime();
1401
- try {
1402
- const { deps, adapter } = makeDeps(runtime);
1403
- adapter.chunk.mockResolvedValue({
1404
- workUnits: [{ unitKey: 'finance-source', rawFiles: ['cards/finance.json'], peerFileIndex: [], dependencyPaths: [] }],
1405
- });
1406
- let currentSession = null;
1407
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1408
- currentSession = toolSession;
1409
- return { toRuntimeTools: vi.fn(() => ({})) };
1410
- });
1411
- deps.agentRunner.runLoop = vi.fn(async () => {
1412
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1413
- await mkdir(join(root, 'semantic-layer/finance'), { recursive: true });
1414
- await writeFile(join(root, 'semantic-layer/finance/orders.yaml'), 'name: orders\ngrain: [id]\ncolumns: [{name: id, type: string}]\njoins: []\nmeasures: []\n');
1415
- addTouchedSlSource(currentSession.touchedSlSources, 'finance', 'orders');
1416
- currentSession.actions.push({
1417
- target: 'sl',
1418
- type: 'created',
1419
- key: 'orders',
1420
- detail: 'Unauthorized target',
1421
- targetConnectionId: 'finance',
1422
- rawPaths: ['cards/finance.json'],
1423
- });
1424
- await currentSession.gitService.commitFiles(['semantic-layer/finance/orders.yaml'], 'wu unauthorized target', 'KTX Test', 'system@ktx.local');
1425
- return { stopReason: 'natural' };
1426
- });
1427
- const runner = new IngestBundleRunner(deps);
1428
- await mockStageRawFiles(runner, runtime, [['cards/finance.json', 'h1']]);
1429
- const preRunHead = await runtime.git.revParseHead();
1430
- await expect(runner.run({
1431
- jobId: 'job-unauthorized-wu-target',
1432
- connectionId: 'warehouse',
1433
- sourceKey: 'metabase',
1434
- trigger: 'upload',
1435
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1436
- })).rejects.toThrow(/isolated diff textual conflict.*semantic-layer target connection not allowed/);
1437
- expect(await runtime.git.revParseHead()).toBe(preRunHead);
1438
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-unauthorized-wu-target/trace.jsonl'), 'utf-8');
1439
- expect(trace).toContain('patch_policy_rejected');
1440
- expect(trace).toContain('semantic-layer/finance/orders.yaml');
1441
- expect(trace).toContain('allowedTargetConnectionIds');
1442
- expect(trace).toContain('ingest_failed');
1443
- expect(trace).toContain('failure_report_created');
1444
- expect(trace).not.toContain('squash_finished');
1445
- const failureReport = deps.reports.create.mock.calls
1446
- .map((call) => call[0])
1447
- .find((report) => report.body.status === 'failed');
1448
- expect(failureReport.body.failure).toMatchObject({
1449
- phase: 'integration',
1450
- message: expect.stringContaining('semantic-layer target connection not allowed'),
1451
- });
1452
- expect(failureReport.body.failure.details).toMatchObject({
1453
- unitKey: 'finance-source',
1454
- allowedTargetConnectionIds: ['warehouse'],
1455
- touchedPaths: ['semantic-layer/finance/orders.yaml'],
1456
- reason: expect.stringContaining('semantic-layer/finance/orders.yaml (finance)'),
1457
- });
1458
- }
1459
- finally {
1460
- await rm(runtime.homeDir, { recursive: true, force: true });
1461
- }
1462
- });
1463
- it('rejects reconciliation mutations that touch unauthorized semantic-layer target connections before squash', async () => {
1464
- const runtime = await makeRealGitRuntime();
1465
- try {
1466
- const { deps, adapter } = makeDeps(runtime);
1467
- adapter.chunk.mockResolvedValue({
1468
- workUnits: [{ unitKey: 'valid-page', rawFiles: ['pages/source.json'], peerFileIndex: [], dependencyPaths: [] }],
1469
- });
1470
- let currentSession = null;
1471
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1472
- currentSession = toolSession;
1473
- return { toRuntimeTools: vi.fn(() => ({})) };
1474
- });
1475
- deps.agentRunner.runLoop = vi.fn(async (params) => {
1476
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1477
- if (params.telemetryTags.operationName === 'ingest-bundle-wu') {
1478
- await mkdir(join(root, 'wiki/global'), { recursive: true });
1479
- await writeFile(join(root, 'wiki/global/valid-page.md'), '---\nsummary: Valid page\nusage_mode: auto\n---\n\nValid\n');
1480
- currentSession.actions.push({
1481
- target: 'wiki',
1482
- type: 'created',
1483
- key: 'valid-page',
1484
- detail: 'Valid page',
1485
- rawPaths: ['pages/source.json'],
1486
- });
1487
- await currentSession.gitService.commitFiles(['wiki/global/valid-page.md'], 'wu valid page', 'KTX Test', 'system@ktx.local');
1488
- }
1489
- else {
1490
- await mkdir(join(root, 'semantic-layer/finance'), { recursive: true });
1491
- await writeFile(join(root, 'semantic-layer/finance/reconcile_orders.yaml'), 'name: reconcile_orders\ngrain: [id]\ncolumns: [{name: id, type: string}]\njoins: []\nmeasures: []\n');
1492
- addTouchedSlSource(currentSession.touchedSlSources, 'finance', 'reconcile_orders');
1493
- currentSession.actions.push({
1494
- target: 'sl',
1495
- type: 'created',
1496
- key: 'reconcile_orders',
1497
- detail: 'Unauthorized reconcile target',
1498
- targetConnectionId: 'finance',
1499
- rawPaths: ['pages/source.json'],
1500
- });
1501
- await currentSession.gitService.commitFiles(['semantic-layer/finance/reconcile_orders.yaml'], 'reconcile unauthorized target', 'KTX Test', 'system@ktx.local');
1502
- }
1503
- return { stopReason: 'natural' };
1504
- });
1505
- const runner = new IngestBundleRunner(deps);
1506
- await mockStageRawFiles(runner, runtime, [['pages/source.json', 'h1']]);
1507
- const preRunHead = await runtime.git.revParseHead();
1508
- await expect(runner.run({
1509
- jobId: 'job-unauthorized-reconcile-target',
1510
- connectionId: 'warehouse',
1511
- sourceKey: 'metabase',
1512
- trigger: 'upload',
1513
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1514
- })).rejects.toThrow(/semantic-layer target connection not allowed/);
1515
- expect(await runtime.git.revParseHead()).toBe(preRunHead);
1516
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-unauthorized-reconcile-target/trace.jsonl'), 'utf-8');
1517
- expect(trace).toContain('semantic_layer_target_policy_started');
1518
- expect(trace).toContain('semantic_layer_target_policy_failed');
1519
- expect(trace).toContain('allowedTargetConnectionIds');
1520
- expect(trace).toContain('semantic-layer/finance/reconcile_orders.yaml');
1521
- expect(trace).toContain('ingest_failed');
1522
- expect(trace).toContain('failure_report_created');
1523
- expect(trace).not.toContain('squash_finished');
1524
- const failureReport = deps.reports.create.mock.calls
1525
- .map((call) => call[0])
1526
- .find((report) => report.body.status === 'failed');
1527
- expect(failureReport.body.failure).toMatchObject({
1528
- phase: 'target_policy',
1529
- message: expect.stringContaining('semantic-layer target connection not allowed'),
1530
- });
1531
- expect(failureReport.body.failure.details).toMatchObject({
1532
- allowedTargetConnectionIds: ['warehouse'],
1533
- touchedPaths: expect.arrayContaining(['semantic-layer/finance/reconcile_orders.yaml']),
1534
- });
1535
- }
1536
- finally {
1537
- await rm(runtime.homeDir, { recursive: true, force: true });
1538
- }
1539
- });
1540
- it('repairs additive same-source textual conflicts before final gates and squash', async () => {
1541
- const runtime = await makeRealGitRuntime();
1542
- try {
1543
- const { deps } = makeDeps(runtime);
1544
- let currentSession = null;
1545
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1546
- currentSession = toolSession;
1547
- return { toRuntimeTools: vi.fn(() => ({})) };
1548
- });
1549
- deps.agentRunner.runLoop = vi.fn(async (params) => {
1550
- if (params.telemetryTags.operationName === 'ingest-isolated-diff-textual-resolver') {
1551
- const current = await params.toolSet.read_integration_file.execute({
1552
- path: 'semantic-layer/warehouse/mart_account_segments.yaml',
1553
- });
1554
- expect(current.markdown).toContain('total_contract_arr_cents');
1555
- const patch = await params.toolSet.read_failed_patch.execute({});
1556
- expect(patch.markdown).toContain('account_count');
1557
- await params.toolSet.write_integration_file.execute({
1558
- path: 'semantic-layer/warehouse/mart_account_segments.yaml',
1559
- content: 'name: mart_account_segments\n' +
1560
- 'grain: [account_id]\n' +
1561
- 'columns: [{name: account_id, type: string}]\n' +
1562
- 'joins: []\n' +
1563
- 'measures:\n' +
1564
- ' - name: total_contract_arr_cents\n' +
1565
- ' expr: sum(contract_arr)\n' +
1566
- ' - name: account_count\n' +
1567
- ' expr: count_distinct(account_id)\n',
1568
- });
1569
- return { stopReason: 'natural' };
1570
- }
1571
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1572
- await mkdir(join(root, 'semantic-layer/warehouse'), { recursive: true });
1573
- if (params.telemetryTags.unitKey === 'card-wiki') {
1574
- await writeFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\n' +
1575
- 'grain: [account_id]\n' +
1576
- 'columns: [{name: account_id, type: string}]\n' +
1577
- 'joins: []\n' +
1578
- 'measures:\n' +
1579
- ' - name: total_contract_arr_cents\n' +
1580
- ' expr: sum(contract_arr)\n');
1581
- }
1582
- else if (params.telemetryTags.unitKey === 'card-source') {
1583
- await writeFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\n' +
1584
- 'grain: [account_id]\n' +
1585
- 'columns: [{name: account_id, type: string}]\n' +
1586
- 'joins: []\n' +
1587
- 'measures:\n' +
1588
- ' - name: account_count\n' +
1589
- ' expr: count_distinct(account_id)\n');
1590
- }
1591
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'mart_account_segments');
1592
- currentSession.actions.push({
1593
- target: 'sl',
1594
- type: 'updated',
1595
- key: 'mart_account_segments',
1596
- detail: 'Updated account segments source',
1597
- targetConnectionId: 'warehouse',
1598
- });
1599
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml'], `wu ${params.telemetryTags.unitKey}`, 'KTX Test', 'system@ktx.local');
1600
- return { stopReason: 'natural' };
1601
- });
1602
- const runner = new IngestBundleRunner(deps);
1603
- await mockStageRawFiles(runner, runtime, [
1604
- ['cards/wiki.json', 'hash-a'],
1605
- ['cards/source.json', 'hash-b'],
1606
- ]);
1607
- const result = await runner.run({
1608
- jobId: 'job-resolver-e2e',
1609
- connectionId: 'warehouse',
1610
- sourceKey: 'metabase',
1611
- trigger: 'manual_resync',
1612
- bundleRef: { kind: 'upload', uploadId: 'upload-1' },
1613
- });
1614
- expect(result.commitSha).toBeTruthy();
1615
- const source = await readFile(join(runtime.configDir, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'utf-8');
1616
- expect(source).toContain('total_contract_arr_cents');
1617
- expect(source).toContain('account_count');
1618
- expect(deps.agentRunner.runLoop).toHaveBeenCalledWith(expect.objectContaining({
1619
- modelRole: 'repair',
1620
- telemetryTags: expect.objectContaining({
1621
- operationName: 'ingest-isolated-diff-textual-resolver',
1622
- unitKey: 'card-source',
1623
- }),
1624
- }));
1625
- const successReport = deps.reports.create.mock.calls.at(-1)?.[0]?.body;
1626
- expect(successReport.isolatedDiff).toMatchObject({
1627
- acceptedPatches: 2,
1628
- textualConflicts: 1,
1629
- semanticConflicts: 0,
1630
- resolverAttempts: 1,
1631
- resolverRepairs: 1,
1632
- resolverFailures: 0,
1633
- });
1634
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-resolver-e2e/trace.jsonl'), 'utf-8');
1635
- expect(trace).toContain('textual_conflict_resolver_repaired');
1636
- expect(trace).toContain('patch_accepted_after_textual_resolution');
1637
- }
1638
- finally {
1639
- await rm(runtime.homeDir, { recursive: true, force: true });
1640
- }
1641
- });
1642
- it('repairs final wiki body refs before squash when the repair agent edits the scoped page', async () => {
1643
- const runtime = await makeRealGitRuntime();
1644
- try {
1645
- await mkdir(join(runtime.configDir, 'semantic-layer/warehouse'), { recursive: true });
1646
- await mkdir(join(runtime.configDir, 'wiki/global'), { recursive: true });
1647
- await writeFile(join(runtime.configDir, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr_cents\n expr: sum(contract_arr)\n');
1648
- await writeFile(join(runtime.configDir, 'wiki/global/account-segments.md'), '---\nsummary: Account segments\nusage_mode: auto\n---\n\nExisting ARR uses `mart_account_segments.total_contract_arr_cents`.\n');
1649
- await runtime.git.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml', 'wiki/global/account-segments.md'], 'seed stale wiki body ref', 'KTX Test', 'system@ktx.local');
1650
- const { deps, adapter } = makeDeps(runtime);
1651
- adapter.chunk.mockResolvedValue({
1652
- workUnits: [{ unitKey: 'source-only', rawFiles: ['cards/source.json'], peerFileIndex: [], dependencyPaths: [] }],
1653
- });
1654
- let currentSession = null;
1655
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1656
- currentSession = toolSession;
1657
- return { toRuntimeTools: vi.fn(() => ({})) };
1658
- });
1659
- deps.agentRunner.runLoop = vi.fn(async (params) => {
1660
- if (params.telemetryTags.operationName === 'ingest-isolated-diff-gate-repair') {
1661
- const gateError = await params.toolSet.read_gate_error.execute({});
1662
- expect(gateError.markdown).toContain('total_contract_arr_cents');
1663
- const page = await params.toolSet.read_repair_file.execute({
1664
- path: 'wiki/global/account-segments.md',
1665
- });
1666
- await params.toolSet.write_repair_file.execute({
1667
- path: 'wiki/global/account-segments.md',
1668
- content: page.markdown.replace('total_contract_arr_cents', 'total_contract_arr'),
1669
- });
1670
- return { stopReason: 'natural' };
1671
- }
1672
- if (params.modelRole === 'reconcile') {
1673
- return { stopReason: 'natural' };
1674
- }
1675
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1676
- await writeFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr\n expr: sum(contract_arr)\n');
1677
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'mart_account_segments');
1678
- currentSession.actions.push({
1679
- target: 'sl',
1680
- type: 'updated',
1681
- key: 'mart_account_segments',
1682
- detail: 'Rename ARR measure',
1683
- targetConnectionId: 'warehouse',
1684
- rawPaths: ['cards/source.json'],
1685
- });
1686
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml'], 'wu source rename', 'KTX Test', 'system@ktx.local');
1687
- return { stopReason: 'natural' };
1688
- });
1689
- const runner = new IngestBundleRunner(deps);
1690
- await mockStageRawFiles(runner, runtime, [['cards/source.json', 'h1']]);
1691
- const result = await runner.run({
1692
- jobId: 'job-final-gate-repair',
1693
- connectionId: 'warehouse',
1694
- sourceKey: 'metabase',
1695
- trigger: 'upload',
1696
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1697
- });
1698
- expect(result.commitSha).toBeTruthy();
1699
- await expect(readFile(join(runtime.configDir, 'wiki/global/account-segments.md'), 'utf-8')).resolves.toContain('mart_account_segments.total_contract_arr');
1700
- await expect(readFile(join(runtime.configDir, 'wiki/global/account-segments.md'), 'utf-8')).resolves.not.toContain('total_contract_arr_cents');
1701
- const reportCreate = vi.mocked(deps.reports.create).mock.calls.at(-1)?.[0];
1702
- expect(reportCreate.body.isolatedDiff).toMatchObject({
1703
- gateRepairAttempts: 1,
1704
- gateRepairs: 1,
1705
- gateRepairFailures: 0,
1706
- });
1707
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-final-gate-repair/trace.jsonl'), 'utf-8');
1708
- expect(trace).toContain('gate_repair_repaired');
1709
- expect(trace).toContain('final_artifact_gates_after_gate_repair_finished');
1710
- expect(trace).toContain('final_gate_repair_committed');
1711
- }
1712
- finally {
1713
- await rm(runtime.homeDir, { recursive: true, force: true });
1714
- }
1715
- });
1716
- it('fails before squash when final gate repair makes no edit', async () => {
1717
- const runtime = await makeRealGitRuntime();
1718
- try {
1719
- await mkdir(join(runtime.configDir, 'semantic-layer/warehouse'), { recursive: true });
1720
- await mkdir(join(runtime.configDir, 'wiki/global'), { recursive: true });
1721
- await writeFile(join(runtime.configDir, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr_cents\n expr: sum(contract_arr)\n');
1722
- await writeFile(join(runtime.configDir, 'wiki/global/account-segments.md'), '---\nsummary: Account segments\nusage_mode: auto\n---\n\nExisting ARR uses `mart_account_segments.total_contract_arr_cents`.\n');
1723
- await runtime.git.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml', 'wiki/global/account-segments.md'], 'seed stale wiki body ref', 'KTX Test', 'system@ktx.local');
1724
- const preRunHead = await runtime.git.revParseHead();
1725
- const { deps, adapter } = makeDeps(runtime);
1726
- adapter.chunk.mockResolvedValue({
1727
- workUnits: [{ unitKey: 'source-only', rawFiles: ['cards/source.json'], peerFileIndex: [], dependencyPaths: [] }],
1728
- });
1729
- let currentSession = null;
1730
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1731
- currentSession = toolSession;
1732
- return { toRuntimeTools: vi.fn(() => ({})) };
1733
- });
1734
- deps.agentRunner.runLoop = vi.fn(async (params) => {
1735
- if (params.telemetryTags.operationName === 'ingest-isolated-diff-gate-repair') {
1736
- return { stopReason: 'natural' };
1737
- }
1738
- if (params.modelRole === 'reconcile') {
1739
- return { stopReason: 'natural' };
1740
- }
1741
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1742
- await writeFile(join(root, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr\n expr: sum(contract_arr)\n');
1743
- addTouchedSlSource(currentSession.touchedSlSources, 'warehouse', 'mart_account_segments');
1744
- currentSession.actions.push({
1745
- target: 'sl',
1746
- type: 'updated',
1747
- key: 'mart_account_segments',
1748
- detail: 'Rename ARR measure',
1749
- targetConnectionId: 'warehouse',
1750
- rawPaths: ['cards/source.json'],
1751
- });
1752
- await currentSession.gitService.commitFiles(['semantic-layer/warehouse/mart_account_segments.yaml'], 'wu source rename', 'KTX Test', 'system@ktx.local');
1753
- return { stopReason: 'natural' };
1754
- });
1755
- const runner = new IngestBundleRunner(deps);
1756
- await mockStageRawFiles(runner, runtime, [['cards/source.json', 'h1']]);
1757
- await expect(runner.run({
1758
- jobId: 'job-final-gate-repair-fails',
1759
- connectionId: 'warehouse',
1760
- sourceKey: 'metabase',
1761
- trigger: 'upload',
1762
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1763
- })).rejects.toThrow(/gate repair completed without editing an allowed path/);
1764
- expect(await runtime.git.revParseHead()).toBe(preRunHead);
1765
- const reportCreate = vi.mocked(deps.reports.create).mock.calls.at(-1)?.[0];
1766
- expect(reportCreate.body.status).toBe('failed');
1767
- expect(reportCreate.body.isolatedDiff).toMatchObject({
1768
- gateRepairAttempts: 1,
1769
- gateRepairs: 0,
1770
- gateRepairFailures: 1,
1771
- });
1772
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-final-gate-repair-fails/trace.jsonl'), 'utf-8');
1773
- expect(trace).toContain('gate_repair_failed');
1774
- expect(trace).not.toContain('squash_finished');
1775
- }
1776
- finally {
1777
- await rm(runtime.homeDir, { recursive: true, force: true });
1778
- }
1779
- });
1780
- it('runs finalization before wiki sl-ref repair and final gates', async () => {
1781
- const runtime = await makeRealGitRuntime();
1782
- try {
1783
- const { deps, adapter } = makeDeps(runtime);
1784
- adapter.chunk.mockResolvedValue({
1785
- workUnits: [{ unitKey: 'wiki-page', rawFiles: ['cards/source.json'], peerFileIndex: [], dependencyPaths: [] }],
1786
- });
1787
- adapter.finalize = vi.fn(async ({ workdir }) => {
1788
- await mkdir(join(workdir, 'semantic-layer/warehouse'), { recursive: true });
1789
- await mkdir(join(workdir, 'wiki/global'), { recursive: true });
1790
- await writeFile(join(workdir, 'semantic-layer/warehouse/mart_account_segments.yaml'), 'name: mart_account_segments\ngrain: [account_id]\ncolumns: [{name: account_id, type: string}]\njoins: []\nmeasures:\n - name: total_contract_arr\n expr: sum(contract_arr)\n');
1791
- await writeFile(join(workdir, 'wiki/global/finalized-accounts.md'), '---\nsummary: Finalized accounts\nusage_mode: auto\nsl_refs:\n - mart_account_segments\n - missing_source\n---\n\nAccounts use `mart_account_segments.total_contract_arr`.\n');
1792
- return {
1793
- warnings: [],
1794
- errors: [],
1795
- touchedSources: [{ connectionId: 'warehouse', sourceName: 'mart_account_segments' }],
1796
- changedWikiPageKeys: ['finalized-accounts'],
1797
- actions: [
1798
- {
1799
- target: 'sl',
1800
- type: 'created',
1801
- key: 'mart_account_segments',
1802
- detail: 'Finalized accounts',
1803
- targetConnectionId: 'warehouse',
1804
- rawPaths: ['cards/source.json'],
1805
- },
1806
- {
1807
- target: 'wiki',
1808
- type: 'created',
1809
- key: 'finalized-accounts',
1810
- detail: 'Finalized wiki',
1811
- rawPaths: ['cards/source.json'],
1812
- },
1813
- ],
1814
- };
1815
- });
1816
- deps.agentRunner.runLoop = vi.fn(async () => ({ stopReason: 'natural' }));
1817
- const runner = new IngestBundleRunner(deps);
1818
- await mockStageRawFiles(runner, runtime, [['cards/source.json', 'h1']]);
1819
- await runner.run({
1820
- jobId: 'job-finalization',
1821
- connectionId: 'warehouse',
1822
- sourceKey: 'metabase',
1823
- trigger: 'upload',
1824
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1825
- });
1826
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-finalization/trace.jsonl'), 'utf-8');
1827
- expect(trace.indexOf('finalization_committed')).toBeLessThan(trace.indexOf('wiki_sl_refs_repaired'));
1828
- expect(trace.indexOf('wiki_sl_refs_repaired')).toBeLessThan(trace.indexOf('final_artifact_gates'));
1829
- await expect(readFile(join(runtime.configDir, 'wiki/global/finalized-accounts.md'), 'utf-8')).resolves.toContain('sl_refs:\n - mart_account_segments');
1830
- }
1831
- finally {
1832
- await rm(runtime.homeDir, { recursive: true, force: true });
1833
- }
1834
- });
1835
- it('fails when finalization edits a path already changed earlier in the run', async () => {
1836
- const runtime = await makeRealGitRuntime();
1837
- try {
1838
- const { deps, adapter } = makeDeps(runtime);
1839
- adapter.chunk.mockResolvedValue({
1840
- workUnits: [{ unitKey: 'wiki-page', rawFiles: ['cards/source.json'], peerFileIndex: [], dependencyPaths: [] }],
1841
- });
1842
- let currentSession = null;
1843
- deps.toolsetFactory.createIngestWuToolset = vi.fn((toolSession) => {
1844
- currentSession = toolSession;
1845
- return { toRuntimeTools: vi.fn(() => ({})) };
1846
- });
1847
- deps.agentRunner.runLoop = vi.fn(async () => {
1848
- const root = rootOfConfig(currentSession.configService, runtime.configDir);
1849
- await mkdir(join(root, 'wiki/global'), { recursive: true });
1850
- await writeFile(join(root, 'wiki/global/orders.md'), '---\nsummary: Orders\nusage_mode: auto\n---\n\nWU body\n');
1851
- currentSession.actions.push({
1852
- target: 'wiki',
1853
- type: 'created',
1854
- key: 'orders',
1855
- detail: 'WU orders',
1856
- rawPaths: ['cards/source.json'],
1857
- });
1858
- await currentSession.gitService.commitFiles(['wiki/global/orders.md'], 'wu orders', 'KTX Test', 'system@ktx.local');
1859
- return { stopReason: 'natural' };
1860
- });
1861
- adapter.finalize = vi.fn(async ({ workdir }) => {
1862
- await writeFile(join(workdir, 'wiki/global/orders.md'), '---\nsummary: Orders\nusage_mode: auto\n---\n\nFinalized body\n');
1863
- return {
1864
- warnings: [],
1865
- errors: [],
1866
- touchedSources: [],
1867
- changedWikiPageKeys: ['orders'],
1868
- actions: [{ target: 'wiki', type: 'updated', key: 'orders', detail: 'Conflicting finalization' }],
1869
- };
1870
- });
1871
- const runner = new IngestBundleRunner(deps);
1872
- await mockStageRawFiles(runner, runtime, [['cards/source.json', 'h1']]);
1873
- await expect(runner.run({
1874
- jobId: 'job-finalization-overlap',
1875
- connectionId: 'warehouse',
1876
- sourceKey: 'metabase',
1877
- trigger: 'upload',
1878
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1879
- })).rejects.toThrow(/finalization modified path\(s\) already changed earlier in this run: wiki\/global\/orders\.md/);
1880
- }
1881
- finally {
1882
- await rm(runtime.homeDir, { recursive: true, force: true });
1883
- }
1884
- });
1885
- it('rejects finalization writes to unauthorized semantic-layer targets', async () => {
1886
- const runtime = await makeRealGitRuntime();
1887
- try {
1888
- const { deps, adapter } = makeDeps(runtime);
1889
- adapter.chunk.mockResolvedValue({ workUnits: [] });
1890
- adapter.finalize = vi.fn(async ({ workdir }) => {
1891
- await mkdir(join(workdir, 'semantic-layer/other-warehouse'), { recursive: true });
1892
- await writeFile(join(workdir, 'semantic-layer/other-warehouse/orders.yaml'), 'name: orders\ngrain: [order_id]\ncolumns: [{name: order_id, type: string}]\njoins: []\nmeasures: []\n');
1893
- return {
1894
- warnings: [],
1895
- errors: [],
1896
- touchedSources: [{ connectionId: 'other-warehouse', sourceName: 'orders' }],
1897
- changedWikiPageKeys: [],
1898
- actions: [
1899
- {
1900
- target: 'sl',
1901
- type: 'created',
1902
- key: 'orders',
1903
- targetConnectionId: 'other-warehouse',
1904
- detail: 'Forbidden target',
1905
- rawPaths: ['cards/source.json'],
1906
- },
1907
- ],
1908
- };
1909
- });
1910
- const runner = new IngestBundleRunner(deps);
1911
- await mockStageRawFiles(runner, runtime, [['cards/source.json', 'h1']]);
1912
- await expect(runner.run({
1913
- jobId: 'job-finalization-target-policy',
1914
- connectionId: 'warehouse',
1915
- sourceKey: 'metabase',
1916
- trigger: 'upload',
1917
- bundleRef: { kind: 'upload', uploadId: 'upload' },
1918
- })).rejects.toThrow(/semantic-layer target connection not allowed/);
1919
- const trace = await readFile(join(runtime.configDir, '.ktx/ingest-traces/job-finalization-target-policy/trace.jsonl'), 'utf-8');
1920
- expect(trace).toContain('finalization_committed');
1921
- expect(trace).toContain('semantic_layer_target_policy');
1922
- expect(trace).toContain('ingest_failed');
1923
- }
1924
- finally {
1925
- await rm(runtime.homeDir, { recursive: true, force: true });
1926
- }
1927
- });
1928
- });