@datarecce/ui 0.1.29 → 0.1.31

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (601) hide show
  1. package/dist/{RecceCheckContext-fAKHgsGz.js → RecceCheckContext-DPpu9nG5.js} +2 -2
  2. package/dist/{RecceCheckContext-fAKHgsGz.js.map → RecceCheckContext-DPpu9nG5.js.map} +1 -1
  3. package/dist/{RecceCheckContext-PT4-g1bW.mjs → RecceCheckContext-bXdfQLGG.mjs} +2 -2
  4. package/dist/{RecceCheckContext-PT4-g1bW.mjs.map → RecceCheckContext-bXdfQLGG.mjs.map} +1 -1
  5. package/dist/api.d.mts +1 -1
  6. package/dist/api.d.ts +1 -1
  7. package/dist/api.js +4 -3
  8. package/dist/api.mjs +4 -3
  9. package/dist/{components-B9F5oJbK.js → components-B-YxuuPz.js} +66 -65
  10. package/dist/{components-B9F5oJbK.js.map → components-B-YxuuPz.js.map} +1 -1
  11. package/dist/{components-D2DRqJsz.css → components-BeAjVBV3.css} +1 -1
  12. package/dist/{components-D2DRqJsz.css.map → components-BeAjVBV3.css.map} +1 -1
  13. package/dist/{components-gDC1ucjo.mjs → components-DCOI1YlQ.mjs} +8 -7
  14. package/dist/{components-gDC1ucjo.mjs.map → components-DCOI1YlQ.mjs.map} +1 -1
  15. package/dist/{components-dVXbmdqd.css → components-iUxcqtUB.css} +1 -1
  16. package/dist/{components-dVXbmdqd.css.map → components-iUxcqtUB.css.map} +1 -1
  17. package/dist/components.d.mts +1 -1
  18. package/dist/components.d.ts +1 -1
  19. package/dist/components.js +6 -5
  20. package/dist/components.mjs +6 -5
  21. package/dist/{hooks-C2jUJ9EN.js → hooks-B9hsc1oD.js} +3 -3
  22. package/dist/{hooks-C2jUJ9EN.js.map → hooks-B9hsc1oD.js.map} +1 -1
  23. package/dist/{hooks-4hRUjy9Q.mjs → hooks-DjBNmTdh.mjs} +3 -3
  24. package/dist/{hooks-4hRUjy9Q.mjs.map → hooks-DjBNmTdh.mjs.map} +1 -1
  25. package/dist/hooks.d.mts +1 -1
  26. package/dist/hooks.d.ts +1 -1
  27. package/dist/hooks.js +5 -4
  28. package/dist/hooks.mjs +5 -4
  29. package/dist/{html2canvas-pro.esm-BR5xeFe-.mjs → html2canvas-pro.esm-BInzOtWO.mjs} +1 -1
  30. package/dist/{html2canvas-pro.esm-BR5xeFe-.mjs.map → html2canvas-pro.esm-BInzOtWO.mjs.map} +1 -1
  31. package/dist/{html2canvas-pro.esm-CVOsBdk0.js → html2canvas-pro.esm-WJxOmKlq.js} +1 -1
  32. package/dist/{html2canvas-pro.esm-CVOsBdk0.js.map → html2canvas-pro.esm-WJxOmKlq.js.map} +1 -1
  33. package/dist/{index-Bv5R8iLo.d.mts → index-B9lSPJTi.d.ts} +192 -12
  34. package/dist/index-B9lSPJTi.d.ts.map +1 -0
  35. package/dist/{index-CUtFlKOo.d.ts → index-IIXVIoOL.d.mts} +262 -78
  36. package/dist/index-IIXVIoOL.d.mts.map +1 -0
  37. package/dist/index.d.mts +2 -2
  38. package/dist/index.d.ts +2 -2
  39. package/dist/index.js +15 -7
  40. package/dist/index.js.map +1 -1
  41. package/dist/index.mjs +10 -8
  42. package/dist/index.mjs.map +1 -1
  43. package/dist/mui-theme-B2wm_cvZ.js +732 -0
  44. package/dist/mui-theme-B2wm_cvZ.js.map +1 -0
  45. package/dist/mui-theme-CUhybmBq.mjs +696 -0
  46. package/dist/mui-theme-CUhybmBq.mjs.map +1 -0
  47. package/dist/{state-CELzQ0tM.mjs → state-B9yzhuKs.mjs} +5 -694
  48. package/dist/state-B9yzhuKs.mjs.map +1 -0
  49. package/dist/{state-eEsMhIy4.css → state-DOUPNifc.css} +1 -1
  50. package/dist/{state-eEsMhIy4.css.map → state-DOUPNifc.css.map} +1 -1
  51. package/dist/{state-CemiRRon.js → state-lPCQsWy5.js} +24 -737
  52. package/dist/state-lPCQsWy5.js.map +1 -0
  53. package/dist/styles.css +4 -0
  54. package/dist/theme.d.mts +3 -0
  55. package/dist/theme.d.ts +3 -0
  56. package/dist/theme.js +9 -0
  57. package/dist/theme.mjs +4 -0
  58. package/dist/{tooltipMessage-CrXjOmVM.mjs → tooltipMessage-B--I3p1V.mjs} +1 -1
  59. package/dist/{tooltipMessage-CrXjOmVM.mjs.map → tooltipMessage-B--I3p1V.mjs.map} +1 -1
  60. package/dist/{tooltipMessage-Dbi1kkfi.js → tooltipMessage-DosF13kZ.js} +1 -1
  61. package/dist/{tooltipMessage-Dbi1kkfi.js.map → tooltipMessage-DosF13kZ.js.map} +1 -1
  62. package/dist/types.d.mts +1 -1
  63. package/dist/types.d.ts +1 -1
  64. package/dist/types.js +2 -2
  65. package/dist/types.mjs +2 -2
  66. package/dist/{urls-D7PrPolY.mjs → urls-B1Ymdoz-.mjs} +1 -1
  67. package/dist/{urls-D7PrPolY.mjs.map → urls-B1Ymdoz-.mjs.map} +1 -1
  68. package/dist/{urls-SazAekCZ.js → urls-C4eAc82S.js} +1 -1
  69. package/dist/{urls-SazAekCZ.js.map → urls-C4eAc82S.js.map} +1 -1
  70. package/dist/{version-bWg7XwOu.js → version-Dh8sZhvs.js} +2 -2
  71. package/dist/{version-bWg7XwOu.js.map → version-Dh8sZhvs.js.map} +1 -1
  72. package/dist/{version-paZ9esBk.mjs → version-OnOKzBeQ.mjs} +2 -2
  73. package/dist/{version-paZ9esBk.mjs.map → version-OnOKzBeQ.mjs.map} +1 -1
  74. package/package.json +9 -2
  75. package/recce-source/.editorconfig +26 -0
  76. package/recce-source/.flake8 +37 -0
  77. package/recce-source/.github/ISSUE_TEMPLATE/bug_report.yml +67 -0
  78. package/recce-source/.github/ISSUE_TEMPLATE/custom.md +10 -0
  79. package/recce-source/.github/ISSUE_TEMPLATE/feature_request.yml +42 -0
  80. package/recce-source/.github/PULL_REQUEST_TEMPLATE.md +21 -0
  81. package/recce-source/.github/copilot-instructions.md +331 -0
  82. package/recce-source/.github/instructions/backend-instructions.md +541 -0
  83. package/recce-source/.github/instructions/frontend-instructions.md +317 -0
  84. package/recce-source/.github/workflows/build-statics.yaml +72 -0
  85. package/recce-source/.github/workflows/bump.yaml +48 -0
  86. package/recce-source/.github/workflows/integration-tests-cloud.yaml +92 -0
  87. package/recce-source/.github/workflows/integration-tests-sqlmesh.yaml +33 -0
  88. package/recce-source/.github/workflows/integration-tests.yaml +52 -0
  89. package/recce-source/.github/workflows/nightly.yaml +246 -0
  90. package/recce-source/.github/workflows/release.yaml +196 -0
  91. package/recce-source/.github/workflows/tests-js.yaml +58 -0
  92. package/recce-source/.github/workflows/tests-python.yaml +128 -0
  93. package/recce-source/.pre-commit-config.yaml +26 -0
  94. package/recce-source/CLAUDE.md +483 -0
  95. package/recce-source/CODE_OF_CONDUCT.md +128 -0
  96. package/recce-source/CONTRIBUTING.md +107 -0
  97. package/recce-source/LICENSE +201 -0
  98. package/recce-source/Makefile +126 -0
  99. package/recce-source/README.md +182 -0
  100. package/recce-source/RECCE_CLOUD.md +81 -0
  101. package/recce-source/SECURITY.md +25 -0
  102. package/recce-source/docs/PACKAGING.md +340 -0
  103. package/recce-source/docs/README.md +1 -0
  104. package/recce-source/integration_tests/dbt/dbt_project.yml +26 -0
  105. package/recce-source/integration_tests/dbt/models/customers.sql +69 -0
  106. package/recce-source/integration_tests/dbt/models/docs.md +14 -0
  107. package/recce-source/integration_tests/dbt/models/orders.sql +56 -0
  108. package/recce-source/integration_tests/dbt/models/schema.yml +82 -0
  109. package/recce-source/integration_tests/dbt/models/staging/schema.yml +31 -0
  110. package/recce-source/integration_tests/dbt/models/staging/stg_customers.sql +22 -0
  111. package/recce-source/integration_tests/dbt/models/staging/stg_orders.sql +23 -0
  112. package/recce-source/integration_tests/dbt/models/staging/stg_payments.sql +25 -0
  113. package/recce-source/integration_tests/dbt/packages.yml +7 -0
  114. package/recce-source/integration_tests/dbt/profiles.yml +8 -0
  115. package/recce-source/integration_tests/dbt/seeds/raw_customers.csv +101 -0
  116. package/recce-source/integration_tests/dbt/seeds/raw_orders.csv +100 -0
  117. package/recce-source/integration_tests/dbt/seeds/raw_payments.csv +114 -0
  118. package/recce-source/integration_tests/dbt/seeds/raw_statuses.csv +5 -0
  119. package/recce-source/integration_tests/dbt/smoke_test.sh +72 -0
  120. package/recce-source/integration_tests/dbt/smoke_test_cloud.sh +71 -0
  121. package/recce-source/integration_tests/sqlmesh/__init__.py +0 -0
  122. package/recce-source/integration_tests/sqlmesh/audits/assert_item_price_above_zero.sql +9 -0
  123. package/recce-source/integration_tests/sqlmesh/audits/items.sql +7 -0
  124. package/recce-source/integration_tests/sqlmesh/audits/order_items.sql +7 -0
  125. package/recce-source/integration_tests/sqlmesh/config.py +171 -0
  126. package/recce-source/integration_tests/sqlmesh/helper.py +20 -0
  127. package/recce-source/integration_tests/sqlmesh/hooks/__init__.py +0 -0
  128. package/recce-source/integration_tests/sqlmesh/macros/__init__.py +0 -0
  129. package/recce-source/integration_tests/sqlmesh/macros/macros.py +8 -0
  130. package/recce-source/integration_tests/sqlmesh/macros/macros.sql +8 -0
  131. package/recce-source/integration_tests/sqlmesh/macros/utils.py +11 -0
  132. package/recce-source/integration_tests/sqlmesh/metrics/metrics.sql +25 -0
  133. package/recce-source/integration_tests/sqlmesh/models/customer_revenue_by_day.sql +41 -0
  134. package/recce-source/integration_tests/sqlmesh/models/customer_revenue_lifetime.sql +60 -0
  135. package/recce-source/integration_tests/sqlmesh/models/customers.sql +32 -0
  136. package/recce-source/integration_tests/sqlmesh/models/items.py +95 -0
  137. package/recce-source/integration_tests/sqlmesh/models/marketing.sql +15 -0
  138. package/recce-source/integration_tests/sqlmesh/models/order_items.py +95 -0
  139. package/recce-source/integration_tests/sqlmesh/models/orders.py +70 -0
  140. package/recce-source/integration_tests/sqlmesh/models/raw_marketing.py +62 -0
  141. package/recce-source/integration_tests/sqlmesh/models/top_waiters.sql +23 -0
  142. package/recce-source/integration_tests/sqlmesh/models/waiter_as_customer_by_day.sql +29 -0
  143. package/recce-source/integration_tests/sqlmesh/models/waiter_names.sql +10 -0
  144. package/recce-source/integration_tests/sqlmesh/models/waiter_revenue_by_day.sql +29 -0
  145. package/recce-source/integration_tests/sqlmesh/models/waiters.py +62 -0
  146. package/recce-source/integration_tests/sqlmesh/prep_env.sh +16 -0
  147. package/recce-source/integration_tests/sqlmesh/schema.yaml +5 -0
  148. package/recce-source/integration_tests/sqlmesh/seeds/waiter_names.csv +11 -0
  149. package/recce-source/integration_tests/sqlmesh/test_server.sh +29 -0
  150. package/recce-source/integration_tests/sqlmesh/tests/test_customer_revenue_by_day.yaml +63 -0
  151. package/recce-source/integration_tests/sqlmesh/tests/test_order_items.yaml +72 -0
  152. package/recce-source/js/.editorconfig +27 -0
  153. package/recce-source/js/.env.development +5 -0
  154. package/recce-source/js/.husky/pre-commit +29 -0
  155. package/recce-source/js/.nvmrc +1 -0
  156. package/recce-source/js/README.md +39 -0
  157. package/recce-source/js/app/(mainComponents)/DisplayModeToggle.tsx +65 -0
  158. package/recce-source/js/app/(mainComponents)/NavBar.tsx +228 -0
  159. package/recce-source/js/app/(mainComponents)/RecceVersionBadge.tsx +107 -0
  160. package/recce-source/js/app/(mainComponents)/TopBar.tsx +252 -0
  161. package/recce-source/js/app/@lineage/default.tsx +20 -0
  162. package/recce-source/js/app/@lineage/page.tsx +14 -0
  163. package/recce-source/js/app/MainLayout.tsx +170 -0
  164. package/recce-source/js/app/Providers.tsx +49 -0
  165. package/recce-source/js/app/checks/page.tsx +296 -0
  166. package/recce-source/js/app/error.tsx +93 -0
  167. package/recce-source/js/app/favicon.ico +0 -0
  168. package/recce-source/js/app/global-error.tsx +115 -0
  169. package/recce-source/js/app/global.css +82 -0
  170. package/recce-source/js/app/layout.tsx +48 -0
  171. package/recce-source/js/app/lineage/page.tsx +15 -0
  172. package/recce-source/js/app/page.tsx +12 -0
  173. package/recce-source/js/app/query/page.tsx +8 -0
  174. package/recce-source/js/biome.json +313 -0
  175. package/recce-source/js/jest.config.js +34 -0
  176. package/recce-source/js/jest.globals.d.ts +32 -0
  177. package/recce-source/js/jest.setup.js +91 -0
  178. package/recce-source/js/next.config.js +16 -0
  179. package/recce-source/js/package-lock.json +13843 -0
  180. package/recce-source/js/package.json +123 -0
  181. package/recce-source/js/pnpm-lock.yaml +9235 -0
  182. package/recce-source/js/pnpm-workspace.yaml +6 -0
  183. package/recce-source/js/postcss.config.js +5 -0
  184. package/recce-source/js/public/auth_callback.html +68 -0
  185. package/recce-source/js/public/imgs/feedback/thumbs-down.png +0 -0
  186. package/recce-source/js/public/imgs/feedback/thumbs-up.png +0 -0
  187. package/recce-source/js/public/imgs/reload-image.svg +4 -0
  188. package/recce-source/js/public/logo/recce-logo-white.png +0 -0
  189. package/recce-source/js/src/components/AuthModal/AuthModal.tsx +202 -0
  190. package/recce-source/js/src/components/app/AvatarDropdown.tsx +159 -0
  191. package/recce-source/js/src/components/app/EnvInfo.tsx +357 -0
  192. package/recce-source/js/src/components/app/Filename.tsx +388 -0
  193. package/recce-source/js/src/components/app/SetupConnectionPopover.tsx +91 -0
  194. package/recce-source/js/src/components/app/StateExporter.tsx +57 -0
  195. package/recce-source/js/src/components/app/StateImporter.tsx +198 -0
  196. package/recce-source/js/src/components/app/StateSharing.tsx +145 -0
  197. package/recce-source/js/src/components/app/StateSynchronizer.tsx +205 -0
  198. package/recce-source/js/src/components/charts/HistogramChart.tsx +291 -0
  199. package/recce-source/js/src/components/charts/SquareIcon.tsx +51 -0
  200. package/recce-source/js/src/components/charts/TopKSummaryList.tsx +457 -0
  201. package/recce-source/js/src/components/charts/chartTheme.ts +74 -0
  202. package/recce-source/js/src/components/check/CheckBreadcrumb.tsx +97 -0
  203. package/recce-source/js/src/components/check/CheckDescription.tsx +134 -0
  204. package/recce-source/js/src/components/check/CheckDetail.tsx +797 -0
  205. package/recce-source/js/src/components/check/CheckEmptyState.tsx +84 -0
  206. package/recce-source/js/src/components/check/CheckList.tsx +320 -0
  207. package/recce-source/js/src/components/check/LineageDiffView.tsx +32 -0
  208. package/recce-source/js/src/components/check/PresetCheckTemplateView.tsx +48 -0
  209. package/recce-source/js/src/components/check/SchemaDiffView.tsx +290 -0
  210. package/recce-source/js/src/components/check/check.ts +25 -0
  211. package/recce-source/js/src/components/check/timeline/CheckTimeline.tsx +163 -0
  212. package/recce-source/js/src/components/check/timeline/CommentInput.tsx +84 -0
  213. package/recce-source/js/src/components/check/timeline/TimelineEvent.tsx +468 -0
  214. package/recce-source/js/src/components/check/timeline/index.ts +12 -0
  215. package/recce-source/js/src/components/check/utils.ts +12 -0
  216. package/recce-source/js/src/components/data-grid/ScreenshotDataGrid.tsx +333 -0
  217. package/recce-source/js/src/components/data-grid/agGridStyles.css +55 -0
  218. package/recce-source/js/src/components/data-grid/agGridTheme.ts +43 -0
  219. package/recce-source/js/src/components/editor/CodeEditor.tsx +107 -0
  220. package/recce-source/js/src/components/editor/DiffEditor.tsx +162 -0
  221. package/recce-source/js/src/components/editor/index.ts +12 -0
  222. package/recce-source/js/src/components/errorboundary/ErrorBoundary.tsx +87 -0
  223. package/recce-source/js/src/components/histogram/HistogramDiffForm.tsx +147 -0
  224. package/recce-source/js/src/components/histogram/HistogramDiffResultView.tsx +63 -0
  225. package/recce-source/js/src/components/icons/index.tsx +142 -0
  226. package/recce-source/js/src/components/lineage/ActionControl.tsx +63 -0
  227. package/recce-source/js/src/components/lineage/ActionTag.tsx +141 -0
  228. package/recce-source/js/src/components/lineage/ChangeStatusLegend.tsx +46 -0
  229. package/recce-source/js/src/components/lineage/ColumnLevelLineageControl.tsx +327 -0
  230. package/recce-source/js/src/components/lineage/ColumnLevelLineageLegend.tsx +57 -0
  231. package/recce-source/js/src/components/lineage/GraphColumnNode.tsx +199 -0
  232. package/recce-source/js/src/components/lineage/GraphEdge.tsx +59 -0
  233. package/recce-source/js/src/components/lineage/GraphNode.tsx +555 -0
  234. package/recce-source/js/src/components/lineage/LineagePage.tsx +10 -0
  235. package/recce-source/js/src/components/lineage/LineageView.tsx +1384 -0
  236. package/recce-source/js/src/components/lineage/LineageViewContext.tsx +86 -0
  237. package/recce-source/js/src/components/lineage/LineageViewContextMenu.tsx +637 -0
  238. package/recce-source/js/src/components/lineage/LineageViewNotification.tsx +64 -0
  239. package/recce-source/js/src/components/lineage/LineageViewTopBar.tsx +596 -0
  240. package/recce-source/js/src/components/lineage/NodeSqlView.tsx +136 -0
  241. package/recce-source/js/src/components/lineage/NodeTag.tsx +278 -0
  242. package/recce-source/js/src/components/lineage/NodeView.tsx +642 -0
  243. package/recce-source/js/src/components/lineage/SandboxView.tsx +436 -0
  244. package/recce-source/js/src/components/lineage/ServerDisconnectedModalContent.tsx +105 -0
  245. package/recce-source/js/src/components/lineage/SetupConnectionBanner.tsx +52 -0
  246. package/recce-source/js/src/components/lineage/SingleEnvironmentQueryView.tsx +152 -0
  247. package/recce-source/js/src/components/lineage/graph.test.ts +31 -0
  248. package/recce-source/js/src/components/lineage/graph.ts +58 -0
  249. package/recce-source/js/src/components/lineage/lineage.test.ts +169 -0
  250. package/recce-source/js/src/components/lineage/lineage.ts +521 -0
  251. package/recce-source/js/src/components/lineage/styles.css +42 -0
  252. package/recce-source/js/src/components/lineage/styles.tsx +165 -0
  253. package/recce-source/js/src/components/lineage/useMultiNodesAction.ts +352 -0
  254. package/recce-source/js/src/components/lineage/useValueDiffAlertDialog.tsx +108 -0
  255. package/recce-source/js/src/components/onboarding-guide/Notification.tsx +62 -0
  256. package/recce-source/js/src/components/profile/ProfileDiffForm.tsx +134 -0
  257. package/recce-source/js/src/components/profile/ProfileDiffResultView.tsx +245 -0
  258. package/recce-source/js/src/components/query/ChangedOnlyCheckbox.tsx +29 -0
  259. package/recce-source/js/src/components/query/DiffText.tsx +120 -0
  260. package/recce-source/js/src/components/query/QueryDiffResultView.tsx +470 -0
  261. package/recce-source/js/src/components/query/QueryForm.tsx +80 -0
  262. package/recce-source/js/src/components/query/QueryPage.tsx +282 -0
  263. package/recce-source/js/src/components/query/QueryResultView.tsx +180 -0
  264. package/recce-source/js/src/components/query/SetupConnectionGuide.tsx +57 -0
  265. package/recce-source/js/src/components/query/SqlEditor.tsx +245 -0
  266. package/recce-source/js/src/components/query/ToggleSwitch.tsx +84 -0
  267. package/recce-source/js/src/components/query/styles.css +21 -0
  268. package/recce-source/js/src/components/routing/DirectUrlAccess.test.tsx +428 -0
  269. package/recce-source/js/src/components/routing/LineageStatePreservation.test.tsx +311 -0
  270. package/recce-source/js/src/components/routing/Navigation.test.tsx +256 -0
  271. package/recce-source/js/src/components/rowcount/RowCountDiffResultView.tsx +109 -0
  272. package/recce-source/js/src/components/rowcount/delta.ts +11 -0
  273. package/recce-source/js/src/components/run/RunList.tsx +303 -0
  274. package/recce-source/js/src/components/run/RunModal.tsx +191 -0
  275. package/recce-source/js/src/components/run/RunPage.tsx +26 -0
  276. package/recce-source/js/src/components/run/RunResultPane.tsx +454 -0
  277. package/recce-source/js/src/components/run/RunStatusAndDate.tsx +106 -0
  278. package/recce-source/js/src/components/run/RunToolbar.tsx +70 -0
  279. package/recce-source/js/src/components/run/RunView.tsx +196 -0
  280. package/recce-source/js/src/components/run/registry.ts +214 -0
  281. package/recce-source/js/src/components/run/types.ts +14 -0
  282. package/recce-source/js/src/components/schema/ColumnNameCell.test.tsx +169 -0
  283. package/recce-source/js/src/components/schema/ColumnNameCell.tsx +198 -0
  284. package/recce-source/js/src/components/schema/SchemaView.tsx +337 -0
  285. package/recce-source/js/src/components/schema/schemaDiff.ts +32 -0
  286. package/recce-source/js/src/components/schema/style.css +134 -0
  287. package/recce-source/js/src/components/screenshot/ScreenshotBox.tsx +39 -0
  288. package/recce-source/js/src/components/shared/HistoryToggle.tsx +35 -0
  289. package/recce-source/js/src/components/split/Split.tsx +40 -0
  290. package/recce-source/js/src/components/split/styles.css +24 -0
  291. package/recce-source/js/src/components/summary/ChangeSummary.tsx +264 -0
  292. package/recce-source/js/src/components/summary/SchemaSummary.tsx +123 -0
  293. package/recce-source/js/src/components/summary/SummaryView.tsx +29 -0
  294. package/recce-source/js/src/components/timeout/IdleTimeoutBadge.tsx +48 -0
  295. package/recce-source/js/src/components/top-k/TopKDiffForm.tsx +58 -0
  296. package/recce-source/js/src/components/top-k/TopKDiffResultView.tsx +73 -0
  297. package/recce-source/js/src/components/ui/dataGrid/DataFrameColumnGroupHeader.tsx +228 -0
  298. package/recce-source/js/src/components/ui/dataGrid/DataFrameColumnHeader.tsx +113 -0
  299. package/recce-source/js/src/components/ui/dataGrid/defaultRenderCell.tsx +72 -0
  300. package/recce-source/js/src/components/ui/dataGrid/index.ts +23 -0
  301. package/recce-source/js/src/components/ui/dataGrid/inlineRenderCell.test.tsx +607 -0
  302. package/recce-source/js/src/components/ui/dataGrid/inlineRenderCell.tsx +211 -0
  303. package/recce-source/js/src/components/ui/dataGrid/schemaCells.test.tsx +452 -0
  304. package/recce-source/js/src/components/ui/dataGrid/schemaCells.tsx +142 -0
  305. package/recce-source/js/src/components/ui/dataGrid/valueDiffCells.test.tsx +178 -0
  306. package/recce-source/js/src/components/ui/dataGrid/valueDiffCells.tsx +275 -0
  307. package/recce-source/js/src/components/ui/markdown/ExternalLinkConfirmDialog.tsx +134 -0
  308. package/recce-source/js/src/components/ui/markdown/MarkdownContent.tsx +364 -0
  309. package/recce-source/js/src/components/ui/mui/index.ts +13 -0
  310. package/recce-source/js/src/components/ui/mui-provider.tsx +67 -0
  311. package/recce-source/js/src/components/ui/mui-theme.ts +1039 -0
  312. package/recce-source/js/src/components/ui/mui-utils.ts +113 -0
  313. package/recce-source/js/src/components/ui/toaster.tsx +288 -0
  314. package/recce-source/js/src/components/valuediff/ValueDiffDetailResultView.tsx +217 -0
  315. package/recce-source/js/src/components/valuediff/ValueDiffForm.tsx +246 -0
  316. package/recce-source/js/src/components/valuediff/ValueDiffResultView.tsx +82 -0
  317. package/recce-source/js/src/components/valuediff/shared.ts +33 -0
  318. package/recce-source/js/src/constants/tooltipMessage.ts +3 -0
  319. package/recce-source/js/src/constants/urls.ts +1 -0
  320. package/recce-source/js/src/lib/UrlHash.ts +12 -0
  321. package/recce-source/js/src/lib/api/adhocQuery.ts +70 -0
  322. package/recce-source/js/src/lib/api/axiosClient.ts +9 -0
  323. package/recce-source/js/src/lib/api/cacheKeys.ts +13 -0
  324. package/recce-source/js/src/lib/api/checkEvents.ts +252 -0
  325. package/recce-source/js/src/lib/api/checks.ts +129 -0
  326. package/recce-source/js/src/lib/api/cll.ts +53 -0
  327. package/recce-source/js/src/lib/api/connectToCloud.ts +13 -0
  328. package/recce-source/js/src/lib/api/flag.ts +37 -0
  329. package/recce-source/js/src/lib/api/info.ts +198 -0
  330. package/recce-source/js/src/lib/api/instanceInfo.ts +25 -0
  331. package/recce-source/js/src/lib/api/keepAlive.ts +108 -0
  332. package/recce-source/js/src/lib/api/lineagecheck.ts +35 -0
  333. package/recce-source/js/src/lib/api/localStorageKeys.ts +7 -0
  334. package/recce-source/js/src/lib/api/models.ts +59 -0
  335. package/recce-source/js/src/lib/api/profile.ts +65 -0
  336. package/recce-source/js/src/lib/api/rowcount.ts +19 -0
  337. package/recce-source/js/src/lib/api/runs.ts +174 -0
  338. package/recce-source/js/src/lib/api/schemacheck.ts +31 -0
  339. package/recce-source/js/src/lib/api/select.ts +25 -0
  340. package/recce-source/js/src/lib/api/sessionStorageKeys.ts +8 -0
  341. package/recce-source/js/src/lib/api/state.ts +117 -0
  342. package/recce-source/js/src/lib/api/track.ts +281 -0
  343. package/recce-source/js/src/lib/api/types.ts +284 -0
  344. package/recce-source/js/src/lib/api/user.ts +42 -0
  345. package/recce-source/js/src/lib/api/valuediff.ts +46 -0
  346. package/recce-source/js/src/lib/api/version.ts +40 -0
  347. package/recce-source/js/src/lib/const.ts +9 -0
  348. package/recce-source/js/src/lib/dataGrid/crossFunctionConsistency.test.ts +626 -0
  349. package/recce-source/js/src/lib/dataGrid/dataGridFactory.test.ts +2140 -0
  350. package/recce-source/js/src/lib/dataGrid/dataGridFactory.ts +397 -0
  351. package/recce-source/js/src/lib/dataGrid/generators/rowCountUtils.test.ts +132 -0
  352. package/recce-source/js/src/lib/dataGrid/generators/rowCountUtils.ts +126 -0
  353. package/recce-source/js/src/lib/dataGrid/generators/toDataDiffGrid.test.ts +1627 -0
  354. package/recce-source/js/src/lib/dataGrid/generators/toDataDiffGrid.ts +140 -0
  355. package/recce-source/js/src/lib/dataGrid/generators/toDataGrid.ts +67 -0
  356. package/recce-source/js/src/lib/dataGrid/generators/toRowCountDataGrid.test.ts +142 -0
  357. package/recce-source/js/src/lib/dataGrid/generators/toRowCountDataGrid.ts +71 -0
  358. package/recce-source/js/src/lib/dataGrid/generators/toRowCountDiffDataGrid.test.ts +258 -0
  359. package/recce-source/js/src/lib/dataGrid/generators/toRowCountDiffDataGrid.ts +153 -0
  360. package/recce-source/js/src/lib/dataGrid/generators/toSchemaDataGrid.test.ts +951 -0
  361. package/recce-source/js/src/lib/dataGrid/generators/toSchemaDataGrid.ts +221 -0
  362. package/recce-source/js/src/lib/dataGrid/generators/toValueDataGrid.test.ts +395 -0
  363. package/recce-source/js/src/lib/dataGrid/generators/toValueDataGrid.ts +184 -0
  364. package/recce-source/js/src/lib/dataGrid/generators/toValueDiffGrid.test.ts +884 -0
  365. package/recce-source/js/src/lib/dataGrid/generators/toValueDiffGrid.ts +113 -0
  366. package/recce-source/js/src/lib/dataGrid/index.ts +51 -0
  367. package/recce-source/js/src/lib/dataGrid/propertyBased.test.ts +858 -0
  368. package/recce-source/js/src/lib/dataGrid/shared/columnBuilders.test.ts +482 -0
  369. package/recce-source/js/src/lib/dataGrid/shared/columnBuilders.ts +345 -0
  370. package/recce-source/js/src/lib/dataGrid/shared/dataTypeEdgeCases.test.ts +698 -0
  371. package/recce-source/js/src/lib/dataGrid/shared/diffColumnBuilder.test.tsx +820 -0
  372. package/recce-source/js/src/lib/dataGrid/shared/diffColumnBuilder.tsx +277 -0
  373. package/recce-source/js/src/lib/dataGrid/shared/gridUtils.test.ts +785 -0
  374. package/recce-source/js/src/lib/dataGrid/shared/gridUtils.ts +370 -0
  375. package/recce-source/js/src/lib/dataGrid/shared/index.ts +81 -0
  376. package/recce-source/js/src/lib/dataGrid/shared/rowBuilders.test.ts +909 -0
  377. package/recce-source/js/src/lib/dataGrid/shared/rowBuilders.ts +325 -0
  378. package/recce-source/js/src/lib/dataGrid/shared/simpleColumnBuilder.tsx +240 -0
  379. package/recce-source/js/src/lib/dataGrid/shared/toDiffColumn.test.tsx +719 -0
  380. package/recce-source/js/src/lib/dataGrid/shared/toDiffColumn.tsx +231 -0
  381. package/recce-source/js/src/lib/dataGrid/shared/validation.test.ts +559 -0
  382. package/recce-source/js/src/lib/dataGrid/shared/validation.ts +367 -0
  383. package/recce-source/js/src/lib/dataGrid/warehouseNamingConventions.test.ts +1117 -0
  384. package/recce-source/js/src/lib/formatSelect.ts +50 -0
  385. package/recce-source/js/src/lib/hooks/ApiConfigContext.tsx +181 -0
  386. package/recce-source/js/src/lib/hooks/IdleTimeoutContext.tsx +177 -0
  387. package/recce-source/js/src/lib/hooks/LineageGraphContext.tsx +512 -0
  388. package/recce-source/js/src/lib/hooks/RecceActionContext.tsx +269 -0
  389. package/recce-source/js/src/lib/hooks/RecceCheckContext.tsx +33 -0
  390. package/recce-source/js/src/lib/hooks/RecceContextProvider.tsx +54 -0
  391. package/recce-source/js/src/lib/hooks/RecceInstanceContext.tsx +129 -0
  392. package/recce-source/js/src/lib/hooks/RecceQueryContext.tsx +98 -0
  393. package/recce-source/js/src/lib/hooks/RecceShareStateContext.tsx +59 -0
  394. package/recce-source/js/src/lib/hooks/ScreenShot.tsx +399 -0
  395. package/recce-source/js/src/lib/hooks/useAppRouter.test.ts +211 -0
  396. package/recce-source/js/src/lib/hooks/useAppRouter.ts +200 -0
  397. package/recce-source/js/src/lib/hooks/useCheckEvents.ts +99 -0
  398. package/recce-source/js/src/lib/hooks/useCheckToast.tsx +14 -0
  399. package/recce-source/js/src/lib/hooks/useClipBoardToast.tsx +27 -0
  400. package/recce-source/js/src/lib/hooks/useCountdownToast.tsx +102 -0
  401. package/recce-source/js/src/lib/hooks/useFeedbackCollectionToast.tsx +130 -0
  402. package/recce-source/js/src/lib/hooks/useGuideToast.tsx +45 -0
  403. package/recce-source/js/src/lib/hooks/useIdleDetection.tsx +185 -0
  404. package/recce-source/js/src/lib/hooks/useModelColumns.tsx +113 -0
  405. package/recce-source/js/src/lib/hooks/useRecceInstanceInfo.tsx +13 -0
  406. package/recce-source/js/src/lib/hooks/useRecceServerFlag.tsx +13 -0
  407. package/recce-source/js/src/lib/hooks/useRun.tsx +89 -0
  408. package/recce-source/js/src/lib/hooks/useThemeColors.ts +115 -0
  409. package/recce-source/js/src/lib/mergeKeys.test.ts +89 -0
  410. package/recce-source/js/src/lib/mergeKeys.ts +86 -0
  411. package/recce-source/js/src/lib/result/ResultErrorFallback.tsx +9 -0
  412. package/recce-source/js/src/lib/utils/formatTime.ts +84 -0
  413. package/recce-source/js/src/lib/utils/urls.ts +16 -0
  414. package/recce-source/js/src/utils/DropdownValuesInput.tsx +297 -0
  415. package/recce-source/js/src/utils/formatters.tsx +237 -0
  416. package/recce-source/js/src/utils/transforms.ts +81 -0
  417. package/recce-source/js/tsconfig.json +47 -0
  418. package/recce-source/macros/README.md +8 -0
  419. package/recce-source/macros/recce_athena.sql +73 -0
  420. package/recce-source/pyproject.toml +109 -0
  421. package/recce-source/recce/VERSION +1 -0
  422. package/recce-source/recce/__init__.py +84 -0
  423. package/recce-source/recce/adapter/__init__.py +0 -0
  424. package/recce-source/recce/adapter/base.py +109 -0
  425. package/recce-source/recce/adapter/dbt_adapter/__init__.py +1699 -0
  426. package/recce-source/recce/adapter/dbt_adapter/dbt_version.py +42 -0
  427. package/recce-source/recce/adapter/sqlmesh_adapter.py +141 -0
  428. package/recce-source/recce/apis/__init__.py +0 -0
  429. package/recce-source/recce/apis/check_api.py +203 -0
  430. package/recce-source/recce/apis/check_events_api.py +353 -0
  431. package/recce-source/recce/apis/check_func.py +130 -0
  432. package/recce-source/recce/apis/run_api.py +130 -0
  433. package/recce-source/recce/apis/run_func.py +258 -0
  434. package/recce-source/recce/artifact.py +266 -0
  435. package/recce-source/recce/cli.py +1846 -0
  436. package/recce-source/recce/config.py +127 -0
  437. package/recce-source/recce/connect_to_cloud.py +138 -0
  438. package/recce-source/recce/core.py +334 -0
  439. package/recce-source/recce/diff.py +26 -0
  440. package/recce-source/recce/event/CONFIG +1 -0
  441. package/recce-source/recce/event/SENTRY_DNS +1 -0
  442. package/recce-source/recce/event/__init__.py +304 -0
  443. package/recce-source/recce/event/collector.py +184 -0
  444. package/recce-source/recce/event/track.py +158 -0
  445. package/recce-source/recce/exceptions.py +21 -0
  446. package/recce-source/recce/git.py +77 -0
  447. package/recce-source/recce/github.py +222 -0
  448. package/recce-source/recce/mcp_server.py +861 -0
  449. package/recce-source/recce/models/__init__.py +6 -0
  450. package/recce-source/recce/models/check.py +473 -0
  451. package/recce-source/recce/models/run.py +46 -0
  452. package/recce-source/recce/models/types.py +218 -0
  453. package/recce-source/recce/pull_request.py +124 -0
  454. package/recce-source/recce/run.py +390 -0
  455. package/recce-source/recce/server.py +877 -0
  456. package/recce-source/recce/state/__init__.py +31 -0
  457. package/recce-source/recce/state/cloud.py +644 -0
  458. package/recce-source/recce/state/const.py +26 -0
  459. package/recce-source/recce/state/local.py +56 -0
  460. package/recce-source/recce/state/state.py +119 -0
  461. package/recce-source/recce/state/state_loader.py +174 -0
  462. package/recce-source/recce/summary.py +575 -0
  463. package/recce-source/recce/tasks/__init__.py +23 -0
  464. package/recce-source/recce/tasks/core.py +134 -0
  465. package/recce-source/recce/tasks/dataframe.py +170 -0
  466. package/recce-source/recce/tasks/histogram.py +433 -0
  467. package/recce-source/recce/tasks/lineage.py +19 -0
  468. package/recce-source/recce/tasks/profile.py +298 -0
  469. package/recce-source/recce/tasks/query.py +450 -0
  470. package/recce-source/recce/tasks/rowcount.py +277 -0
  471. package/recce-source/recce/tasks/schema.py +65 -0
  472. package/recce-source/recce/tasks/top_k.py +172 -0
  473. package/recce-source/recce/tasks/utils.py +147 -0
  474. package/recce-source/recce/tasks/valuediff.py +497 -0
  475. package/recce-source/recce/util/__init__.py +4 -0
  476. package/recce-source/recce/util/api_token.py +80 -0
  477. package/recce-source/recce/util/breaking.py +330 -0
  478. package/recce-source/recce/util/cache.py +25 -0
  479. package/recce-source/recce/util/cll.py +355 -0
  480. package/recce-source/recce/util/cloud/__init__.py +15 -0
  481. package/recce-source/recce/util/cloud/base.py +115 -0
  482. package/recce-source/recce/util/cloud/check_events.py +190 -0
  483. package/recce-source/recce/util/cloud/checks.py +242 -0
  484. package/recce-source/recce/util/io.py +120 -0
  485. package/recce-source/recce/util/lineage.py +83 -0
  486. package/recce-source/recce/util/logger.py +25 -0
  487. package/recce-source/recce/util/onboarding_state.py +45 -0
  488. package/recce-source/recce/util/perf_tracking.py +85 -0
  489. package/recce-source/recce/util/pydantic_model.py +22 -0
  490. package/recce-source/recce/util/recce_cloud.py +454 -0
  491. package/recce-source/recce/util/singleton.py +18 -0
  492. package/recce-source/recce/util/startup_perf.py +121 -0
  493. package/recce-source/recce/yaml/__init__.py +58 -0
  494. package/recce-source/recce_cloud/README.md +780 -0
  495. package/recce-source/recce_cloud/VERSION +1 -0
  496. package/recce-source/recce_cloud/__init__.py +24 -0
  497. package/recce-source/recce_cloud/api/__init__.py +17 -0
  498. package/recce-source/recce_cloud/api/base.py +132 -0
  499. package/recce-source/recce_cloud/api/client.py +186 -0
  500. package/recce-source/recce_cloud/api/exceptions.py +26 -0
  501. package/recce-source/recce_cloud/api/factory.py +63 -0
  502. package/recce-source/recce_cloud/api/github.py +106 -0
  503. package/recce-source/recce_cloud/api/gitlab.py +111 -0
  504. package/recce-source/recce_cloud/artifact.py +57 -0
  505. package/recce-source/recce_cloud/ci_providers/__init__.py +9 -0
  506. package/recce-source/recce_cloud/ci_providers/base.py +82 -0
  507. package/recce-source/recce_cloud/ci_providers/detector.py +147 -0
  508. package/recce-source/recce_cloud/ci_providers/github_actions.py +136 -0
  509. package/recce-source/recce_cloud/ci_providers/gitlab_ci.py +130 -0
  510. package/recce-source/recce_cloud/cli.py +434 -0
  511. package/recce-source/recce_cloud/download.py +230 -0
  512. package/recce-source/recce_cloud/hatch_build.py +20 -0
  513. package/recce-source/recce_cloud/pyproject.toml +49 -0
  514. package/recce-source/recce_cloud/upload.py +214 -0
  515. package/recce-source/test.py +0 -0
  516. package/recce-source/tests/__init__.py +0 -0
  517. package/recce-source/tests/adapter/__init__.py +0 -0
  518. package/recce-source/tests/adapter/dbt_adapter/__init__.py +0 -0
  519. package/recce-source/tests/adapter/dbt_adapter/conftest.py +17 -0
  520. package/recce-source/tests/adapter/dbt_adapter/dbt_test_helper.py +298 -0
  521. package/recce-source/tests/adapter/dbt_adapter/test_dbt_adapter.py +25 -0
  522. package/recce-source/tests/adapter/dbt_adapter/test_dbt_cll.py +717 -0
  523. package/recce-source/tests/adapter/dbt_adapter/test_proj/dbt_project.yml +4 -0
  524. package/recce-source/tests/adapter/dbt_adapter/test_proj/manifest.json +1 -0
  525. package/recce-source/tests/adapter/dbt_adapter/test_proj/package-lock.yml +8 -0
  526. package/recce-source/tests/adapter/dbt_adapter/test_proj/packages.yml +7 -0
  527. package/recce-source/tests/adapter/dbt_adapter/test_proj/profiles.yml +6 -0
  528. package/recce-source/tests/adapter/dbt_adapter/test_selector.py +205 -0
  529. package/recce-source/tests/apis/__init__.py +0 -0
  530. package/recce-source/tests/apis/row_count_diff.json +59 -0
  531. package/recce-source/tests/apis/test_check_events_api.py +615 -0
  532. package/recce-source/tests/apis/test_run_func.py +433 -0
  533. package/recce-source/tests/catalog.json +527 -0
  534. package/recce-source/tests/data/manifest/base/catalog.json +1 -0
  535. package/recce-source/tests/data/manifest/base/manifest.json +1 -0
  536. package/recce-source/tests/data/manifest/pr2/catalog.json +1 -0
  537. package/recce-source/tests/data/manifest/pr2/manifest.json +1 -0
  538. package/recce-source/tests/manifest.json +10655 -0
  539. package/recce-source/tests/models/__init__.py +0 -0
  540. package/recce-source/tests/models/test_check.py +731 -0
  541. package/recce-source/tests/models/test_run_models.py +295 -0
  542. package/recce-source/tests/recce_cloud/__init__.py +0 -0
  543. package/recce-source/tests/recce_cloud/test_ci_providers.py +351 -0
  544. package/recce-source/tests/recce_cloud/test_cli.py +735 -0
  545. package/recce-source/tests/recce_cloud/test_client.py +379 -0
  546. package/recce-source/tests/recce_cloud/test_platform_clients.py +483 -0
  547. package/recce-source/tests/recce_state.json +1 -0
  548. package/recce-source/tests/state/test_cloud.py +719 -0
  549. package/recce-source/tests/state/test_local.py +164 -0
  550. package/recce-source/tests/state/test_state_loader.py +211 -0
  551. package/recce-source/tests/tasks/__init__.py +0 -0
  552. package/recce-source/tests/tasks/conftest.py +4 -0
  553. package/recce-source/tests/tasks/test_histogram.py +129 -0
  554. package/recce-source/tests/tasks/test_lineage.py +55 -0
  555. package/recce-source/tests/tasks/test_preset_checks.py +64 -0
  556. package/recce-source/tests/tasks/test_profile.py +397 -0
  557. package/recce-source/tests/tasks/test_query.py +528 -0
  558. package/recce-source/tests/tasks/test_row_count.py +133 -0
  559. package/recce-source/tests/tasks/test_schema.py +122 -0
  560. package/recce-source/tests/tasks/test_top_k.py +77 -0
  561. package/recce-source/tests/tasks/test_utils.py +439 -0
  562. package/recce-source/tests/tasks/test_valuediff.py +361 -0
  563. package/recce-source/tests/test_cli.py +236 -0
  564. package/recce-source/tests/test_cli_mcp_optional.py +45 -0
  565. package/recce-source/tests/test_cloud_listing_cli.py +324 -0
  566. package/recce-source/tests/test_config.py +43 -0
  567. package/recce-source/tests/test_connect_to_cloud.py +82 -0
  568. package/recce-source/tests/test_core.py +174 -0
  569. package/recce-source/tests/test_dbt.py +36 -0
  570. package/recce-source/tests/test_mcp_server.py +505 -0
  571. package/recce-source/tests/test_pull_request.py +130 -0
  572. package/recce-source/tests/test_server.py +202 -0
  573. package/recce-source/tests/test_server_lifespan.py +138 -0
  574. package/recce-source/tests/test_summary.py +73 -0
  575. package/recce-source/tests/util/__init__.py +0 -0
  576. package/recce-source/tests/util/cloud/__init__.py +0 -0
  577. package/recce-source/tests/util/cloud/test_check_events.py +255 -0
  578. package/recce-source/tests/util/cloud/test_checks.py +204 -0
  579. package/recce-source/tests/util/test_api_token.py +119 -0
  580. package/recce-source/tests/util/test_breaking.py +1427 -0
  581. package/recce-source/tests/util/test_cll.py +706 -0
  582. package/recce-source/tests/util/test_lineage.py +122 -0
  583. package/recce-source/tests/util/test_onboarding_state.py +84 -0
  584. package/recce-source/tests/util/test_recce_cloud.py +231 -0
  585. package/recce-source/tox.ini +40 -0
  586. package/recce-source/uv.lock +3928 -0
  587. package/src/api/index.ts +32 -0
  588. package/src/components/index.ts +154 -0
  589. package/src/global.d.ts +14 -0
  590. package/src/hooks/index.ts +56 -0
  591. package/src/index.ts +17 -0
  592. package/src/lib/hooks/RouteConfigContext.ts +139 -0
  593. package/src/lib/hooks/useAppRouter.ts +240 -0
  594. package/src/mui-augmentation.d.ts +139 -0
  595. package/src/theme/index.ts +13 -0
  596. package/src/theme.ts +23 -0
  597. package/src/types/index.ts +23 -0
  598. package/dist/index-Bv5R8iLo.d.mts.map +0 -1
  599. package/dist/index-CUtFlKOo.d.ts.map +0 -1
  600. package/dist/state-CELzQ0tM.mjs.map +0 -1
  601. package/dist/state-CemiRRon.js.map +0 -1
