@datarecce/ui 0.1.30 → 0.1.32
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.
- package/dist/{state-lPCQsWy5.js → RecceCheckContext-BOGmP80K.js} +4367 -5089
- package/dist/RecceCheckContext-BOGmP80K.js.map +1 -0
- package/dist/{state-B9yzhuKs.mjs → RecceCheckContext-CwUCo6AW.mjs} +4273 -4541
- package/dist/RecceCheckContext-CwUCo6AW.mjs.map +1 -0
- package/dist/{state-DOUPNifc.css → RecceCheckContext-DPnWB_aU.css} +57 -57
- package/dist/RecceCheckContext-DPnWB_aU.css.map +1 -0
- package/dist/{state-IA7HWYOs.css → RecceCheckContext-DyxOeUsX.css} +57 -57
- package/dist/RecceCheckContext-DyxOeUsX.css.map +1 -0
- package/dist/api.d.mts +1 -1
- package/dist/api.d.ts +1 -1
- package/dist/api.js +65 -26
- package/dist/api.js.map +1 -0
- package/dist/api.mjs +44 -5
- package/dist/api.mjs.map +1 -0
- package/dist/{components-DCOI1YlQ.mjs → components-DQUwJlNQ.mjs} +20 -21
- package/dist/{components-DCOI1YlQ.mjs.map → components-DQUwJlNQ.mjs.map} +1 -1
- package/dist/{components-B-YxuuPz.js → components-uVp0e4cH.js} +231 -241
- package/dist/{components-B-YxuuPz.js.map → components-uVp0e4cH.js.map} +1 -1
- package/dist/components.d.mts +1 -1
- package/dist/components.d.ts +1 -1
- package/dist/components.js +38 -38
- package/dist/components.mjs +5 -5
- package/dist/{urls-B1Ymdoz-.mjs → const-CaIm1Z8g.mjs} +2 -6
- package/dist/const-CaIm1Z8g.mjs.map +1 -0
- package/dist/{urls-C4eAc82S.js → const-CvdZO0FN.js} +1 -11
- package/dist/const-CvdZO0FN.js.map +1 -0
- package/dist/global-styles.css +84 -0
- package/dist/global-styles.css.map +1 -0
- package/dist/global-styles.js +1 -0
- package/dist/global-styles.mjs +1 -0
- package/dist/{hooks-DjBNmTdh.mjs → hooks-C6Kma5ld.mjs} +2 -3
- package/dist/{hooks-DjBNmTdh.mjs.map → hooks-C6Kma5ld.mjs.map} +1 -1
- package/dist/hooks-CfZ4UBBF.js +40 -0
- package/dist/{hooks-B9hsc1oD.js.map → hooks-CfZ4UBBF.js.map} +1 -1
- package/dist/hooks.d.mts +1 -1
- package/dist/hooks.d.ts +1 -1
- package/dist/hooks.js +26 -26
- package/dist/hooks.mjs +4 -4
- package/dist/{html2canvas-pro.esm-WJxOmKlq.js → html2canvas-pro.esm-CsuSOHXp.js} +1 -1
- package/dist/{html2canvas-pro.esm-WJxOmKlq.js.map → html2canvas-pro.esm-CsuSOHXp.js.map} +1 -1
- package/dist/{html2canvas-pro.esm-BInzOtWO.mjs → html2canvas-pro.esm-E7kpobrC.mjs} +1 -1
- package/dist/{html2canvas-pro.esm-BInzOtWO.mjs.map → html2canvas-pro.esm-E7kpobrC.mjs.map} +1 -1
- package/dist/{index-BNUP2V_N.d.ts → index-B9lSPJTi.d.ts} +184 -2
- package/dist/index-B9lSPJTi.d.ts.map +1 -0
- package/dist/{index-DOPZuhD8.d.mts → index-DsCU2oGL.d.mts} +259 -77
- package/dist/index-DsCU2oGL.d.mts.map +1 -0
- package/dist/index.d.mts +2 -2
- package/dist/index.d.ts +2 -2
- package/dist/index.js +92 -84
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +12 -10
- package/dist/index.mjs.map +1 -1
- package/dist/{mui-theme-B2wm_cvZ.js → mui-theme-iBHkjXJq.js} +2 -2
- package/dist/{mui-theme-B2wm_cvZ.js.map → mui-theme-iBHkjXJq.js.map} +1 -1
- package/dist/state-CTITyT0R.js +795 -0
- package/dist/state-CTITyT0R.js.map +1 -0
- package/dist/state-Sc2b4jri.mjs +382 -0
- package/dist/state-Sc2b4jri.mjs.map +1 -0
- package/dist/styles.css +5 -0
- package/dist/theme.d.mts +2 -185
- package/dist/theme.d.ts +2 -185
- package/dist/theme.js +1 -1
- package/dist/{tooltipMessage-DosF13kZ.js → tooltipMessage-BC5W7H3X.js} +1 -1
- package/dist/{tooltipMessage-DosF13kZ.js.map → tooltipMessage-BC5W7H3X.js.map} +1 -1
- package/dist/{tooltipMessage-B--I3p1V.mjs → tooltipMessage-B_xMIKWL.mjs} +1 -1
- package/dist/{tooltipMessage-B--I3p1V.mjs.map → tooltipMessage-B_xMIKWL.mjs.map} +1 -1
- package/dist/types.d.mts +1 -1
- package/dist/types.d.ts +1 -1
- package/dist/types.js +5 -4
- package/dist/types.mjs +3 -2
- package/dist/urls-BQW5wjg-.js +13 -0
- package/dist/urls-BQW5wjg-.js.map +1 -0
- package/dist/urls-DT7FVEcS.mjs +7 -0
- package/dist/urls-DT7FVEcS.mjs.map +1 -0
- package/dist/version-B9s8yne-.js +300 -0
- package/dist/version-B9s8yne-.js.map +1 -0
- package/dist/version-DP1kU_7v.mjs +162 -0
- package/dist/version-DP1kU_7v.mjs.map +1 -0
- package/package.json +5 -3
- package/recce-source/.editorconfig +26 -0
- package/recce-source/.flake8 +37 -0
- package/recce-source/.github/ISSUE_TEMPLATE/bug_report.yml +67 -0
- package/recce-source/.github/ISSUE_TEMPLATE/custom.md +10 -0
- package/recce-source/.github/ISSUE_TEMPLATE/feature_request.yml +42 -0
- package/recce-source/.github/PULL_REQUEST_TEMPLATE.md +21 -0
- package/recce-source/.github/copilot-instructions.md +331 -0
- package/recce-source/.github/instructions/backend-instructions.md +541 -0
- package/recce-source/.github/instructions/frontend-instructions.md +317 -0
- package/recce-source/.github/workflows/build-statics.yaml +72 -0
- package/recce-source/.github/workflows/bump.yaml +48 -0
- package/recce-source/.github/workflows/integration-tests-cloud.yaml +92 -0
- package/recce-source/.github/workflows/integration-tests-sqlmesh.yaml +33 -0
- package/recce-source/.github/workflows/integration-tests.yaml +52 -0
- package/recce-source/.github/workflows/nightly.yaml +246 -0
- package/recce-source/.github/workflows/release.yaml +196 -0
- package/recce-source/.github/workflows/tests-js.yaml +58 -0
- package/recce-source/.github/workflows/tests-python.yaml +128 -0
- package/recce-source/.pre-commit-config.yaml +26 -0
- package/recce-source/CLAUDE.md +483 -0
- package/recce-source/CODE_OF_CONDUCT.md +128 -0
- package/recce-source/CONTRIBUTING.md +107 -0
- package/recce-source/LICENSE +201 -0
- package/recce-source/Makefile +126 -0
- package/recce-source/README.md +182 -0
- package/recce-source/RECCE_CLOUD.md +81 -0
- package/recce-source/SECURITY.md +25 -0
- package/recce-source/docs/PACKAGING.md +340 -0
- package/recce-source/docs/README.md +1 -0
- package/recce-source/integration_tests/dbt/dbt_project.yml +26 -0
- package/recce-source/integration_tests/dbt/models/customers.sql +69 -0
- package/recce-source/integration_tests/dbt/models/docs.md +14 -0
- package/recce-source/integration_tests/dbt/models/orders.sql +56 -0
- package/recce-source/integration_tests/dbt/models/schema.yml +82 -0
- package/recce-source/integration_tests/dbt/models/staging/schema.yml +31 -0
- package/recce-source/integration_tests/dbt/models/staging/stg_customers.sql +22 -0
- package/recce-source/integration_tests/dbt/models/staging/stg_orders.sql +23 -0
- package/recce-source/integration_tests/dbt/models/staging/stg_payments.sql +25 -0
- package/recce-source/integration_tests/dbt/packages.yml +7 -0
- package/recce-source/integration_tests/dbt/profiles.yml +8 -0
- package/recce-source/integration_tests/dbt/seeds/raw_customers.csv +101 -0
- package/recce-source/integration_tests/dbt/seeds/raw_orders.csv +100 -0
- package/recce-source/integration_tests/dbt/seeds/raw_payments.csv +114 -0
- package/recce-source/integration_tests/dbt/seeds/raw_statuses.csv +5 -0
- package/recce-source/integration_tests/dbt/smoke_test.sh +72 -0
- package/recce-source/integration_tests/dbt/smoke_test_cloud.sh +71 -0
- package/recce-source/integration_tests/sqlmesh/__init__.py +0 -0
- package/recce-source/integration_tests/sqlmesh/audits/assert_item_price_above_zero.sql +9 -0
- package/recce-source/integration_tests/sqlmesh/audits/items.sql +7 -0
- package/recce-source/integration_tests/sqlmesh/audits/order_items.sql +7 -0
- package/recce-source/integration_tests/sqlmesh/config.py +171 -0
- package/recce-source/integration_tests/sqlmesh/helper.py +20 -0
- package/recce-source/integration_tests/sqlmesh/hooks/__init__.py +0 -0
- package/recce-source/integration_tests/sqlmesh/macros/__init__.py +0 -0
- package/recce-source/integration_tests/sqlmesh/macros/macros.py +8 -0
- package/recce-source/integration_tests/sqlmesh/macros/macros.sql +8 -0
- package/recce-source/integration_tests/sqlmesh/macros/utils.py +11 -0
- package/recce-source/integration_tests/sqlmesh/metrics/metrics.sql +25 -0
- package/recce-source/integration_tests/sqlmesh/models/customer_revenue_by_day.sql +41 -0
- package/recce-source/integration_tests/sqlmesh/models/customer_revenue_lifetime.sql +60 -0
- package/recce-source/integration_tests/sqlmesh/models/customers.sql +32 -0
- package/recce-source/integration_tests/sqlmesh/models/items.py +95 -0
- package/recce-source/integration_tests/sqlmesh/models/marketing.sql +15 -0
- package/recce-source/integration_tests/sqlmesh/models/order_items.py +95 -0
- package/recce-source/integration_tests/sqlmesh/models/orders.py +70 -0
- package/recce-source/integration_tests/sqlmesh/models/raw_marketing.py +62 -0
- package/recce-source/integration_tests/sqlmesh/models/top_waiters.sql +23 -0
- package/recce-source/integration_tests/sqlmesh/models/waiter_as_customer_by_day.sql +29 -0
- package/recce-source/integration_tests/sqlmesh/models/waiter_names.sql +10 -0
- package/recce-source/integration_tests/sqlmesh/models/waiter_revenue_by_day.sql +29 -0
- package/recce-source/integration_tests/sqlmesh/models/waiters.py +62 -0
- package/recce-source/integration_tests/sqlmesh/prep_env.sh +16 -0
- package/recce-source/integration_tests/sqlmesh/schema.yaml +5 -0
- package/recce-source/integration_tests/sqlmesh/seeds/waiter_names.csv +11 -0
- package/recce-source/integration_tests/sqlmesh/test_server.sh +29 -0
- package/recce-source/integration_tests/sqlmesh/tests/test_customer_revenue_by_day.yaml +63 -0
- package/recce-source/integration_tests/sqlmesh/tests/test_order_items.yaml +72 -0
- package/recce-source/js/.editorconfig +27 -0
- package/recce-source/js/.env.development +5 -0
- package/recce-source/js/.husky/pre-commit +29 -0
- package/recce-source/js/.nvmrc +1 -0
- package/recce-source/js/README.md +39 -0
- package/recce-source/js/app/(mainComponents)/DisplayModeToggle.tsx +65 -0
- package/recce-source/js/app/(mainComponents)/NavBar.tsx +228 -0
- package/recce-source/js/app/(mainComponents)/RecceVersionBadge.tsx +107 -0
- package/recce-source/js/app/(mainComponents)/TopBar.tsx +252 -0
- package/recce-source/js/app/@lineage/default.tsx +20 -0
- package/recce-source/js/app/@lineage/page.tsx +14 -0
- package/recce-source/js/app/MainLayout.tsx +170 -0
- package/recce-source/js/app/Providers.tsx +49 -0
- package/recce-source/js/app/checks/page.tsx +296 -0
- package/recce-source/js/app/error.tsx +93 -0
- package/recce-source/js/app/favicon.ico +0 -0
- package/recce-source/js/app/global-error.tsx +115 -0
- package/recce-source/js/app/global.css +82 -0
- package/recce-source/js/app/layout.tsx +48 -0
- package/recce-source/js/app/lineage/page.tsx +15 -0
- package/recce-source/js/app/page.tsx +12 -0
- package/recce-source/js/app/query/page.tsx +8 -0
- package/recce-source/js/biome.json +313 -0
- package/recce-source/js/jest.config.js +34 -0
- package/recce-source/js/jest.globals.d.ts +32 -0
- package/recce-source/js/jest.setup.js +91 -0
- package/recce-source/js/next.config.js +16 -0
- package/recce-source/js/package-lock.json +13843 -0
- package/recce-source/js/package.json +123 -0
- package/recce-source/js/pnpm-lock.yaml +9235 -0
- package/recce-source/js/pnpm-workspace.yaml +6 -0
- package/recce-source/js/postcss.config.js +5 -0
- package/recce-source/js/public/auth_callback.html +68 -0
- package/recce-source/js/public/imgs/feedback/thumbs-down.png +0 -0
- package/recce-source/js/public/imgs/feedback/thumbs-up.png +0 -0
- package/recce-source/js/public/imgs/reload-image.svg +4 -0
- package/recce-source/js/public/logo/recce-logo-white.png +0 -0
- package/recce-source/js/src/components/AuthModal/AuthModal.tsx +202 -0
- package/recce-source/js/src/components/app/AvatarDropdown.tsx +159 -0
- package/recce-source/js/src/components/app/EnvInfo.tsx +357 -0
- package/recce-source/js/src/components/app/Filename.tsx +388 -0
- package/recce-source/js/src/components/app/SetupConnectionPopover.tsx +91 -0
- package/recce-source/js/src/components/app/StateExporter.tsx +57 -0
- package/recce-source/js/src/components/app/StateImporter.tsx +198 -0
- package/recce-source/js/src/components/app/StateSharing.tsx +145 -0
- package/recce-source/js/src/components/app/StateSynchronizer.tsx +205 -0
- package/recce-source/js/src/components/charts/HistogramChart.tsx +291 -0
- package/recce-source/js/src/components/charts/SquareIcon.tsx +51 -0
- package/recce-source/js/src/components/charts/TopKSummaryList.tsx +457 -0
- package/recce-source/js/src/components/charts/chartTheme.ts +74 -0
- package/recce-source/js/src/components/check/CheckBreadcrumb.tsx +97 -0
- package/recce-source/js/src/components/check/CheckDescription.tsx +134 -0
- package/recce-source/js/src/components/check/CheckDetail.tsx +797 -0
- package/recce-source/js/src/components/check/CheckEmptyState.tsx +84 -0
- package/recce-source/js/src/components/check/CheckList.tsx +320 -0
- package/recce-source/js/src/components/check/LineageDiffView.tsx +32 -0
- package/recce-source/js/src/components/check/PresetCheckTemplateView.tsx +48 -0
- package/recce-source/js/src/components/check/SchemaDiffView.tsx +290 -0
- package/recce-source/js/src/components/check/check.ts +25 -0
- package/recce-source/js/src/components/check/timeline/CheckTimeline.tsx +163 -0
- package/recce-source/js/src/components/check/timeline/CommentInput.tsx +84 -0
- package/recce-source/js/src/components/check/timeline/TimelineEvent.tsx +468 -0
- package/recce-source/js/src/components/check/timeline/index.ts +12 -0
- package/recce-source/js/src/components/check/utils.ts +12 -0
- package/recce-source/js/src/components/data-grid/ScreenshotDataGrid.tsx +333 -0
- package/recce-source/js/src/components/data-grid/agGridStyles.css +55 -0
- package/recce-source/js/src/components/data-grid/agGridTheme.ts +43 -0
- package/recce-source/js/src/components/editor/CodeEditor.tsx +107 -0
- package/recce-source/js/src/components/editor/DiffEditor.tsx +162 -0
- package/recce-source/js/src/components/editor/index.ts +12 -0
- package/recce-source/js/src/components/errorboundary/ErrorBoundary.tsx +87 -0
- package/recce-source/js/src/components/histogram/HistogramDiffForm.tsx +147 -0
- package/recce-source/js/src/components/histogram/HistogramDiffResultView.tsx +63 -0
- package/recce-source/js/src/components/icons/index.tsx +142 -0
- package/recce-source/js/src/components/lineage/ActionControl.tsx +63 -0
- package/recce-source/js/src/components/lineage/ActionTag.tsx +141 -0
- package/recce-source/js/src/components/lineage/ChangeStatusLegend.tsx +46 -0
- package/recce-source/js/src/components/lineage/ColumnLevelLineageControl.tsx +327 -0
- package/recce-source/js/src/components/lineage/ColumnLevelLineageLegend.tsx +57 -0
- package/recce-source/js/src/components/lineage/GraphColumnNode.tsx +199 -0
- package/recce-source/js/src/components/lineage/GraphEdge.tsx +59 -0
- package/recce-source/js/src/components/lineage/GraphNode.tsx +555 -0
- package/recce-source/js/src/components/lineage/LineagePage.tsx +10 -0
- package/recce-source/js/src/components/lineage/LineageView.tsx +1384 -0
- package/recce-source/js/src/components/lineage/LineageViewContext.tsx +86 -0
- package/recce-source/js/src/components/lineage/LineageViewContextMenu.tsx +637 -0
- package/recce-source/js/src/components/lineage/LineageViewNotification.tsx +64 -0
- package/recce-source/js/src/components/lineage/LineageViewTopBar.tsx +596 -0
- package/recce-source/js/src/components/lineage/NodeSqlView.tsx +136 -0
- package/recce-source/js/src/components/lineage/NodeTag.tsx +278 -0
- package/recce-source/js/src/components/lineage/NodeView.tsx +642 -0
- package/recce-source/js/src/components/lineage/SandboxView.tsx +436 -0
- package/recce-source/js/src/components/lineage/ServerDisconnectedModalContent.tsx +105 -0
- package/recce-source/js/src/components/lineage/SetupConnectionBanner.tsx +52 -0
- package/recce-source/js/src/components/lineage/SingleEnvironmentQueryView.tsx +152 -0
- package/recce-source/js/src/components/lineage/graph.test.ts +31 -0
- package/recce-source/js/src/components/lineage/graph.ts +58 -0
- package/recce-source/js/src/components/lineage/lineage.test.ts +169 -0
- package/recce-source/js/src/components/lineage/lineage.ts +521 -0
- package/recce-source/js/src/components/lineage/styles.css +42 -0
- package/recce-source/js/src/components/lineage/styles.tsx +165 -0
- package/recce-source/js/src/components/lineage/useMultiNodesAction.ts +352 -0
- package/recce-source/js/src/components/lineage/useValueDiffAlertDialog.tsx +108 -0
- package/recce-source/js/src/components/onboarding-guide/Notification.tsx +62 -0
- package/recce-source/js/src/components/profile/ProfileDiffForm.tsx +134 -0
- package/recce-source/js/src/components/profile/ProfileDiffResultView.tsx +245 -0
- package/recce-source/js/src/components/query/ChangedOnlyCheckbox.tsx +29 -0
- package/recce-source/js/src/components/query/DiffText.tsx +120 -0
- package/recce-source/js/src/components/query/QueryDiffResultView.tsx +470 -0
- package/recce-source/js/src/components/query/QueryForm.tsx +80 -0
- package/recce-source/js/src/components/query/QueryPage.tsx +282 -0
- package/recce-source/js/src/components/query/QueryResultView.tsx +180 -0
- package/recce-source/js/src/components/query/SetupConnectionGuide.tsx +57 -0
- package/recce-source/js/src/components/query/SqlEditor.tsx +245 -0
- package/recce-source/js/src/components/query/ToggleSwitch.tsx +84 -0
- package/recce-source/js/src/components/query/styles.css +21 -0
- package/recce-source/js/src/components/routing/DirectUrlAccess.test.tsx +428 -0
- package/recce-source/js/src/components/routing/LineageStatePreservation.test.tsx +311 -0
- package/recce-source/js/src/components/routing/Navigation.test.tsx +256 -0
- package/recce-source/js/src/components/rowcount/RowCountDiffResultView.tsx +109 -0
- package/recce-source/js/src/components/rowcount/delta.ts +11 -0
- package/recce-source/js/src/components/run/RunList.tsx +303 -0
- package/recce-source/js/src/components/run/RunModal.tsx +191 -0
- package/recce-source/js/src/components/run/RunPage.tsx +26 -0
- package/recce-source/js/src/components/run/RunResultPane.tsx +454 -0
- package/recce-source/js/src/components/run/RunStatusAndDate.tsx +106 -0
- package/recce-source/js/src/components/run/RunToolbar.tsx +70 -0
- package/recce-source/js/src/components/run/RunView.tsx +196 -0
- package/recce-source/js/src/components/run/registry.ts +214 -0
- package/recce-source/js/src/components/run/types.ts +14 -0
- package/recce-source/js/src/components/schema/ColumnNameCell.test.tsx +169 -0
- package/recce-source/js/src/components/schema/ColumnNameCell.tsx +198 -0
- package/recce-source/js/src/components/schema/SchemaView.tsx +337 -0
- package/recce-source/js/src/components/schema/schemaDiff.ts +32 -0
- package/recce-source/js/src/components/schema/style.css +134 -0
- package/recce-source/js/src/components/screenshot/ScreenshotBox.tsx +39 -0
- package/recce-source/js/src/components/shared/HistoryToggle.tsx +35 -0
- package/recce-source/js/src/components/split/Split.tsx +40 -0
- package/recce-source/js/src/components/split/styles.css +24 -0
- package/recce-source/js/src/components/summary/ChangeSummary.tsx +264 -0
- package/recce-source/js/src/components/summary/SchemaSummary.tsx +123 -0
- package/recce-source/js/src/components/summary/SummaryView.tsx +29 -0
- package/recce-source/js/src/components/timeout/IdleTimeoutBadge.tsx +48 -0
- package/recce-source/js/src/components/top-k/TopKDiffForm.tsx +58 -0
- package/recce-source/js/src/components/top-k/TopKDiffResultView.tsx +73 -0
- package/recce-source/js/src/components/ui/dataGrid/DataFrameColumnGroupHeader.tsx +228 -0
- package/recce-source/js/src/components/ui/dataGrid/DataFrameColumnHeader.tsx +113 -0
- package/recce-source/js/src/components/ui/dataGrid/defaultRenderCell.tsx +72 -0
- package/recce-source/js/src/components/ui/dataGrid/index.ts +23 -0
- package/recce-source/js/src/components/ui/dataGrid/inlineRenderCell.test.tsx +607 -0
- package/recce-source/js/src/components/ui/dataGrid/inlineRenderCell.tsx +211 -0
- package/recce-source/js/src/components/ui/dataGrid/schemaCells.test.tsx +452 -0
- package/recce-source/js/src/components/ui/dataGrid/schemaCells.tsx +142 -0
- package/recce-source/js/src/components/ui/dataGrid/valueDiffCells.test.tsx +178 -0
- package/recce-source/js/src/components/ui/dataGrid/valueDiffCells.tsx +275 -0
- package/recce-source/js/src/components/ui/markdown/ExternalLinkConfirmDialog.tsx +134 -0
- package/recce-source/js/src/components/ui/markdown/MarkdownContent.tsx +364 -0
- package/recce-source/js/src/components/ui/mui/index.ts +13 -0
- package/recce-source/js/src/components/ui/mui-provider.tsx +67 -0
- package/recce-source/js/src/components/ui/mui-theme.ts +1039 -0
- package/recce-source/js/src/components/ui/mui-utils.ts +113 -0
- package/recce-source/js/src/components/ui/toaster.tsx +288 -0
- package/recce-source/js/src/components/valuediff/ValueDiffDetailResultView.tsx +217 -0
- package/recce-source/js/src/components/valuediff/ValueDiffForm.tsx +246 -0
- package/recce-source/js/src/components/valuediff/ValueDiffResultView.tsx +82 -0
- package/recce-source/js/src/components/valuediff/shared.ts +33 -0
- package/recce-source/js/src/constants/tooltipMessage.ts +3 -0
- package/recce-source/js/src/constants/urls.ts +1 -0
- package/recce-source/js/src/lib/UrlHash.ts +12 -0
- package/recce-source/js/src/lib/api/adhocQuery.ts +70 -0
- package/recce-source/js/src/lib/api/axiosClient.ts +9 -0
- package/recce-source/js/src/lib/api/cacheKeys.ts +13 -0
- package/recce-source/js/src/lib/api/checkEvents.ts +252 -0
- package/recce-source/js/src/lib/api/checks.ts +129 -0
- package/recce-source/js/src/lib/api/cll.ts +53 -0
- package/recce-source/js/src/lib/api/connectToCloud.ts +13 -0
- package/recce-source/js/src/lib/api/flag.ts +37 -0
- package/recce-source/js/src/lib/api/info.ts +198 -0
- package/recce-source/js/src/lib/api/instanceInfo.ts +25 -0
- package/recce-source/js/src/lib/api/keepAlive.ts +108 -0
- package/recce-source/js/src/lib/api/lineagecheck.ts +35 -0
- package/recce-source/js/src/lib/api/localStorageKeys.ts +7 -0
- package/recce-source/js/src/lib/api/models.ts +59 -0
- package/recce-source/js/src/lib/api/profile.ts +65 -0
- package/recce-source/js/src/lib/api/rowcount.ts +19 -0
- package/recce-source/js/src/lib/api/runs.ts +174 -0
- package/recce-source/js/src/lib/api/schemacheck.ts +31 -0
- package/recce-source/js/src/lib/api/select.ts +25 -0
- package/recce-source/js/src/lib/api/sessionStorageKeys.ts +8 -0
- package/recce-source/js/src/lib/api/state.ts +117 -0
- package/recce-source/js/src/lib/api/track.ts +281 -0
- package/recce-source/js/src/lib/api/types.ts +284 -0
- package/recce-source/js/src/lib/api/user.ts +42 -0
- package/recce-source/js/src/lib/api/valuediff.ts +46 -0
- package/recce-source/js/src/lib/api/version.ts +40 -0
- package/recce-source/js/src/lib/const.ts +9 -0
- package/recce-source/js/src/lib/dataGrid/crossFunctionConsistency.test.ts +626 -0
- package/recce-source/js/src/lib/dataGrid/dataGridFactory.test.ts +2140 -0
- package/recce-source/js/src/lib/dataGrid/dataGridFactory.ts +397 -0
- package/recce-source/js/src/lib/dataGrid/generators/rowCountUtils.test.ts +132 -0
- package/recce-source/js/src/lib/dataGrid/generators/rowCountUtils.ts +126 -0
- package/recce-source/js/src/lib/dataGrid/generators/toDataDiffGrid.test.ts +1627 -0
- package/recce-source/js/src/lib/dataGrid/generators/toDataDiffGrid.ts +140 -0
- package/recce-source/js/src/lib/dataGrid/generators/toDataGrid.ts +67 -0
- package/recce-source/js/src/lib/dataGrid/generators/toRowCountDataGrid.test.ts +142 -0
- package/recce-source/js/src/lib/dataGrid/generators/toRowCountDataGrid.ts +71 -0
- package/recce-source/js/src/lib/dataGrid/generators/toRowCountDiffDataGrid.test.ts +258 -0
- package/recce-source/js/src/lib/dataGrid/generators/toRowCountDiffDataGrid.ts +153 -0
- package/recce-source/js/src/lib/dataGrid/generators/toSchemaDataGrid.test.ts +951 -0
- package/recce-source/js/src/lib/dataGrid/generators/toSchemaDataGrid.ts +221 -0
- package/recce-source/js/src/lib/dataGrid/generators/toValueDataGrid.test.ts +395 -0
- package/recce-source/js/src/lib/dataGrid/generators/toValueDataGrid.ts +184 -0
- package/recce-source/js/src/lib/dataGrid/generators/toValueDiffGrid.test.ts +884 -0
- package/recce-source/js/src/lib/dataGrid/generators/toValueDiffGrid.ts +113 -0
- package/recce-source/js/src/lib/dataGrid/index.ts +51 -0
- package/recce-source/js/src/lib/dataGrid/propertyBased.test.ts +858 -0
- package/recce-source/js/src/lib/dataGrid/shared/columnBuilders.test.ts +482 -0
- package/recce-source/js/src/lib/dataGrid/shared/columnBuilders.ts +345 -0
- package/recce-source/js/src/lib/dataGrid/shared/dataTypeEdgeCases.test.ts +698 -0
- package/recce-source/js/src/lib/dataGrid/shared/diffColumnBuilder.test.tsx +820 -0
- package/recce-source/js/src/lib/dataGrid/shared/diffColumnBuilder.tsx +277 -0
- package/recce-source/js/src/lib/dataGrid/shared/gridUtils.test.ts +785 -0
- package/recce-source/js/src/lib/dataGrid/shared/gridUtils.ts +370 -0
- package/recce-source/js/src/lib/dataGrid/shared/index.ts +81 -0
- package/recce-source/js/src/lib/dataGrid/shared/rowBuilders.test.ts +909 -0
- package/recce-source/js/src/lib/dataGrid/shared/rowBuilders.ts +325 -0
- package/recce-source/js/src/lib/dataGrid/shared/simpleColumnBuilder.tsx +240 -0
- package/recce-source/js/src/lib/dataGrid/shared/toDiffColumn.test.tsx +719 -0
- package/recce-source/js/src/lib/dataGrid/shared/toDiffColumn.tsx +231 -0
- package/recce-source/js/src/lib/dataGrid/shared/validation.test.ts +559 -0
- package/recce-source/js/src/lib/dataGrid/shared/validation.ts +367 -0
- package/recce-source/js/src/lib/dataGrid/warehouseNamingConventions.test.ts +1117 -0
- package/recce-source/js/src/lib/formatSelect.ts +50 -0
- package/recce-source/js/src/lib/hooks/ApiConfigContext.tsx +181 -0
- package/recce-source/js/src/lib/hooks/IdleTimeoutContext.tsx +177 -0
- package/recce-source/js/src/lib/hooks/LineageGraphContext.tsx +512 -0
- package/recce-source/js/src/lib/hooks/RecceActionContext.tsx +269 -0
- package/recce-source/js/src/lib/hooks/RecceCheckContext.tsx +33 -0
- package/recce-source/js/src/lib/hooks/RecceContextProvider.tsx +54 -0
- package/recce-source/js/src/lib/hooks/RecceInstanceContext.tsx +129 -0
- package/recce-source/js/src/lib/hooks/RecceQueryContext.tsx +98 -0
- package/recce-source/js/src/lib/hooks/RecceShareStateContext.tsx +59 -0
- package/recce-source/js/src/lib/hooks/ScreenShot.tsx +399 -0
- package/recce-source/js/src/lib/hooks/useAppRouter.test.ts +211 -0
- package/recce-source/js/src/lib/hooks/useAppRouter.ts +200 -0
- package/recce-source/js/src/lib/hooks/useCheckEvents.ts +99 -0
- package/recce-source/js/src/lib/hooks/useCheckToast.tsx +14 -0
- package/recce-source/js/src/lib/hooks/useClipBoardToast.tsx +27 -0
- package/recce-source/js/src/lib/hooks/useCountdownToast.tsx +102 -0
- package/recce-source/js/src/lib/hooks/useFeedbackCollectionToast.tsx +130 -0
- package/recce-source/js/src/lib/hooks/useGuideToast.tsx +45 -0
- package/recce-source/js/src/lib/hooks/useIdleDetection.tsx +185 -0
- package/recce-source/js/src/lib/hooks/useModelColumns.tsx +113 -0
- package/recce-source/js/src/lib/hooks/useRecceInstanceInfo.tsx +13 -0
- package/recce-source/js/src/lib/hooks/useRecceServerFlag.tsx +13 -0
- package/recce-source/js/src/lib/hooks/useRun.tsx +89 -0
- package/recce-source/js/src/lib/hooks/useThemeColors.ts +115 -0
- package/recce-source/js/src/lib/mergeKeys.test.ts +89 -0
- package/recce-source/js/src/lib/mergeKeys.ts +86 -0
- package/recce-source/js/src/lib/result/ResultErrorFallback.tsx +9 -0
- package/recce-source/js/src/lib/utils/formatTime.ts +84 -0
- package/recce-source/js/src/lib/utils/urls.ts +16 -0
- package/recce-source/js/src/utils/DropdownValuesInput.tsx +297 -0
- package/recce-source/js/src/utils/formatters.tsx +237 -0
- package/recce-source/js/src/utils/transforms.ts +81 -0
- package/recce-source/js/tsconfig.json +47 -0
- package/recce-source/macros/README.md +8 -0
- package/recce-source/macros/recce_athena.sql +73 -0
- package/recce-source/pyproject.toml +109 -0
- package/recce-source/recce/VERSION +1 -0
- package/recce-source/recce/__init__.py +84 -0
- package/recce-source/recce/adapter/__init__.py +0 -0
- package/recce-source/recce/adapter/base.py +109 -0
- package/recce-source/recce/adapter/dbt_adapter/__init__.py +1699 -0
- package/recce-source/recce/adapter/dbt_adapter/dbt_version.py +42 -0
- package/recce-source/recce/adapter/sqlmesh_adapter.py +141 -0
- package/recce-source/recce/apis/__init__.py +0 -0
- package/recce-source/recce/apis/check_api.py +203 -0
- package/recce-source/recce/apis/check_events_api.py +353 -0
- package/recce-source/recce/apis/check_func.py +130 -0
- package/recce-source/recce/apis/run_api.py +130 -0
- package/recce-source/recce/apis/run_func.py +258 -0
- package/recce-source/recce/artifact.py +266 -0
- package/recce-source/recce/cli.py +1846 -0
- package/recce-source/recce/config.py +127 -0
- package/recce-source/recce/connect_to_cloud.py +138 -0
- package/recce-source/recce/core.py +334 -0
- package/recce-source/recce/diff.py +26 -0
- package/recce-source/recce/event/CONFIG +1 -0
- package/recce-source/recce/event/SENTRY_DNS +1 -0
- package/recce-source/recce/event/__init__.py +304 -0
- package/recce-source/recce/event/collector.py +184 -0
- package/recce-source/recce/event/track.py +158 -0
- package/recce-source/recce/exceptions.py +21 -0
- package/recce-source/recce/git.py +77 -0
- package/recce-source/recce/github.py +222 -0
- package/recce-source/recce/mcp_server.py +861 -0
- package/recce-source/recce/models/__init__.py +6 -0
- package/recce-source/recce/models/check.py +473 -0
- package/recce-source/recce/models/run.py +46 -0
- package/recce-source/recce/models/types.py +218 -0
- package/recce-source/recce/pull_request.py +124 -0
- package/recce-source/recce/run.py +390 -0
- package/recce-source/recce/server.py +877 -0
- package/recce-source/recce/state/__init__.py +31 -0
- package/recce-source/recce/state/cloud.py +644 -0
- package/recce-source/recce/state/const.py +26 -0
- package/recce-source/recce/state/local.py +56 -0
- package/recce-source/recce/state/state.py +119 -0
- package/recce-source/recce/state/state_loader.py +174 -0
- package/recce-source/recce/summary.py +575 -0
- package/recce-source/recce/tasks/__init__.py +23 -0
- package/recce-source/recce/tasks/core.py +134 -0
- package/recce-source/recce/tasks/dataframe.py +170 -0
- package/recce-source/recce/tasks/histogram.py +433 -0
- package/recce-source/recce/tasks/lineage.py +19 -0
- package/recce-source/recce/tasks/profile.py +298 -0
- package/recce-source/recce/tasks/query.py +450 -0
- package/recce-source/recce/tasks/rowcount.py +277 -0
- package/recce-source/recce/tasks/schema.py +65 -0
- package/recce-source/recce/tasks/top_k.py +172 -0
- package/recce-source/recce/tasks/utils.py +147 -0
- package/recce-source/recce/tasks/valuediff.py +497 -0
- package/recce-source/recce/util/__init__.py +4 -0
- package/recce-source/recce/util/api_token.py +80 -0
- package/recce-source/recce/util/breaking.py +330 -0
- package/recce-source/recce/util/cache.py +25 -0
- package/recce-source/recce/util/cll.py +355 -0
- package/recce-source/recce/util/cloud/__init__.py +15 -0
- package/recce-source/recce/util/cloud/base.py +115 -0
- package/recce-source/recce/util/cloud/check_events.py +190 -0
- package/recce-source/recce/util/cloud/checks.py +242 -0
- package/recce-source/recce/util/io.py +120 -0
- package/recce-source/recce/util/lineage.py +83 -0
- package/recce-source/recce/util/logger.py +25 -0
- package/recce-source/recce/util/onboarding_state.py +45 -0
- package/recce-source/recce/util/perf_tracking.py +85 -0
- package/recce-source/recce/util/pydantic_model.py +22 -0
- package/recce-source/recce/util/recce_cloud.py +454 -0
- package/recce-source/recce/util/singleton.py +18 -0
- package/recce-source/recce/util/startup_perf.py +121 -0
- package/recce-source/recce/yaml/__init__.py +58 -0
- package/recce-source/recce_cloud/README.md +780 -0
- package/recce-source/recce_cloud/VERSION +1 -0
- package/recce-source/recce_cloud/__init__.py +24 -0
- package/recce-source/recce_cloud/api/__init__.py +17 -0
- package/recce-source/recce_cloud/api/base.py +132 -0
- package/recce-source/recce_cloud/api/client.py +186 -0
- package/recce-source/recce_cloud/api/exceptions.py +26 -0
- package/recce-source/recce_cloud/api/factory.py +63 -0
- package/recce-source/recce_cloud/api/github.py +106 -0
- package/recce-source/recce_cloud/api/gitlab.py +111 -0
- package/recce-source/recce_cloud/artifact.py +57 -0
- package/recce-source/recce_cloud/ci_providers/__init__.py +9 -0
- package/recce-source/recce_cloud/ci_providers/base.py +82 -0
- package/recce-source/recce_cloud/ci_providers/detector.py +147 -0
- package/recce-source/recce_cloud/ci_providers/github_actions.py +136 -0
- package/recce-source/recce_cloud/ci_providers/gitlab_ci.py +130 -0
- package/recce-source/recce_cloud/cli.py +434 -0
- package/recce-source/recce_cloud/download.py +230 -0
- package/recce-source/recce_cloud/hatch_build.py +20 -0
- package/recce-source/recce_cloud/pyproject.toml +49 -0
- package/recce-source/recce_cloud/upload.py +214 -0
- package/recce-source/test.py +0 -0
- package/recce-source/tests/__init__.py +0 -0
- package/recce-source/tests/adapter/__init__.py +0 -0
- package/recce-source/tests/adapter/dbt_adapter/__init__.py +0 -0
- package/recce-source/tests/adapter/dbt_adapter/conftest.py +17 -0
- package/recce-source/tests/adapter/dbt_adapter/dbt_test_helper.py +298 -0
- package/recce-source/tests/adapter/dbt_adapter/test_dbt_adapter.py +25 -0
- package/recce-source/tests/adapter/dbt_adapter/test_dbt_cll.py +717 -0
- package/recce-source/tests/adapter/dbt_adapter/test_proj/dbt_project.yml +4 -0
- package/recce-source/tests/adapter/dbt_adapter/test_proj/manifest.json +1 -0
- package/recce-source/tests/adapter/dbt_adapter/test_proj/package-lock.yml +8 -0
- package/recce-source/tests/adapter/dbt_adapter/test_proj/packages.yml +7 -0
- package/recce-source/tests/adapter/dbt_adapter/test_proj/profiles.yml +6 -0
- package/recce-source/tests/adapter/dbt_adapter/test_selector.py +205 -0
- package/recce-source/tests/apis/__init__.py +0 -0
- package/recce-source/tests/apis/row_count_diff.json +59 -0
- package/recce-source/tests/apis/test_check_events_api.py +615 -0
- package/recce-source/tests/apis/test_run_func.py +433 -0
- package/recce-source/tests/catalog.json +527 -0
- package/recce-source/tests/data/manifest/base/catalog.json +1 -0
- package/recce-source/tests/data/manifest/base/manifest.json +1 -0
- package/recce-source/tests/data/manifest/pr2/catalog.json +1 -0
- package/recce-source/tests/data/manifest/pr2/manifest.json +1 -0
- package/recce-source/tests/manifest.json +10655 -0
- package/recce-source/tests/models/__init__.py +0 -0
- package/recce-source/tests/models/test_check.py +731 -0
- package/recce-source/tests/models/test_run_models.py +295 -0
- package/recce-source/tests/recce_cloud/__init__.py +0 -0
- package/recce-source/tests/recce_cloud/test_ci_providers.py +351 -0
- package/recce-source/tests/recce_cloud/test_cli.py +735 -0
- package/recce-source/tests/recce_cloud/test_client.py +379 -0
- package/recce-source/tests/recce_cloud/test_platform_clients.py +483 -0
- package/recce-source/tests/recce_state.json +1 -0
- package/recce-source/tests/state/test_cloud.py +719 -0
- package/recce-source/tests/state/test_local.py +164 -0
- package/recce-source/tests/state/test_state_loader.py +211 -0
- package/recce-source/tests/tasks/__init__.py +0 -0
- package/recce-source/tests/tasks/conftest.py +4 -0
- package/recce-source/tests/tasks/test_histogram.py +129 -0
- package/recce-source/tests/tasks/test_lineage.py +55 -0
- package/recce-source/tests/tasks/test_preset_checks.py +64 -0
- package/recce-source/tests/tasks/test_profile.py +397 -0
- package/recce-source/tests/tasks/test_query.py +528 -0
- package/recce-source/tests/tasks/test_row_count.py +133 -0
- package/recce-source/tests/tasks/test_schema.py +122 -0
- package/recce-source/tests/tasks/test_top_k.py +77 -0
- package/recce-source/tests/tasks/test_utils.py +439 -0
- package/recce-source/tests/tasks/test_valuediff.py +361 -0
- package/recce-source/tests/test_cli.py +236 -0
- package/recce-source/tests/test_cli_mcp_optional.py +45 -0
- package/recce-source/tests/test_cloud_listing_cli.py +324 -0
- package/recce-source/tests/test_config.py +43 -0
- package/recce-source/tests/test_connect_to_cloud.py +82 -0
- package/recce-source/tests/test_core.py +174 -0
- package/recce-source/tests/test_dbt.py +36 -0
- package/recce-source/tests/test_mcp_server.py +505 -0
- package/recce-source/tests/test_pull_request.py +130 -0
- package/recce-source/tests/test_server.py +202 -0
- package/recce-source/tests/test_server_lifespan.py +138 -0
- package/recce-source/tests/test_summary.py +73 -0
- package/recce-source/tests/util/__init__.py +0 -0
- package/recce-source/tests/util/cloud/__init__.py +0 -0
- package/recce-source/tests/util/cloud/test_check_events.py +255 -0
- package/recce-source/tests/util/cloud/test_checks.py +204 -0
- package/recce-source/tests/util/test_api_token.py +119 -0
- package/recce-source/tests/util/test_breaking.py +1427 -0
- package/recce-source/tests/util/test_cll.py +706 -0
- package/recce-source/tests/util/test_lineage.py +122 -0
- package/recce-source/tests/util/test_onboarding_state.py +84 -0
- package/recce-source/tests/util/test_recce_cloud.py +231 -0
- package/recce-source/tox.ini +40 -0
- package/recce-source/uv.lock +3928 -0
- package/src/api/index.ts +32 -0
- package/src/components/index.ts +154 -0
- package/src/global-styles.css +81 -0
- package/src/global.d.ts +14 -0
- package/src/hooks/index.ts +56 -0
- package/src/index.ts +17 -0
- package/src/lib/hooks/RouteConfigContext.ts +139 -0
- package/src/lib/hooks/useAppRouter.ts +240 -0
- package/src/mui-augmentation.d.ts +139 -0
- package/src/theme/index.ts +13 -0
- package/src/theme.ts +23 -0
- package/src/types/index.ts +23 -0
- package/dist/RecceCheckContext-DPpu9nG5.js +0 -303
- package/dist/RecceCheckContext-DPpu9nG5.js.map +0 -1
- package/dist/RecceCheckContext-bXdfQLGG.mjs +0 -229
- package/dist/RecceCheckContext-bXdfQLGG.mjs.map +0 -1
- package/dist/hooks-B9hsc1oD.js +0 -40
- package/dist/index-BNUP2V_N.d.ts.map +0 -1
- package/dist/index-DOPZuhD8.d.mts.map +0 -1
- package/dist/state-B9yzhuKs.mjs.map +0 -1
- package/dist/state-DOUPNifc.css.map +0 -1
- package/dist/state-IA7HWYOs.css.map +0 -1
- package/dist/state-lPCQsWy5.js.map +0 -1
- package/dist/theme.d.mts.map +0 -1
- package/dist/theme.d.ts.map +0 -1
- package/dist/urls-B1Ymdoz-.mjs.map +0 -1
- package/dist/urls-C4eAc82S.js.map +0 -1
- package/dist/version-Dh8sZhvs.js +0 -147
- package/dist/version-Dh8sZhvs.js.map +0 -1
- package/dist/version-OnOKzBeQ.mjs +0 -93
- package/dist/version-OnOKzBeQ.mjs.map +0 -1
|
@@ -0,0 +1,454 @@
|
|
|
1
|
+
import json
|
|
2
|
+
import logging
|
|
3
|
+
import os
|
|
4
|
+
import typing
|
|
5
|
+
from typing import IO, Dict
|
|
6
|
+
|
|
7
|
+
import requests
|
|
8
|
+
|
|
9
|
+
from recce import get_version
|
|
10
|
+
from recce.event import get_user_id, is_anonymous_tracking
|
|
11
|
+
from recce.pull_request import PullRequestInfo
|
|
12
|
+
|
|
13
|
+
if typing.TYPE_CHECKING:
|
|
14
|
+
from recce.util.cloud import ChecksCloud
|
|
15
|
+
|
|
16
|
+
RECCE_CLOUD_API_HOST = os.environ.get("RECCE_CLOUD_API_HOST", "https://cloud.datarecce.io")
|
|
17
|
+
RECCE_CLOUD_BASE_URL = os.environ.get("RECCE_CLOUD_BASE_URL", RECCE_CLOUD_API_HOST)
|
|
18
|
+
|
|
19
|
+
DOCKER_INTERNAL_URL_PREFIX = "http://host.docker.internal"
|
|
20
|
+
LOCALHOST_URL_PREFIX = "http://localhost"
|
|
21
|
+
|
|
22
|
+
logger = logging.getLogger("uvicorn")
|
|
23
|
+
|
|
24
|
+
|
|
25
|
+
class PresignedUrlMethod:
|
|
26
|
+
UPLOAD = "upload"
|
|
27
|
+
DOWNLOAD = "download"
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
class RecceCloudException(Exception):
|
|
31
|
+
def __init__(self, message: str, reason: str, status_code: int):
|
|
32
|
+
super().__init__(message)
|
|
33
|
+
self.status_code = status_code
|
|
34
|
+
|
|
35
|
+
try:
|
|
36
|
+
reason = json.loads(reason).get("detail", "")
|
|
37
|
+
except json.JSONDecodeError:
|
|
38
|
+
pass
|
|
39
|
+
self.reason = reason
|
|
40
|
+
|
|
41
|
+
|
|
42
|
+
class RecceCloud:
|
|
43
|
+
def __init__(self, token: str):
|
|
44
|
+
if token is None:
|
|
45
|
+
raise ValueError("Token cannot be None.")
|
|
46
|
+
self.token = token
|
|
47
|
+
self.token_type = "github_token" if token.startswith(("ghp_", "gho_", "ghu_", "ghs_", "ghr_")) else "api_token"
|
|
48
|
+
self.base_url = f"{RECCE_CLOUD_API_HOST}/api/v1"
|
|
49
|
+
self.base_url_v2 = f"{RECCE_CLOUD_API_HOST}/api/v2"
|
|
50
|
+
|
|
51
|
+
# Initialize modular clients
|
|
52
|
+
self._checks_client = None
|
|
53
|
+
|
|
54
|
+
@property
|
|
55
|
+
def checks(self) -> "ChecksCloud":
|
|
56
|
+
"""
|
|
57
|
+
Get the checks client for check operations.
|
|
58
|
+
|
|
59
|
+
Returns:
|
|
60
|
+
ChecksCloud instance for check operations
|
|
61
|
+
|
|
62
|
+
Example:
|
|
63
|
+
>>> cloud = RecceCloud(token="your-token")
|
|
64
|
+
>>> checks = cloud.checks.list_checks("org", "proj", "sess")
|
|
65
|
+
"""
|
|
66
|
+
if self._checks_client is None:
|
|
67
|
+
from recce.util.cloud import ChecksCloud
|
|
68
|
+
|
|
69
|
+
self._checks_client = ChecksCloud(self.token)
|
|
70
|
+
return self._checks_client
|
|
71
|
+
|
|
72
|
+
def _request(self, method, url, headers: Dict = None, **kwargs):
|
|
73
|
+
headers = {
|
|
74
|
+
**(headers or {}),
|
|
75
|
+
"Authorization": f"Bearer {self.token}",
|
|
76
|
+
}
|
|
77
|
+
return requests.request(method, url, headers=headers, **kwargs)
|
|
78
|
+
|
|
79
|
+
def verify_token(self) -> bool:
|
|
80
|
+
if self.token_type == "github_token":
|
|
81
|
+
return True
|
|
82
|
+
# Verify the Recce Cloud API token
|
|
83
|
+
api_url = f"{self.base_url}/verify-token"
|
|
84
|
+
try:
|
|
85
|
+
headers: Dict = None
|
|
86
|
+
if is_anonymous_tracking():
|
|
87
|
+
headers = {
|
|
88
|
+
"X-Recce-Oss-User-Id": get_user_id(),
|
|
89
|
+
"X-Recce-Oss-Version": get_version(),
|
|
90
|
+
}
|
|
91
|
+
response = self._request("GET", api_url, headers=headers)
|
|
92
|
+
if response.status_code == 200:
|
|
93
|
+
return True
|
|
94
|
+
except Exception:
|
|
95
|
+
pass
|
|
96
|
+
return False
|
|
97
|
+
|
|
98
|
+
def get_presigned_url_by_github_repo(
|
|
99
|
+
self,
|
|
100
|
+
method: PresignedUrlMethod,
|
|
101
|
+
repository: str,
|
|
102
|
+
artifact_name: str,
|
|
103
|
+
metadata: dict = None,
|
|
104
|
+
pr_id: int = None,
|
|
105
|
+
branch: str = None,
|
|
106
|
+
) -> str:
|
|
107
|
+
response = self._fetch_presigned_url(method, repository, artifact_name, metadata, pr_id, branch)
|
|
108
|
+
return response.get("presigned_url")
|
|
109
|
+
|
|
110
|
+
def _replace_localhost_with_docker_internal(self, url: str) -> str:
|
|
111
|
+
if url is None:
|
|
112
|
+
return None
|
|
113
|
+
if (
|
|
114
|
+
os.environ.get("RECCE_SHARE_INSTANCE_ENV") == "docker"
|
|
115
|
+
or os.environ.get("RECCE_TASK_INSTANCE_ENV") == "docker"
|
|
116
|
+
or os.environ.get("RECCE_INSTANCE_ENV") == "docker"
|
|
117
|
+
):
|
|
118
|
+
# For local development, convert the presigned URL from localhost to host.docker.internal
|
|
119
|
+
if url.startswith(LOCALHOST_URL_PREFIX):
|
|
120
|
+
return url.replace(LOCALHOST_URL_PREFIX, DOCKER_INTERNAL_URL_PREFIX)
|
|
121
|
+
return url
|
|
122
|
+
|
|
123
|
+
def get_presigned_url_by_share_id(
|
|
124
|
+
self,
|
|
125
|
+
method: PresignedUrlMethod,
|
|
126
|
+
share_id: str,
|
|
127
|
+
metadata: dict = None,
|
|
128
|
+
) -> str:
|
|
129
|
+
response = self._fetch_presigned_url_by_share_id(method, share_id, metadata=metadata)
|
|
130
|
+
presigned_url = response.get("presigned_url")
|
|
131
|
+
if not presigned_url:
|
|
132
|
+
raise RecceCloudException(
|
|
133
|
+
message="Failed to get presigned URL from Recce Cloud.",
|
|
134
|
+
reason="No presigned URL returned from the server.",
|
|
135
|
+
status_code=404,
|
|
136
|
+
)
|
|
137
|
+
presigned_url = self._replace_localhost_with_docker_internal(presigned_url)
|
|
138
|
+
return presigned_url
|
|
139
|
+
|
|
140
|
+
def get_download_presigned_url_by_github_repo_with_tags(
|
|
141
|
+
self, repository: str, artifact_name: str, branch: str = None
|
|
142
|
+
) -> (str, dict):
|
|
143
|
+
response = self._fetch_presigned_url(PresignedUrlMethod.DOWNLOAD, repository, artifact_name, branch=branch)
|
|
144
|
+
return response.get("presigned_url"), response.get("tags", {})
|
|
145
|
+
|
|
146
|
+
def _fetch_presigned_url(
|
|
147
|
+
self,
|
|
148
|
+
method: PresignedUrlMethod,
|
|
149
|
+
repository: str,
|
|
150
|
+
artifact_name: str,
|
|
151
|
+
metadata: dict = None,
|
|
152
|
+
pr_id: int = None,
|
|
153
|
+
branch: str = None,
|
|
154
|
+
) -> str:
|
|
155
|
+
if pr_id is not None:
|
|
156
|
+
api_url = f"{self.base_url}/{repository}/pulls/{pr_id}/artifacts/{method}?artifact_name={artifact_name}&enable_ssec=true"
|
|
157
|
+
elif branch is not None:
|
|
158
|
+
api_url = f"{self.base_url}/{repository}/commits/{branch}/artifacts/{method}?artifact_name={artifact_name}&enable_ssec=true"
|
|
159
|
+
else:
|
|
160
|
+
raise ValueError("Either pr_id or sha must be provided.")
|
|
161
|
+
response = self._request("POST", api_url, json=metadata)
|
|
162
|
+
if response.status_code != 200:
|
|
163
|
+
raise RecceCloudException(
|
|
164
|
+
message="Failed to {method} artifact {preposition} Recce Cloud.".format(
|
|
165
|
+
method=method, preposition="from" if method == PresignedUrlMethod.DOWNLOAD else "to"
|
|
166
|
+
),
|
|
167
|
+
reason=response.text,
|
|
168
|
+
status_code=response.status_code,
|
|
169
|
+
)
|
|
170
|
+
return response.json()
|
|
171
|
+
|
|
172
|
+
def _fetch_presigned_url_by_share_id(
|
|
173
|
+
self,
|
|
174
|
+
method: PresignedUrlMethod,
|
|
175
|
+
share_id: str,
|
|
176
|
+
metadata: dict = None,
|
|
177
|
+
):
|
|
178
|
+
api_url = f"{self.base_url}/shares/{share_id}/presigned/{method}"
|
|
179
|
+
data = None
|
|
180
|
+
# Only provide metadata for upload requests
|
|
181
|
+
if method == PresignedUrlMethod.UPLOAD:
|
|
182
|
+
# Covert metadata values to strings to ensure JSON serializability
|
|
183
|
+
data = {"metadata": {key: str(value) for key, value in metadata.items()}} if metadata else None
|
|
184
|
+
response = self._request(
|
|
185
|
+
"POST",
|
|
186
|
+
api_url,
|
|
187
|
+
json=data,
|
|
188
|
+
)
|
|
189
|
+
if response.status_code != 200:
|
|
190
|
+
raise RecceCloudException(
|
|
191
|
+
message="Failed to {method} artifact {preposition} Recce Cloud.".format(
|
|
192
|
+
method=method, preposition="from" if method == PresignedUrlMethod.DOWNLOAD else "to"
|
|
193
|
+
),
|
|
194
|
+
reason=response.text,
|
|
195
|
+
status_code=response.status_code,
|
|
196
|
+
)
|
|
197
|
+
return response.json()
|
|
198
|
+
|
|
199
|
+
def get_artifact_metadata(self, pr_info: PullRequestInfo) -> dict:
|
|
200
|
+
api_url = f"{self.base_url}/{pr_info.repository}/pulls/{pr_info.id}/metadata"
|
|
201
|
+
response = self._request("GET", api_url)
|
|
202
|
+
if response.status_code == 204:
|
|
203
|
+
return None
|
|
204
|
+
if response.status_code != 200:
|
|
205
|
+
raise RecceCloudException(
|
|
206
|
+
message="Failed to get artifact metadata from Recce Cloud.",
|
|
207
|
+
reason=response.text,
|
|
208
|
+
status_code=response.status_code,
|
|
209
|
+
)
|
|
210
|
+
return response.json()
|
|
211
|
+
|
|
212
|
+
def purge_artifacts(self, repository: str, pr_id: int = None, branch: str = None):
|
|
213
|
+
if pr_id is not None:
|
|
214
|
+
api_url = f"{self.base_url}/{repository}/pulls/{pr_id}/artifacts"
|
|
215
|
+
error_message = "Failed to purge artifacts from Recce Cloud."
|
|
216
|
+
elif branch is not None:
|
|
217
|
+
api_url = f"{self.base_url}/{repository}/commits/{branch}/artifacts"
|
|
218
|
+
error_message = "Failed to delete artifacts from Recce Cloud."
|
|
219
|
+
else:
|
|
220
|
+
raise ValueError(
|
|
221
|
+
"Please either run this command from within a pull request context "
|
|
222
|
+
"or specify a branch using the --branch option."
|
|
223
|
+
)
|
|
224
|
+
response = self._request("DELETE", api_url)
|
|
225
|
+
if response.status_code != 204:
|
|
226
|
+
raise RecceCloudException(
|
|
227
|
+
message=error_message,
|
|
228
|
+
reason=response.text,
|
|
229
|
+
status_code=response.status_code,
|
|
230
|
+
)
|
|
231
|
+
|
|
232
|
+
def check_artifacts_exists(self, pr_info: PullRequestInfo) -> bool:
|
|
233
|
+
api_url = f"{self.base_url}/{pr_info.repository}/pulls/{pr_info.id}/metadata"
|
|
234
|
+
response = self._request("GET", api_url)
|
|
235
|
+
if response.status_code == 200:
|
|
236
|
+
return True
|
|
237
|
+
elif response.status_code == 204:
|
|
238
|
+
return False
|
|
239
|
+
else:
|
|
240
|
+
raise RecceCloudException(
|
|
241
|
+
message="Failed to check if artifacts exist in Recce Cloud.",
|
|
242
|
+
reason=response.text,
|
|
243
|
+
status_code=response.status_code,
|
|
244
|
+
)
|
|
245
|
+
|
|
246
|
+
def share_state(self, file_name: str, file_io: IO):
|
|
247
|
+
api_url = f"{self.base_url}/recce-state/upload"
|
|
248
|
+
files = {"file": (file_name, file_io, "application/json")}
|
|
249
|
+
response = self._request("POST", api_url, files=files)
|
|
250
|
+
if response.status_code == 403:
|
|
251
|
+
return {"status": "error", "message": response.json().get("detail")}
|
|
252
|
+
if response.status_code != 200:
|
|
253
|
+
raise RecceCloudException(
|
|
254
|
+
message="Failed to share Recce state.", reason=response.text, status_code=response.status_code
|
|
255
|
+
)
|
|
256
|
+
return response.json()
|
|
257
|
+
|
|
258
|
+
def update_github_pull_request_check(self, pr_info: PullRequestInfo, metadata: dict = None):
|
|
259
|
+
api_url = f"{self.base_url}/{pr_info.repository}/pulls/{pr_info.id}/github/checks"
|
|
260
|
+
try:
|
|
261
|
+
self._request("POST", api_url, json=metadata)
|
|
262
|
+
except Exception as e:
|
|
263
|
+
# We don't care the response of this request, so we don't need to raise any exception.
|
|
264
|
+
logger.debug(f"Failed to update the GitHub PR check. Reason: {str(e)}")
|
|
265
|
+
|
|
266
|
+
def get_user_info(self) -> Dict:
|
|
267
|
+
api_url = f"{self.base_url}/users"
|
|
268
|
+
response = self._request("GET", api_url)
|
|
269
|
+
if response.status_code != 200:
|
|
270
|
+
raise RecceCloudException(
|
|
271
|
+
message="Failed to get user info from Recce Cloud.",
|
|
272
|
+
reason=response.text,
|
|
273
|
+
status_code=response.status_code,
|
|
274
|
+
)
|
|
275
|
+
return response.json().get("user")
|
|
276
|
+
|
|
277
|
+
def set_onboarding_state(self, state: str):
|
|
278
|
+
api_url = f"{self.base_url}/users/onboarding-state"
|
|
279
|
+
try:
|
|
280
|
+
response = self._request("PUT", api_url, json={"state": state})
|
|
281
|
+
response.raise_for_status()
|
|
282
|
+
except requests.exceptions.HTTPError as e:
|
|
283
|
+
# Don't Raise an exception if setting onboarding_state fails
|
|
284
|
+
logger.warning(f"Failed to set Onboarding State in Recce Cloud. Reason: {str(e)}")
|
|
285
|
+
return
|
|
286
|
+
|
|
287
|
+
def get_session(self, session_id: str):
|
|
288
|
+
api_url = f"{self.base_url_v2}/sessions/{session_id}"
|
|
289
|
+
response = self._request("GET", api_url)
|
|
290
|
+
if response.status_code == 403:
|
|
291
|
+
return {"status": "error", "message": response.json().get("detail")}
|
|
292
|
+
if response.status_code != 200:
|
|
293
|
+
raise RecceCloudException(
|
|
294
|
+
message="Failed to get session from Recce Cloud.",
|
|
295
|
+
reason=response.text,
|
|
296
|
+
status_code=response.status_code,
|
|
297
|
+
)
|
|
298
|
+
data = response.json()
|
|
299
|
+
if data["success"] is not True:
|
|
300
|
+
raise RecceCloudException(
|
|
301
|
+
message="Failed to get session from Recce Cloud.",
|
|
302
|
+
reason=data.get("message", "Unknown error"),
|
|
303
|
+
status_code=response.status_code,
|
|
304
|
+
)
|
|
305
|
+
return data["session"]
|
|
306
|
+
|
|
307
|
+
def update_session(self, org_id: str, project_id: str, session_id: str, adapter_type: str):
|
|
308
|
+
api_url = f"{self.base_url_v2}/organizations/{org_id}/projects/{project_id}/sessions/{session_id}"
|
|
309
|
+
data = {"adapter_type": adapter_type}
|
|
310
|
+
response = self._request("PATCH", api_url, json=data)
|
|
311
|
+
if response.status_code == 403:
|
|
312
|
+
return {"status": "error", "message": response.json().get("detail")}
|
|
313
|
+
if response.status_code != 200:
|
|
314
|
+
raise RecceCloudException(
|
|
315
|
+
message="Failed to update session in Recce Cloud.",
|
|
316
|
+
reason=response.text,
|
|
317
|
+
status_code=response.status_code,
|
|
318
|
+
)
|
|
319
|
+
return response.json()
|
|
320
|
+
|
|
321
|
+
def get_download_urls_by_session_id(self, org_id: str, project_id: str, session_id: str) -> dict[str, str]:
|
|
322
|
+
api_url = f"{self.base_url_v2}/organizations/{org_id}/projects/{project_id}/sessions/{session_id}/download-url"
|
|
323
|
+
response = self._request("GET", api_url)
|
|
324
|
+
if response.status_code != 200:
|
|
325
|
+
raise RecceCloudException(
|
|
326
|
+
message="Failed to download session from Recce Cloud.",
|
|
327
|
+
reason=response.text,
|
|
328
|
+
status_code=response.status_code,
|
|
329
|
+
)
|
|
330
|
+
data = response.json()
|
|
331
|
+
if data["presigned_urls"] is None:
|
|
332
|
+
raise RecceCloudException(
|
|
333
|
+
message="No presigned URLs returned from the server.",
|
|
334
|
+
reason="",
|
|
335
|
+
status_code=404,
|
|
336
|
+
)
|
|
337
|
+
|
|
338
|
+
presigned_urls = data["presigned_urls"]
|
|
339
|
+
for key, url in presigned_urls.items():
|
|
340
|
+
presigned_urls[key] = self._replace_localhost_with_docker_internal(url)
|
|
341
|
+
return presigned_urls
|
|
342
|
+
|
|
343
|
+
def get_base_session_download_urls(self, org_id: str, project_id: str) -> dict[str, str]:
|
|
344
|
+
"""Get download URLs for the base session of a project."""
|
|
345
|
+
api_url = f"{self.base_url_v2}/organizations/{org_id}/projects/{project_id}/base-session/download-url"
|
|
346
|
+
response = self._request("GET", api_url)
|
|
347
|
+
if response.status_code != 200:
|
|
348
|
+
raise RecceCloudException(
|
|
349
|
+
message="Failed to download base session from Recce Cloud.",
|
|
350
|
+
reason=response.text,
|
|
351
|
+
status_code=response.status_code,
|
|
352
|
+
)
|
|
353
|
+
data = response.json()
|
|
354
|
+
if data["presigned_urls"] is None:
|
|
355
|
+
raise RecceCloudException(
|
|
356
|
+
message="No presigned URLs returned from the server.",
|
|
357
|
+
reason="",
|
|
358
|
+
status_code=404,
|
|
359
|
+
)
|
|
360
|
+
|
|
361
|
+
presigned_urls = data["presigned_urls"]
|
|
362
|
+
for key, url in presigned_urls.items():
|
|
363
|
+
presigned_urls[key] = self._replace_localhost_with_docker_internal(url)
|
|
364
|
+
return presigned_urls
|
|
365
|
+
|
|
366
|
+
def get_upload_urls_by_session_id(self, org_id: str, project_id: str, session_id: str) -> dict[str, str]:
|
|
367
|
+
api_url = f"{self.base_url_v2}/organizations/{org_id}/projects/{project_id}/sessions/{session_id}/upload-url"
|
|
368
|
+
response = self._request("GET", api_url)
|
|
369
|
+
if response.status_code != 200:
|
|
370
|
+
raise RecceCloudException(
|
|
371
|
+
message="Failed to get upload URLs for session from Recce Cloud.",
|
|
372
|
+
reason=response.text,
|
|
373
|
+
status_code=response.status_code,
|
|
374
|
+
)
|
|
375
|
+
data = response.json()
|
|
376
|
+
if data["presigned_urls"] is None:
|
|
377
|
+
raise RecceCloudException(
|
|
378
|
+
message="No presigned URLs returned from the server.",
|
|
379
|
+
reason="",
|
|
380
|
+
status_code=404,
|
|
381
|
+
)
|
|
382
|
+
|
|
383
|
+
presigned_urls = data["presigned_urls"]
|
|
384
|
+
for key, url in presigned_urls.items():
|
|
385
|
+
presigned_urls[key] = self._replace_localhost_with_docker_internal(url)
|
|
386
|
+
return presigned_urls
|
|
387
|
+
|
|
388
|
+
def post_recce_state_uploaded_by_session_id(self, org_id: str, project_id: str, session_id: str):
|
|
389
|
+
api_url = f"{self.base_url_v2}/organizations/{org_id}/projects/{project_id}/sessions/{session_id}/recce-state-uploaded"
|
|
390
|
+
response = self._request("POST", api_url)
|
|
391
|
+
if response.status_code != 204:
|
|
392
|
+
raise RecceCloudException(
|
|
393
|
+
message="Failed to notify state uploaded for session in Recce Cloud.",
|
|
394
|
+
reason=response.text,
|
|
395
|
+
status_code=response.status_code,
|
|
396
|
+
)
|
|
397
|
+
|
|
398
|
+
def list_organizations(self) -> list:
|
|
399
|
+
"""List all organizations the user has access to."""
|
|
400
|
+
api_url = f"{self.base_url_v2}/organizations"
|
|
401
|
+
response = self._request("GET", api_url)
|
|
402
|
+
if response.status_code != 200:
|
|
403
|
+
raise RecceCloudException(
|
|
404
|
+
message="Failed to list organizations from Recce Cloud.",
|
|
405
|
+
reason=response.text,
|
|
406
|
+
status_code=response.status_code,
|
|
407
|
+
)
|
|
408
|
+
data = response.json()
|
|
409
|
+
return data.get("organizations", [])
|
|
410
|
+
|
|
411
|
+
def list_projects(self, org_id: str) -> list:
|
|
412
|
+
"""List all projects in an organization."""
|
|
413
|
+
api_url = f"{self.base_url_v2}/organizations/{org_id}/projects"
|
|
414
|
+
response = self._request("GET", api_url)
|
|
415
|
+
if response.status_code != 200:
|
|
416
|
+
raise RecceCloudException(
|
|
417
|
+
message="Failed to list projects from Recce Cloud.",
|
|
418
|
+
reason=response.text,
|
|
419
|
+
status_code=response.status_code,
|
|
420
|
+
)
|
|
421
|
+
data = response.json()
|
|
422
|
+
return data.get("projects", [])
|
|
423
|
+
|
|
424
|
+
def list_sessions(self, org_id: str, project_id: str) -> list:
|
|
425
|
+
"""List all sessions in a project."""
|
|
426
|
+
api_url = f"{self.base_url_v2}/organizations/{org_id}/projects/{project_id}/sessions"
|
|
427
|
+
response = self._request("GET", api_url)
|
|
428
|
+
if response.status_code != 200:
|
|
429
|
+
raise RecceCloudException(
|
|
430
|
+
message="Failed to list sessions from Recce Cloud.",
|
|
431
|
+
reason=response.text,
|
|
432
|
+
status_code=response.status_code,
|
|
433
|
+
)
|
|
434
|
+
data = response.json()
|
|
435
|
+
return data.get("sessions", [])
|
|
436
|
+
|
|
437
|
+
|
|
438
|
+
def get_recce_cloud_onboarding_state(token: str) -> str:
|
|
439
|
+
if token and token.startswith("rct-"):
|
|
440
|
+
return "undefined"
|
|
441
|
+
|
|
442
|
+
try:
|
|
443
|
+
recce_cloud = RecceCloud(token)
|
|
444
|
+
user_info = recce_cloud.get_user_info()
|
|
445
|
+
if user_info:
|
|
446
|
+
return user_info.get("onboarding_state")
|
|
447
|
+
except Exception as e:
|
|
448
|
+
logger.debug(str(e))
|
|
449
|
+
return "undefined"
|
|
450
|
+
|
|
451
|
+
|
|
452
|
+
def set_recce_cloud_onboarding_state(token: str, new_state: str):
|
|
453
|
+
recce_cloud = RecceCloud(token)
|
|
454
|
+
recce_cloud.set_onboarding_state(new_state)
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
class SingletonMeta(type):
|
|
2
|
+
"""
|
|
3
|
+
The Singleton class can be implemented in different ways in Python. Some
|
|
4
|
+
possible methods include: base class, decorator, metaclass. We will use the
|
|
5
|
+
metaclass because it is best suited for this purpose.
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
_instances = {}
|
|
9
|
+
|
|
10
|
+
def __call__(cls, *args, **kwargs):
|
|
11
|
+
"""
|
|
12
|
+
Possible changes to the value of the `__init__` argument do not affect
|
|
13
|
+
the returned instance.
|
|
14
|
+
"""
|
|
15
|
+
if cls not in cls._instances:
|
|
16
|
+
instance = super().__call__(*args, **kwargs)
|
|
17
|
+
cls._instances[cls] = instance
|
|
18
|
+
return cls._instances[cls]
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
import functools
|
|
2
|
+
import os
|
|
3
|
+
import time
|
|
4
|
+
from dataclasses import dataclass, field
|
|
5
|
+
from typing import Dict, Optional
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
@dataclass
|
|
9
|
+
class StartupPerfTracker:
|
|
10
|
+
"""
|
|
11
|
+
Tracks startup performance metrics for Recce server.
|
|
12
|
+
All timing values are in milliseconds.
|
|
13
|
+
"""
|
|
14
|
+
|
|
15
|
+
# All timings in ms (populated by @track_timing decorator)
|
|
16
|
+
timings: Dict[str, float] = field(default_factory=dict)
|
|
17
|
+
|
|
18
|
+
# Artifact sizes in bytes
|
|
19
|
+
artifact_sizes: Dict[str, int] = field(default_factory=dict)
|
|
20
|
+
|
|
21
|
+
# Metadata
|
|
22
|
+
cloud_mode: bool = False
|
|
23
|
+
catalog_type: Optional[str] = None # github, preview, session
|
|
24
|
+
adapter_type: Optional[str] = None
|
|
25
|
+
node_count: Optional[int] = None
|
|
26
|
+
command: Optional[str] = None # server, read-only, preview
|
|
27
|
+
|
|
28
|
+
def record_timing(self, name: str, elapsed_ms: float):
|
|
29
|
+
"""Record timing for a named phase or artifact"""
|
|
30
|
+
self.timings[name] = elapsed_ms
|
|
31
|
+
|
|
32
|
+
def set_cloud_mode(self, cloud_mode: bool):
|
|
33
|
+
self.cloud_mode = cloud_mode
|
|
34
|
+
|
|
35
|
+
def set_catalog_type(self, catalog_type: str):
|
|
36
|
+
self.catalog_type = catalog_type
|
|
37
|
+
|
|
38
|
+
def set_artifact_size(self, name: str, size_bytes: int):
|
|
39
|
+
"""Set artifact size by name"""
|
|
40
|
+
self.artifact_sizes[name] = size_bytes
|
|
41
|
+
|
|
42
|
+
def to_dict(self) -> Dict:
|
|
43
|
+
return {
|
|
44
|
+
"timings": self.timings if self.timings else None,
|
|
45
|
+
"artifact_sizes": self.artifact_sizes if self.artifact_sizes else None,
|
|
46
|
+
"cloud_mode": self.cloud_mode,
|
|
47
|
+
"catalog_type": self.catalog_type,
|
|
48
|
+
"adapter_type": self.adapter_type,
|
|
49
|
+
"node_count": self.node_count,
|
|
50
|
+
"command": self.command,
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
# Module-level singleton for tracking startup across the call stack
|
|
55
|
+
_startup_tracker: Optional[StartupPerfTracker] = None
|
|
56
|
+
|
|
57
|
+
|
|
58
|
+
def get_startup_tracker() -> Optional[StartupPerfTracker]:
|
|
59
|
+
"""Get the global startup tracker instance"""
|
|
60
|
+
return _startup_tracker
|
|
61
|
+
|
|
62
|
+
|
|
63
|
+
def set_startup_tracker(tracker: StartupPerfTracker):
|
|
64
|
+
"""Set the global startup tracker instance"""
|
|
65
|
+
global _startup_tracker
|
|
66
|
+
_startup_tracker = tracker
|
|
67
|
+
|
|
68
|
+
|
|
69
|
+
def clear_startup_tracker():
|
|
70
|
+
"""Clear the global startup tracker instance"""
|
|
71
|
+
global _startup_tracker
|
|
72
|
+
_startup_tracker = None
|
|
73
|
+
|
|
74
|
+
|
|
75
|
+
def track_timing(timing_name: str = None, *, record_size: bool = False):
|
|
76
|
+
"""
|
|
77
|
+
Decorator factory to track timing for any operation.
|
|
78
|
+
|
|
79
|
+
Args:
|
|
80
|
+
timing_name: Name for the timing. If None, expects 'timing_name' kwarg at call time.
|
|
81
|
+
record_size: If True, record file size from 'path' kwarg.
|
|
82
|
+
|
|
83
|
+
Usage:
|
|
84
|
+
# Name at decoration time
|
|
85
|
+
@track_timing("state_loader_init")
|
|
86
|
+
def create_state_loader_by_args(...):
|
|
87
|
+
...
|
|
88
|
+
|
|
89
|
+
# Name at call time (for reusable functions)
|
|
90
|
+
@track_timing(record_size=True)
|
|
91
|
+
def load_manifest(path=None, data=None):
|
|
92
|
+
...
|
|
93
|
+
|
|
94
|
+
load_manifest(path=p, timing_name="curr_manifest")
|
|
95
|
+
"""
|
|
96
|
+
|
|
97
|
+
def decorator(func):
|
|
98
|
+
@functools.wraps(func)
|
|
99
|
+
def wrapper(*args, **kwargs):
|
|
100
|
+
# Get timing name from decorator arg or from kwargs
|
|
101
|
+
name = timing_name
|
|
102
|
+
if name is None:
|
|
103
|
+
name = kwargs.pop("timing_name", None)
|
|
104
|
+
|
|
105
|
+
path = kwargs.get("path") or (args[0] if args else None)
|
|
106
|
+
|
|
107
|
+
start = time.perf_counter_ns()
|
|
108
|
+
result = func(*args, **kwargs)
|
|
109
|
+
elapsed_ms = (time.perf_counter_ns() - start) / 1_000_000
|
|
110
|
+
|
|
111
|
+
if tracker := get_startup_tracker():
|
|
112
|
+
if name:
|
|
113
|
+
tracker.record_timing(name, elapsed_ms)
|
|
114
|
+
if record_size and name and path and os.path.exists(path):
|
|
115
|
+
tracker.set_artifact_size(name, os.path.getsize(path))
|
|
116
|
+
|
|
117
|
+
return result
|
|
118
|
+
|
|
119
|
+
return wrapper
|
|
120
|
+
|
|
121
|
+
return decorator
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
from typing import Any, Callable
|
|
2
|
+
|
|
3
|
+
from ruamel import yaml
|
|
4
|
+
from ruamel.yaml import CommentedMap as _cm
|
|
5
|
+
from ruamel.yaml import CommentedSeq as _cs
|
|
6
|
+
|
|
7
|
+
_yaml = yaml.YAML()
|
|
8
|
+
_safe_yaml = yaml.YAML(typ="safe")
|
|
9
|
+
|
|
10
|
+
CommentedMap = _cm
|
|
11
|
+
CommentedSeq = _cs
|
|
12
|
+
YAMLError = yaml.YAMLError
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
def load(stream) -> Any:
|
|
16
|
+
return _yaml.load(stream)
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def allow_duplicate_keys_loader() -> Callable:
|
|
20
|
+
yml = yaml.YAML()
|
|
21
|
+
yml.allow_duplicate_keys = True
|
|
22
|
+
return yml.load
|
|
23
|
+
|
|
24
|
+
|
|
25
|
+
def safe_load(stream, version=None) -> Any:
|
|
26
|
+
if version is not None:
|
|
27
|
+
_safe_yaml.version = version
|
|
28
|
+
return _safe_yaml.load(stream)
|
|
29
|
+
|
|
30
|
+
|
|
31
|
+
def dump(data, stream: Any = None, *, transform: Any = None) -> Any:
|
|
32
|
+
return _yaml.dump(data, stream, transform=transform)
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
def safe_load_yaml(file_path):
|
|
36
|
+
try:
|
|
37
|
+
with open(file_path, "r", encoding="utf-8") as f:
|
|
38
|
+
payload = safe_load(f)
|
|
39
|
+
except yaml.YAMLError as e:
|
|
40
|
+
print(e)
|
|
41
|
+
return None
|
|
42
|
+
except FileNotFoundError:
|
|
43
|
+
return None
|
|
44
|
+
return payload
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
def round_trip_load_yaml(file_path):
|
|
48
|
+
with open(file_path, "r", encoding="utf-8") as f:
|
|
49
|
+
try:
|
|
50
|
+
payload = load(f)
|
|
51
|
+
except yaml.YAMLError as e:
|
|
52
|
+
print(e)
|
|
53
|
+
return None
|
|
54
|
+
return payload
|
|
55
|
+
|
|
56
|
+
|
|
57
|
+
def round_trip_dump(data: Any, stream=None):
|
|
58
|
+
return yaml.round_trip_dump(data, stream)
|