@libredb/studio 0.9.7 → 0.9.13
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/chunk-34YQUUCM.mjs +319 -0
- package/dist/chunk-34YQUUCM.mjs.map +1 -0
- package/dist/chunk-4LVB3K53.mjs +37 -0
- package/dist/chunk-4LVB3K53.mjs.map +1 -0
- package/dist/chunk-6DRZXXNT.mjs +100 -0
- package/dist/chunk-6DRZXXNT.mjs.map +1 -0
- package/dist/chunk-CPF7XWV5.mjs +1289 -0
- package/dist/chunk-CPF7XWV5.mjs.map +1 -0
- package/dist/chunk-CZVV3JJB.mjs +160 -0
- package/dist/chunk-CZVV3JJB.mjs.map +1 -0
- package/dist/chunk-D4WVWWWF.js +332 -0
- package/dist/chunk-D4WVWWWF.js.map +1 -0
- package/dist/chunk-DY3KXE44.mjs +3 -0
- package/dist/chunk-DY3KXE44.mjs.map +1 -0
- package/dist/chunk-FYSE52VB.js +242 -0
- package/dist/chunk-FYSE52VB.js.map +1 -0
- package/dist/chunk-G3S66G64.mjs +6673 -0
- package/dist/chunk-G3S66G64.mjs.map +1 -0
- package/dist/chunk-G4WYE6TI.js +4 -0
- package/dist/chunk-G4WYE6TI.js.map +1 -0
- package/dist/chunk-HGPD6PWV.js +1310 -0
- package/dist/chunk-HGPD6PWV.js.map +1 -0
- package/dist/chunk-JZO5KRZN.js +165 -0
- package/dist/chunk-JZO5KRZN.js.map +1 -0
- package/dist/chunk-KV356UXJ.js +253 -0
- package/dist/chunk-KV356UXJ.js.map +1 -0
- package/dist/chunk-PPODO6HX.mjs +237 -0
- package/dist/chunk-PPODO6HX.mjs.map +1 -0
- package/dist/chunk-PTIRB2JO.js +258 -0
- package/dist/chunk-PTIRB2JO.js.map +1 -0
- package/dist/chunk-Q6LRDBK7.js +42 -0
- package/dist/chunk-Q6LRDBK7.js.map +1 -0
- package/dist/chunk-QJP5FZRY.mjs +255 -0
- package/dist/chunk-QJP5FZRY.mjs.map +1 -0
- package/dist/chunk-R3POCJK6.mjs +248 -0
- package/dist/chunk-R3POCJK6.mjs.map +1 -0
- package/dist/chunk-RCQB4FCE.js +186 -0
- package/dist/chunk-RCQB4FCE.js.map +1 -0
- package/dist/chunk-SR5DRGBX.mjs +174 -0
- package/dist/chunk-SR5DRGBX.mjs.map +1 -0
- package/dist/chunk-VLCRUZX7.js +102 -0
- package/dist/chunk-VLCRUZX7.js.map +1 -0
- package/dist/chunk-Y52UIFEX.js +6741 -0
- package/dist/chunk-Y52UIFEX.js.map +1 -0
- package/dist/components.d.mts +273 -0
- package/dist/components.d.ts +273 -0
- package/dist/components.js +59 -0
- package/dist/components.js.map +1 -0
- package/dist/components.mjs +6 -0
- package/dist/components.mjs.map +1 -0
- package/dist/custom-BNDOYC5P.js +134 -0
- package/dist/custom-BNDOYC5P.js.map +1 -0
- package/dist/custom-S2EKFMP3.mjs +132 -0
- package/dist/custom-S2EKFMP3.mjs.map +1 -0
- package/dist/gemini-4ASHNK4H.js +81 -0
- package/dist/gemini-4ASHNK4H.js.map +1 -0
- package/dist/gemini-C5RBLQEJ.mjs +79 -0
- package/dist/gemini-C5RBLQEJ.mjs.map +1 -0
- package/dist/index.d.mts +6 -0
- package/dist/index.d.ts +6 -0
- package/dist/index.js +95 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +10 -0
- package/dist/index.mjs.map +1 -0
- package/dist/mongodb-XMZEZA4A.mjs +748 -0
- package/dist/mongodb-XMZEZA4A.mjs.map +1 -0
- package/dist/mongodb-YQJJTLX3.js +750 -0
- package/dist/mongodb-YQJJTLX3.js.map +1 -0
- package/dist/mssql-PMOU4D36.js +916 -0
- package/dist/mssql-PMOU4D36.js.map +1 -0
- package/{src/lib/db/providers/sql/mssql.ts → dist/mssql-ZH5VP2C5.mjs} +268 -423
- package/dist/mssql-ZH5VP2C5.mjs.map +1 -0
- package/{src/lib/db/providers/sql/mysql.ts → dist/mysql-I3WJQXN2.mjs} +277 -428
- package/dist/mysql-I3WJQXN2.mjs.map +1 -0
- package/dist/mysql-Y3MSA5QY.js +833 -0
- package/dist/mysql-Y3MSA5QY.js.map +1 -0
- package/dist/ollama-26BYLVEV.mjs +115 -0
- package/dist/ollama-26BYLVEV.mjs.map +1 -0
- package/dist/ollama-HVWAGKQC.js +117 -0
- package/dist/ollama-HVWAGKQC.js.map +1 -0
- package/dist/openai-4U56KPG7.mjs +111 -0
- package/dist/openai-4U56KPG7.mjs.map +1 -0
- package/dist/openai-AK3R37BS.js +113 -0
- package/dist/openai-AK3R37BS.js.map +1 -0
- package/dist/oracle-L6VEAVXO.js +917 -0
- package/dist/oracle-L6VEAVXO.js.map +1 -0
- package/{src/lib/db/providers/sql/oracle.ts → dist/oracle-P2G7T4P4.mjs} +321 -454
- package/dist/oracle-P2G7T4P4.mjs.map +1 -0
- package/{src/lib/db/providers/sql/postgres.ts → dist/postgres-O5KOQUVP.mjs} +261 -471
- package/dist/postgres-O5KOQUVP.mjs.map +1 -0
- package/dist/postgres-RLCWNFFX.js +971 -0
- package/dist/postgres-RLCWNFFX.js.map +1 -0
- package/dist/providers.d.mts +149 -0
- package/dist/providers.d.ts +149 -0
- package/dist/providers.js +44 -0
- package/dist/providers.js.map +1 -0
- package/dist/providers.mjs +7 -0
- package/dist/providers.mjs.map +1 -0
- package/dist/redis-4WMQOVLX.mjs +435 -0
- package/dist/redis-4WMQOVLX.mjs.map +1 -0
- package/dist/redis-QVQ6YU62.js +441 -0
- package/dist/redis-QVQ6YU62.js.map +1 -0
- package/dist/sqlite-4I2P2OGQ.js +554 -0
- package/dist/sqlite-4I2P2OGQ.js.map +1 -0
- package/dist/sqlite-OA4YJX5S.mjs +531 -0
- package/dist/sqlite-OA4YJX5S.mjs.map +1 -0
- package/dist/types-BJvJfxSY.d.mts +141 -0
- package/dist/types-BJvJfxSY.d.ts +141 -0
- package/dist/types-ClAg_v5k.d.mts +343 -0
- package/dist/types-Der_X8E8.d.ts +343 -0
- package/dist/types.d.mts +2 -0
- package/dist/types.d.ts +2 -0
- package/dist/types.js +6 -0
- package/dist/types.js.map +1 -0
- package/dist/types.mjs +3 -0
- package/dist/types.mjs.map +1 -0
- package/dist/workspace.d.mts +80 -0
- package/dist/workspace.d.ts +80 -0
- package/dist/workspace.js +4174 -0
- package/dist/workspace.js.map +1 -0
- package/dist/workspace.mjs +4147 -0
- package/dist/workspace.mjs.map +1 -0
- package/package.json +60 -5
- package/.claude/settings.local.json +0 -127
- package/.cursorrules +0 -426
- package/.devin/wiki.json +0 -143
- package/.dockerignore +0 -80
- package/.env.example +0 -159
- package/.github/ISSUE_TEMPLATE/bug_report.md +0 -49
- package/.github/ISSUE_TEMPLATE/feature_request.md +0 -29
- package/.github/PULL_REQUEST_TEMPLATE.md +0 -57
- package/.github/workflows/ci.yml +0 -185
- package/.github/workflows/codeql.yml +0 -57
- package/.github/workflows/docker-build-push.yml +0 -118
- package/.github/workflows/helm-release.yml +0 -113
- package/CLAUDE.md +0 -265
- package/CODE_OF_CONDUCT.md +0 -124
- package/CONTRIBUTING.md +0 -154
- package/Dockerfile +0 -73
- package/SECURITY.md +0 -107
- package/artifacthub-repo.yml +0 -4
- package/bun.lock +0 -1714
- package/bunfig.toml +0 -3
- package/charts/libredb-studio/.helmignore +0 -11
- package/charts/libredb-studio/Chart.lock +0 -6
- package/charts/libredb-studio/Chart.yaml +0 -50
- package/charts/libredb-studio/README.md +0 -206
- package/charts/libredb-studio/templates/NOTES.txt +0 -59
- package/charts/libredb-studio/templates/_helpers.tpl +0 -135
- package/charts/libredb-studio/templates/configmap.yaml +0 -37
- package/charts/libredb-studio/templates/deployment.yaml +0 -184
- package/charts/libredb-studio/templates/hpa.yaml +0 -32
- package/charts/libredb-studio/templates/ingress.yaml +0 -41
- package/charts/libredb-studio/templates/networkpolicy.yaml +0 -50
- package/charts/libredb-studio/templates/pdb.yaml +0 -18
- package/charts/libredb-studio/templates/pvc.yaml +0 -23
- package/charts/libredb-studio/templates/secret.yaml +0 -30
- package/charts/libredb-studio/templates/seed-configmap.yaml +0 -11
- package/charts/libredb-studio/templates/service.yaml +0 -22
- package/charts/libredb-studio/templates/serviceaccount.yaml +0 -13
- package/charts/libredb-studio/values.schema.json +0 -246
- package/charts/libredb-studio/values.yaml +0 -286
- package/components.json +0 -22
- package/conductor/code_styleguides/typescript.md +0 -43
- package/conductor/product-guidelines.md +0 -43
- package/conductor/product.md +0 -3
- package/conductor/setup_state.json +0 -1
- package/conductor/tech-stack.md +0 -39
- package/conductor/tracks/enhance_postgres_monitoring_20251227/metadata.json +0 -8
- package/conductor/tracks/enhance_postgres_monitoring_20251227/plan.md +0 -44
- package/conductor/tracks/enhance_postgres_monitoring_20251227/spec.md +0 -31
- package/conductor/tracks.md +0 -8
- package/conductor/workflow.md +0 -333
- package/database-compose.yml +0 -55
- package/docker/postgres-init/01-extensions.sql +0 -10
- package/docker/postgres-init/02-sample-data.sql +0 -585
- package/docker/postgres.yml +0 -68
- package/docker-compose.yml +0 -38
- package/docs/AI_PLAN.md +0 -74
- package/docs/API_DOCS.md +0 -875
- package/docs/ARCHITECTURE.md +0 -218
- package/docs/DATABASE_PROVIDERS.md +0 -358
- package/docs/FEATURES.md +0 -116
- package/docs/HELM_CHART.md +0 -252
- package/docs/LOGIN_PAGE.md +0 -178
- package/docs/MONACO_EDITOR_PERFORMANCE.md +0 -315
- package/docs/OIDC_ARCH.md +0 -681
- package/docs/OIDC_SETUP.md +0 -322
- package/docs/POSTGRES_METRICS.md +0 -516
- package/docs/QUERY_OPTIMIZATION.md +0 -370
- package/docs/SEED_CONNECTIONS.md +0 -468
- package/docs/SQL_ALIAS_COMPLETION.md +0 -190
- package/docs/STORAGE_ARCHITECTURE.md +0 -565
- package/docs/STORAGE_QUICK_SETUP.md +0 -419
- package/docs/TECHNICAL_PLAN.md +0 -36
- package/docs/THEMING.md +0 -345
- package/docs/adding-a-new-database-provider.md +0 -642
- package/docs/backlogs/000-PLATFORM_DATA_SYNC_DATABASE.md +0 -360
- package/docs/backlogs/001-INLINE_DATA_EDITING.md +0 -118
- package/docs/backlogs/002-DATA_IMPORT.md +0 -215
- package/docs/backlogs/003-QUERY_TIME_MACHINE.md +0 -183
- package/docs/backlogs/004-AI_DATA_STORYTELLER.md +0 -292
- package/docs/backlogs/005-QUERY_PLAYGROUND.md +0 -352
- package/docs/backlogs/006-DATA_MASKING.md +0 -418
- package/docs/enterprise-features.md +0 -718
- package/docs/kubernetes-helm-chart-artifacthub-plan.md +0 -803
- package/docs/medium-koyeb-article-en.md +0 -215
- package/docs/plans/test-plans.md +0 -445
- package/docs/releases/RELEASE.V0.3.0.md +0 -22
- package/docs/releases/RELEASE.V0.4.0.md +0 -154
- package/docs/releases/RELEASE.V0.5.0.md +0 -252
- package/docs/releases/RELEASE_v0.5.6.md +0 -145
- package/docs/releases/RELEASE_v0.6.1.md +0 -303
- package/docs/releases/RELEASE_v0.6.7.md +0 -292
- package/docs/releases/RELEASE_v0.7.0.md +0 -332
- package/docs/releases/RELEASE_v0.8.0.md +0 -521
- package/docs/sampledb/titanic.sql +0 -1379
- package/docs/superpowers/plans/2026-03-25-seed-connections.md +0 -1362
- package/docs/superpowers/specs/2026-03-25-seed-connections-design.md +0 -590
- package/e2e/admin-dashboard.spec.ts +0 -64
- package/e2e/connection-management.spec.ts +0 -58
- package/e2e/export.spec.ts +0 -34
- package/e2e/login.spec.ts +0 -85
- package/e2e/query-execution.spec.ts +0 -35
- package/e2e/tab-management.spec.ts +0 -64
- package/eslint.config.mjs +0 -28
- package/fly.toml +0 -43
- package/next.config.ts +0 -32
- package/playwright.config.ts +0 -34
- package/postcss.config.mjs +0 -7
- package/public/favicon-32x32.png +0 -0
- package/public/favicon.ico +0 -0
- package/public/file.svg +0 -1
- package/public/globe.svg +0 -1
- package/public/logo.svg +0 -32
- package/public/next.svg +0 -1
- package/public/screenshots/code-generator.png +0 -0
- package/public/screenshots/connection-modal.png +0 -0
- package/public/screenshots/data-profiler.png +0 -0
- package/public/screenshots/erd-diagram.png +0 -0
- package/public/screenshots/hero-editor.png +0 -0
- package/public/screenshots/nl2sql.png +0 -0
- package/public/vercel.svg +0 -1
- package/public/window.svg +0 -1
- package/render.yaml +0 -58
- package/scripts/merge-lcov.mjs +0 -239
- package/sonar-project.properties +0 -16
- package/src/app/admin/error.tsx +0 -46
- package/src/app/admin/page.tsx +0 -10
- package/src/app/api/admin/audit/route.ts +0 -52
- package/src/app/api/admin/fleet-health/route.ts +0 -81
- package/src/app/api/ai/autopilot/route.ts +0 -105
- package/src/app/api/ai/chat/route.ts +0 -132
- package/src/app/api/ai/describe-schema/route.ts +0 -52
- package/src/app/api/ai/explain/route.ts +0 -86
- package/src/app/api/ai/impact/route.ts +0 -97
- package/src/app/api/ai/index-advisor/route.ts +0 -98
- package/src/app/api/ai/nl2sql/route.ts +0 -87
- package/src/app/api/ai/query-safety/route.ts +0 -87
- package/src/app/api/auth/login/route.ts +0 -62
- package/src/app/api/auth/logout/route.ts +0 -25
- package/src/app/api/auth/me/route.ts +0 -10
- package/src/app/api/auth/oidc/callback/route.ts +0 -82
- package/src/app/api/auth/oidc/login/route.ts +0 -43
- package/src/app/api/connections/managed/route.ts +0 -35
- package/src/app/api/db/cancel/route.ts +0 -42
- package/src/app/api/db/disconnect/route.ts +0 -28
- package/src/app/api/db/health/route.ts +0 -49
- package/src/app/api/db/maintenance/route.ts +0 -72
- package/src/app/api/db/monitoring/route.ts +0 -62
- package/src/app/api/db/multi-query/route.ts +0 -116
- package/src/app/api/db/pool-stats/route.ts +0 -37
- package/src/app/api/db/profile/route.ts +0 -144
- package/src/app/api/db/provider-meta/route.ts +0 -49
- package/src/app/api/db/query/route.ts +0 -50
- package/src/app/api/db/schema/route.ts +0 -47
- package/src/app/api/db/schema-snapshot/route.ts +0 -42
- package/src/app/api/db/test-connection/route.ts +0 -55
- package/src/app/api/db/transaction/route.ts +0 -111
- package/src/app/api/storage/[collection]/route.ts +0 -67
- package/src/app/api/storage/config/route.ts +0 -17
- package/src/app/api/storage/migrate/route.ts +0 -45
- package/src/app/api/storage/route.ts +0 -32
- package/src/app/error.tsx +0 -49
- package/src/app/global-error.tsx +0 -55
- package/src/app/globals.css +0 -146
- package/src/app/icon.svg +0 -42
- package/src/app/layout.tsx +0 -34
- package/src/app/login/login-form.tsx +0 -301
- package/src/app/login/page.tsx +0 -11
- package/src/app/monitoring/page.tsx +0 -8
- package/src/app/not-found.tsx +0 -29
- package/src/app/page.tsx +0 -5
- package/src/components/AIAutopilotPanel.tsx +0 -238
- package/src/components/CodeGenerator.tsx +0 -271
- package/src/components/CommandPalette.tsx +0 -227
- package/src/components/ConnectionModal.tsx +0 -759
- package/src/components/CreateTableModal.tsx +0 -281
- package/src/components/DataCharts.tsx +0 -962
- package/src/components/DataImportModal.tsx +0 -582
- package/src/components/DataProfiler.tsx +0 -335
- package/src/components/DatabaseDocs.tsx +0 -251
- package/src/components/MaskingSettings.tsx +0 -414
- package/src/components/MobileNav.tsx +0 -50
- package/src/components/NL2SQLPanel.tsx +0 -281
- package/src/components/PivotTable.tsx +0 -257
- package/src/components/QueryEditor.tsx +0 -760
- package/src/components/QueryHistory.tsx +0 -344
- package/src/components/QuerySafetyDialog.tsx +0 -290
- package/src/components/ResultsGrid.tsx +0 -644
- package/src/components/SaveQueryModal.tsx +0 -104
- package/src/components/SavedQueries.tsx +0 -128
- package/src/components/SchemaDiagram.tsx +0 -473
- package/src/components/SchemaDiff.tsx +0 -473
- package/src/components/SnapshotTimeline.tsx +0 -116
- package/src/components/Studio.tsx +0 -639
- package/src/components/TestDataGenerator.tsx +0 -261
- package/src/components/VisualExplain.tsx +0 -820
- package/src/components/admin/AdminDashboard.tsx +0 -163
- package/src/components/admin/tabs/AuditTab.tsx +0 -531
- package/src/components/admin/tabs/MonitoringEmbed.tsx +0 -11
- package/src/components/admin/tabs/OperationsTab.tsx +0 -646
- package/src/components/admin/tabs/OverviewTab.tsx +0 -1328
- package/src/components/admin/tabs/SecurityTab.tsx +0 -284
- package/src/components/community-section.tsx +0 -92
- package/src/components/icons/db-icons.tsx +0 -84
- package/src/components/libredb-logo.tsx +0 -61
- package/src/components/monitoring/MonitoringDashboard.tsx +0 -345
- package/src/components/monitoring/tabs/MetricChart.tsx +0 -82
- package/src/components/monitoring/tabs/OverviewTab.tsx +0 -263
- package/src/components/monitoring/tabs/PerformanceTab.tsx +0 -254
- package/src/components/monitoring/tabs/PoolTab.tsx +0 -174
- package/src/components/monitoring/tabs/QueriesTab.tsx +0 -287
- package/src/components/monitoring/tabs/SessionsTab.tsx +0 -316
- package/src/components/monitoring/tabs/StorageTab.tsx +0 -335
- package/src/components/monitoring/tabs/TablesTab.tsx +0 -300
- package/src/components/results-grid/ResultCard.tsx +0 -111
- package/src/components/results-grid/RowDetailSheet.tsx +0 -178
- package/src/components/results-grid/StatsBar.tsx +0 -201
- package/src/components/results-grid/index.ts +0 -1
- package/src/components/results-grid/utils.ts +0 -23
- package/src/components/schema-explorer/ColumnList.tsx +0 -53
- package/src/components/schema-explorer/SchemaExplorer.tsx +0 -182
- package/src/components/schema-explorer/TableItem.tsx +0 -210
- package/src/components/schema-explorer/index.ts +0 -1
- package/src/components/sidebar/ConnectionItem.tsx +0 -105
- package/src/components/sidebar/ConnectionsList.tsx +0 -62
- package/src/components/sidebar/Sidebar.tsx +0 -130
- package/src/components/sidebar/index.ts +0 -2
- package/src/components/studio/BottomPanel.tsx +0 -286
- package/src/components/studio/QueryToolbar.tsx +0 -180
- package/src/components/studio/StudioDesktopHeader.tsx +0 -114
- package/src/components/studio/StudioMobileHeader.tsx +0 -340
- package/src/components/studio/StudioTabBar.tsx +0 -82
- package/src/components/studio/index.ts +0 -5
- package/src/components/ui/accordion.tsx +0 -66
- package/src/components/ui/alert-dialog.tsx +0 -157
- package/src/components/ui/alert.tsx +0 -66
- package/src/components/ui/aspect-ratio.tsx +0 -11
- package/src/components/ui/avatar.tsx +0 -53
- package/src/components/ui/badge.tsx +0 -46
- package/src/components/ui/breadcrumb.tsx +0 -109
- package/src/components/ui/button-group.tsx +0 -83
- package/src/components/ui/button.tsx +0 -60
- package/src/components/ui/calendar.tsx +0 -216
- package/src/components/ui/card.tsx +0 -92
- package/src/components/ui/carousel.tsx +0 -241
- package/src/components/ui/chart.tsx +0 -357
- package/src/components/ui/checkbox.tsx +0 -32
- package/src/components/ui/collapsible.tsx +0 -33
- package/src/components/ui/command.tsx +0 -184
- package/src/components/ui/context-menu.tsx +0 -252
- package/src/components/ui/dialog.tsx +0 -143
- package/src/components/ui/drawer.tsx +0 -135
- package/src/components/ui/dropdown-menu.tsx +0 -257
- package/src/components/ui/empty.tsx +0 -104
- package/src/components/ui/field.tsx +0 -248
- package/src/components/ui/form.tsx +0 -167
- package/src/components/ui/hover-card.tsx +0 -44
- package/src/components/ui/input-group.tsx +0 -170
- package/src/components/ui/input-otp.tsx +0 -77
- package/src/components/ui/input.tsx +0 -21
- package/src/components/ui/item.tsx +0 -193
- package/src/components/ui/kbd.tsx +0 -28
- package/src/components/ui/label.tsx +0 -24
- package/src/components/ui/menubar.tsx +0 -276
- package/src/components/ui/navigation-menu.tsx +0 -168
- package/src/components/ui/pagination.tsx +0 -127
- package/src/components/ui/popover.tsx +0 -48
- package/src/components/ui/progress.tsx +0 -31
- package/src/components/ui/radio-group.tsx +0 -45
- package/src/components/ui/resizable.tsx +0 -56
- package/src/components/ui/scroll-area.tsx +0 -58
- package/src/components/ui/select.tsx +0 -187
- package/src/components/ui/separator.tsx +0 -28
- package/src/components/ui/sheet.tsx +0 -139
- package/src/components/ui/sidebar.tsx +0 -726
- package/src/components/ui/skeleton.tsx +0 -13
- package/src/components/ui/slider.tsx +0 -63
- package/src/components/ui/sonner.tsx +0 -40
- package/src/components/ui/spinner.tsx +0 -16
- package/src/components/ui/switch.tsx +0 -31
- package/src/components/ui/table.tsx +0 -116
- package/src/components/ui/tabs.tsx +0 -66
- package/src/components/ui/textarea.tsx +0 -18
- package/src/components/ui/toggle-group.tsx +0 -83
- package/src/components/ui/toggle.tsx +0 -47
- package/src/components/ui/tooltip.tsx +0 -61
- package/src/exports/components.ts +0 -15
- package/src/exports/index.ts +0 -4
- package/src/exports/providers.ts +0 -4
- package/src/exports/types.ts +0 -26
- package/src/hooks/use-ai-chat.ts +0 -182
- package/src/hooks/use-all-connections.ts +0 -66
- package/src/hooks/use-api-call.ts +0 -71
- package/src/hooks/use-auth.ts +0 -51
- package/src/hooks/use-connection-form.ts +0 -349
- package/src/hooks/use-connection-manager.ts +0 -169
- package/src/hooks/use-connection-payload.ts +0 -15
- package/src/hooks/use-inline-editing.ts +0 -109
- package/src/hooks/use-mobile.ts +0 -20
- package/src/hooks/use-monitoring-data.ts +0 -270
- package/src/hooks/use-provider-metadata.ts +0 -62
- package/src/hooks/use-query-execution.ts +0 -478
- package/src/hooks/use-storage-sync.ts +0 -259
- package/src/hooks/use-tab-manager.ts +0 -231
- package/src/hooks/use-toast.ts +0 -20
- package/src/hooks/use-transaction-control.ts +0 -64
- package/src/lib/api/error-codes.ts +0 -30
- package/src/lib/api/errors.ts +0 -236
- package/src/lib/api/with-error-handler.ts +0 -41
- package/src/lib/audit.ts +0 -105
- package/src/lib/auth.ts +0 -87
- package/src/lib/connection-string-parser.ts +0 -172
- package/src/lib/data-masking.ts +0 -385
- package/src/lib/db/base-provider.ts +0 -325
- package/src/lib/db/errors.ts +0 -317
- package/src/lib/db/factory.ts +0 -324
- package/src/lib/db/index.ts +0 -123
- package/src/lib/db/providers/document/index.ts +0 -6
- package/src/lib/db/providers/document/mongodb.ts +0 -992
- package/src/lib/db/providers/keyvalue/redis.ts +0 -554
- package/src/lib/db/providers/sql/index.ts +0 -11
- package/src/lib/db/providers/sql/sql-base.ts +0 -174
- package/src/lib/db/providers/sql/sqlite.ts +0 -721
- package/src/lib/db/types.ts +0 -437
- package/src/lib/db/utils/pool-manager.ts +0 -287
- package/src/lib/db/utils/query-limiter.ts +0 -239
- package/src/lib/db-ui-config.ts +0 -86
- package/src/lib/editor/mongodb-completions.ts +0 -172
- package/src/lib/editor/sql-completions.ts +0 -280
- package/src/lib/llm/base-provider.ts +0 -117
- package/src/lib/llm/factory.ts +0 -102
- package/src/lib/llm/index.ts +0 -90
- package/src/lib/llm/providers/custom.ts +0 -181
- package/src/lib/llm/providers/gemini.ts +0 -126
- package/src/lib/llm/providers/ollama.ts +0 -154
- package/src/lib/llm/providers/openai.ts +0 -146
- package/src/lib/llm/types.ts +0 -173
- package/src/lib/llm/utils/config.ts +0 -187
- package/src/lib/llm/utils/retry.ts +0 -119
- package/src/lib/llm/utils/streaming.ts +0 -202
- package/src/lib/logger.ts +0 -127
- package/src/lib/monitoring-thresholds.ts +0 -44
- package/src/lib/oidc.ts +0 -262
- package/src/lib/query-generators.ts +0 -61
- package/src/lib/schema-diff/diff-engine.ts +0 -273
- package/src/lib/schema-diff/migration-generator.ts +0 -208
- package/src/lib/schema-diff/types.ts +0 -55
- package/src/lib/seed/config-loader.ts +0 -79
- package/src/lib/seed/connection-filter.ts +0 -49
- package/src/lib/seed/credential-resolver.ts +0 -62
- package/src/lib/seed/index.ts +0 -40
- package/src/lib/seed/resolve-connection.ts +0 -57
- package/src/lib/seed/types.ts +0 -69
- package/src/lib/sql/alias-extractor.ts +0 -267
- package/src/lib/sql/index.ts +0 -8
- package/src/lib/sql/statement-splitter.ts +0 -167
- package/src/lib/sql/types.ts +0 -40
- package/src/lib/ssh/tunnel.ts +0 -142
- package/src/lib/storage/factory.ts +0 -84
- package/src/lib/storage/index.ts +0 -14
- package/src/lib/storage/local-storage.ts +0 -99
- package/src/lib/storage/providers/postgres.ts +0 -225
- package/src/lib/storage/providers/sqlite.ts +0 -153
- package/src/lib/storage/storage-facade.ts +0 -272
- package/src/lib/storage/types.ts +0 -75
- package/src/lib/time-series-buffer.ts +0 -58
- package/src/lib/types.ts +0 -173
- package/src/lib/utils.ts +0 -6
- package/src/proxy.ts +0 -104
- package/src/types/db-drivers.d.ts +0 -23
- package/src/types/html2canvas.d.ts +0 -9
- package/tests/api/admin/audit.test.ts +0 -178
- package/tests/api/admin/fleet-health.test.ts +0 -183
- package/tests/api/ai/autopilot.test.ts +0 -174
- package/tests/api/ai/chat.test.ts +0 -250
- package/tests/api/ai/describe-schema.test.ts +0 -266
- package/tests/api/ai/explain.test.ts +0 -199
- package/tests/api/ai/impact.test.ts +0 -168
- package/tests/api/ai/index-advisor.test.ts +0 -171
- package/tests/api/ai/nl2sql.test.ts +0 -202
- package/tests/api/ai/query-safety.test.ts +0 -196
- package/tests/api/auth/login.test.ts +0 -170
- package/tests/api/auth/logout.test.ts +0 -140
- package/tests/api/auth/me.test.ts +0 -73
- package/tests/api/auth/oidc-callback.test.ts +0 -215
- package/tests/api/auth/oidc-login.test.ts +0 -127
- package/tests/api/db/cancel.test.ts +0 -198
- package/tests/api/db/disconnect.test.ts +0 -124
- package/tests/api/db/health.test.ts +0 -222
- package/tests/api/db/maintenance.test.ts +0 -263
- package/tests/api/db/monitoring.test.ts +0 -221
- package/tests/api/db/multi-query.test.ts +0 -316
- package/tests/api/db/pool-stats.test.ts +0 -135
- package/tests/api/db/profile.test.ts +0 -330
- package/tests/api/db/provider-meta.test.ts +0 -193
- package/tests/api/db/query.test.ts +0 -314
- package/tests/api/db/schema-snapshot.test.ts +0 -170
- package/tests/api/db/schema.test.ts +0 -191
- package/tests/api/db/test-connection.test.ts +0 -185
- package/tests/api/db/transaction.test.ts +0 -314
- package/tests/api/proxy.test.ts +0 -191
- package/tests/api/seed/managed-route.test.ts +0 -113
- package/tests/api/storage/config.test.ts +0 -42
- package/tests/api/storage/storage-routes.test.ts +0 -309
- package/tests/components/AIAutopilotPanel.test.tsx +0 -756
- package/tests/components/AdminPage.test.tsx +0 -33
- package/tests/components/CodeGenerator.test.tsx +0 -182
- package/tests/components/CommandPalette.test.tsx +0 -428
- package/tests/components/CommunitySection.test.tsx +0 -91
- package/tests/components/ConnectionModal.mobile.test.tsx +0 -284
- package/tests/components/ConnectionModal.test.tsx +0 -570
- package/tests/components/CreateTableModal.test.tsx +0 -383
- package/tests/components/DataCharts.test.tsx +0 -739
- package/tests/components/DataImportModal.test.tsx +0 -751
- package/tests/components/DataProfiler.test.tsx +0 -589
- package/tests/components/DatabaseDocs.test.tsx +0 -353
- package/tests/components/LoginPage.test.tsx +0 -163
- package/tests/components/LoginPageOIDC.test.tsx +0 -92
- package/tests/components/MaskingSettings.test.tsx +0 -498
- package/tests/components/MobileNav.test.tsx +0 -30
- package/tests/components/MonitoringPage.test.tsx +0 -32
- package/tests/components/NL2SQLPanel.test.tsx +0 -621
- package/tests/components/Page.test.tsx +0 -33
- package/tests/components/PivotTable.test.tsx +0 -350
- package/tests/components/QueryEditor.test.tsx +0 -1730
- package/tests/components/QueryHistory.test.tsx +0 -572
- package/tests/components/QuerySafetyDialog.test.tsx +0 -586
- package/tests/components/ResultsGrid.test.tsx +0 -804
- package/tests/components/RootLayout.test.tsx +0 -83
- package/tests/components/SaveQueryModal.test.tsx +0 -25
- package/tests/components/SavedQueries.test.tsx +0 -43
- package/tests/components/SchemaDiagram.test.tsx +0 -1034
- package/tests/components/SchemaDiff.test.tsx +0 -906
- package/tests/components/SnapshotTimeline.test.tsx +0 -174
- package/tests/components/Studio.test.tsx +0 -1030
- package/tests/components/TestDataGenerator.test.tsx +0 -291
- package/tests/components/VisualExplain.test.tsx +0 -704
- package/tests/components/admin/AdminDashboard.test.tsx +0 -205
- package/tests/components/admin/AuditTab.test.tsx +0 -220
- package/tests/components/admin/MonitoringEmbed.test.tsx +0 -58
- package/tests/components/admin/OperationsTab.test.tsx +0 -975
- package/tests/components/admin/OverviewTab.test.tsx +0 -254
- package/tests/components/admin/SecurityTab.test.tsx +0 -467
- package/tests/components/monitoring/MetricChart.test.tsx +0 -111
- package/tests/components/monitoring/MonitoringDashboard.test.tsx +0 -259
- package/tests/components/monitoring/OverviewTab.test.tsx +0 -78
- package/tests/components/monitoring/PerformanceTab.test.tsx +0 -87
- package/tests/components/monitoring/PoolTab.test.tsx +0 -42
- package/tests/components/monitoring/QueriesTab.test.tsx +0 -80
- package/tests/components/monitoring/SessionsTab.test.tsx +0 -154
- package/tests/components/monitoring/StorageTab.test.tsx +0 -127
- package/tests/components/monitoring/TablesTab.test.tsx +0 -153
- package/tests/components/results-grid/ResultCard.test.tsx +0 -105
- package/tests/components/results-grid/RowDetailSheet.test.tsx +0 -308
- package/tests/components/results-grid/StatsBar.test.tsx +0 -162
- package/tests/components/schema-explorer/ColumnList.test.tsx +0 -151
- package/tests/components/schema-explorer/SchemaExplorer.test.tsx +0 -461
- package/tests/components/schema-explorer/TableItem.test.tsx +0 -415
- package/tests/components/sidebar/ConnectionItem.test.tsx +0 -201
- package/tests/components/sidebar/ConnectionsList.test.tsx +0 -176
- package/tests/components/sidebar/Sidebar.test.tsx +0 -187
- package/tests/components/studio/BottomPanel.test.tsx +0 -383
- package/tests/components/studio/QueryToolbar.test.tsx +0 -321
- package/tests/components/studio/StudioDesktopHeader.test.tsx +0 -377
- package/tests/components/studio/StudioMobileHeader.test.tsx +0 -198
- package/tests/components/studio/StudioTabBar.test.tsx +0 -331
- package/tests/fixtures/connections.ts +0 -96
- package/tests/fixtures/masking-configs.ts +0 -86
- package/tests/fixtures/query-results.ts +0 -71
- package/tests/fixtures/schemas.ts +0 -64
- package/tests/fixtures/seed-connections/invalid-config.yaml +0 -7
- package/tests/fixtures/seed-connections/minimal-config.yaml +0 -8
- package/tests/fixtures/seed-connections/mixed-credentials.yaml +0 -23
- package/tests/fixtures/seed-connections/multi-role-config.yaml +0 -30
- package/tests/fixtures/seed-connections/valid-config.json +0 -15
- package/tests/fixtures/seed-connections/valid-config.yaml +0 -51
- package/tests/helpers/mock-fetch.ts +0 -59
- package/tests/helpers/mock-monaco.ts +0 -112
- package/tests/helpers/mock-navigation.ts +0 -28
- package/tests/helpers/mock-next.ts +0 -80
- package/tests/helpers/mock-provider.ts +0 -133
- package/tests/helpers/mock-sonner.ts +0 -29
- package/tests/helpers/render-with-providers.tsx +0 -19
- package/tests/hooks/use-ai-chat.test.ts +0 -600
- package/tests/hooks/use-auth.test.ts +0 -371
- package/tests/hooks/use-connection-form.test.ts +0 -743
- package/tests/hooks/use-connection-manager.test.ts +0 -466
- package/tests/hooks/use-inline-editing.test.ts +0 -321
- package/tests/hooks/use-mobile.test.ts +0 -177
- package/tests/hooks/use-monitoring-data.test.ts +0 -819
- package/tests/hooks/use-provider-metadata.test.ts +0 -228
- package/tests/hooks/use-query-execution.test.ts +0 -1212
- package/tests/hooks/use-tab-manager.test.ts +0 -756
- package/tests/hooks/use-toast.test.ts +0 -74
- package/tests/hooks/use-transaction-control.test.ts +0 -211
- package/tests/integration/db/mongodb-provider.test.ts +0 -698
- package/tests/integration/db/mssql-provider.test.ts +0 -840
- package/tests/integration/db/mysql-provider.test.ts +0 -872
- package/tests/integration/db/oracle-provider.test.ts +0 -843
- package/tests/integration/db/postgres-provider.test.ts +0 -1382
- package/tests/integration/db/redis-provider.test.ts +0 -526
- package/tests/integration/db/sqlite-provider.test.ts +0 -480
- package/tests/integration/seed/seed-pipeline.test.ts +0 -102
- package/tests/isolated/factory-singleton.test.ts +0 -150
- package/tests/isolated/use-storage-sync.test.ts +0 -389
- package/tests/run-components.sh +0 -196
- package/tests/setup-dom.ts +0 -58
- package/tests/setup.ts +0 -40
- package/tests/unit/api-errors.test.ts +0 -210
- package/tests/unit/code-generator-functions.test.ts +0 -271
- package/tests/unit/components/column-list.test.tsx +0 -190
- package/tests/unit/components/data-import-modal.test.tsx +0 -441
- package/tests/unit/components/studio-mobile-header.test.tsx +0 -327
- package/tests/unit/data-charts-functions.test.ts +0 -496
- package/tests/unit/data-import-functions.test.ts +0 -320
- package/tests/unit/data-import-utils.test.ts +0 -125
- package/tests/unit/db/base-provider.test.ts +0 -517
- package/tests/unit/db/errors.test.ts +0 -403
- package/tests/unit/db/factory.test.ts +0 -436
- package/tests/unit/db/pool-manager.test.ts +0 -440
- package/tests/unit/db/query-limiter.test.ts +0 -387
- package/tests/unit/db/sql-base.test.ts +0 -438
- package/tests/unit/lib/api/error-codes.test.ts +0 -39
- package/tests/unit/lib/audit.test.ts +0 -326
- package/tests/unit/lib/auth.test.ts +0 -146
- package/tests/unit/lib/connection-string-parser.test.ts +0 -424
- package/tests/unit/lib/data-masking.test.ts +0 -583
- package/tests/unit/lib/db-icons.test.tsx +0 -41
- package/tests/unit/lib/monitoring-thresholds.test.ts +0 -133
- package/tests/unit/lib/oidc.test.ts +0 -509
- package/tests/unit/lib/query-generators.test.ts +0 -127
- package/tests/unit/lib/storage/factory.test.ts +0 -71
- package/tests/unit/lib/storage/local-storage.test.ts +0 -114
- package/tests/unit/lib/storage/providers/postgres.test.ts +0 -312
- package/tests/unit/lib/storage/providers/sqlite.test.ts +0 -232
- package/tests/unit/lib/storage/storage-facade-extended.test.ts +0 -331
- package/tests/unit/lib/storage/storage-facade.test.ts +0 -184
- package/tests/unit/lib/storage.test.ts +0 -317
- package/tests/unit/lib/time-series-buffer.test.ts +0 -212
- package/tests/unit/lib/utils.test.ts +0 -24
- package/tests/unit/llm/base-provider.test.ts +0 -238
- package/tests/unit/llm/config.test.ts +0 -262
- package/tests/unit/llm/custom-provider.test.ts +0 -281
- package/tests/unit/llm/gemini-provider.test.ts +0 -248
- package/tests/unit/llm/llm-factory.test.ts +0 -155
- package/tests/unit/llm/ollama-provider.test.ts +0 -288
- package/tests/unit/llm/openai-provider.test.ts +0 -324
- package/tests/unit/llm/retry.test.ts +0 -180
- package/tests/unit/llm/streaming.test.ts +0 -355
- package/tests/unit/logger.test.ts +0 -198
- package/tests/unit/mongodb-completions.test.ts +0 -516
- package/tests/unit/pivot-table-functions.test.ts +0 -76
- package/tests/unit/query-cancelled-error.test.ts +0 -81
- package/tests/unit/schema-diff/diff-engine.test.ts +0 -367
- package/tests/unit/schema-diff/migration-generator.test.ts +0 -513
- package/tests/unit/seed/config-loader.test.ts +0 -73
- package/tests/unit/seed/connection-filter.test.ts +0 -91
- package/tests/unit/seed/credential-resolver.test.ts +0 -85
- package/tests/unit/seed/index.test.ts +0 -72
- package/tests/unit/seed/resolve-connection.test.ts +0 -74
- package/tests/unit/seed/types.test.ts +0 -129
- package/tests/unit/sql/alias-extractor.test.ts +0 -444
- package/tests/unit/sql/statement-splitter.test.ts +0 -348
- package/tests/unit/sql-completions.test.ts +0 -463
- package/tests/unit/ssh-tunnel.test.ts +0 -465
- package/tsconfig.json +0 -42
|
@@ -1,704 +0,0 @@
|
|
|
1
|
-
import '../setup-dom';
|
|
2
|
-
import '../helpers/mock-sonner';
|
|
3
|
-
import '../helpers/mock-navigation';
|
|
4
|
-
|
|
5
|
-
import React from 'react';
|
|
6
|
-
import { afterEach, beforeEach, describe, expect, mock, test } from 'bun:test';
|
|
7
|
-
import { cleanup, fireEvent, render, waitFor } from '@testing-library/react';
|
|
8
|
-
import userEvent from '@testing-library/user-event';
|
|
9
|
-
import { VisualExplain, type ExplainPlanResult } from '@/components/VisualExplain';
|
|
10
|
-
|
|
11
|
-
let originalFetch: typeof globalThis.fetch;
|
|
12
|
-
|
|
13
|
-
function mockFetchStream(body: string, ok = true, errorBody?: { error: string }) {
|
|
14
|
-
const encoder = new TextEncoder();
|
|
15
|
-
const stream = new ReadableStream({
|
|
16
|
-
start(controller) {
|
|
17
|
-
controller.enqueue(encoder.encode(body));
|
|
18
|
-
controller.close();
|
|
19
|
-
},
|
|
20
|
-
});
|
|
21
|
-
return mock(() =>
|
|
22
|
-
Promise.resolve({
|
|
23
|
-
ok,
|
|
24
|
-
body: ok ? stream : null,
|
|
25
|
-
json: () => Promise.resolve(errorBody || {}),
|
|
26
|
-
})
|
|
27
|
-
);
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
// Sample plan with Seq Scan on large table + row estimate mismatch
|
|
31
|
-
const samplePlan: ExplainPlanResult[] = [
|
|
32
|
-
{
|
|
33
|
-
Plan: {
|
|
34
|
-
'Node Type': 'Seq Scan',
|
|
35
|
-
'Relation Name': 'users',
|
|
36
|
-
'Actual Rows': 15000,
|
|
37
|
-
'Plan Rows': 100,
|
|
38
|
-
'Actual Total Time': 42.5,
|
|
39
|
-
'Total Cost': 120,
|
|
40
|
-
'Shared Hit Blocks': 80,
|
|
41
|
-
'Shared Read Blocks': 5,
|
|
42
|
-
Filter: 'status = active',
|
|
43
|
-
Plans: [
|
|
44
|
-
{
|
|
45
|
-
'Node Type': 'Index Scan',
|
|
46
|
-
'Relation Name': 'orders',
|
|
47
|
-
'Actual Rows': 200,
|
|
48
|
-
'Plan Rows': 200,
|
|
49
|
-
'Actual Total Time': 1.2,
|
|
50
|
-
'Total Cost': 10,
|
|
51
|
-
'Index Name': 'idx_orders_user_id',
|
|
52
|
-
},
|
|
53
|
-
],
|
|
54
|
-
},
|
|
55
|
-
'Execution Time': 42.5,
|
|
56
|
-
'Planning Time': 0.8,
|
|
57
|
-
},
|
|
58
|
-
];
|
|
59
|
-
|
|
60
|
-
// Healthy plan — no warnings
|
|
61
|
-
const healthyPlan: ExplainPlanResult[] = [
|
|
62
|
-
{
|
|
63
|
-
Plan: {
|
|
64
|
-
'Node Type': 'Index Scan',
|
|
65
|
-
'Actual Rows': 10,
|
|
66
|
-
'Plan Rows': 10,
|
|
67
|
-
'Actual Total Time': 0.5,
|
|
68
|
-
'Total Cost': 5,
|
|
69
|
-
'Index Name': 'idx_pk',
|
|
70
|
-
},
|
|
71
|
-
'Execution Time': 0.5,
|
|
72
|
-
'Planning Time': 0.1,
|
|
73
|
-
},
|
|
74
|
-
];
|
|
75
|
-
|
|
76
|
-
// Plan with expensive sort
|
|
77
|
-
const sortPlan: ExplainPlanResult[] = [
|
|
78
|
-
{
|
|
79
|
-
Plan: {
|
|
80
|
-
'Node Type': 'Sort',
|
|
81
|
-
'Actual Rows': 5000,
|
|
82
|
-
'Plan Rows': 5000,
|
|
83
|
-
'Actual Total Time': 250,
|
|
84
|
-
'Total Cost': 300,
|
|
85
|
-
},
|
|
86
|
-
'Execution Time': 250,
|
|
87
|
-
'Planning Time': 0.5,
|
|
88
|
-
},
|
|
89
|
-
];
|
|
90
|
-
|
|
91
|
-
// Plan with high nested loop count
|
|
92
|
-
const nestedLoopPlan: ExplainPlanResult[] = [
|
|
93
|
-
{
|
|
94
|
-
Plan: {
|
|
95
|
-
'Node Type': 'Nested Loop',
|
|
96
|
-
'Actual Rows': 50000,
|
|
97
|
-
'Plan Rows': 100,
|
|
98
|
-
'Actual Total Time': 800,
|
|
99
|
-
'Total Cost': 900,
|
|
100
|
-
'Actual Loops': 5000,
|
|
101
|
-
},
|
|
102
|
-
'Execution Time': 800,
|
|
103
|
-
'Planning Time': 1.0,
|
|
104
|
-
},
|
|
105
|
-
];
|
|
106
|
-
|
|
107
|
-
describe('VisualExplain', () => {
|
|
108
|
-
beforeEach(() => {
|
|
109
|
-
originalFetch = globalThis.fetch;
|
|
110
|
-
globalThis.fetch = mock(() => Promise.resolve(new Response('', { status: 200 }))) as never;
|
|
111
|
-
});
|
|
112
|
-
|
|
113
|
-
afterEach(() => {
|
|
114
|
-
globalThis.fetch = originalFetch;
|
|
115
|
-
cleanup();
|
|
116
|
-
});
|
|
117
|
-
|
|
118
|
-
// -----------------------------------------------------------------------
|
|
119
|
-
// Empty state
|
|
120
|
-
// -----------------------------------------------------------------------
|
|
121
|
-
|
|
122
|
-
test('shows empty state when plan is null', () => {
|
|
123
|
-
const { queryByText } = render(<VisualExplain plan={null} />);
|
|
124
|
-
expect(queryByText('No execution plan')).not.toBeNull();
|
|
125
|
-
expect(queryByText(/Run a SELECT query/)).not.toBeNull();
|
|
126
|
-
});
|
|
127
|
-
|
|
128
|
-
test('shows empty state when plan is empty array', () => {
|
|
129
|
-
const { queryByText } = render(<VisualExplain plan={[]} />);
|
|
130
|
-
expect(queryByText('No execution plan')).not.toBeNull();
|
|
131
|
-
});
|
|
132
|
-
|
|
133
|
-
test('shows empty state when plan is undefined', () => {
|
|
134
|
-
const { queryByText } = render(<VisualExplain plan={undefined} />);
|
|
135
|
-
expect(queryByText('No execution plan')).not.toBeNull();
|
|
136
|
-
});
|
|
137
|
-
|
|
138
|
-
// -----------------------------------------------------------------------
|
|
139
|
-
// Header stats
|
|
140
|
-
// -----------------------------------------------------------------------
|
|
141
|
-
|
|
142
|
-
test('renders execution time, rows, and cost in header', () => {
|
|
143
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
144
|
-
expect(queryByText('execution')).not.toBeNull();
|
|
145
|
-
expect(queryByText('rows')).not.toBeNull();
|
|
146
|
-
expect(queryByText('cost')).not.toBeNull();
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
test('displays formatted execution time', () => {
|
|
150
|
-
const { queryAllByText } = render(<VisualExplain plan={samplePlan} />);
|
|
151
|
-
// 42.50ms appears in header + insights + plan node
|
|
152
|
-
expect(queryAllByText('42.50ms').length).toBeGreaterThan(0);
|
|
153
|
-
});
|
|
154
|
-
|
|
155
|
-
test('displays formatted row count', () => {
|
|
156
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
157
|
-
// 15000 + 200 = 15200 → "15.2K"
|
|
158
|
-
expect(queryByText('15.2K')).not.toBeNull();
|
|
159
|
-
});
|
|
160
|
-
|
|
161
|
-
// -----------------------------------------------------------------------
|
|
162
|
-
// Insights tab (default)
|
|
163
|
-
// -----------------------------------------------------------------------
|
|
164
|
-
|
|
165
|
-
test('shows Sequential Scan warning for large table', () => {
|
|
166
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
167
|
-
expect(queryByText('Performance Issues')).not.toBeNull();
|
|
168
|
-
expect(queryByText('Sequential Scan')).not.toBeNull();
|
|
169
|
-
expect(queryByText(/Full table scan on "users"/)).not.toBeNull();
|
|
170
|
-
});
|
|
171
|
-
|
|
172
|
-
test('shows Estimate Mismatch warning', () => {
|
|
173
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
174
|
-
expect(queryByText('Estimate Mismatch')).not.toBeNull();
|
|
175
|
-
expect(queryByText(/Statistics may be outdated/)).not.toBeNull();
|
|
176
|
-
});
|
|
177
|
-
|
|
178
|
-
test('shows "Query looks good" for healthy plan', () => {
|
|
179
|
-
const { queryByText } = render(<VisualExplain plan={healthyPlan} />);
|
|
180
|
-
expect(queryByText('Query looks good')).not.toBeNull();
|
|
181
|
-
expect(queryByText(/No obvious performance issues/)).not.toBeNull();
|
|
182
|
-
});
|
|
183
|
-
|
|
184
|
-
test('shows Expensive Sort warning', () => {
|
|
185
|
-
const { queryByText } = render(<VisualExplain plan={sortPlan} />);
|
|
186
|
-
expect(queryByText('Expensive Sort')).not.toBeNull();
|
|
187
|
-
expect(queryByText(/Sort operation took/)).not.toBeNull();
|
|
188
|
-
});
|
|
189
|
-
|
|
190
|
-
test('shows High Loop Count warning for nested loops', () => {
|
|
191
|
-
const { queryByText } = render(<VisualExplain plan={nestedLoopPlan} />);
|
|
192
|
-
expect(queryByText('High Loop Count')).not.toBeNull();
|
|
193
|
-
expect(queryByText(/N\+1 problem/)).not.toBeNull();
|
|
194
|
-
});
|
|
195
|
-
|
|
196
|
-
test('renders insight metrics (Cache Hit Rate, Operations, Execution)', () => {
|
|
197
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
198
|
-
expect(queryByText('Cache Hit Rate')).not.toBeNull();
|
|
199
|
-
expect(queryByText('Operations')).not.toBeNull();
|
|
200
|
-
expect(queryByText('Execution')).not.toBeNull();
|
|
201
|
-
});
|
|
202
|
-
|
|
203
|
-
test('calculates cache hit rate', () => {
|
|
204
|
-
// 80 hits / (80+5) = 94.1%
|
|
205
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
206
|
-
expect(queryByText('94.1%')).not.toBeNull();
|
|
207
|
-
});
|
|
208
|
-
|
|
209
|
-
test('shows "Execution Plan" section with plan tree', () => {
|
|
210
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
211
|
-
expect(queryByText('Execution Plan')).not.toBeNull();
|
|
212
|
-
expect(queryByText('Seq Scan')).not.toBeNull();
|
|
213
|
-
});
|
|
214
|
-
|
|
215
|
-
// -----------------------------------------------------------------------
|
|
216
|
-
// Tree tab
|
|
217
|
-
// -----------------------------------------------------------------------
|
|
218
|
-
|
|
219
|
-
test('switches to tree tab and shows plan nodes', () => {
|
|
220
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
221
|
-
fireEvent.click(queryByText('tree')!);
|
|
222
|
-
expect(queryByText('Seq Scan')).not.toBeNull();
|
|
223
|
-
expect(queryByText('users')).not.toBeNull();
|
|
224
|
-
});
|
|
225
|
-
|
|
226
|
-
test('tree shows filter info for nodes with filters', () => {
|
|
227
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
228
|
-
fireEvent.click(queryByText('tree')!);
|
|
229
|
-
expect(queryByText('status = active')).not.toBeNull();
|
|
230
|
-
});
|
|
231
|
-
|
|
232
|
-
test('tree shows index name for index scans', () => {
|
|
233
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
234
|
-
fireEvent.click(queryByText('tree')!);
|
|
235
|
-
expect(queryByText('idx_orders_user_id')).not.toBeNull();
|
|
236
|
-
});
|
|
237
|
-
|
|
238
|
-
test('tree shows buffer stats', () => {
|
|
239
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
240
|
-
fireEvent.click(queryByText('tree')!);
|
|
241
|
-
expect(queryByText(/Cache hits: 80/)).not.toBeNull();
|
|
242
|
-
expect(queryByText(/Disk reads: 5/)).not.toBeNull();
|
|
243
|
-
});
|
|
244
|
-
|
|
245
|
-
// -----------------------------------------------------------------------
|
|
246
|
-
// Raw tab
|
|
247
|
-
// -----------------------------------------------------------------------
|
|
248
|
-
|
|
249
|
-
test('switches to raw tab and shows JSON', () => {
|
|
250
|
-
const { container, queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
251
|
-
fireEvent.click(queryByText('raw')!);
|
|
252
|
-
expect(container.textContent).toContain('"Node Type"');
|
|
253
|
-
expect(container.textContent).toContain('"Seq Scan"');
|
|
254
|
-
});
|
|
255
|
-
|
|
256
|
-
// -----------------------------------------------------------------------
|
|
257
|
-
// AI Explain tab
|
|
258
|
-
// -----------------------------------------------------------------------
|
|
259
|
-
|
|
260
|
-
test('switches to AI tab and shows initial state', () => {
|
|
261
|
-
const { queryByText } = render(
|
|
262
|
-
<VisualExplain plan={samplePlan} query="SELECT * FROM users" />
|
|
263
|
-
);
|
|
264
|
-
fireEvent.click(queryByText('AI Explain')!);
|
|
265
|
-
expect(queryByText('AI Query Analysis')).not.toBeNull();
|
|
266
|
-
expect(queryByText('Analyze with AI')).not.toBeNull();
|
|
267
|
-
});
|
|
268
|
-
|
|
269
|
-
test('AI tab shows disabled state when no query', () => {
|
|
270
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
271
|
-
fireEvent.click(queryByText('AI Explain')!);
|
|
272
|
-
expect(queryByText(/Run a query first/)).not.toBeNull();
|
|
273
|
-
});
|
|
274
|
-
|
|
275
|
-
test('AI Analyze button calls /api/ai/explain', async () => {
|
|
276
|
-
const user = userEvent.setup();
|
|
277
|
-
globalThis.fetch = mockFetchStream('## Analysis\nThis query performs a seq scan.') as unknown as typeof fetch;
|
|
278
|
-
|
|
279
|
-
const { queryByText } = render(
|
|
280
|
-
<VisualExplain plan={samplePlan} query="SELECT * FROM users" databaseType="postgres" />
|
|
281
|
-
);
|
|
282
|
-
fireEvent.click(queryByText('AI Explain')!);
|
|
283
|
-
await user.click(queryByText('Analyze with AI')!);
|
|
284
|
-
|
|
285
|
-
await waitFor(() => {
|
|
286
|
-
expect(globalThis.fetch).toHaveBeenCalled();
|
|
287
|
-
});
|
|
288
|
-
|
|
289
|
-
const fetchCall = (globalThis.fetch as unknown as ReturnType<typeof mock>).mock.calls[0] as unknown as [string, RequestInit];
|
|
290
|
-
expect(fetchCall[0]).toBe('/api/ai/explain');
|
|
291
|
-
const body = JSON.parse(fetchCall[1].body as string);
|
|
292
|
-
expect(body.query).toBe('SELECT * FROM users');
|
|
293
|
-
expect(body.databaseType).toBe('postgres');
|
|
294
|
-
});
|
|
295
|
-
|
|
296
|
-
test('AI tab displays streamed response', async () => {
|
|
297
|
-
const user = userEvent.setup();
|
|
298
|
-
globalThis.fetch = mockFetchStream('## Performance\nThe query uses a sequential scan.') as unknown as typeof fetch;
|
|
299
|
-
|
|
300
|
-
const { queryByText } = render(
|
|
301
|
-
<VisualExplain plan={samplePlan} query="SELECT * FROM users" />
|
|
302
|
-
);
|
|
303
|
-
fireEvent.click(queryByText('AI Explain')!);
|
|
304
|
-
await user.click(queryByText('Analyze with AI')!);
|
|
305
|
-
|
|
306
|
-
await waitFor(() => {
|
|
307
|
-
expect(queryByText('Performance')).not.toBeNull();
|
|
308
|
-
expect(queryByText(/sequential scan/)).not.toBeNull();
|
|
309
|
-
});
|
|
310
|
-
});
|
|
311
|
-
|
|
312
|
-
test('AI tab shows error on API failure', async () => {
|
|
313
|
-
const user = userEvent.setup();
|
|
314
|
-
globalThis.fetch = mockFetchStream('', false, { error: 'Model unavailable' }) as unknown as typeof fetch;
|
|
315
|
-
|
|
316
|
-
const { queryByText } = render(
|
|
317
|
-
<VisualExplain plan={samplePlan} query="SELECT 1" />
|
|
318
|
-
);
|
|
319
|
-
fireEvent.click(queryByText('AI Explain')!);
|
|
320
|
-
await user.click(queryByText('Analyze with AI')!);
|
|
321
|
-
|
|
322
|
-
await waitFor(() => {
|
|
323
|
-
expect(queryByText('Model unavailable')).not.toBeNull();
|
|
324
|
-
});
|
|
325
|
-
});
|
|
326
|
-
|
|
327
|
-
test('AI tab shows Re-analyze button after first analysis', async () => {
|
|
328
|
-
const user = userEvent.setup();
|
|
329
|
-
globalThis.fetch = mockFetchStream('Analysis result') as unknown as typeof fetch;
|
|
330
|
-
|
|
331
|
-
const { queryByText } = render(
|
|
332
|
-
<VisualExplain plan={samplePlan} query="SELECT 1" />
|
|
333
|
-
);
|
|
334
|
-
fireEvent.click(queryByText('AI Explain')!);
|
|
335
|
-
await user.click(queryByText('Analyze with AI')!);
|
|
336
|
-
|
|
337
|
-
await waitFor(() => {
|
|
338
|
-
expect(queryByText('Re-analyze')).not.toBeNull();
|
|
339
|
-
});
|
|
340
|
-
});
|
|
341
|
-
|
|
342
|
-
test('AI tab renders code blocks with "Try This" button for SQL', async () => {
|
|
343
|
-
const user = userEvent.setup();
|
|
344
|
-
const onLoadQuery = mock(() => {});
|
|
345
|
-
globalThis.fetch = mockFetchStream('Try this:\n```sql\nSELECT id FROM users WHERE active;\n```\nDone.') as unknown as typeof fetch;
|
|
346
|
-
|
|
347
|
-
const { queryByText, container } = render(
|
|
348
|
-
<VisualExplain plan={samplePlan} query="SELECT * FROM users" onLoadQuery={onLoadQuery} />
|
|
349
|
-
);
|
|
350
|
-
fireEvent.click(queryByText('AI Explain')!);
|
|
351
|
-
await user.click(queryByText('Analyze with AI')!);
|
|
352
|
-
|
|
353
|
-
await waitFor(() => {
|
|
354
|
-
const pre = container.querySelector('pre');
|
|
355
|
-
expect(pre).not.toBeNull();
|
|
356
|
-
expect(pre!.textContent).toContain('SELECT id FROM users WHERE active;');
|
|
357
|
-
expect(queryByText('Try This')).not.toBeNull();
|
|
358
|
-
});
|
|
359
|
-
});
|
|
360
|
-
|
|
361
|
-
// -----------------------------------------------------------------------
|
|
362
|
-
// Tab navigation
|
|
363
|
-
// -----------------------------------------------------------------------
|
|
364
|
-
|
|
365
|
-
test('all 4 tabs are rendered: insights, AI Explain, tree, raw', () => {
|
|
366
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
367
|
-
expect(queryByText('insights')).not.toBeNull();
|
|
368
|
-
expect(queryByText('AI Explain')).not.toBeNull();
|
|
369
|
-
expect(queryByText('tree')).not.toBeNull();
|
|
370
|
-
expect(queryByText('raw')).not.toBeNull();
|
|
371
|
-
});
|
|
372
|
-
|
|
373
|
-
// -----------------------------------------------------------------------
|
|
374
|
-
// Format helpers (tested through rendered output)
|
|
375
|
-
// -----------------------------------------------------------------------
|
|
376
|
-
|
|
377
|
-
test('formatTime: seconds for >= 1000ms', () => {
|
|
378
|
-
const bigPlan: ExplainPlanResult[] = [{
|
|
379
|
-
Plan: { 'Node Type': 'Seq Scan', 'Actual Rows': 1, 'Plan Rows': 1, 'Actual Total Time': 2500, 'Total Cost': 1 },
|
|
380
|
-
'Execution Time': 2500,
|
|
381
|
-
'Planning Time': 0,
|
|
382
|
-
}];
|
|
383
|
-
const { queryAllByText } = render(<VisualExplain plan={bigPlan} />);
|
|
384
|
-
expect(queryAllByText('2.50s').length).toBeGreaterThan(0);
|
|
385
|
-
});
|
|
386
|
-
|
|
387
|
-
test('formatTime: microseconds for < 1ms', () => {
|
|
388
|
-
const tinyPlan: ExplainPlanResult[] = [{
|
|
389
|
-
Plan: { 'Node Type': 'Index Scan', 'Actual Rows': 1, 'Plan Rows': 1, 'Actual Total Time': 0.05, 'Total Cost': 1 },
|
|
390
|
-
'Execution Time': 0.05,
|
|
391
|
-
'Planning Time': 0,
|
|
392
|
-
}];
|
|
393
|
-
const { queryAllByText } = render(<VisualExplain plan={tinyPlan} />);
|
|
394
|
-
expect(queryAllByText('50μs').length).toBeGreaterThan(0);
|
|
395
|
-
});
|
|
396
|
-
|
|
397
|
-
test('formatNumber: millions', () => {
|
|
398
|
-
const bigRowPlan: ExplainPlanResult[] = [{
|
|
399
|
-
Plan: { 'Node Type': 'Seq Scan', 'Actual Rows': 2500000, 'Plan Rows': 2500000, 'Actual Total Time': 100, 'Total Cost': 500 },
|
|
400
|
-
'Execution Time': 100,
|
|
401
|
-
'Planning Time': 0,
|
|
402
|
-
}];
|
|
403
|
-
const { queryAllByText } = render(<VisualExplain plan={bigRowPlan} />);
|
|
404
|
-
expect(queryAllByText('2.5M').length).toBeGreaterThan(0);
|
|
405
|
-
});
|
|
406
|
-
|
|
407
|
-
test('shows N/A for cache hit rate when no buffer data', () => {
|
|
408
|
-
const { queryByText } = render(<VisualExplain plan={healthyPlan} />);
|
|
409
|
-
expect(queryByText('N/A')).not.toBeNull();
|
|
410
|
-
});
|
|
411
|
-
|
|
412
|
-
// -----------------------------------------------------------------------
|
|
413
|
-
// PlanNode collapse / expand behaviour
|
|
414
|
-
// -----------------------------------------------------------------------
|
|
415
|
-
|
|
416
|
-
test('PlanNode at depth 0 starts expanded and collapses on click', () => {
|
|
417
|
-
// samplePlan root has a child "Index Scan" — it should be visible initially
|
|
418
|
-
const { queryByText } = render(<VisualExplain plan={samplePlan} />);
|
|
419
|
-
// Switch to tree tab for a clean view of the plan
|
|
420
|
-
fireEvent.click(queryByText('tree')!);
|
|
421
|
-
|
|
422
|
-
// Child should be visible because root (depth 0) starts expanded
|
|
423
|
-
expect(queryByText('Index Scan')).not.toBeNull();
|
|
424
|
-
|
|
425
|
-
// Click on the root node to collapse it
|
|
426
|
-
fireEvent.click(queryByText('Seq Scan')!);
|
|
427
|
-
|
|
428
|
-
// After collapsing, the child should disappear
|
|
429
|
-
expect(queryByText('Index Scan')).toBeNull();
|
|
430
|
-
});
|
|
431
|
-
|
|
432
|
-
test('PlanNode at depth >= 2 starts collapsed', () => {
|
|
433
|
-
// Build a 3-level deep plan: root (depth 0) → child (depth 1) → grandchild (depth 2)
|
|
434
|
-
const deepPlan: ExplainPlanResult[] = [
|
|
435
|
-
{
|
|
436
|
-
Plan: {
|
|
437
|
-
'Node Type': 'Nested Loop',
|
|
438
|
-
'Actual Rows': 100,
|
|
439
|
-
'Plan Rows': 100,
|
|
440
|
-
'Actual Total Time': 10,
|
|
441
|
-
'Total Cost': 50,
|
|
442
|
-
Plans: [
|
|
443
|
-
{
|
|
444
|
-
'Node Type': 'Hash Join',
|
|
445
|
-
'Actual Rows': 50,
|
|
446
|
-
'Plan Rows': 50,
|
|
447
|
-
'Actual Total Time': 5,
|
|
448
|
-
'Total Cost': 25,
|
|
449
|
-
Plans: [
|
|
450
|
-
{
|
|
451
|
-
'Node Type': 'Seq Scan',
|
|
452
|
-
'Relation Name': 'deep_table',
|
|
453
|
-
'Actual Rows': 10,
|
|
454
|
-
'Plan Rows': 10,
|
|
455
|
-
'Actual Total Time': 1,
|
|
456
|
-
'Total Cost': 5,
|
|
457
|
-
},
|
|
458
|
-
],
|
|
459
|
-
},
|
|
460
|
-
],
|
|
461
|
-
},
|
|
462
|
-
'Execution Time': 10,
|
|
463
|
-
'Planning Time': 0.1,
|
|
464
|
-
},
|
|
465
|
-
];
|
|
466
|
-
const { queryByText } = render(<VisualExplain plan={deepPlan} />);
|
|
467
|
-
fireEvent.click(queryByText('tree')!);
|
|
468
|
-
|
|
469
|
-
// depth 0 (Nested Loop) is expanded → depth 1 (Hash Join) is visible
|
|
470
|
-
expect(queryByText('Hash Join')).not.toBeNull();
|
|
471
|
-
|
|
472
|
-
// depth 1 (Hash Join) is also expanded (depth < 2) → depth 2 node visible
|
|
473
|
-
// But depth 2 (Seq Scan on deep_table) starts collapsed, so its details are irrelevant —
|
|
474
|
-
// the node itself IS rendered by its parent (depth 1 expanded).
|
|
475
|
-
// The key point: depth 2 node is rendered but its OWN children would be collapsed.
|
|
476
|
-
// Since the Seq Scan has no children, we verify the grandchild is visible
|
|
477
|
-
// because its parent (depth 1) is expanded.
|
|
478
|
-
expect(queryByText('deep_table')).not.toBeNull();
|
|
479
|
-
|
|
480
|
-
// Now build a 4-level plan to truly test depth 2 collapse:
|
|
481
|
-
const deeperPlan: ExplainPlanResult[] = [
|
|
482
|
-
{
|
|
483
|
-
Plan: {
|
|
484
|
-
'Node Type': 'Nested Loop',
|
|
485
|
-
'Actual Rows': 100,
|
|
486
|
-
'Plan Rows': 100,
|
|
487
|
-
'Actual Total Time': 10,
|
|
488
|
-
'Total Cost': 50,
|
|
489
|
-
Plans: [
|
|
490
|
-
{
|
|
491
|
-
'Node Type': 'Hash Join',
|
|
492
|
-
'Actual Rows': 50,
|
|
493
|
-
'Plan Rows': 50,
|
|
494
|
-
'Actual Total Time': 5,
|
|
495
|
-
'Total Cost': 25,
|
|
496
|
-
Plans: [
|
|
497
|
-
{
|
|
498
|
-
'Node Type': 'Merge Join',
|
|
499
|
-
'Actual Rows': 20,
|
|
500
|
-
'Plan Rows': 20,
|
|
501
|
-
'Actual Total Time': 2,
|
|
502
|
-
'Total Cost': 10,
|
|
503
|
-
Plans: [
|
|
504
|
-
{
|
|
505
|
-
'Node Type': 'Index Scan',
|
|
506
|
-
'Relation Name': 'hidden_table',
|
|
507
|
-
'Actual Rows': 5,
|
|
508
|
-
'Plan Rows': 5,
|
|
509
|
-
'Actual Total Time': 0.5,
|
|
510
|
-
'Total Cost': 2,
|
|
511
|
-
},
|
|
512
|
-
],
|
|
513
|
-
},
|
|
514
|
-
],
|
|
515
|
-
},
|
|
516
|
-
],
|
|
517
|
-
},
|
|
518
|
-
'Execution Time': 10,
|
|
519
|
-
'Planning Time': 0.1,
|
|
520
|
-
},
|
|
521
|
-
];
|
|
522
|
-
cleanup();
|
|
523
|
-
const { queryByText: q2 } = render(<VisualExplain plan={deeperPlan} />);
|
|
524
|
-
fireEvent.click(q2('tree')!);
|
|
525
|
-
|
|
526
|
-
// depth 0 = Nested Loop (expanded), depth 1 = Hash Join (expanded)
|
|
527
|
-
// depth 2 = Merge Join (collapsed!) → depth 3 child should NOT be visible
|
|
528
|
-
expect(q2('Merge Join')).not.toBeNull();
|
|
529
|
-
expect(q2('hidden_table')).toBeNull();
|
|
530
|
-
});
|
|
531
|
-
|
|
532
|
-
// -----------------------------------------------------------------------
|
|
533
|
-
// NodeIcon variants
|
|
534
|
-
// -----------------------------------------------------------------------
|
|
535
|
-
|
|
536
|
-
test('NodeIcon renders Layers icon for Join type', () => {
|
|
537
|
-
const joinPlan: ExplainPlanResult[] = [
|
|
538
|
-
{
|
|
539
|
-
Plan: {
|
|
540
|
-
'Node Type': 'Hash Join',
|
|
541
|
-
'Actual Rows': 10,
|
|
542
|
-
'Plan Rows': 10,
|
|
543
|
-
'Actual Total Time': 1,
|
|
544
|
-
'Total Cost': 5,
|
|
545
|
-
},
|
|
546
|
-
'Execution Time': 1,
|
|
547
|
-
'Planning Time': 0.1,
|
|
548
|
-
},
|
|
549
|
-
];
|
|
550
|
-
const { container } = render(<VisualExplain plan={joinPlan} />);
|
|
551
|
-
// Layers icon for Join has text-purple-400 class
|
|
552
|
-
const purpleIcon = container.querySelector('.text-purple-400');
|
|
553
|
-
expect(purpleIcon).not.toBeNull();
|
|
554
|
-
});
|
|
555
|
-
|
|
556
|
-
test('NodeIcon renders ArrowDown icon for Sort type', () => {
|
|
557
|
-
// sortPlan already has Node Type = 'Sort'
|
|
558
|
-
const { container } = render(<VisualExplain plan={sortPlan} />);
|
|
559
|
-
// Sort nodes use ArrowDown icon with text-amber-400
|
|
560
|
-
// Seq Scan also uses amber-400, but sortPlan has 'Sort' not 'Seq Scan'
|
|
561
|
-
const amberIcon = container.querySelector('.text-amber-400');
|
|
562
|
-
expect(amberIcon).not.toBeNull();
|
|
563
|
-
});
|
|
564
|
-
|
|
565
|
-
test('NodeIcon renders Zap icon for Aggregate type', () => {
|
|
566
|
-
const aggPlan: ExplainPlanResult[] = [
|
|
567
|
-
{
|
|
568
|
-
Plan: {
|
|
569
|
-
'Node Type': 'Aggregate',
|
|
570
|
-
'Actual Rows': 1,
|
|
571
|
-
'Plan Rows': 1,
|
|
572
|
-
'Actual Total Time': 2,
|
|
573
|
-
'Total Cost': 10,
|
|
574
|
-
},
|
|
575
|
-
'Execution Time': 2,
|
|
576
|
-
'Planning Time': 0.1,
|
|
577
|
-
},
|
|
578
|
-
];
|
|
579
|
-
const { container } = render(<VisualExplain plan={aggPlan} />);
|
|
580
|
-
// Aggregate uses Zap icon with text-pink-400
|
|
581
|
-
const pinkIcon = container.querySelector('.text-pink-400');
|
|
582
|
-
expect(pinkIcon).not.toBeNull();
|
|
583
|
-
});
|
|
584
|
-
|
|
585
|
-
test('NodeIcon renders HardDrive icon for Hash type', () => {
|
|
586
|
-
const hashPlan: ExplainPlanResult[] = [
|
|
587
|
-
{
|
|
588
|
-
Plan: {
|
|
589
|
-
'Node Type': 'Hash',
|
|
590
|
-
'Actual Rows': 100,
|
|
591
|
-
'Plan Rows': 100,
|
|
592
|
-
'Actual Total Time': 3,
|
|
593
|
-
'Total Cost': 15,
|
|
594
|
-
},
|
|
595
|
-
'Execution Time': 3,
|
|
596
|
-
'Planning Time': 0.1,
|
|
597
|
-
},
|
|
598
|
-
];
|
|
599
|
-
const { container } = render(<VisualExplain plan={hashPlan} />);
|
|
600
|
-
// Hash uses HardDrive icon with text-cyan-400
|
|
601
|
-
const cyanIcon = container.querySelector('.text-cyan-400');
|
|
602
|
-
expect(cyanIcon).not.toBeNull();
|
|
603
|
-
});
|
|
604
|
-
|
|
605
|
-
test('NodeIcon renders Database icon for unknown type', () => {
|
|
606
|
-
const unknownPlan: ExplainPlanResult[] = [
|
|
607
|
-
{
|
|
608
|
-
Plan: {
|
|
609
|
-
'Node Type': 'Materialize',
|
|
610
|
-
'Actual Rows': 10,
|
|
611
|
-
'Plan Rows': 10,
|
|
612
|
-
'Actual Total Time': 1,
|
|
613
|
-
'Total Cost': 5,
|
|
614
|
-
},
|
|
615
|
-
'Execution Time': 1,
|
|
616
|
-
'Planning Time': 0.1,
|
|
617
|
-
},
|
|
618
|
-
];
|
|
619
|
-
const { container } = render(<VisualExplain plan={unknownPlan} />);
|
|
620
|
-
// Unknown type uses Database icon with text-zinc-500
|
|
621
|
-
// Need to find the icon inside the node icon wrapper (p-1 rounded div)
|
|
622
|
-
const iconWrappers = container.querySelectorAll('.bg-white\\/5');
|
|
623
|
-
// The node icon container has bg-white/5 for non-scan types
|
|
624
|
-
let foundZincIcon = false;
|
|
625
|
-
iconWrappers.forEach((wrapper) => {
|
|
626
|
-
const icon = wrapper.querySelector('.text-zinc-500');
|
|
627
|
-
if (icon) foundZincIcon = true;
|
|
628
|
-
});
|
|
629
|
-
expect(foundZincIcon).toBe(true);
|
|
630
|
-
});
|
|
631
|
-
|
|
632
|
-
// -----------------------------------------------------------------------
|
|
633
|
-
// AI tab: onLoadQuery via "Try This" button
|
|
634
|
-
// -----------------------------------------------------------------------
|
|
635
|
-
|
|
636
|
-
test('clicking "Try This" calls onLoadQuery with the SQL code', async () => {
|
|
637
|
-
const user = userEvent.setup();
|
|
638
|
-
const onLoadQuery = mock(() => {});
|
|
639
|
-
globalThis.fetch = mockFetchStream('Suggestion:\n```sql\nCREATE INDEX idx_active ON users(active);\n```\nDone.') as unknown as typeof fetch;
|
|
640
|
-
|
|
641
|
-
const { queryByText } = render(
|
|
642
|
-
<VisualExplain plan={samplePlan} query="SELECT * FROM users" onLoadQuery={onLoadQuery} />
|
|
643
|
-
);
|
|
644
|
-
fireEvent.click(queryByText('AI Explain')!);
|
|
645
|
-
await user.click(queryByText('Analyze with AI')!);
|
|
646
|
-
|
|
647
|
-
await waitFor(() => {
|
|
648
|
-
expect(queryByText('Try This')).not.toBeNull();
|
|
649
|
-
});
|
|
650
|
-
|
|
651
|
-
await user.click(queryByText('Try This')!);
|
|
652
|
-
expect(onLoadQuery).toHaveBeenCalledTimes(1);
|
|
653
|
-
expect(onLoadQuery).toHaveBeenCalledWith('CREATE INDEX idx_active ON users(active);');
|
|
654
|
-
});
|
|
655
|
-
|
|
656
|
-
// -----------------------------------------------------------------------
|
|
657
|
-
// formatTime: microsecond branch (ms < 1)
|
|
658
|
-
// -----------------------------------------------------------------------
|
|
659
|
-
|
|
660
|
-
test('formatTime shows microseconds for sub-millisecond execution', () => {
|
|
661
|
-
const microPlan: ExplainPlanResult[] = [
|
|
662
|
-
{
|
|
663
|
-
Plan: {
|
|
664
|
-
'Node Type': 'Result',
|
|
665
|
-
'Actual Rows': 1,
|
|
666
|
-
'Plan Rows': 1,
|
|
667
|
-
'Actual Total Time': 0.002,
|
|
668
|
-
'Total Cost': 0.01,
|
|
669
|
-
},
|
|
670
|
-
'Execution Time': 0.002,
|
|
671
|
-
'Planning Time': 0,
|
|
672
|
-
},
|
|
673
|
-
];
|
|
674
|
-
const { queryAllByText } = render(<VisualExplain plan={microPlan} />);
|
|
675
|
-
// 0.002ms * 1000 = 2μs
|
|
676
|
-
expect(queryAllByText('2μs').length).toBeGreaterThan(0);
|
|
677
|
-
});
|
|
678
|
-
|
|
679
|
-
// -----------------------------------------------------------------------
|
|
680
|
-
// Leaf nodes show spacer instead of chevron
|
|
681
|
-
// -----------------------------------------------------------------------
|
|
682
|
-
|
|
683
|
-
test('leaf nodes show spacer div instead of chevron icon', () => {
|
|
684
|
-
// healthyPlan has a single Index Scan with no children → leaf node
|
|
685
|
-
const { container, queryByText } = render(<VisualExplain plan={healthyPlan} />);
|
|
686
|
-
fireEvent.click(queryByText('tree')!);
|
|
687
|
-
|
|
688
|
-
// The leaf node should have a <div class="w-3"> spacer instead of a ChevronRight svg
|
|
689
|
-
// PlanNode renders: children.length === 0 → <div className="w-3" />
|
|
690
|
-
// ChevronRight has the class rotate-90 or is a ChevronRight svg
|
|
691
|
-
const planNodeContainer = container.querySelector('.rounded-lg.border');
|
|
692
|
-
expect(planNodeContainer).not.toBeNull();
|
|
693
|
-
|
|
694
|
-
// Within the plan node, find the spacer div (w-3 without svg child)
|
|
695
|
-
const spacers = planNodeContainer!.querySelectorAll('div.w-3');
|
|
696
|
-
expect(spacers.length).toBeGreaterThan(0);
|
|
697
|
-
|
|
698
|
-
// Verify there's no chevron (which would have the rotate-90 or transition-transform classes on an svg)
|
|
699
|
-
// For a leaf node, there should be no ChevronRight rendered
|
|
700
|
-
const chevrons = planNodeContainer!.querySelectorAll('svg.transition-transform');
|
|
701
|
-
// healthyPlan has a single node with no children, so no chevrons at all
|
|
702
|
-
expect(chevrons.length).toBe(0);
|
|
703
|
-
});
|
|
704
|
-
});
|