@@ -0,0 +1,861 @@
1
+ """
2
+ Recce MCP (Model Context Protocol) Server
3
+
4
+ This module implements a stdio-based MCP server that provides tools for
5
+ interacting with Recce's data validation capabilities.
6
+ """
7
+
8
+ import asyncio
9
+ import json
10
+ import logging
11
+ import os
12
+ import textwrap
13
+ import time
14
+ from datetime import datetime, timezone
15
+ from typing import Any, Dict, List, Optional
16
+
17
+ from mcp.server import Server
18
+ from mcp.server.stdio import stdio_server
19
+ from mcp.types import TextContent, Tool
20
+
21
+ from recce.core import RecceContext, load_context
22
+ from recce.exceptions import RecceException
23
+ from recce.server import RecceServerMode
24
+ from recce.tasks.dataframe import DataFrame
25
+ from recce.tasks.profile import ProfileDiffTask
26
+ from recce.tasks.query import QueryDiffTask, QueryTask
27
+ from recce.tasks.rowcount import RowCountDiffTask
28
+
29
+ logger = logging.getLogger(__name__)
30
+
31
+
32
+ def _truncate_strings(obj: Any, max_length: int = 200) -> Any:
33
+ """Recursively truncate strings longer than max_length in nested dicts and lists"""
34
+ if isinstance(obj, dict):
35
+ return {k: _truncate_strings(v, max_length) for k, v in obj.items()}
36
+ elif isinstance(obj, list):
37
+ return [_truncate_strings(item, max_length) for item in obj]
38
+ elif isinstance(obj, str) and len(obj) > max_length:
39
+ return obj[:max_length] + "..."
40
+ return obj
41
+
42
+
43
+ class MCPLogger:
44
+ """JSON logger for MCP server request/response logging"""
45
+
46
+ def __init__(self, debug: bool = False, log_file: str = "logs/recce-mcp.json"):
47
+ self.debug = debug
48
+ self.log_file = log_file
49
+
50
+ if self.debug:
51
+ # Create logs directory if it doesn't exist
52
+ log_dir = os.path.dirname(log_file)
53
+ if log_dir:
54
+ os.makedirs(log_dir, exist_ok=True)
55
+
56
+ # Overwrite log file on initialization
57
+ try:
58
+ with open(log_file, "w") as f:
59
+ f.write("") # Clear existing content
60
+ except Exception as e:
61
+ logger.warning(f"Failed to initialize log file {log_file}: {e}")
62
+
63
+ def _write_log(self, log_entry: Dict[str, Any]) -> None:
64
+ """Write a log entry to the JSON file"""
65
+ if not self.debug:
66
+ return
67
+
68
+ try:
69
+ with open(self.log_file, "a") as f:
70
+ f.write(json.dumps(log_entry) + "\n")
71
+ except Exception as e:
72
+ logger.warning(f"Failed to write to log file {self.log_file}: {e}")
73
+
74
+ def log_list_tools(self, tools: List[Tool]) -> None:
75
+ """Log a list_tools call"""
76
+ tool_names = [tool.name for tool in tools]
77
+ log_entry = {
78
+ "timestamp": datetime.now(timezone.utc).strftime("%Y-%m-%dT%H:%M:%SZ"),
79
+ "type": "list_tools",
80
+ "tools": tool_names,
81
+ }
82
+ self._write_log(log_entry)
83
+
84
+ def log_tool_call(
85
+ self,
86
+ tool_name: str,
87
+ arguments: Dict[str, Any],
88
+ response: Dict[str, Any],
89
+ duration_ms: float,
90
+ error: Optional[str] = None,
91
+ ) -> None:
92
+ """Log a tool call with request and response"""
93
+ log_entry = {
94
+ "timestamp": datetime.now(timezone.utc).strftime("%Y-%m-%dT%H:%M:%SZ"),
95
+ "type": "call_tool",
96
+ "tool": tool_name,
97
+ "request": arguments,
98
+ "duration_ms": round(duration_ms, 2),
99
+ }
100
+
101
+ if error:
102
+ log_entry["error"] = error
103
+ else:
104
+ log_entry["response"] = _truncate_strings(response)
105
+
106
+ self._write_log(log_entry)
107
+
108
+
109
+ class RecceMCPServer:
110
+ """MCP Server for Recce data validation tools"""
111
+
112
+ def __init__(
113
+ self,
114
+ context: RecceContext,
115
+ mode: Optional[RecceServerMode] = None,
116
+ debug: bool = False,
117
+ log_file: str = "logs/recce-mcp.json",
118
+ ):
119
+ self.context = context
120
+ self.mode = mode or RecceServerMode.server
121
+ self.server = Server("recce")
122
+ self.mcp_logger = MCPLogger(debug=debug, log_file=log_file)
123
+ self._setup_handlers()
124
+
125
+ def _setup_handlers(self):
126
+ """Register all tool handlers"""
127
+
128
+ @self.server.list_tools()
129
+ async def list_tools() -> List[Tool]:
130
+ """List all available tools based on server mode"""
131
+ logger.info(f"[MCP] list_tools called (mode: {self.mode.value if self.mode else 'server'})")
132
+ tools = []
133
+
134
+ # Always available in all modes
135
+ tools.append(
136
+ Tool(
137
+ name="lineage_diff",
138
+ description=textwrap.dedent(
139
+ """
140
+ Get the lineage diff between production(base) and session(current) for changed models.
141
+ Returns nodes and edges (node dependencies) in compact dataframe format.
142
+
143
+ Nodes dataframe includes: idx, id, name, resource_type, materialized, change_status, impacted.
144
+ Edges dataframe includes: from (parent node idx), to (child node idx).
145
+
146
+ Rendering guidance for Mermaid diagram:
147
+ Use graph LR and apply these styles based on change_status and impacted:
148
+ - change_status="added": fill:#d4edda, stroke:#28a745, color:#000000
149
+ - change_status="removed": fill:#f8d7da, stroke:#dc3545, color:#000000
150
+ - change_status="modified" AND impacted=true: fill:#fff3cd, stroke:#ffc107, color:#000000
151
+ - change_status=null AND impacted=true: fill:#ffffff, stroke:#ffc107, color:#000000
152
+ - change_status=null AND impacted=false: fill:#ffffff, stroke:#d3d3d3, color:#999999
153
+ """
154
+ ).strip(),
155
+ inputSchema={
156
+ "type": "object",
157
+ "properties": {
158
+ "select": {
159
+ "type": "string",
160
+ "description": "dbt selector syntax to filter models (optional)",
161
+ },
162
+ "exclude": {
163
+ "type": "string",
164
+ "description": "dbt selector syntax to exclude models (optional)",
165
+ },
166
+ "packages": {
167
+ "type": "array",
168
+ "items": {"type": "string"},
169
+ "description": "List of packages to filter (optional)",
170
+ },
171
+ "view_mode": {
172
+ "type": "string",
173
+ "enum": ["changed_models", "all"],
174
+ "default": "changed_models",
175
+ "description": "View mode: 'changed_models' for only changed models (default), 'all' for all models",
176
+ },
177
+ },
178
+ },
179
+ )
180
+ )
181
+ tools.append(
182
+ Tool(
183
+ name="schema_diff",
184
+ description="Get the schema diff (column changes) between base and current environments. "
185
+ "Shows added, removed, and type-changed columns in compact dataframe format.",
186
+ inputSchema={
187
+ "type": "object",
188
+ "properties": {
189
+ "select": {
190
+ "type": "string",
191
+ "description": "dbt selector syntax to filter models (optional)",
192
+ },
193
+ "exclude": {
194
+ "type": "string",
195
+ "description": "dbt selector syntax to exclude models (optional)",
196
+ },
197
+ "packages": {
198
+ "type": "array",
199
+ "items": {"type": "string"},
200
+ "description": "List of packages to filter (optional)",
201
+ },
202
+ },
203
+ },
204
+ )
205
+ )
206
+
207
+ # Diff tools only available in server mode, not in preview or read-only mode
208
+ if self.mode == RecceServerMode.server:
209
+ tools.extend(
210
+ [
211
+ Tool(
212
+ name="row_count_diff",
213
+ description="Compare row counts between base and current environments for specified models.",
214
+ inputSchema={
215
+ "type": "object",
216
+ "properties": {
217
+ "node_names": {
218
+ "type": "array",
219
+ "items": {"type": "string"},
220
+ "description": "List of model names to check row counts (optional)",
221
+ },
222
+ "node_ids": {
223
+ "type": "array",
224
+ "items": {"type": "string"},
225
+ "description": "List of node IDs to check row counts (optional)",
226
+ },
227
+ "select": {
228
+ "type": "string",
229
+ "description": "dbt selector syntax to filter models (optional)",
230
+ },
231
+ "exclude": {
232
+ "type": "string",
233
+ "description": "dbt selector syntax to exclude models (optional)",
234
+ },
235
+ },
236
+ },
237
+ ),
238
+ Tool(
239
+ name="query",
240
+ description="Execute a SQL query on the current environment. "
241
+ "Supports Jinja templates with dbt macros like {{ ref('model_name') }}.",
242
+ inputSchema={
243
+ "type": "object",
244
+ "properties": {
245
+ "sql_template": {
246
+ "type": "string",
247
+ "description": "SQL query template with optional Jinja syntax",
248
+ },
249
+ "base": {
250
+ "type": "boolean",
251
+ "description": "Whether to run on base environment (default: false)",
252
+ "default": False,
253
+ },
254
+ },
255
+ "required": ["sql_template"],
256
+ },
257
+ ),
258
+ Tool(
259
+ name="query_diff",
260
+ description="Execute SQL queries on both base and current environments and compare results. "
261
+ "Supports primary keys for row-level comparison.",
262
+ inputSchema={
263
+ "type": "object",
264
+ "properties": {
265
+ "sql_template": {
266
+ "type": "string",
267
+ "description": "SQL query template for current environment",
268
+ },
269
+ "base_sql_template": {
270
+ "type": "string",
271
+ "description": "SQL query template for base environment (optional, defaults to sql_template)",
272
+ },
273
+ "primary_keys": {
274
+ "type": "array",
275
+ "items": {"type": "string"},
276
+ "description": "List of primary key columns for row comparison (optional)",
277
+ },
278
+ },
279
+ "required": ["sql_template"],
280
+ },
281
+ ),
282
+ Tool(
283
+ name="profile_diff",
284
+ description="Generate and compare statistical profiles (min, max, avg, distinct count, etc.) "
285
+ "for columns in a model between base and current environments.",
286
+ inputSchema={
287
+ "type": "object",
288
+ "properties": {
289
+ "model": {
290
+ "type": "string",
291
+ "description": "Model name to profile",
292
+ },
293
+ "columns": {
294
+ "type": "array",
295
+ "items": {"type": "string"},
296
+ "description": "List of column names to profile (optional, profiles all columns if not specified)",
297
+ },
298
+ },
299
+ "required": ["model"],
300
+ },
301
+ ),
302
+ Tool(
303
+ name="list_checks",
304
+ description="List all checks in the current session. Returns check metadata including check IDs, names, types, parameters, and approval status.",
305
+ inputSchema={
306
+ "type": "object",
307
+ "properties": {},
308
+ },
309
+ ),
310
+ Tool(
311
+ name="run_check",
312
+ description="Run a single check by ID and wait for completion. Returns execution status, results, and approval status.",
313
+ inputSchema={
314
+ "type": "object",
315
+ "properties": {
316
+ "check_id": {
317
+ "type": "string",
318
+ "description": "The ID of the check to run",
319
+ },
320
+ },
321
+ "required": ["check_id"],
322
+ },
323
+ ),
324
+ ]
325
+ )
326
+
327
+ self.mcp_logger.log_list_tools(tools)
328
+
329
+ # Log available tools to console
330
+ tool_names = [tool.name for tool in tools]
331
+ logger.info(f"[MCP] Returning {len(tools)} tools: {', '.join(tool_names)}")
332
+
333
+ return tools
334
+
335
+ @self.server.call_tool()
336
+ async def call_tool(name: str, arguments: Dict[str, Any]) -> List[TextContent]:
337
+ """Handle tool calls"""
338
+ start_time = time.perf_counter()
339
+
340
+ # Log incoming request
341
+ logger.info(f"[MCP] Tool call received: {name}")
342
+ logger.info(f"[MCP] Arguments: {json.dumps(arguments, indent=2)}")
343
+
344
+ try:
345
+ # Check if tool is blocked in non-server mode
346
+ blocked_tools_in_non_server = {
347
+ "row_count_diff",
348
+ "query",
349
+ "query_diff",
350
+ "profile_diff",
351
+ "list_checks",
352
+ "run_check",
353
+ }
354
+ if self.mode != RecceServerMode.server and name in blocked_tools_in_non_server:
355
+ raise ValueError(
356
+ f"Tool '{name}' is not available in {self.mode.value} mode. "
357
+ "Only 'lineage_diff' and 'schema_diff' are available in this mode."
358
+ )
359
+
360
+ if name == "lineage_diff":
361
+ result = await self._tool_lineage_diff(arguments)
362
+ elif name == "schema_diff":
363
+ result = await self._tool_schema_diff(arguments)
364
+ elif name == "row_count_diff":
365
+ result = await self._tool_row_count_diff(arguments)
366
+ elif name == "query":
367
+ result = await self._tool_query(arguments)
368
+ elif name == "query_diff":
369
+ result = await self._tool_query_diff(arguments)
370
+ elif name == "profile_diff":
371
+ result = await self._tool_profile_diff(arguments)
372
+ elif name == "list_checks":
373
+ result = await self._tool_list_checks(arguments)
374
+ elif name == "run_check":
375
+ result = await self._tool_run_check(arguments)
376
+ else:
377
+ raise ValueError(f"Unknown tool: {name}")
378
+
379
+ duration_ms = (time.perf_counter() - start_time) * 1000
380
+ self.mcp_logger.log_tool_call(name, arguments, result, duration_ms)
381
+
382
+ # Log outgoing response
383
+ response_json = json.dumps(result, indent=2)
384
+ logger.info(f"[MCP] Tool response for {name} ({duration_ms:.2f}ms):")
385
+ # Truncate large responses for console readability
386
+ if len(response_json) > 1000:
387
+ logger.debug(f"[MCP] {response_json[:1000]}... (truncated, {len(response_json)} chars total)")
388
+ else:
389
+ logger.debug(f"[MCP] {response_json}")
390
+
391
+ return [TextContent(type="text", text=response_json)]
392
+ except Exception as e:
393
+ duration_ms = (time.perf_counter() - start_time) * 1000
394
+ self.mcp_logger.log_tool_call(name, arguments, {}, duration_ms, error=str(e))
395
+ logger.error(f"[MCP] Error executing tool {name} ({duration_ms:.2f}ms): {str(e)}")
396
+ logger.exception("[MCP] Full traceback:")
397
+ error_response = json.dumps({"error": str(e)}, indent=2)
398
+ return [TextContent(type="text", text=error_response)]
399
+
400
+ async def _tool_lineage_diff(self, arguments: Dict[str, Any]) -> Dict[str, Any]:
401
+ """Get lineage diff between base and current"""
402
+ try:
403
+ # Extract filter arguments
404
+ select = arguments.get("select")
405
+ exclude = arguments.get("exclude")
406
+ packages = arguments.get("packages")
407
+ view_mode = arguments.get("view_mode", "changed_models")
408
+
409
+ # Get lineage diff from adapter (returns a Pydantic LineageDiff model)
410
+ lineage_diff = self.context.get_lineage_diff().model_dump(mode="json")
411
+
412
+ # Apply node selection filtering if arguments provided
413
+ selected_node_ids = self.context.adapter.select_nodes(
414
+ select=select,
415
+ exclude=exclude,
416
+ packages=packages,
417
+ view_mode=view_mode,
418
+ )
419
+ impacted_node_ids = self.context.adapter.select_nodes(
420
+ select="state:modified+",
421
+ )
422
+
423
+ # Get diff information for change_status
424
+ diff_info = lineage_diff.get("diff", {})
425
+
426
+ # Extract parent_map and simplified nodes from both base and current
427
+ parent_map = {}
428
+ nodes = {}
429
+
430
+ # Merge parent_map and nodes: base first, then current overrides
431
+ for env_key in ["base", "current"]:
432
+ if env_key not in lineage_diff:
433
+ continue
434
+
435
+ env_data = lineage_diff[env_key]
436
+
437
+ # Merge parent_map (filtering by selected nodes)
438
+ if "parent_map" in env_data:
439
+ for node_id, parents in env_data["parent_map"].items():
440
+ if node_id in selected_node_ids:
441
+ parent_map[node_id] = parents
442
+
443
+ # Merge nodes (filtering by selected nodes)
444
+ if "nodes" in env_data:
445
+ for node_id, node_info in env_data["nodes"].items():
446
+ if node_id in selected_node_ids:
447
+ nodes[node_id] = {
448
+ "name": node_info.get("name"),
449
+ "resource_type": node_info.get("resource_type"),
450
+ }
451
+
452
+ materialized = node_info.get("config", {}).get("materialized")
453
+ if materialized is not None:
454
+ nodes[node_id]["materialized"] = materialized
455
+
456
+ # Create id to idx mapping
457
+ id_to_idx = {node_id: idx for idx, node_id in enumerate(nodes.keys())}
458
+
459
+ # Prepare node data for DataFrame
460
+ nodes_data = [
461
+ (
462
+ id_to_idx[node_id],
463
+ node_id,
464
+ node_info.get("name"),
465
+ node_info.get("resource_type"),
466
+ node_info.get("materialized"),
467
+ diff_info.get(node_id, {}).get("change_status"),
468
+ node_id in impacted_node_ids,
469
+ )
470
+ for node_id, node_info in nodes.items()
471
+ ]
472
+
473
+ # Create nodes DataFrame using from_data with simple dict format
474
+ nodes_df = DataFrame.from_data(
475
+ columns={
476
+ "idx": "integer",
477
+ "id": "text",
478
+ "name": "text",
479
+ "resource_type": "text",
480
+ "materialized": "text",
481
+ "change_status": "text",
482
+ "impacted": "boolean",
483
+ },
484
+ data=nodes_data,
485
+ )
486
+
487
+ # Build edges from parent_map
488
+ edges_data = []
489
+ for node_id, parents in parent_map.items():
490
+ if node_id in id_to_idx:
491
+ for parent_id in parents:
492
+ if parent_id in id_to_idx:
493
+ edges_data.append((id_to_idx[parent_id], id_to_idx[node_id]))
494
+
495
+ # Create edges DataFrame
496
+ edges_df = DataFrame.from_data(
497
+ columns={
498
+ "from": "integer",
499
+ "to": "integer",
500
+ },
501
+ data=edges_data,
502
+ )
503
+
504
+ # Build simplified result
505
+ result = {"nodes": nodes_df.model_dump(mode="json"), "edges": edges_df.model_dump(mode="json")}
506
+
507
+ return result
508
+
509
+ except Exception:
510
+ logger.exception("Error getting lineage diff")
511
+ raise
512
+
513
+ async def _tool_schema_diff(self, arguments: Dict[str, Any]) -> Dict[str, Any]:
514
+ """Get schema diff (column changes) between base and current"""
515
+ try:
516
+ # Extract filter arguments
517
+ select = arguments.get("select")
518
+ exclude = arguments.get("exclude")
519
+ packages = arguments.get("packages")
520
+
521
+ # Get lineage diff from adapter
522
+ lineage_diff = self.context.get_lineage_diff().model_dump(mode="json")
523
+
524
+ # Get all nodes from current environment
525
+ current_nodes = {}
526
+ if "current" in lineage_diff and "nodes" in lineage_diff["current"]:
527
+ current_nodes = lineage_diff["current"]["nodes"]
528
+
529
+ # Filter to only nodes that exist in both base and current (exclude added nodes)
530
+ base_nodes = lineage_diff.get("base", {}).get("nodes", {})
531
+ nodes_to_compare = set(current_nodes.keys()) & set(base_nodes.keys())
532
+
533
+ # Apply filtering if arguments provided
534
+ if select or exclude or packages:
535
+ selected_node_ids = self.context.adapter.select_nodes(
536
+ select=select,
537
+ exclude=exclude,
538
+ packages=packages,
539
+ )
540
+ nodes_to_compare = nodes_to_compare & selected_node_ids
541
+
542
+ # Build schema changes
543
+ schema_changes = []
544
+
545
+ for node_id in nodes_to_compare:
546
+ base_node = base_nodes.get(node_id, {})
547
+ current_node = current_nodes.get(node_id, {})
548
+
549
+ base_columns = base_node.get("columns", {})
550
+ current_columns = current_node.get("columns", {})
551
+
552
+ # Get column names in base and current
553
+ base_col_names = set(base_columns.keys())
554
+ current_col_names = set(current_columns.keys())
555
+
556
+ # Find added columns (in current but not in base)
557
+ for col_name in current_col_names - base_col_names:
558
+ schema_changes.append((node_id, col_name, "added"))
559
+
560
+ # Find removed columns (in base but not in current)
561
+ for col_name in base_col_names - current_col_names:
562
+ schema_changes.append((node_id, col_name, "removed"))
563
+
564
+ # Find modified columns (in both but with different types)
565
+ for col_name in base_col_names & current_col_names:
566
+ base_col_type = base_columns[col_name].get("type")
567
+ current_col_type = current_columns[col_name].get("type")
568
+ if base_col_type != current_col_type:
569
+ schema_changes.append((node_id, col_name, "modified"))
570
+
571
+ # Check if there are more than 100 rows
572
+ limit = 100
573
+ has_more = len(schema_changes) > limit
574
+ limited_schema_changes = schema_changes[:limit]
575
+
576
+ # Convert schema changes to dataframe format using DataFrame.from_data()
577
+ diff_df = DataFrame.from_data(
578
+ columns={
579
+ "node_id": "text",
580
+ "column": "text",
581
+ "change_status": "text",
582
+ },
583
+ data=limited_schema_changes,
584
+ limit=limit,
585
+ more=has_more,
586
+ )
587
+ return diff_df.model_dump(mode="json")
588
+
589
+ except Exception:
590
+ logger.exception("Error getting schema diff")
591
+ raise
592
+
593
+ async def _tool_row_count_diff(self, arguments: Dict[str, Any]) -> Dict[str, Any]:
594
+ """Execute row count diff task"""
595
+ try:
596
+ task = RowCountDiffTask(params=arguments)
597
+
598
+ # Execute task synchronously (it's already sync)
599
+ result = await asyncio.get_event_loop().run_in_executor(None, task.execute)
600
+
601
+ return result
602
+ except Exception:
603
+ logger.exception("Error executing row count diff")
604
+ raise
605
+
606
+ async def _tool_query(self, arguments: Dict[str, Any]) -> Dict[str, Any]:
607
+ """Execute a query"""
608
+ try:
609
+ sql_template = arguments.get("sql_template")
610
+ is_base = arguments.get("base", False)
611
+
612
+ params = {"sql_template": sql_template}
613
+ task = QueryTask(params=params)
614
+ task.is_base = is_base
615
+
616
+ # Execute task
617
+ result = await asyncio.get_event_loop().run_in_executor(None, task.execute)
618
+
619
+ # Convert to dict if it's a model
620
+ if hasattr(result, "model_dump"):
621
+ return result.model_dump(mode="json")
622
+ return result
623
+ except Exception:
624
+ logger.exception("Error executing query")
625
+ raise
626
+
627
+ async def _tool_query_diff(self, arguments: Dict[str, Any]) -> Dict[str, Any]:
628
+ """Execute query diff task"""
629
+ try:
630
+ task = QueryDiffTask(params=arguments)
631
+
632
+ # Execute task
633
+ result = await asyncio.get_event_loop().run_in_executor(None, task.execute)
634
+
635
+ # Convert to dict if it's a model
636
+ if hasattr(result, "model_dump"):
637
+ return result.model_dump(mode="json")
638
+ return result
639
+ except Exception:
640
+ logger.exception("Error executing query diff")
641
+ raise
642
+
643
+ async def _tool_profile_diff(self, arguments: Dict[str, Any]) -> Dict[str, Any]:
644
+ """Execute profile diff task"""
645
+ try:
646
+ task = ProfileDiffTask(params=arguments)
647
+
648
+ # Execute task
649
+ result = await asyncio.get_event_loop().run_in_executor(None, task.execute)
650
+
651
+ # Convert to dict if it's a model
652
+ if hasattr(result, "model_dump"):
653
+ return result.model_dump(mode="json")
654
+ return result
655
+ except Exception:
656
+ logger.exception("Error executing profile diff")
657
+ raise
658
+
659
+ async def _tool_list_checks(self, arguments: Dict[str, Any]) -> Dict[str, Any]:
660
+ """List all checks in the current session"""
661
+ try:
662
+ from recce.models import CheckDAO
663
+
664
+ # Get all checks
665
+ checks_dao = CheckDAO()
666
+ checks = checks_dao.list()
667
+
668
+ # Build checks list with relevant metadata
669
+ checks_list = []
670
+ for check in checks:
671
+ checks_list.append(
672
+ {
673
+ "check_id": str(check.check_id),
674
+ "name": check.name,
675
+ "type": check.type.value,
676
+ "description": check.description or "",
677
+ "params": check.params or {},
678
+ "is_checked": check.is_checked,
679
+ "is_preset": check.is_preset,
680
+ }
681
+ )
682
+
683
+ # Get statistics
684
+ stats = checks_dao.status()
685
+
686
+ result = {
687
+ "checks": checks_list,
688
+ "total": stats["total"],
689
+ "approved": stats["approved"],
690
+ }
691
+
692
+ return result
693
+ except Exception:
694
+ logger.exception("Error listing checks")
695
+ raise
696
+
697
+ async def _tool_run_check(self, arguments: Dict[str, Any]) -> Dict[str, Any]:
698
+ """Run a single check by ID"""
699
+ try:
700
+ from recce.apis.run_func import submit_run
701
+ from recce.models import CheckDAO
702
+ from recce.models.types import RunType
703
+
704
+ check_id = arguments.get("check_id")
705
+ if not check_id:
706
+ raise ValueError("check_id is required")
707
+
708
+ # Get the check from database
709
+ check = CheckDAO().find_check_by_id(check_id)
710
+ if not check:
711
+ raise ValueError(f"Check with ID {check_id} not found")
712
+
713
+ # For lineage_diff and schema_diff, call the corresponding diff tools
714
+ if check.type == RunType.LINEAGE_DIFF:
715
+ return await self._tool_lineage_diff(check.params or {})
716
+
717
+ if check.type == RunType.SCHEMA_DIFF:
718
+ return await self._tool_schema_diff(check.params or {})
719
+
720
+ try:
721
+ # Submit and execute the run for other check types
722
+ run, future = submit_run(check.type, params=check.params or {}, check_id=check_id)
723
+ run.result = await future
724
+
725
+ # Return run object in JSON format (same as run_check_handler in check_api.py)
726
+ return run.model_dump(mode="json")
727
+ except RecceException as e:
728
+ raise ValueError(str(e))
729
+
730
+ except Exception:
731
+ logger.exception("Error running check")
732
+ raise
733
+
734
+ async def run(self):
735
+ """Run the MCP server in stdio mode"""
736
+ async with stdio_server() as (read_stream, write_stream):
737
+ await self.server.run(read_stream, write_stream, self.server.create_initialization_options())
738
+
739
+ async def run_sse(self, host: str = "localhost", port: int = 8000):
740
+ """Run the MCP server in HTTP mode using Server-Sent Events (SSE)
741
+
742
+ Args:
743
+ host: Host to bind to (default: localhost)
744
+ port: Port to bind to (default: 8000)
745
+ """
746
+ import uvicorn
747
+ from mcp.server.sse import SseServerTransport
748
+ from starlette.applications import Starlette
749
+ from starlette.requests import Request
750
+ from starlette.responses import Response
751
+ from starlette.routing import Mount, Route
752
+
753
+ # Create SSE transport - endpoint where clients POST messages
754
+ sse = SseServerTransport("/")
755
+
756
+ async def handle_sse_request(request: Request):
757
+ """Handle SSE connection (GET /sse) following official MCP example"""
758
+ client_info = f"{request.client.host}:{request.client.port}" if request.client else "unknown"
759
+ logger.info(f"[MCP HTTP] SSE connection established from {client_info}")
760
+ try:
761
+ async with sse.connect_sse(request.scope, request.receive, request._send) as streams:
762
+ await self.server.run(streams[0], streams[1], self.server.create_initialization_options())
763
+ finally:
764
+ logger.info(f"[MCP HTTP] SSE connection closed from {client_info}")
765
+ return Response() # Required to avoid NoneType error
766
+
767
+ async def handle_post_message(scope, receive, send):
768
+ """Handle POST messages (POST /) for MCP protocol"""
769
+ # Log POST message (session_id will be in query params)
770
+ query_string = scope.get("query_string", b"").decode("utf-8")
771
+ logger.debug(f"[MCP HTTP] POST message received with query: {query_string}")
772
+ await sse.handle_post_message(scope, receive, send)
773
+
774
+ async def handle_health_check(request: Request):
775
+ """Handle health check endpoint (GET /health)"""
776
+ return Response(content='{"status":"ok"}', media_type="application/json")
777
+
778
+ # Create Starlette app
779
+ app = Starlette(
780
+ debug=self.mcp_logger.debug,
781
+ routes=[
782
+ Route("/health", endpoint=handle_health_check, methods=["GET"]),
783
+ Route("/sse", endpoint=handle_sse_request, methods=["GET"]),
784
+ Mount("/", app=handle_post_message),
785
+ ],
786
+ )
787
+
788
+ # Run with uvicorn
789
+ logger.info(f"Starting Recce MCP Server in HTTP mode on {host}:{port}")
790
+ logger.info(f"Connection URL: http://{host}:{port}/sse")
791
+ config = uvicorn.Config(app, host=host, port=port, log_level="info")
792
+ server = uvicorn.Server(config)
793
+ await server.serve()
794
+
795
+
796
+ async def run_mcp_server(
797
+ sse: bool = False,
798
+ host: str = "localhost",
799
+ port: int = 8000,
800
+ **kwargs,
801
+ ):
802
+ """
803
+ Entry point for running the MCP server
804
+
805
+ Args:
806
+ sse: Whether to run in HTTP/SSE mode (default: False for stdio mode)
807
+ host: Host to bind to in SSE mode (default: localhost)
808
+ port: Port to bind to in SSE mode (default: 8000)
809
+ **kwargs: Arguments for loading RecceContext (dbt options, etc.)
810
+ Optionally includes 'state_loader' for persisting state on shutdown
811
+ Optionally includes 'mode' for server mode (server, preview, read-only)
812
+ Optionally includes 'debug' flag for enabling MCP logging
813
+ """
814
+ state_loader = kwargs.get("state_loader", None)
815
+ # Setup logging
816
+ logging.basicConfig(level=logging.INFO, format="%(asctime)s - %(name)s - %(levelname)s - %(message)s")
817
+
818
+ # Load Recce context
819
+ context = load_context(**kwargs)
820
+
821
+ # Extract mode from kwargs (defaults to server mode)
822
+ mode_str = kwargs.get("mode")
823
+ mode = None
824
+ if mode_str:
825
+ # Convert string mode to RecceServerMode enum
826
+ try:
827
+ mode = RecceServerMode(mode_str)
828
+ except ValueError:
829
+ logger.warning(f"Invalid mode '{mode_str}', using default server mode")
830
+
831
+ # Extract debug flag from kwargs
832
+ debug = kwargs.get("debug", False)
833
+
834
+ # Create MCP server
835
+ server = RecceMCPServer(context, mode=mode, debug=debug)
836
+
837
+ try:
838
+ # Run in either stdio or SSE mode
839
+ if sse:
840
+ await server.run_sse(host=host, port=port)
841
+ else:
842
+ await server.run()
843
+ finally:
844
+ # Export state on shutdown if state_loader is available
845
+ if state_loader:
846
+ try:
847
+ from rich.console import Console
848
+
849
+ console = Console()
850
+
851
+ # Export the state
852
+ msg = state_loader.export(context.export_state())
853
+ if msg is not None:
854
+ console.print(f"[yellow]On shutdown:[/yellow] {msg}")
855
+ else:
856
+ if hasattr(state_loader, "state_file") and state_loader.state_file:
857
+ console.print(f"[yellow]On shutdown:[/yellow] State exported to '{state_loader.state_file}'")
858
+ else:
859
+ console.print("[yellow]On shutdown:[/yellow] State exported successfully")
860
+ except Exception as e:
861
+ logger.exception(f"Failed to export state on shutdown: {e}")