@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,122 @@
1
+ import os
2
+ from unittest import TestCase
3
+ from unittest.mock import MagicMock
4
+
5
+ import pytest
6
+
7
+ from recce.adapter.dbt_adapter import DbtAdapter, load_catalog, load_manifest
8
+ from recce.core import RecceContext, set_default_context
9
+ from recce.run import schema_diff_should_be_approved
10
+
11
+
12
+ def test_validator():
13
+ from recce.tasks.schema import SchemaDiffCheckValidator
14
+
15
+ validator = SchemaDiffCheckValidator()
16
+
17
+ def validate(params: dict):
18
+ validator.validate(
19
+ {
20
+ "name": "test",
21
+ "type": "schema_diff",
22
+ "params": params,
23
+ }
24
+ )
25
+
26
+ # Select all models
27
+ validate({})
28
+
29
+ # Select by node name
30
+ validate(
31
+ {
32
+ "node_id": "abc",
33
+ }
34
+ )
35
+ validate(
36
+ {
37
+ "node_id": ["abc"],
38
+ }
39
+ )
40
+
41
+ # Select by selector
42
+ validate(
43
+ {
44
+ "select": "customers",
45
+ "exclude": "customers",
46
+ "packages": ["jaffle_shop"],
47
+ "view_mode": "all",
48
+ }
49
+ )
50
+
51
+ # packages should be an array
52
+ with pytest.raises(ValueError):
53
+ validate(
54
+ {
55
+ "packages": "jaffle_shop",
56
+ }
57
+ )
58
+
59
+ # view_mode should be 'all' or 'changed_models'
60
+ validate(
61
+ {
62
+ "view_mode": None,
63
+ }
64
+ )
65
+ validate(
66
+ {
67
+ "view_mode": "all",
68
+ }
69
+ )
70
+ with pytest.raises(ValueError):
71
+ validate(
72
+ {
73
+ "view_mode": "abc",
74
+ }
75
+ )
76
+
77
+
78
+ test_root_path = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
79
+
80
+
81
+ class TestSchemaDiffAutoApprove(TestCase):
82
+
83
+ def setUp(self):
84
+ self.default_context = MagicMock(spec=RecceContext)
85
+ manifest = load_manifest(path=os.path.join(test_root_path, "manifest.json"))
86
+ catalog = load_catalog(path=os.path.join(test_root_path, "catalog.json"))
87
+ dbt_adapter = DbtAdapter(curr_manifest=manifest, curr_catalog=catalog)
88
+ self.default_context.adapter = dbt_adapter
89
+
90
+ dbt_adapter.adapter = MagicMock()
91
+ dbt_adapter.adapter.type.return_value = None
92
+
93
+ dbt_adapter.select_nodes = MagicMock()
94
+ # Base and Current will be the same
95
+ self.default_context.get_lineage.return_value = dbt_adapter.get_lineage()
96
+ set_default_context(self.default_context)
97
+
98
+ def test_schema_diff_should_be_approved(self):
99
+ # Node_id is string
100
+ is_approved = schema_diff_should_be_approved(
101
+ {
102
+ "node_id": "model.jaffle_shop.customers",
103
+ }
104
+ )
105
+ assert is_approved is True
106
+
107
+ # Node_id is list
108
+ is_approved = schema_diff_should_be_approved(
109
+ {
110
+ "node_id": ["model.jaffle_shop.customers"],
111
+ }
112
+ )
113
+ assert is_approved is True
114
+
115
+ # Select all models
116
+ self.default_context.adapter.select_nodes.return_value = ["model.jaffle_shop.customers"]
117
+ is_approved = schema_diff_should_be_approved(
118
+ {
119
+ "select": "customers",
120
+ }
121
+ )
122
+ assert is_approved is True
@@ -0,0 +1,77 @@
1
+ import pytest
2
+
3
+ from recce.tasks import TopKDiffTask
4
+ from recce.tasks.top_k import TopKDiffCheckValidator
5
+
6
+
7
+ def test_top_k(dbt_test_helper):
8
+ csv_data_curr = """
9
+ customer_id,name,age
10
+ 1,Alice,30
11
+ 2,Bob,25
12
+ 3,Charlie,35
13
+ 4,Bob,35
14
+ """
15
+
16
+ csv_data_base = """
17
+ customer_id,name,age
18
+ 1,Alice,35
19
+ 2,Bob,25
20
+ 3,Charlie,35
21
+ 4,,35
22
+ """
23
+
24
+ dbt_test_helper.create_model("customers", csv_data_base, csv_data_curr)
25
+
26
+ params = dict(
27
+ model="customers",
28
+ column_name="name",
29
+ k=50,
30
+ )
31
+
32
+ task = TopKDiffTask(params)
33
+ run_result = task.execute()
34
+
35
+ # {
36
+ # 'values': ['Bob', 'Alice', 'Charlie'],
37
+ # 'counts': [2, 1, 1],
38
+ # 'valids': 4,
39
+ # 'total': 4
40
+ # }
41
+ assert run_result["current"]["values"][0] == "Bob"
42
+ assert run_result["current"]["counts"][0] == 2
43
+ assert run_result["current"]["valids"] == 4
44
+ assert run_result["current"]["total"] == 4
45
+
46
+ # {
47
+ # 'values': ['Bob', 'Alice', 'Charlie'],
48
+ # 'counts': [1, 1, 1],
49
+ # 'valids': 3,
50
+ # 'total': 4
51
+ # }
52
+ assert run_result["base"]["counts"][0] == 1
53
+ assert run_result["base"]["valids"] == 3
54
+ assert run_result["base"]["total"] == 4
55
+
56
+
57
+ def test_validator():
58
+ def validate(params: dict = {}, view_options: dict = {}):
59
+ TopKDiffCheckValidator().validate(
60
+ {
61
+ "name": "test",
62
+ "type": "top_k_diff",
63
+ "params": params,
64
+ "view_options": view_options,
65
+ }
66
+ )
67
+
68
+ validate(
69
+ {
70
+ "model": "customers",
71
+ "column_name": "name",
72
+ "k": 50,
73
+ }
74
+ )
75
+
76
+ with pytest.raises(ValueError):
77
+ validate({})
@@ -0,0 +1,439 @@
1
+ """Tests for task utility functions."""
2
+
3
+ import unittest
4
+
5
+ from recce.tasks.utils import normalize_keys_to_columns, strip_identifier_quotes
6
+
7
+
8
+ class TestStripIdentifierQuotes(unittest.TestCase):
9
+ """Tests for strip_identifier_quotes function."""
10
+
11
+ def test_double_quotes(self):
12
+ """Double quotes (PostgreSQL, Snowflake) are stripped."""
13
+ assert strip_identifier_quotes('"myColumn"') == "myColumn"
14
+ assert strip_identifier_quotes('"customerID"') == "customerID"
15
+
16
+ def test_backticks(self):
17
+ """Backticks (MySQL, BigQuery) are stripped."""
18
+ assert strip_identifier_quotes("`my_column`") == "my_column"
19
+ assert strip_identifier_quotes("`Column Name`") == "Column Name"
20
+
21
+ def test_square_brackets(self):
22
+ """Square brackets (SQL Server) are stripped."""
23
+ assert strip_identifier_quotes("[Column Name]") == "Column Name"
24
+ assert strip_identifier_quotes("[my_column]") == "my_column"
25
+
26
+ def test_unquoted_passthrough(self):
27
+ """Unquoted identifiers pass through unchanged."""
28
+ assert strip_identifier_quotes("regular_column") == "regular_column"
29
+ assert strip_identifier_quotes("myColumn") == "myColumn"
30
+ assert strip_identifier_quotes("PAYMENT_ID") == "PAYMENT_ID"
31
+
32
+ def test_empty_and_short_strings(self):
33
+ """Empty and short strings pass through unchanged."""
34
+ assert strip_identifier_quotes("") == ""
35
+ assert strip_identifier_quotes("a") == "a"
36
+ assert strip_identifier_quotes('"') == '"' # Single quote char, not a pair
37
+
38
+ def test_mismatched_quotes(self):
39
+ """Mismatched quotes are not stripped."""
40
+ assert strip_identifier_quotes('"column`') == '"column`'
41
+ assert strip_identifier_quotes('`column"') == '`column"'
42
+ assert strip_identifier_quotes('"column]') == '"column]'
43
+
44
+
45
+ class TestNormalizeKeysToColumns(unittest.TestCase):
46
+ """Tests for normalize_keys_to_columns function."""
47
+
48
+ # ========================================================================
49
+ # Warehouse-specific casing tests
50
+ # ========================================================================
51
+
52
+ def test_snowflake_uppercase(self):
53
+ """Snowflake returns UPPERCASE column names."""
54
+ keys = ["payment_id", "order_id"]
55
+ columns = ["PAYMENT_ID", "ORDER_ID", "AMOUNT"]
56
+
57
+ result = normalize_keys_to_columns(keys, columns)
58
+
59
+ assert result == ["PAYMENT_ID", "ORDER_ID"]
60
+
61
+ def test_postgres_lowercase(self):
62
+ """PostgreSQL/Redshift returns lowercase column names."""
63
+ keys = ["Payment_ID", "Order_ID"]
64
+ columns = ["payment_id", "order_id", "amount"]
65
+
66
+ result = normalize_keys_to_columns(keys, columns)
67
+
68
+ assert result == ["payment_id", "order_id"]
69
+
70
+ def test_bigquery_preserves_case(self):
71
+ """BigQuery preserves original case."""
72
+ keys = ["PaymentId", "OrderId"]
73
+ columns = ["PaymentId", "OrderId", "Amount"]
74
+
75
+ result = normalize_keys_to_columns(keys, columns)
76
+
77
+ assert result == ["PaymentId", "OrderId"]
78
+
79
+ def test_mixed_case_matching(self):
80
+ """Mixed case input matches correctly via case-insensitive fallback."""
81
+ keys = ["PAYMENT_id", "order_ID"]
82
+ columns = ["Payment_Id", "Order_Id", "Amount"]
83
+
84
+ result = normalize_keys_to_columns(keys, columns)
85
+
86
+ assert result == ["Payment_Id", "Order_Id"]
87
+
88
+ # ========================================================================
89
+ # Quoted column tests (exact match first)
90
+ # ========================================================================
91
+
92
+ def test_quoted_column_exact_match(self):
93
+ """Quoted columns with exact match are preserved."""
94
+ keys = ["preCommitID"]
95
+ columns = ["preCommitID", "order_id", "amount"]
96
+
97
+ result = normalize_keys_to_columns(keys, columns)
98
+
99
+ assert result == ["preCommitID"]
100
+
101
+ def test_quoted_column_exact_match_mixed_with_unquoted(self):
102
+ """Mix of quoted (exact match) and unquoted (case-insensitive) columns."""
103
+ keys = ["preCommitID", "ORDER_ID"]
104
+ columns = ["preCommitID", "order_id", "amount"]
105
+
106
+ result = normalize_keys_to_columns(keys, columns)
107
+
108
+ assert result == ["preCommitID", "order_id"]
109
+
110
+ def test_quoted_column_case_insensitive_fallback(self):
111
+ """
112
+ If quoted column doesn't match exactly, falls back to case-insensitive.
113
+ Note: This is a last-ditch fallback to capture potential mistakes the user
114
+ made when typing their primary key. It is a fallback measure, not meant to
115
+ be a fully robust solution.
116
+ """
117
+ keys = ["precommitid"] # User provides lowercase
118
+ columns = ["preCommitID", "order_id"] # But warehouse has mixed case
119
+
120
+ result = normalize_keys_to_columns(keys, columns)
121
+
122
+ assert result == ["preCommitID"] # Falls back to case-insensitive match
123
+
124
+ def test_multiple_quoted_columns(self):
125
+ """Multiple quoted columns with preserved casing."""
126
+ keys = ["preCommitID", "postCommitID", "userName"]
127
+ columns = ["preCommitID", "postCommitID", "userName", "created_at"]
128
+
129
+ result = normalize_keys_to_columns(keys, columns)
130
+
131
+ assert result == ["preCommitID", "postCommitID", "userName"]
132
+
133
+ # ========================================================================
134
+ # SQL identifier quote stripping tests
135
+ # ========================================================================
136
+
137
+ def test_double_quoted_key_stripped_and_matched(self):
138
+ """Double-quoted keys are stripped before matching (PostgreSQL, Snowflake)."""
139
+ keys = ['"customerID"']
140
+ columns = ["customerID", "amount", "created_at"]
141
+
142
+ result = normalize_keys_to_columns(keys, columns)
143
+
144
+ assert result == ["customerID"]
145
+
146
+ def test_backtick_quoted_key_stripped_and_matched(self):
147
+ """Backtick-quoted keys are stripped before matching (MySQL, BigQuery)."""
148
+ keys = ["`my_column`"]
149
+ columns = ["MY_COLUMN", "other_col"]
150
+
151
+ result = normalize_keys_to_columns(keys, columns)
152
+
153
+ assert result == ["MY_COLUMN"] # Case-insensitive match after stripping
154
+
155
+ def test_square_bracket_quoted_key_stripped_and_matched(self):
156
+ """Square bracket-quoted keys are stripped before matching (SQL Server)."""
157
+ keys = ["[Column Name]"]
158
+ columns = ["Column Name", "other_col"]
159
+
160
+ result = normalize_keys_to_columns(keys, columns)
161
+
162
+ assert result == ["Column Name"]
163
+
164
+ def test_quoted_key_empty_columns_returns_unquoted(self):
165
+ """Quoted keys with empty columns list returns unquoted keys."""
166
+ keys = ['"customerID"', "`another_col`"]
167
+
168
+ result = normalize_keys_to_columns(keys, [])
169
+
170
+ assert result == ["customerID", "another_col"]
171
+
172
+ def test_mixed_quoted_and_unquoted_keys(self):
173
+ """Mix of quoted and unquoted keys are handled correctly."""
174
+ keys = ['"customerID"', "order_id", "`created_at`"]
175
+ columns = ["customerID", "ORDER_ID", "CREATED_AT"]
176
+
177
+ result = normalize_keys_to_columns(keys, columns)
178
+
179
+ assert result == ["customerID", "ORDER_ID", "CREATED_AT"]
180
+
181
+ def test_quoted_key_case_insensitive_match(self):
182
+ """Quoted key after stripping matches case-insensitively."""
183
+ keys = ['"my_column"'] # Lowercase inside quotes
184
+ columns = ["MY_COLUMN", "other"] # Uppercase in warehouse (Snowflake)
185
+
186
+ result = normalize_keys_to_columns(keys, columns)
187
+
188
+ assert result == ["MY_COLUMN"]
189
+
190
+ # ========================================================================
191
+ # Edge cases and null handling
192
+ # ========================================================================
193
+
194
+ def test_none_keys_returns_none(self):
195
+ """None keys input returns None."""
196
+ result = normalize_keys_to_columns(None, ["col1", "col2"])
197
+
198
+ assert result is None
199
+
200
+ def test_empty_keys_returns_empty(self):
201
+ """Empty keys list returns empty list."""
202
+ result = normalize_keys_to_columns([], ["col1", "col2"])
203
+
204
+ assert result == []
205
+
206
+ def test_empty_columns_preserves_keys(self):
207
+ """Empty columns list preserves original keys."""
208
+ keys = ["payment_id", "order_id"]
209
+
210
+ result = normalize_keys_to_columns(keys, [])
211
+
212
+ assert result == ["payment_id", "order_id"]
213
+
214
+ def test_unmatched_key_preserved(self):
215
+ """Keys not found in columns are preserved as-is."""
216
+ keys = ["payment_id", "nonexistent_col"]
217
+ columns = ["PAYMENT_ID", "ORDER_ID"]
218
+
219
+ result = normalize_keys_to_columns(keys, columns)
220
+
221
+ assert result == ["PAYMENT_ID", "nonexistent_col"]
222
+
223
+ def test_single_key(self):
224
+ """Single key normalization works."""
225
+ result = normalize_keys_to_columns(["id"], ["ID", "NAME"])
226
+
227
+ assert result == ["ID"]
228
+
229
+ def test_special_columns_in_a_in_b(self):
230
+ """IN_A and IN_B columns normalize correctly."""
231
+ keys = ["in_a", "in_b"]
232
+ columns = ["ID", "IN_A", "IN_B"]
233
+
234
+ result = normalize_keys_to_columns(keys, columns)
235
+
236
+ assert result == ["IN_A", "IN_B"]
237
+
238
+ # ========================================================================
239
+ # Case collision edge cases
240
+ # ========================================================================
241
+
242
+ def test_exact_match_takes_priority_over_case_insensitive(self):
243
+ """When exact match exists, it takes priority even if case-insensitive would match differently."""
244
+ # This tests the scenario where there might be columns that differ only by case
245
+ keys = ["ID"]
246
+ columns = ["ID", "id"] # Both exist (rare but possible with quoting)
247
+
248
+ result = normalize_keys_to_columns(keys, columns)
249
+
250
+ # Should use exact match "ID", not fall back to case-insensitive
251
+ assert result == ["ID"]
252
+
253
+ def test_case_insensitive_when_no_exact_match(self):
254
+ """Case-insensitive matching when exact match doesn't exist."""
255
+ keys = ["Id"] # Mixed case, won't match exactly
256
+ columns = ["ID", "id"] # Both exist
257
+
258
+ result = normalize_keys_to_columns(keys, columns)
259
+
260
+ # Falls back to case-insensitive, last one in the map wins
261
+ # (This is acceptable - having same-name columns with different case is an edge case)
262
+ assert result[0] in ["ID", "id"]
263
+
264
+ # Add to tests/tasks/test_utils.py
265
+
266
+
267
+ class TestNormalizeBooleanFlagColumns(unittest.TestCase):
268
+ """Tests for normalize_boolean_flag_columns function."""
269
+
270
+ def test_normalizes_uppercase_in_a_in_b(self):
271
+ """Snowflake returns UPPERCASE - should normalize to lowercase."""
272
+ from recce.tasks.dataframe import (
273
+ DataFrame,
274
+ DataFrameColumn,
275
+ DataFrameColumnType,
276
+ )
277
+ from recce.tasks.utils import normalize_boolean_flag_columns
278
+
279
+ df = DataFrame(
280
+ columns=[
281
+ DataFrameColumn(key="ID", name="ID", type=DataFrameColumnType.INTEGER),
282
+ DataFrameColumn(key="IN_A", name="IN_A", type=DataFrameColumnType.BOOLEAN),
283
+ DataFrameColumn(key="IN_B", name="IN_B", type=DataFrameColumnType.BOOLEAN),
284
+ DataFrameColumn(key="VALUE", name="VALUE", type=DataFrameColumnType.NUMBER),
285
+ ],
286
+ data=[(1, True, True, 100), (2, True, False, 200)],
287
+ )
288
+
289
+ result = normalize_boolean_flag_columns(df)
290
+
291
+ column_keys = [col.key for col in result.columns]
292
+ self.assertEqual(column_keys, ["ID", "in_a", "in_b", "VALUE"])
293
+
294
+ column_names = [col.name for col in result.columns]
295
+ self.assertEqual(column_names, ["ID", "in_a", "in_b", "VALUE"])
296
+
297
+ def test_preserves_lowercase_in_a_in_b(self):
298
+ """PostgreSQL returns lowercase - should remain unchanged."""
299
+ from recce.tasks.dataframe import (
300
+ DataFrame,
301
+ DataFrameColumn,
302
+ DataFrameColumnType,
303
+ )
304
+ from recce.tasks.utils import normalize_boolean_flag_columns
305
+
306
+ df = DataFrame(
307
+ columns=[
308
+ DataFrameColumn(key="id", name="id", type=DataFrameColumnType.INTEGER),
309
+ DataFrameColumn(key="in_a", name="in_a", type=DataFrameColumnType.BOOLEAN),
310
+ DataFrameColumn(key="in_b", name="in_b", type=DataFrameColumnType.BOOLEAN),
311
+ DataFrameColumn(key="value", name="value", type=DataFrameColumnType.NUMBER),
312
+ ],
313
+ data=[(1, True, True, 100)],
314
+ )
315
+
316
+ result = normalize_boolean_flag_columns(df)
317
+
318
+ column_keys = [col.key for col in result.columns]
319
+ self.assertEqual(column_keys, ["id", "in_a", "in_b", "value"])
320
+
321
+ def test_handles_mixed_case_in_a_in_b(self):
322
+ """Mixed case like In_A should normalize to lowercase."""
323
+ from recce.tasks.dataframe import (
324
+ DataFrame,
325
+ DataFrameColumn,
326
+ DataFrameColumnType,
327
+ )
328
+ from recce.tasks.utils import normalize_boolean_flag_columns
329
+
330
+ df = DataFrame(
331
+ columns=[
332
+ DataFrameColumn(key="id", name="id", type=DataFrameColumnType.INTEGER),
333
+ DataFrameColumn(key="In_A", name="In_A", type=DataFrameColumnType.BOOLEAN),
334
+ DataFrameColumn(key="In_B", name="In_B", type=DataFrameColumnType.BOOLEAN),
335
+ ],
336
+ data=[(1, True, False)],
337
+ )
338
+
339
+ result = normalize_boolean_flag_columns(df)
340
+
341
+ column_keys = [col.key for col in result.columns]
342
+ self.assertEqual(column_keys, ["id", "in_a", "in_b"])
343
+
344
+ def test_preserves_other_columns(self):
345
+ """Non in_a/in_b columns should remain unchanged."""
346
+ from recce.tasks.dataframe import (
347
+ DataFrame,
348
+ DataFrameColumn,
349
+ DataFrameColumnType,
350
+ )
351
+ from recce.tasks.utils import normalize_boolean_flag_columns
352
+
353
+ df = DataFrame(
354
+ columns=[
355
+ DataFrameColumn(key="USER_ID", name="USER_ID", type=DataFrameColumnType.INTEGER),
356
+ DataFrameColumn(key="IN_A", name="IN_A", type=DataFrameColumnType.BOOLEAN),
357
+ DataFrameColumn(key="IN_B", name="IN_B", type=DataFrameColumnType.BOOLEAN),
358
+ DataFrameColumn(key="base__VALUE", name="base__VALUE", type=DataFrameColumnType.NUMBER),
359
+ DataFrameColumn(key="current__VALUE", name="current__VALUE", type=DataFrameColumnType.NUMBER),
360
+ ],
361
+ data=[(1, True, True, 100, 150)],
362
+ )
363
+
364
+ result = normalize_boolean_flag_columns(df)
365
+
366
+ column_keys = [col.key for col in result.columns]
367
+ # Only in_a/in_b should be lowercased
368
+ self.assertEqual(column_keys, ["USER_ID", "in_a", "in_b", "base__VALUE", "current__VALUE"])
369
+
370
+ def test_preserves_data(self):
371
+ """Data should remain unchanged."""
372
+ from recce.tasks.dataframe import (
373
+ DataFrame,
374
+ DataFrameColumn,
375
+ DataFrameColumnType,
376
+ )
377
+ from recce.tasks.utils import normalize_boolean_flag_columns
378
+
379
+ original_data = [(1, True, False, 100), (2, False, True, 200)]
380
+ df = DataFrame(
381
+ columns=[
382
+ DataFrameColumn(key="ID", name="ID", type=DataFrameColumnType.INTEGER),
383
+ DataFrameColumn(key="IN_A", name="IN_A", type=DataFrameColumnType.BOOLEAN),
384
+ DataFrameColumn(key="IN_B", name="IN_B", type=DataFrameColumnType.BOOLEAN),
385
+ DataFrameColumn(key="VALUE", name="VALUE", type=DataFrameColumnType.NUMBER),
386
+ ],
387
+ data=original_data,
388
+ )
389
+
390
+ result = normalize_boolean_flag_columns(df)
391
+
392
+ self.assertEqual(result.data, original_data)
393
+
394
+ def test_preserves_limit_and_more(self):
395
+ """limit and more fields should be preserved."""
396
+ from recce.tasks.dataframe import (
397
+ DataFrame,
398
+ DataFrameColumn,
399
+ DataFrameColumnType,
400
+ )
401
+ from recce.tasks.utils import normalize_boolean_flag_columns
402
+
403
+ df = DataFrame(
404
+ columns=[
405
+ DataFrameColumn(key="id", name="id", type=DataFrameColumnType.INTEGER),
406
+ DataFrameColumn(key="IN_A", name="IN_A", type=DataFrameColumnType.BOOLEAN),
407
+ DataFrameColumn(key="IN_B", name="IN_B", type=DataFrameColumnType.BOOLEAN),
408
+ ],
409
+ data=[(1, True, True)],
410
+ limit=100,
411
+ more=True,
412
+ )
413
+
414
+ result = normalize_boolean_flag_columns(df)
415
+
416
+ self.assertEqual(result.limit, 100)
417
+ self.assertEqual(result.more, True)
418
+
419
+ def test_handles_dataframe_without_in_columns(self):
420
+ """DataFrames without in_a/in_b should pass through unchanged."""
421
+ from recce.tasks.dataframe import (
422
+ DataFrame,
423
+ DataFrameColumn,
424
+ DataFrameColumnType,
425
+ )
426
+ from recce.tasks.utils import normalize_boolean_flag_columns
427
+
428
+ df = DataFrame(
429
+ columns=[
430
+ DataFrameColumn(key="id", name="id", type=DataFrameColumnType.INTEGER),
431
+ DataFrameColumn(key="name", name="name", type=DataFrameColumnType.TEXT),
432
+ ],
433
+ data=[(1, "Alice"), (2, "Bob")],
434
+ )
435
+
436
+ result = normalize_boolean_flag_columns(df)
437
+
438
+ column_keys = [col.key for col in result.columns]
439
+ self.assertEqual(column_keys, ["id", "name"])