@libredb/studio 0.9.7
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/.claude/settings.local.json +127 -0
- package/.cursorrules +426 -0
- package/.devin/wiki.json +143 -0
- package/.dockerignore +80 -0
- package/.env.example +159 -0
- package/.github/ISSUE_TEMPLATE/bug_report.md +49 -0
- package/.github/ISSUE_TEMPLATE/feature_request.md +29 -0
- package/.github/PULL_REQUEST_TEMPLATE.md +57 -0
- package/.github/workflows/ci.yml +185 -0
- package/.github/workflows/codeql.yml +57 -0
- package/.github/workflows/docker-build-push.yml +118 -0
- package/.github/workflows/helm-release.yml +113 -0
- package/CLAUDE.md +265 -0
- package/CODE_OF_CONDUCT.md +124 -0
- package/CONTRIBUTING.md +154 -0
- package/Dockerfile +73 -0
- package/LICENSE +21 -0
- package/README.md +614 -0
- package/SECURITY.md +107 -0
- package/artifacthub-repo.yml +4 -0
- package/bun.lock +1714 -0
- package/bunfig.toml +3 -0
- package/charts/libredb-studio/.helmignore +11 -0
- package/charts/libredb-studio/Chart.lock +6 -0
- package/charts/libredb-studio/Chart.yaml +50 -0
- package/charts/libredb-studio/README.md +206 -0
- package/charts/libredb-studio/templates/NOTES.txt +59 -0
- package/charts/libredb-studio/templates/_helpers.tpl +135 -0
- package/charts/libredb-studio/templates/configmap.yaml +37 -0
- package/charts/libredb-studio/templates/deployment.yaml +184 -0
- package/charts/libredb-studio/templates/hpa.yaml +32 -0
- package/charts/libredb-studio/templates/ingress.yaml +41 -0
- package/charts/libredb-studio/templates/networkpolicy.yaml +50 -0
- package/charts/libredb-studio/templates/pdb.yaml +18 -0
- package/charts/libredb-studio/templates/pvc.yaml +23 -0
- package/charts/libredb-studio/templates/secret.yaml +30 -0
- package/charts/libredb-studio/templates/seed-configmap.yaml +11 -0
- package/charts/libredb-studio/templates/service.yaml +22 -0
- package/charts/libredb-studio/templates/serviceaccount.yaml +13 -0
- package/charts/libredb-studio/values.schema.json +246 -0
- package/charts/libredb-studio/values.yaml +286 -0
- package/components.json +22 -0
- package/conductor/code_styleguides/typescript.md +43 -0
- package/conductor/product-guidelines.md +43 -0
- package/conductor/product.md +3 -0
- package/conductor/setup_state.json +1 -0
- package/conductor/tech-stack.md +39 -0
- package/conductor/tracks/enhance_postgres_monitoring_20251227/metadata.json +8 -0
- package/conductor/tracks/enhance_postgres_monitoring_20251227/plan.md +44 -0
- package/conductor/tracks/enhance_postgres_monitoring_20251227/spec.md +31 -0
- package/conductor/tracks.md +8 -0
- package/conductor/workflow.md +333 -0
- package/database-compose.yml +55 -0
- package/docker/postgres-init/01-extensions.sql +10 -0
- package/docker/postgres-init/02-sample-data.sql +585 -0
- package/docker/postgres.yml +68 -0
- package/docker-compose.yml +38 -0
- package/docs/AI_PLAN.md +74 -0
- package/docs/API_DOCS.md +875 -0
- package/docs/ARCHITECTURE.md +218 -0
- package/docs/DATABASE_PROVIDERS.md +358 -0
- package/docs/FEATURES.md +116 -0
- package/docs/HELM_CHART.md +252 -0
- package/docs/LOGIN_PAGE.md +178 -0
- package/docs/MONACO_EDITOR_PERFORMANCE.md +315 -0
- package/docs/OIDC_ARCH.md +681 -0
- package/docs/OIDC_SETUP.md +322 -0
- package/docs/POSTGRES_METRICS.md +516 -0
- package/docs/QUERY_OPTIMIZATION.md +370 -0
- package/docs/SEED_CONNECTIONS.md +468 -0
- package/docs/SQL_ALIAS_COMPLETION.md +190 -0
- package/docs/STORAGE_ARCHITECTURE.md +565 -0
- package/docs/STORAGE_QUICK_SETUP.md +419 -0
- package/docs/TECHNICAL_PLAN.md +36 -0
- package/docs/THEMING.md +345 -0
- package/docs/adding-a-new-database-provider.md +642 -0
- package/docs/backlogs/000-PLATFORM_DATA_SYNC_DATABASE.md +360 -0
- package/docs/backlogs/001-INLINE_DATA_EDITING.md +118 -0
- package/docs/backlogs/002-DATA_IMPORT.md +215 -0
- package/docs/backlogs/003-QUERY_TIME_MACHINE.md +183 -0
- package/docs/backlogs/004-AI_DATA_STORYTELLER.md +292 -0
- package/docs/backlogs/005-QUERY_PLAYGROUND.md +352 -0
- package/docs/backlogs/006-DATA_MASKING.md +418 -0
- package/docs/enterprise-features.md +718 -0
- package/docs/kubernetes-helm-chart-artifacthub-plan.md +803 -0
- package/docs/medium-koyeb-article-en.md +215 -0
- package/docs/plans/test-plans.md +445 -0
- package/docs/releases/RELEASE.V0.3.0.md +22 -0
- package/docs/releases/RELEASE.V0.4.0.md +154 -0
- package/docs/releases/RELEASE.V0.5.0.md +252 -0
- package/docs/releases/RELEASE_v0.5.6.md +145 -0
- package/docs/releases/RELEASE_v0.6.1.md +303 -0
- package/docs/releases/RELEASE_v0.6.7.md +292 -0
- package/docs/releases/RELEASE_v0.7.0.md +332 -0
- package/docs/releases/RELEASE_v0.8.0.md +521 -0
- package/docs/sampledb/titanic.sql +1379 -0
- package/docs/superpowers/plans/2026-03-25-seed-connections.md +1362 -0
- package/docs/superpowers/specs/2026-03-25-seed-connections-design.md +590 -0
- package/e2e/admin-dashboard.spec.ts +64 -0
- package/e2e/connection-management.spec.ts +58 -0
- package/e2e/export.spec.ts +34 -0
- package/e2e/login.spec.ts +85 -0
- package/e2e/query-execution.spec.ts +35 -0
- package/e2e/tab-management.spec.ts +64 -0
- package/eslint.config.mjs +28 -0
- package/fly.toml +43 -0
- package/next.config.ts +32 -0
- package/package.json +130 -0
- package/playwright.config.ts +34 -0
- package/postcss.config.mjs +7 -0
- package/public/favicon-32x32.png +0 -0
- package/public/favicon.ico +0 -0
- package/public/file.svg +1 -0
- package/public/globe.svg +1 -0
- package/public/logo.svg +32 -0
- package/public/next.svg +1 -0
- 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 +1 -0
- package/public/window.svg +1 -0
- package/render.yaml +58 -0
- package/scripts/merge-lcov.mjs +239 -0
- package/sonar-project.properties +16 -0
- package/src/app/admin/error.tsx +46 -0
- package/src/app/admin/page.tsx +10 -0
- package/src/app/api/admin/audit/route.ts +52 -0
- package/src/app/api/admin/fleet-health/route.ts +81 -0
- package/src/app/api/ai/autopilot/route.ts +105 -0
- package/src/app/api/ai/chat/route.ts +132 -0
- package/src/app/api/ai/describe-schema/route.ts +52 -0
- package/src/app/api/ai/explain/route.ts +86 -0
- package/src/app/api/ai/impact/route.ts +97 -0
- package/src/app/api/ai/index-advisor/route.ts +98 -0
- package/src/app/api/ai/nl2sql/route.ts +87 -0
- package/src/app/api/ai/query-safety/route.ts +87 -0
- package/src/app/api/auth/login/route.ts +62 -0
- package/src/app/api/auth/logout/route.ts +25 -0
- package/src/app/api/auth/me/route.ts +10 -0
- package/src/app/api/auth/oidc/callback/route.ts +82 -0
- package/src/app/api/auth/oidc/login/route.ts +43 -0
- package/src/app/api/connections/managed/route.ts +35 -0
- package/src/app/api/db/cancel/route.ts +42 -0
- package/src/app/api/db/disconnect/route.ts +28 -0
- package/src/app/api/db/health/route.ts +49 -0
- package/src/app/api/db/maintenance/route.ts +72 -0
- package/src/app/api/db/monitoring/route.ts +62 -0
- package/src/app/api/db/multi-query/route.ts +116 -0
- package/src/app/api/db/pool-stats/route.ts +37 -0
- package/src/app/api/db/profile/route.ts +144 -0
- package/src/app/api/db/provider-meta/route.ts +49 -0
- package/src/app/api/db/query/route.ts +50 -0
- package/src/app/api/db/schema/route.ts +47 -0
- package/src/app/api/db/schema-snapshot/route.ts +42 -0
- package/src/app/api/db/test-connection/route.ts +55 -0
- package/src/app/api/db/transaction/route.ts +111 -0
- package/src/app/api/storage/[collection]/route.ts +67 -0
- package/src/app/api/storage/config/route.ts +17 -0
- package/src/app/api/storage/migrate/route.ts +45 -0
- package/src/app/api/storage/route.ts +32 -0
- package/src/app/error.tsx +49 -0
- package/src/app/global-error.tsx +55 -0
- package/src/app/globals.css +146 -0
- package/src/app/icon.svg +42 -0
- package/src/app/layout.tsx +34 -0
- package/src/app/login/login-form.tsx +301 -0
- package/src/app/login/page.tsx +11 -0
- package/src/app/monitoring/page.tsx +8 -0
- package/src/app/not-found.tsx +29 -0
- package/src/app/page.tsx +5 -0
- package/src/components/AIAutopilotPanel.tsx +238 -0
- package/src/components/CodeGenerator.tsx +271 -0
- package/src/components/CommandPalette.tsx +227 -0
- package/src/components/ConnectionModal.tsx +759 -0
- package/src/components/CreateTableModal.tsx +281 -0
- package/src/components/DataCharts.tsx +962 -0
- package/src/components/DataImportModal.tsx +582 -0
- package/src/components/DataProfiler.tsx +335 -0
- package/src/components/DatabaseDocs.tsx +251 -0
- package/src/components/MaskingSettings.tsx +414 -0
- package/src/components/MobileNav.tsx +50 -0
- package/src/components/NL2SQLPanel.tsx +281 -0
- package/src/components/PivotTable.tsx +257 -0
- package/src/components/QueryEditor.tsx +760 -0
- package/src/components/QueryHistory.tsx +344 -0
- package/src/components/QuerySafetyDialog.tsx +290 -0
- package/src/components/ResultsGrid.tsx +644 -0
- package/src/components/SaveQueryModal.tsx +104 -0
- package/src/components/SavedQueries.tsx +128 -0
- package/src/components/SchemaDiagram.tsx +473 -0
- package/src/components/SchemaDiff.tsx +473 -0
- package/src/components/SnapshotTimeline.tsx +116 -0
- package/src/components/Studio.tsx +639 -0
- package/src/components/TestDataGenerator.tsx +261 -0
- package/src/components/VisualExplain.tsx +820 -0
- package/src/components/admin/AdminDashboard.tsx +163 -0
- package/src/components/admin/tabs/AuditTab.tsx +531 -0
- package/src/components/admin/tabs/MonitoringEmbed.tsx +11 -0
- package/src/components/admin/tabs/OperationsTab.tsx +646 -0
- package/src/components/admin/tabs/OverviewTab.tsx +1328 -0
- package/src/components/admin/tabs/SecurityTab.tsx +284 -0
- package/src/components/community-section.tsx +92 -0
- package/src/components/icons/db-icons.tsx +84 -0
- package/src/components/libredb-logo.tsx +61 -0
- package/src/components/monitoring/MonitoringDashboard.tsx +345 -0
- package/src/components/monitoring/tabs/MetricChart.tsx +82 -0
- package/src/components/monitoring/tabs/OverviewTab.tsx +263 -0
- package/src/components/monitoring/tabs/PerformanceTab.tsx +254 -0
- package/src/components/monitoring/tabs/PoolTab.tsx +174 -0
- package/src/components/monitoring/tabs/QueriesTab.tsx +287 -0
- package/src/components/monitoring/tabs/SessionsTab.tsx +316 -0
- package/src/components/monitoring/tabs/StorageTab.tsx +335 -0
- package/src/components/monitoring/tabs/TablesTab.tsx +300 -0
- package/src/components/results-grid/ResultCard.tsx +111 -0
- package/src/components/results-grid/RowDetailSheet.tsx +178 -0
- package/src/components/results-grid/StatsBar.tsx +201 -0
- package/src/components/results-grid/index.ts +1 -0
- package/src/components/results-grid/utils.ts +23 -0
- package/src/components/schema-explorer/ColumnList.tsx +53 -0
- package/src/components/schema-explorer/SchemaExplorer.tsx +182 -0
- package/src/components/schema-explorer/TableItem.tsx +210 -0
- package/src/components/schema-explorer/index.ts +1 -0
- package/src/components/sidebar/ConnectionItem.tsx +105 -0
- package/src/components/sidebar/ConnectionsList.tsx +62 -0
- package/src/components/sidebar/Sidebar.tsx +130 -0
- package/src/components/sidebar/index.ts +2 -0
- package/src/components/studio/BottomPanel.tsx +286 -0
- package/src/components/studio/QueryToolbar.tsx +180 -0
- package/src/components/studio/StudioDesktopHeader.tsx +114 -0
- package/src/components/studio/StudioMobileHeader.tsx +340 -0
- package/src/components/studio/StudioTabBar.tsx +82 -0
- package/src/components/studio/index.ts +5 -0
- package/src/components/ui/accordion.tsx +66 -0
- package/src/components/ui/alert-dialog.tsx +157 -0
- package/src/components/ui/alert.tsx +66 -0
- package/src/components/ui/aspect-ratio.tsx +11 -0
- package/src/components/ui/avatar.tsx +53 -0
- package/src/components/ui/badge.tsx +46 -0
- package/src/components/ui/breadcrumb.tsx +109 -0
- package/src/components/ui/button-group.tsx +83 -0
- package/src/components/ui/button.tsx +60 -0
- package/src/components/ui/calendar.tsx +216 -0
- package/src/components/ui/card.tsx +92 -0
- package/src/components/ui/carousel.tsx +241 -0
- package/src/components/ui/chart.tsx +357 -0
- package/src/components/ui/checkbox.tsx +32 -0
- package/src/components/ui/collapsible.tsx +33 -0
- package/src/components/ui/command.tsx +184 -0
- package/src/components/ui/context-menu.tsx +252 -0
- package/src/components/ui/dialog.tsx +143 -0
- package/src/components/ui/drawer.tsx +135 -0
- package/src/components/ui/dropdown-menu.tsx +257 -0
- package/src/components/ui/empty.tsx +104 -0
- package/src/components/ui/field.tsx +248 -0
- package/src/components/ui/form.tsx +167 -0
- package/src/components/ui/hover-card.tsx +44 -0
- package/src/components/ui/input-group.tsx +170 -0
- package/src/components/ui/input-otp.tsx +77 -0
- package/src/components/ui/input.tsx +21 -0
- package/src/components/ui/item.tsx +193 -0
- package/src/components/ui/kbd.tsx +28 -0
- package/src/components/ui/label.tsx +24 -0
- package/src/components/ui/menubar.tsx +276 -0
- package/src/components/ui/navigation-menu.tsx +168 -0
- package/src/components/ui/pagination.tsx +127 -0
- package/src/components/ui/popover.tsx +48 -0
- package/src/components/ui/progress.tsx +31 -0
- package/src/components/ui/radio-group.tsx +45 -0
- package/src/components/ui/resizable.tsx +56 -0
- package/src/components/ui/scroll-area.tsx +58 -0
- package/src/components/ui/select.tsx +187 -0
- package/src/components/ui/separator.tsx +28 -0
- package/src/components/ui/sheet.tsx +139 -0
- package/src/components/ui/sidebar.tsx +726 -0
- package/src/components/ui/skeleton.tsx +13 -0
- package/src/components/ui/slider.tsx +63 -0
- package/src/components/ui/sonner.tsx +40 -0
- package/src/components/ui/spinner.tsx +16 -0
- package/src/components/ui/switch.tsx +31 -0
- package/src/components/ui/table.tsx +116 -0
- package/src/components/ui/tabs.tsx +66 -0
- package/src/components/ui/textarea.tsx +18 -0
- package/src/components/ui/toggle-group.tsx +83 -0
- package/src/components/ui/toggle.tsx +47 -0
- package/src/components/ui/tooltip.tsx +61 -0
- package/src/exports/components.ts +15 -0
- package/src/exports/index.ts +4 -0
- package/src/exports/providers.ts +4 -0
- package/src/exports/types.ts +26 -0
- package/src/hooks/use-ai-chat.ts +182 -0
- package/src/hooks/use-all-connections.ts +66 -0
- package/src/hooks/use-api-call.ts +71 -0
- package/src/hooks/use-auth.ts +51 -0
- package/src/hooks/use-connection-form.ts +349 -0
- package/src/hooks/use-connection-manager.ts +169 -0
- package/src/hooks/use-connection-payload.ts +15 -0
- package/src/hooks/use-inline-editing.ts +109 -0
- package/src/hooks/use-mobile.ts +20 -0
- package/src/hooks/use-monitoring-data.ts +270 -0
- package/src/hooks/use-provider-metadata.ts +62 -0
- package/src/hooks/use-query-execution.ts +478 -0
- package/src/hooks/use-storage-sync.ts +259 -0
- package/src/hooks/use-tab-manager.ts +231 -0
- package/src/hooks/use-toast.ts +20 -0
- package/src/hooks/use-transaction-control.ts +64 -0
- package/src/lib/api/error-codes.ts +30 -0
- package/src/lib/api/errors.ts +236 -0
- package/src/lib/api/with-error-handler.ts +41 -0
- package/src/lib/audit.ts +105 -0
- package/src/lib/auth.ts +87 -0
- package/src/lib/connection-string-parser.ts +172 -0
- package/src/lib/data-masking.ts +385 -0
- package/src/lib/db/base-provider.ts +325 -0
- package/src/lib/db/errors.ts +317 -0
- package/src/lib/db/factory.ts +324 -0
- package/src/lib/db/index.ts +123 -0
- package/src/lib/db/providers/document/index.ts +6 -0
- package/src/lib/db/providers/document/mongodb.ts +992 -0
- package/src/lib/db/providers/keyvalue/redis.ts +554 -0
- package/src/lib/db/providers/sql/index.ts +11 -0
- package/src/lib/db/providers/sql/mssql.ts +1065 -0
- package/src/lib/db/providers/sql/mysql.ts +978 -0
- package/src/lib/db/providers/sql/oracle.ts +1044 -0
- package/src/lib/db/providers/sql/postgres.ts +1179 -0
- package/src/lib/db/providers/sql/sql-base.ts +174 -0
- package/src/lib/db/providers/sql/sqlite.ts +721 -0
- package/src/lib/db/types.ts +437 -0
- package/src/lib/db/utils/pool-manager.ts +287 -0
- package/src/lib/db/utils/query-limiter.ts +239 -0
- package/src/lib/db-ui-config.ts +86 -0
- package/src/lib/editor/mongodb-completions.ts +172 -0
- package/src/lib/editor/sql-completions.ts +280 -0
- package/src/lib/llm/base-provider.ts +117 -0
- package/src/lib/llm/factory.ts +102 -0
- package/src/lib/llm/index.ts +90 -0
- package/src/lib/llm/providers/custom.ts +181 -0
- package/src/lib/llm/providers/gemini.ts +126 -0
- package/src/lib/llm/providers/ollama.ts +154 -0
- package/src/lib/llm/providers/openai.ts +146 -0
- package/src/lib/llm/types.ts +173 -0
- package/src/lib/llm/utils/config.ts +187 -0
- package/src/lib/llm/utils/retry.ts +119 -0
- package/src/lib/llm/utils/streaming.ts +202 -0
- package/src/lib/logger.ts +127 -0
- package/src/lib/monitoring-thresholds.ts +44 -0
- package/src/lib/oidc.ts +262 -0
- package/src/lib/query-generators.ts +61 -0
- package/src/lib/schema-diff/diff-engine.ts +273 -0
- package/src/lib/schema-diff/migration-generator.ts +208 -0
- package/src/lib/schema-diff/types.ts +55 -0
- package/src/lib/seed/config-loader.ts +79 -0
- package/src/lib/seed/connection-filter.ts +49 -0
- package/src/lib/seed/credential-resolver.ts +62 -0
- package/src/lib/seed/index.ts +40 -0
- package/src/lib/seed/resolve-connection.ts +57 -0
- package/src/lib/seed/types.ts +69 -0
- package/src/lib/sql/alias-extractor.ts +267 -0
- package/src/lib/sql/index.ts +8 -0
- package/src/lib/sql/statement-splitter.ts +167 -0
- package/src/lib/sql/types.ts +40 -0
- package/src/lib/ssh/tunnel.ts +142 -0
- package/src/lib/storage/factory.ts +84 -0
- package/src/lib/storage/index.ts +14 -0
- package/src/lib/storage/local-storage.ts +99 -0
- package/src/lib/storage/providers/postgres.ts +225 -0
- package/src/lib/storage/providers/sqlite.ts +153 -0
- package/src/lib/storage/storage-facade.ts +272 -0
- package/src/lib/storage/types.ts +75 -0
- package/src/lib/time-series-buffer.ts +58 -0
- package/src/lib/types.ts +173 -0
- package/src/lib/utils.ts +6 -0
- package/src/proxy.ts +104 -0
- package/src/types/db-drivers.d.ts +23 -0
- package/src/types/html2canvas.d.ts +9 -0
- package/tests/api/admin/audit.test.ts +178 -0
- package/tests/api/admin/fleet-health.test.ts +183 -0
- package/tests/api/ai/autopilot.test.ts +174 -0
- package/tests/api/ai/chat.test.ts +250 -0
- package/tests/api/ai/describe-schema.test.ts +266 -0
- package/tests/api/ai/explain.test.ts +199 -0
- package/tests/api/ai/impact.test.ts +168 -0
- package/tests/api/ai/index-advisor.test.ts +171 -0
- package/tests/api/ai/nl2sql.test.ts +202 -0
- package/tests/api/ai/query-safety.test.ts +196 -0
- package/tests/api/auth/login.test.ts +170 -0
- package/tests/api/auth/logout.test.ts +140 -0
- package/tests/api/auth/me.test.ts +73 -0
- package/tests/api/auth/oidc-callback.test.ts +215 -0
- package/tests/api/auth/oidc-login.test.ts +127 -0
- package/tests/api/db/cancel.test.ts +198 -0
- package/tests/api/db/disconnect.test.ts +124 -0
- package/tests/api/db/health.test.ts +222 -0
- package/tests/api/db/maintenance.test.ts +263 -0
- package/tests/api/db/monitoring.test.ts +221 -0
- package/tests/api/db/multi-query.test.ts +316 -0
- package/tests/api/db/pool-stats.test.ts +135 -0
- package/tests/api/db/profile.test.ts +330 -0
- package/tests/api/db/provider-meta.test.ts +193 -0
- package/tests/api/db/query.test.ts +314 -0
- package/tests/api/db/schema-snapshot.test.ts +170 -0
- package/tests/api/db/schema.test.ts +191 -0
- package/tests/api/db/test-connection.test.ts +185 -0
- package/tests/api/db/transaction.test.ts +314 -0
- package/tests/api/proxy.test.ts +191 -0
- package/tests/api/seed/managed-route.test.ts +113 -0
- package/tests/api/storage/config.test.ts +42 -0
- package/tests/api/storage/storage-routes.test.ts +309 -0
- package/tests/components/AIAutopilotPanel.test.tsx +756 -0
- package/tests/components/AdminPage.test.tsx +33 -0
- package/tests/components/CodeGenerator.test.tsx +182 -0
- package/tests/components/CommandPalette.test.tsx +428 -0
- package/tests/components/CommunitySection.test.tsx +91 -0
- package/tests/components/ConnectionModal.mobile.test.tsx +284 -0
- package/tests/components/ConnectionModal.test.tsx +570 -0
- package/tests/components/CreateTableModal.test.tsx +383 -0
- package/tests/components/DataCharts.test.tsx +739 -0
- package/tests/components/DataImportModal.test.tsx +751 -0
- package/tests/components/DataProfiler.test.tsx +589 -0
- package/tests/components/DatabaseDocs.test.tsx +353 -0
- package/tests/components/LoginPage.test.tsx +163 -0
- package/tests/components/LoginPageOIDC.test.tsx +92 -0
- package/tests/components/MaskingSettings.test.tsx +498 -0
- package/tests/components/MobileNav.test.tsx +30 -0
- package/tests/components/MonitoringPage.test.tsx +32 -0
- package/tests/components/NL2SQLPanel.test.tsx +621 -0
- package/tests/components/Page.test.tsx +33 -0
- package/tests/components/PivotTable.test.tsx +350 -0
- package/tests/components/QueryEditor.test.tsx +1730 -0
- package/tests/components/QueryHistory.test.tsx +572 -0
- package/tests/components/QuerySafetyDialog.test.tsx +586 -0
- package/tests/components/ResultsGrid.test.tsx +804 -0
- package/tests/components/RootLayout.test.tsx +83 -0
- package/tests/components/SaveQueryModal.test.tsx +25 -0
- package/tests/components/SavedQueries.test.tsx +43 -0
- package/tests/components/SchemaDiagram.test.tsx +1034 -0
- package/tests/components/SchemaDiff.test.tsx +906 -0
- package/tests/components/SnapshotTimeline.test.tsx +174 -0
- package/tests/components/Studio.test.tsx +1030 -0
- package/tests/components/TestDataGenerator.test.tsx +291 -0
- package/tests/components/VisualExplain.test.tsx +704 -0
- package/tests/components/admin/AdminDashboard.test.tsx +205 -0
- package/tests/components/admin/AuditTab.test.tsx +220 -0
- package/tests/components/admin/MonitoringEmbed.test.tsx +58 -0
- package/tests/components/admin/OperationsTab.test.tsx +975 -0
- package/tests/components/admin/OverviewTab.test.tsx +254 -0
- package/tests/components/admin/SecurityTab.test.tsx +467 -0
- package/tests/components/monitoring/MetricChart.test.tsx +111 -0
- package/tests/components/monitoring/MonitoringDashboard.test.tsx +259 -0
- package/tests/components/monitoring/OverviewTab.test.tsx +78 -0
- package/tests/components/monitoring/PerformanceTab.test.tsx +87 -0
- package/tests/components/monitoring/PoolTab.test.tsx +42 -0
- package/tests/components/monitoring/QueriesTab.test.tsx +80 -0
- package/tests/components/monitoring/SessionsTab.test.tsx +154 -0
- package/tests/components/monitoring/StorageTab.test.tsx +127 -0
- package/tests/components/monitoring/TablesTab.test.tsx +153 -0
- package/tests/components/results-grid/ResultCard.test.tsx +105 -0
- package/tests/components/results-grid/RowDetailSheet.test.tsx +308 -0
- package/tests/components/results-grid/StatsBar.test.tsx +162 -0
- package/tests/components/schema-explorer/ColumnList.test.tsx +151 -0
- package/tests/components/schema-explorer/SchemaExplorer.test.tsx +461 -0
- package/tests/components/schema-explorer/TableItem.test.tsx +415 -0
- package/tests/components/sidebar/ConnectionItem.test.tsx +201 -0
- package/tests/components/sidebar/ConnectionsList.test.tsx +176 -0
- package/tests/components/sidebar/Sidebar.test.tsx +187 -0
- package/tests/components/studio/BottomPanel.test.tsx +383 -0
- package/tests/components/studio/QueryToolbar.test.tsx +321 -0
- package/tests/components/studio/StudioDesktopHeader.test.tsx +377 -0
- package/tests/components/studio/StudioMobileHeader.test.tsx +198 -0
- package/tests/components/studio/StudioTabBar.test.tsx +331 -0
- package/tests/fixtures/connections.ts +96 -0
- package/tests/fixtures/masking-configs.ts +86 -0
- package/tests/fixtures/query-results.ts +71 -0
- package/tests/fixtures/schemas.ts +64 -0
- package/tests/fixtures/seed-connections/invalid-config.yaml +7 -0
- package/tests/fixtures/seed-connections/minimal-config.yaml +8 -0
- package/tests/fixtures/seed-connections/mixed-credentials.yaml +23 -0
- package/tests/fixtures/seed-connections/multi-role-config.yaml +30 -0
- package/tests/fixtures/seed-connections/valid-config.json +15 -0
- package/tests/fixtures/seed-connections/valid-config.yaml +51 -0
- package/tests/helpers/mock-fetch.ts +59 -0
- package/tests/helpers/mock-monaco.ts +112 -0
- package/tests/helpers/mock-navigation.ts +28 -0
- package/tests/helpers/mock-next.ts +80 -0
- package/tests/helpers/mock-provider.ts +133 -0
- package/tests/helpers/mock-sonner.ts +29 -0
- package/tests/helpers/render-with-providers.tsx +19 -0
- package/tests/hooks/use-ai-chat.test.ts +600 -0
- package/tests/hooks/use-auth.test.ts +371 -0
- package/tests/hooks/use-connection-form.test.ts +743 -0
- package/tests/hooks/use-connection-manager.test.ts +466 -0
- package/tests/hooks/use-inline-editing.test.ts +321 -0
- package/tests/hooks/use-mobile.test.ts +177 -0
- package/tests/hooks/use-monitoring-data.test.ts +819 -0
- package/tests/hooks/use-provider-metadata.test.ts +228 -0
- package/tests/hooks/use-query-execution.test.ts +1212 -0
- package/tests/hooks/use-tab-manager.test.ts +756 -0
- package/tests/hooks/use-toast.test.ts +74 -0
- package/tests/hooks/use-transaction-control.test.ts +211 -0
- package/tests/integration/db/mongodb-provider.test.ts +698 -0
- package/tests/integration/db/mssql-provider.test.ts +840 -0
- package/tests/integration/db/mysql-provider.test.ts +872 -0
- package/tests/integration/db/oracle-provider.test.ts +843 -0
- package/tests/integration/db/postgres-provider.test.ts +1382 -0
- package/tests/integration/db/redis-provider.test.ts +526 -0
- package/tests/integration/db/sqlite-provider.test.ts +480 -0
- package/tests/integration/seed/seed-pipeline.test.ts +102 -0
- package/tests/isolated/factory-singleton.test.ts +150 -0
- package/tests/isolated/use-storage-sync.test.ts +389 -0
- package/tests/run-components.sh +196 -0
- package/tests/setup-dom.ts +58 -0
- package/tests/setup.ts +40 -0
- package/tests/unit/api-errors.test.ts +210 -0
- package/tests/unit/code-generator-functions.test.ts +271 -0
- package/tests/unit/components/column-list.test.tsx +190 -0
- package/tests/unit/components/data-import-modal.test.tsx +441 -0
- package/tests/unit/components/studio-mobile-header.test.tsx +327 -0
- package/tests/unit/data-charts-functions.test.ts +496 -0
- package/tests/unit/data-import-functions.test.ts +320 -0
- package/tests/unit/data-import-utils.test.ts +125 -0
- package/tests/unit/db/base-provider.test.ts +517 -0
- package/tests/unit/db/errors.test.ts +403 -0
- package/tests/unit/db/factory.test.ts +436 -0
- package/tests/unit/db/pool-manager.test.ts +440 -0
- package/tests/unit/db/query-limiter.test.ts +387 -0
- package/tests/unit/db/sql-base.test.ts +438 -0
- package/tests/unit/lib/api/error-codes.test.ts +39 -0
- package/tests/unit/lib/audit.test.ts +326 -0
- package/tests/unit/lib/auth.test.ts +146 -0
- package/tests/unit/lib/connection-string-parser.test.ts +424 -0
- package/tests/unit/lib/data-masking.test.ts +583 -0
- package/tests/unit/lib/db-icons.test.tsx +41 -0
- package/tests/unit/lib/monitoring-thresholds.test.ts +133 -0
- package/tests/unit/lib/oidc.test.ts +509 -0
- package/tests/unit/lib/query-generators.test.ts +127 -0
- package/tests/unit/lib/storage/factory.test.ts +71 -0
- package/tests/unit/lib/storage/local-storage.test.ts +114 -0
- package/tests/unit/lib/storage/providers/postgres.test.ts +312 -0
- package/tests/unit/lib/storage/providers/sqlite.test.ts +232 -0
- package/tests/unit/lib/storage/storage-facade-extended.test.ts +331 -0
- package/tests/unit/lib/storage/storage-facade.test.ts +184 -0
- package/tests/unit/lib/storage.test.ts +317 -0
- package/tests/unit/lib/time-series-buffer.test.ts +212 -0
- package/tests/unit/lib/utils.test.ts +24 -0
- package/tests/unit/llm/base-provider.test.ts +238 -0
- package/tests/unit/llm/config.test.ts +262 -0
- package/tests/unit/llm/custom-provider.test.ts +281 -0
- package/tests/unit/llm/gemini-provider.test.ts +248 -0
- package/tests/unit/llm/llm-factory.test.ts +155 -0
- package/tests/unit/llm/ollama-provider.test.ts +288 -0
- package/tests/unit/llm/openai-provider.test.ts +324 -0
- package/tests/unit/llm/retry.test.ts +180 -0
- package/tests/unit/llm/streaming.test.ts +355 -0
- package/tests/unit/logger.test.ts +198 -0
- package/tests/unit/mongodb-completions.test.ts +516 -0
- package/tests/unit/pivot-table-functions.test.ts +76 -0
- package/tests/unit/query-cancelled-error.test.ts +81 -0
- package/tests/unit/schema-diff/diff-engine.test.ts +367 -0
- package/tests/unit/schema-diff/migration-generator.test.ts +513 -0
- package/tests/unit/seed/config-loader.test.ts +73 -0
- package/tests/unit/seed/connection-filter.test.ts +91 -0
- package/tests/unit/seed/credential-resolver.test.ts +85 -0
- package/tests/unit/seed/index.test.ts +72 -0
- package/tests/unit/seed/resolve-connection.test.ts +74 -0
- package/tests/unit/seed/types.test.ts +129 -0
- package/tests/unit/sql/alias-extractor.test.ts +444 -0
- package/tests/unit/sql/statement-splitter.test.ts +348 -0
- package/tests/unit/sql-completions.test.ts +463 -0
- package/tests/unit/ssh-tunnel.test.ts +465 -0
- package/tsconfig.json +42 -0
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import '../setup-dom';
|
|
2
|
+
import { mock } from 'bun:test';
|
|
3
|
+
import React from 'react';
|
|
4
|
+
|
|
5
|
+
// Mock AdminDashboard to avoid its massive dependency tree
|
|
6
|
+
mock.module('@/components/admin/AdminDashboard', () => ({
|
|
7
|
+
default: () => React.createElement('div', { 'data-testid': 'admin-dashboard' }, 'AdminDashboard Mock'),
|
|
8
|
+
}));
|
|
9
|
+
|
|
10
|
+
const { default: AdminPage } = await import('@/app/admin/page');
|
|
11
|
+
|
|
12
|
+
import { afterEach, describe, expect, test } from 'bun:test';
|
|
13
|
+
import { cleanup, render } from '@testing-library/react';
|
|
14
|
+
|
|
15
|
+
describe('AdminPage', () => {
|
|
16
|
+
afterEach(() => { cleanup(); });
|
|
17
|
+
|
|
18
|
+
test('renders AdminDashboard component', () => {
|
|
19
|
+
const { getByTestId } = render(<AdminPage />);
|
|
20
|
+
expect(getByTestId('admin-dashboard')).not.toBeNull();
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
test('renders AdminDashboard content', () => {
|
|
24
|
+
const { getByText } = render(<AdminPage />);
|
|
25
|
+
expect(getByText('AdminDashboard Mock')).not.toBeNull();
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
test('wraps AdminDashboard in Suspense', () => {
|
|
29
|
+
// Verify the component renders without throwing (Suspense boundary works)
|
|
30
|
+
const element = AdminPage();
|
|
31
|
+
expect(element.type).toBe(React.Suspense);
|
|
32
|
+
});
|
|
33
|
+
});
|
|
@@ -0,0 +1,182 @@
|
|
|
1
|
+
import '../setup-dom';
|
|
2
|
+
import '../helpers/mock-sonner';
|
|
3
|
+
import '../helpers/mock-navigation';
|
|
4
|
+
|
|
5
|
+
import React from 'react';
|
|
6
|
+
import { afterEach, describe, expect, mock, test } from 'bun:test';
|
|
7
|
+
import { cleanup, fireEvent, render } from '@testing-library/react';
|
|
8
|
+
import { CodeGenerator } from '@/components/CodeGenerator';
|
|
9
|
+
import type { TableSchema } from '@/lib/types';
|
|
10
|
+
|
|
11
|
+
const schema: TableSchema = {
|
|
12
|
+
name: 'users',
|
|
13
|
+
indexes: [],
|
|
14
|
+
columns: [
|
|
15
|
+
{ name: 'id', type: 'SERIAL', nullable: false, isPrimary: true },
|
|
16
|
+
{ name: 'email', type: 'VARCHAR(255)', nullable: false, isPrimary: false },
|
|
17
|
+
{ name: 'age', type: 'INTEGER', nullable: true, isPrimary: false },
|
|
18
|
+
{ name: 'is_active', type: 'BOOLEAN', nullable: false, isPrimary: false },
|
|
19
|
+
{ name: 'created_at', type: 'TIMESTAMP', nullable: true, isPrimary: false },
|
|
20
|
+
],
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
describe('CodeGenerator', () => {
|
|
24
|
+
afterEach(() => { cleanup(); });
|
|
25
|
+
|
|
26
|
+
test('does not render when isOpen is false', () => {
|
|
27
|
+
const { container } = render(
|
|
28
|
+
<CodeGenerator isOpen={false} onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
29
|
+
);
|
|
30
|
+
expect(container.textContent).toBe('');
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
test('renders TypeScript interface by default', () => {
|
|
34
|
+
const { queryByText, container } = render(
|
|
35
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
36
|
+
);
|
|
37
|
+
expect(queryByText('Code Generator')).not.toBeNull();
|
|
38
|
+
expect(container.textContent).toContain('export interface User');
|
|
39
|
+
expect(container.textContent).toContain('email: string');
|
|
40
|
+
expect(container.textContent).toContain('age: number | null');
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
test('switches language via dropdown', () => {
|
|
44
|
+
const { queryByText, container } = render(
|
|
45
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
46
|
+
);
|
|
47
|
+
fireEvent.click(queryByText('TypeScript Interface')!);
|
|
48
|
+
fireEvent.click(queryByText('Go Struct')!);
|
|
49
|
+
expect(container.textContent).toContain('type User struct');
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
test('copy button works', () => {
|
|
53
|
+
const writeText = mock(async (t: string) => { void t; });
|
|
54
|
+
Object.defineProperty(globalThis.navigator, 'clipboard', { value: { writeText }, configurable: true });
|
|
55
|
+
|
|
56
|
+
const { queryByText } = render(
|
|
57
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
58
|
+
);
|
|
59
|
+
fireEvent.click(queryByText('Copy')!);
|
|
60
|
+
expect(writeText).toHaveBeenCalledTimes(1);
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
test('close button fires onClose', () => {
|
|
64
|
+
const onClose = mock(() => {});
|
|
65
|
+
const { container } = render(
|
|
66
|
+
<CodeGenerator isOpen onClose={onClose} tableName="users" tableSchema={schema} />
|
|
67
|
+
);
|
|
68
|
+
const closeBtn = container.querySelector('button');
|
|
69
|
+
fireEvent.click(closeBtn!);
|
|
70
|
+
expect(onClose).toHaveBeenCalledTimes(1);
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
test('shows table name in header', () => {
|
|
74
|
+
const { queryAllByText } = render(
|
|
75
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="orders" tableSchema={schema} />
|
|
76
|
+
);
|
|
77
|
+
// Table name appears in header and footer
|
|
78
|
+
expect(queryAllByText('orders').length).toBeGreaterThanOrEqual(1);
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
test('shows database type when provided', () => {
|
|
82
|
+
const { queryByText } = render(
|
|
83
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} databaseType="postgres" />
|
|
84
|
+
);
|
|
85
|
+
expect(queryByText('postgres')).not.toBeNull();
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
test('does not show database type badge when not provided', () => {
|
|
89
|
+
const { queryByText } = render(
|
|
90
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
91
|
+
);
|
|
92
|
+
expect(queryByText('postgres')).toBeNull();
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
test('shows no schema message when tableSchema is null', () => {
|
|
96
|
+
const { container } = render(
|
|
97
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={null} />
|
|
98
|
+
);
|
|
99
|
+
expect(container.textContent).toContain('No schema available');
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
test('switches to Zod schema', () => {
|
|
103
|
+
const { queryByText, container } = render(
|
|
104
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
105
|
+
);
|
|
106
|
+
fireEvent.click(queryByText('TypeScript Interface')!);
|
|
107
|
+
fireEvent.click(queryByText('Zod Schema')!);
|
|
108
|
+
expect(container.textContent).toContain('z.object');
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
test('switches to Prisma model', () => {
|
|
112
|
+
const { queryByText, container } = render(
|
|
113
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
114
|
+
);
|
|
115
|
+
fireEvent.click(queryByText('TypeScript Interface')!);
|
|
116
|
+
fireEvent.click(queryByText('Prisma Model')!);
|
|
117
|
+
expect(container.textContent).toContain('model User');
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
test('switches to Python dataclass', () => {
|
|
121
|
+
const { queryByText, container } = render(
|
|
122
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
123
|
+
);
|
|
124
|
+
fireEvent.click(queryByText('TypeScript Interface')!);
|
|
125
|
+
fireEvent.click(queryByText('Python Dataclass')!);
|
|
126
|
+
expect(container.textContent).toContain('@dataclass');
|
|
127
|
+
expect(container.textContent).toContain('class User:');
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
test('switches to Java POJO', () => {
|
|
131
|
+
const { queryByText, container } = render(
|
|
132
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
133
|
+
);
|
|
134
|
+
fireEvent.click(queryByText('TypeScript Interface')!);
|
|
135
|
+
fireEvent.click(queryByText('Java POJO')!);
|
|
136
|
+
expect(container.textContent).toContain('public class User');
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
test('footer shows column count and format', () => {
|
|
140
|
+
const { queryByText } = render(
|
|
141
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
142
|
+
);
|
|
143
|
+
expect(queryByText(/5 columns/)).not.toBeNull();
|
|
144
|
+
expect(queryByText(/ts format/)).not.toBeNull();
|
|
145
|
+
});
|
|
146
|
+
|
|
147
|
+
test('footer format changes with language', () => {
|
|
148
|
+
const { queryByText } = render(
|
|
149
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
150
|
+
);
|
|
151
|
+
fireEvent.click(queryByText('TypeScript Interface')!);
|
|
152
|
+
fireEvent.click(queryByText('Go Struct')!);
|
|
153
|
+
expect(queryByText(/go format/)).not.toBeNull();
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
test('copy button shows Copied! after click', () => {
|
|
157
|
+
const writeText = mock(async (t: string) => { void t; });
|
|
158
|
+
Object.defineProperty(globalThis.navigator, 'clipboard', { value: { writeText }, configurable: true });
|
|
159
|
+
|
|
160
|
+
const { queryByText } = render(
|
|
161
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
162
|
+
);
|
|
163
|
+
fireEvent.click(queryByText('Copy')!);
|
|
164
|
+
expect(queryByText('Copied!')).not.toBeNull();
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
test('dropdown closes after selecting language', () => {
|
|
168
|
+
const { queryByText, queryAllByText } = render(
|
|
169
|
+
<CodeGenerator isOpen onClose={mock(() => {})} tableName="users" tableSchema={schema} />
|
|
170
|
+
);
|
|
171
|
+
// Open dropdown
|
|
172
|
+
fireEvent.click(queryByText('TypeScript Interface')!);
|
|
173
|
+
// All languages should be visible
|
|
174
|
+
expect(queryByText('Go Struct')).not.toBeNull();
|
|
175
|
+
// Select one
|
|
176
|
+
fireEvent.click(queryByText('Go Struct')!);
|
|
177
|
+
// Dropdown should close — 'Python Dataclass' should not be in dropdown (only in button would be Go Struct)
|
|
178
|
+
const pythonItems = queryAllByText('Python Dataclass');
|
|
179
|
+
// After closing, the dropdown items are gone
|
|
180
|
+
expect(pythonItems.length).toBe(0);
|
|
181
|
+
});
|
|
182
|
+
});
|
|
@@ -0,0 +1,428 @@
|
|
|
1
|
+
import '../setup-dom';
|
|
2
|
+
import '../helpers/mock-sonner';
|
|
3
|
+
import '../helpers/mock-navigation';
|
|
4
|
+
|
|
5
|
+
import { mock } from 'bun:test';
|
|
6
|
+
|
|
7
|
+
// Mock cmdk before component import
|
|
8
|
+
mock.module('cmdk', () => {
|
|
9
|
+
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
|
10
|
+
const React = require('react');
|
|
11
|
+
const Command = React.forwardRef(({ children, ...props }: Record<string, unknown>, ref: React.Ref<HTMLElement>) =>
|
|
12
|
+
React.createElement('div', { ...props, ref, 'data-testid': 'command' }, children));
|
|
13
|
+
Command.displayName = 'Command';
|
|
14
|
+
|
|
15
|
+
const CommandInput = React.forwardRef((props: Record<string, unknown>, ref: React.Ref<HTMLElement>) =>
|
|
16
|
+
React.createElement('input', { ...props, ref, 'data-testid': 'command-input' }));
|
|
17
|
+
CommandInput.displayName = 'CommandInput';
|
|
18
|
+
Command.Input = CommandInput;
|
|
19
|
+
|
|
20
|
+
const CommandList = ({ children, ...props }: Record<string, unknown>) =>
|
|
21
|
+
React.createElement('div', { ...props, 'data-testid': 'command-list' }, children);
|
|
22
|
+
CommandList.displayName = 'CommandList';
|
|
23
|
+
Command.List = CommandList;
|
|
24
|
+
|
|
25
|
+
const CommandEmpty = ({ children }: Record<string, unknown>) =>
|
|
26
|
+
React.createElement('div', { 'data-testid': 'command-empty' }, children);
|
|
27
|
+
CommandEmpty.displayName = 'CommandEmpty';
|
|
28
|
+
Command.Empty = CommandEmpty;
|
|
29
|
+
|
|
30
|
+
const CommandGroup = ({ children, heading, ...props }: Record<string, unknown>) =>
|
|
31
|
+
React.createElement('div', { ...props, 'data-testid': `command-group-${heading}` },
|
|
32
|
+
React.createElement('div', null, heading), children);
|
|
33
|
+
CommandGroup.displayName = 'CommandGroup';
|
|
34
|
+
Command.Group = CommandGroup;
|
|
35
|
+
|
|
36
|
+
const CommandItem = ({ children, onSelect, ...props }: Record<string, unknown>) =>
|
|
37
|
+
React.createElement('div', { ...props, onClick: onSelect, role: 'option', 'data-testid': 'command-item' }, children);
|
|
38
|
+
CommandItem.displayName = 'CommandItem';
|
|
39
|
+
Command.Item = CommandItem;
|
|
40
|
+
|
|
41
|
+
const CommandSeparator = () => null;
|
|
42
|
+
CommandSeparator.displayName = 'CommandSeparator';
|
|
43
|
+
Command.Separator = CommandSeparator;
|
|
44
|
+
|
|
45
|
+
return { Command };
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
// Mock storage
|
|
49
|
+
mock.module('@/lib/storage', () => ({
|
|
50
|
+
storage: {
|
|
51
|
+
getSavedQueries: mock(() => []),
|
|
52
|
+
getHistory: mock(() => []),
|
|
53
|
+
},
|
|
54
|
+
}));
|
|
55
|
+
|
|
56
|
+
// Mock db-ui-config
|
|
57
|
+
mock.module('@/lib/db-ui-config', () => ({
|
|
58
|
+
getDBIcon: () => {
|
|
59
|
+
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
|
60
|
+
const React = require('react');
|
|
61
|
+
const MockDBIcon = (props: Record<string, unknown>) => React.createElement('span', { ...props, 'data-testid': 'db-icon' });
|
|
62
|
+
MockDBIcon.displayName = 'MockDBIcon';
|
|
63
|
+
return MockDBIcon;
|
|
64
|
+
},
|
|
65
|
+
getDBConfig: () => ({ icon: () => null, color: 'text-blue-400', label: 'PostgreSQL', defaultPort: '5432' }),
|
|
66
|
+
getDBColor: () => 'text-blue-400',
|
|
67
|
+
}));
|
|
68
|
+
|
|
69
|
+
import { describe, test, expect, afterEach } from 'bun:test';
|
|
70
|
+
import { render, fireEvent, cleanup } from '@testing-library/react';
|
|
71
|
+
import React from 'react';
|
|
72
|
+
|
|
73
|
+
import { CommandPalette } from '@/components/CommandPalette';
|
|
74
|
+
import { storage } from '@/lib/storage';
|
|
75
|
+
import { mockPostgresConnection, mockMySQLConnection, mockSQLiteConnection, mockMongoDBConnection, mockRedisConnection, mockOracleConnection } from '../fixtures/connections';
|
|
76
|
+
import { mockSchema } from '../fixtures/schemas';
|
|
77
|
+
|
|
78
|
+
// =============================================================================
|
|
79
|
+
// CommandPalette Tests
|
|
80
|
+
// =============================================================================
|
|
81
|
+
|
|
82
|
+
function createDefaultProps(overrides: Partial<Parameters<typeof CommandPalette>[0]> = {}) {
|
|
83
|
+
return {
|
|
84
|
+
connections: [mockPostgresConnection, mockMySQLConnection],
|
|
85
|
+
activeConnection: mockPostgresConnection,
|
|
86
|
+
schema: mockSchema,
|
|
87
|
+
onSelectConnection: mock(() => {}),
|
|
88
|
+
onTableClick: mock(() => {}),
|
|
89
|
+
onAddConnection: mock(() => {}),
|
|
90
|
+
onExecuteQuery: mock(() => {}),
|
|
91
|
+
onLoadSavedQuery: mock(() => {}),
|
|
92
|
+
onLoadHistoryQuery: mock(() => {}),
|
|
93
|
+
onNavigateHealth: mock(() => {}),
|
|
94
|
+
onNavigateMonitoring: mock(() => {}),
|
|
95
|
+
onShowDiagram: mock(() => {}),
|
|
96
|
+
onFormatQuery: mock(() => {}),
|
|
97
|
+
onSaveQuery: mock(() => {}),
|
|
98
|
+
onToggleAI: mock(() => {}),
|
|
99
|
+
onLogout: mock(() => {}),
|
|
100
|
+
...overrides,
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
describe('CommandPalette', () => {
|
|
105
|
+
afterEach(() => {
|
|
106
|
+
cleanup();
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
test('renders without crashing (dialog not visible initially)', () => {
|
|
110
|
+
const props = createDefaultProps();
|
|
111
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
112
|
+
// Dialog starts closed (open=false), so content is not rendered
|
|
113
|
+
expect(queryByText('Run Query')).toBeNull();
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
test('Cmd+K keyboard shortcut opens dialog', () => {
|
|
117
|
+
const props = createDefaultProps();
|
|
118
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
119
|
+
|
|
120
|
+
// Initially, dialog content is not visible
|
|
121
|
+
expect(queryByText('Run Query')).toBeNull();
|
|
122
|
+
|
|
123
|
+
// Fire Cmd+K to open the dialog
|
|
124
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
125
|
+
|
|
126
|
+
// After opening, the actions should be visible
|
|
127
|
+
expect(queryByText('Run Query')).not.toBeNull();
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
test('shows connections when dialog is open', () => {
|
|
131
|
+
const props = createDefaultProps();
|
|
132
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
133
|
+
|
|
134
|
+
// Open dialog
|
|
135
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
136
|
+
|
|
137
|
+
// Connection names should appear
|
|
138
|
+
expect(queryByText('Test PostgreSQL')).not.toBeNull();
|
|
139
|
+
expect(queryByText('Test MySQL')).not.toBeNull();
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
test('shows tables from schema when dialog is open', () => {
|
|
143
|
+
const props = createDefaultProps();
|
|
144
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
145
|
+
|
|
146
|
+
// Open dialog
|
|
147
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
148
|
+
|
|
149
|
+
// Table names from mockSchema
|
|
150
|
+
expect(queryByText('users')).not.toBeNull();
|
|
151
|
+
expect(queryByText('orders')).not.toBeNull();
|
|
152
|
+
expect(queryByText('products')).not.toBeNull();
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
test('active connection gets "Active" badge', () => {
|
|
156
|
+
const props = createDefaultProps({
|
|
157
|
+
activeConnection: mockPostgresConnection,
|
|
158
|
+
});
|
|
159
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
160
|
+
|
|
161
|
+
// Open dialog
|
|
162
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
163
|
+
|
|
164
|
+
// The active connection should have "Active" text
|
|
165
|
+
expect(queryByText('Active')).not.toBeNull();
|
|
166
|
+
});
|
|
167
|
+
|
|
168
|
+
test('no "Active" badge when no connection is active', () => {
|
|
169
|
+
const props = createDefaultProps({
|
|
170
|
+
activeConnection: null,
|
|
171
|
+
});
|
|
172
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
173
|
+
|
|
174
|
+
// Open dialog
|
|
175
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
176
|
+
|
|
177
|
+
expect(queryByText('Active')).toBeNull();
|
|
178
|
+
});
|
|
179
|
+
|
|
180
|
+
test('callbacks fire when action items clicked', () => {
|
|
181
|
+
const onExecuteQuery = mock(() => {});
|
|
182
|
+
const props = createDefaultProps({ onExecuteQuery });
|
|
183
|
+
const { getByText } = render(<CommandPalette {...props} />);
|
|
184
|
+
|
|
185
|
+
// Open dialog
|
|
186
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
187
|
+
|
|
188
|
+
// Find and click "Run Query" item
|
|
189
|
+
const runQueryText = getByText('Run Query');
|
|
190
|
+
const commandItem = runQueryText.closest('[role="option"]');
|
|
191
|
+
expect(commandItem).not.toBeNull();
|
|
192
|
+
fireEvent.click(commandItem!);
|
|
193
|
+
|
|
194
|
+
// runAction calls setOpen(false) then setTimeout(action, 100)
|
|
195
|
+
// Verify dialog closed (content disappears since open becomes false)
|
|
196
|
+
});
|
|
197
|
+
|
|
198
|
+
test('dialog closes after Cmd+K toggle', () => {
|
|
199
|
+
const props = createDefaultProps();
|
|
200
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
201
|
+
|
|
202
|
+
// Open dialog
|
|
203
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
204
|
+
expect(queryByText('Run Query')).not.toBeNull();
|
|
205
|
+
|
|
206
|
+
// Close dialog by pressing Cmd+K again
|
|
207
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
208
|
+
expect(queryByText('Run Query')).toBeNull();
|
|
209
|
+
});
|
|
210
|
+
|
|
211
|
+
// ===========================================================================
|
|
212
|
+
// Additional Tests
|
|
213
|
+
// ===========================================================================
|
|
214
|
+
|
|
215
|
+
test('Ctrl+K keyboard shortcut opens dialog (Windows)', () => {
|
|
216
|
+
const props = createDefaultProps();
|
|
217
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
218
|
+
|
|
219
|
+
// Initially closed
|
|
220
|
+
expect(queryByText('Run Query')).toBeNull();
|
|
221
|
+
|
|
222
|
+
// Fire Ctrl+K (Windows shortcut) to open the dialog
|
|
223
|
+
fireEvent.keyDown(document, { key: 'k', ctrlKey: true });
|
|
224
|
+
|
|
225
|
+
// After opening, the actions should be visible
|
|
226
|
+
expect(queryByText('Run Query')).not.toBeNull();
|
|
227
|
+
});
|
|
228
|
+
|
|
229
|
+
test('Schema Diagram is hidden when no activeConnection', () => {
|
|
230
|
+
const props = createDefaultProps({ activeConnection: null });
|
|
231
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
232
|
+
|
|
233
|
+
// Open dialog
|
|
234
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
235
|
+
|
|
236
|
+
// Schema Diagram (ERD) should not be rendered
|
|
237
|
+
expect(queryByText('Schema Diagram (ERD)')).toBeNull();
|
|
238
|
+
});
|
|
239
|
+
|
|
240
|
+
test('Schema Diagram is visible with activeConnection', () => {
|
|
241
|
+
const props = createDefaultProps({ activeConnection: mockPostgresConnection });
|
|
242
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
243
|
+
|
|
244
|
+
// Open dialog
|
|
245
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
246
|
+
|
|
247
|
+
// Schema Diagram (ERD) should be rendered
|
|
248
|
+
expect(queryByText('Schema Diagram (ERD)')).not.toBeNull();
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
test('Saved Queries group renders from storage', () => {
|
|
252
|
+
const mockSaved = [
|
|
253
|
+
{ id: 'sq-1', name: 'Get active users', query: 'SELECT * FROM users WHERE is_active = true', connectionType: 'postgres' as const, createdAt: new Date(), updatedAt: new Date() },
|
|
254
|
+
{ id: 'sq-2', name: 'Count orders', query: 'SELECT COUNT(*) FROM orders', connectionType: 'postgres' as const, createdAt: new Date(), updatedAt: new Date() },
|
|
255
|
+
];
|
|
256
|
+
(storage.getSavedQueries as ReturnType<typeof mock>).mockReturnValue(mockSaved);
|
|
257
|
+
|
|
258
|
+
const props = createDefaultProps();
|
|
259
|
+
const { queryByText, getByTestId } = render(<CommandPalette {...props} />);
|
|
260
|
+
|
|
261
|
+
// Open dialog
|
|
262
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
263
|
+
|
|
264
|
+
// Saved Queries group heading should appear
|
|
265
|
+
expect(getByTestId('command-group-Saved Queries')).not.toBeNull();
|
|
266
|
+
expect(queryByText('Get active users')).not.toBeNull();
|
|
267
|
+
expect(queryByText('Count orders')).not.toBeNull();
|
|
268
|
+
|
|
269
|
+
// Restore default
|
|
270
|
+
(storage.getSavedQueries as ReturnType<typeof mock>).mockReturnValue([]);
|
|
271
|
+
});
|
|
272
|
+
|
|
273
|
+
test('Recent Queries group renders from history', () => {
|
|
274
|
+
const mockHistory = [
|
|
275
|
+
{ id: 'h-1', connectionId: 'test-pg-1', query: 'SELECT * FROM users LIMIT 10', executionTime: 42, status: 'success' as const, executedAt: new Date() },
|
|
276
|
+
{ id: 'h-2', connectionId: 'test-pg-1', query: 'SELECT * FROM orders WHERE total > 100', executionTime: 78, status: 'success' as const, executedAt: new Date() },
|
|
277
|
+
];
|
|
278
|
+
(storage.getHistory as ReturnType<typeof mock>).mockReturnValue(mockHistory);
|
|
279
|
+
|
|
280
|
+
const props = createDefaultProps();
|
|
281
|
+
const { getByTestId, queryByText } = render(<CommandPalette {...props} />);
|
|
282
|
+
|
|
283
|
+
// Open dialog
|
|
284
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
285
|
+
|
|
286
|
+
// Recent Queries group heading should appear
|
|
287
|
+
expect(getByTestId('command-group-Recent Queries')).not.toBeNull();
|
|
288
|
+
expect(queryByText('42ms')).not.toBeNull();
|
|
289
|
+
expect(queryByText('78ms')).not.toBeNull();
|
|
290
|
+
|
|
291
|
+
// Restore default
|
|
292
|
+
(storage.getHistory as ReturnType<typeof mock>).mockReturnValue([]);
|
|
293
|
+
});
|
|
294
|
+
|
|
295
|
+
test('table column and row count display', () => {
|
|
296
|
+
const props = createDefaultProps();
|
|
297
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
298
|
+
|
|
299
|
+
// Open dialog
|
|
300
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
301
|
+
|
|
302
|
+
// users table: 6 columns, 100 rows
|
|
303
|
+
expect(queryByText('6 cols / 100 rows')).not.toBeNull();
|
|
304
|
+
// orders table: 5 columns, 500 rows
|
|
305
|
+
expect(queryByText('5 cols / 500 rows')).not.toBeNull();
|
|
306
|
+
// products table: 4 columns, 50 rows
|
|
307
|
+
expect(queryByText('4 cols / 50 rows')).not.toBeNull();
|
|
308
|
+
});
|
|
309
|
+
|
|
310
|
+
test('Format Query action callback fires via runAction', () => {
|
|
311
|
+
const onFormatQuery = mock(() => {});
|
|
312
|
+
const props = createDefaultProps({ onFormatQuery });
|
|
313
|
+
const { getByText } = render(<CommandPalette {...props} />);
|
|
314
|
+
|
|
315
|
+
// Open dialog
|
|
316
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
317
|
+
|
|
318
|
+
// Click Format Query item
|
|
319
|
+
const formatItem = getByText('Format Query').closest('[role="option"]');
|
|
320
|
+
expect(formatItem).not.toBeNull();
|
|
321
|
+
fireEvent.click(formatItem!);
|
|
322
|
+
|
|
323
|
+
// Dialog should close (content disappears)
|
|
324
|
+
expect(getByText).toBeDefined();
|
|
325
|
+
});
|
|
326
|
+
|
|
327
|
+
test('Save Query action callback fires via runAction', () => {
|
|
328
|
+
const onSaveQuery = mock(() => {});
|
|
329
|
+
const props = createDefaultProps({ onSaveQuery });
|
|
330
|
+
const { getByText } = render(<CommandPalette {...props} />);
|
|
331
|
+
|
|
332
|
+
// Open dialog
|
|
333
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
334
|
+
|
|
335
|
+
// Click Save Current Query item
|
|
336
|
+
const saveItem = getByText('Save Current Query').closest('[role="option"]');
|
|
337
|
+
expect(saveItem).not.toBeNull();
|
|
338
|
+
fireEvent.click(saveItem!);
|
|
339
|
+
});
|
|
340
|
+
|
|
341
|
+
test('AI Assistant action callback fires via runAction', () => {
|
|
342
|
+
const onToggleAI = mock(() => {});
|
|
343
|
+
const props = createDefaultProps({ onToggleAI });
|
|
344
|
+
const { getByText } = render(<CommandPalette {...props} />);
|
|
345
|
+
|
|
346
|
+
// Open dialog
|
|
347
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
348
|
+
|
|
349
|
+
// Click AI Assistant item
|
|
350
|
+
const aiItem = getByText('AI Assistant').closest('[role="option"]');
|
|
351
|
+
expect(aiItem).not.toBeNull();
|
|
352
|
+
fireEvent.click(aiItem!);
|
|
353
|
+
});
|
|
354
|
+
|
|
355
|
+
test('Logout action callback fires via runAction', () => {
|
|
356
|
+
const onLogout = mock(() => {});
|
|
357
|
+
const props = createDefaultProps({ onLogout });
|
|
358
|
+
const { getByText } = render(<CommandPalette {...props} />);
|
|
359
|
+
|
|
360
|
+
// Open dialog
|
|
361
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
362
|
+
|
|
363
|
+
// Click Logout item
|
|
364
|
+
const logoutItem = getByText('Logout').closest('[role="option"]');
|
|
365
|
+
expect(logoutItem).not.toBeNull();
|
|
366
|
+
fireEvent.click(logoutItem!);
|
|
367
|
+
});
|
|
368
|
+
|
|
369
|
+
test('connection item click calls onSelectConnection', () => {
|
|
370
|
+
const onSelectConnection = mock(() => {});
|
|
371
|
+
const props = createDefaultProps({ onSelectConnection });
|
|
372
|
+
const { getByText } = render(<CommandPalette {...props} />);
|
|
373
|
+
|
|
374
|
+
// Open dialog
|
|
375
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
376
|
+
|
|
377
|
+
// Click the MySQL connection item
|
|
378
|
+
const mysqlItem = getByText('Test MySQL').closest('[role="option"]');
|
|
379
|
+
expect(mysqlItem).not.toBeNull();
|
|
380
|
+
fireEvent.click(mysqlItem!);
|
|
381
|
+
|
|
382
|
+
// runAction calls setOpen(false) then setTimeout(callback, 100)
|
|
383
|
+
// The dialog should close immediately
|
|
384
|
+
});
|
|
385
|
+
|
|
386
|
+
test('table item click calls onTableClick', () => {
|
|
387
|
+
const onTableClick = mock(() => {});
|
|
388
|
+
const props = createDefaultProps({ onTableClick });
|
|
389
|
+
const { getByText } = render(<CommandPalette {...props} />);
|
|
390
|
+
|
|
391
|
+
// Open dialog
|
|
392
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
393
|
+
|
|
394
|
+
// Click the "orders" table item
|
|
395
|
+
const ordersItem = getByText('orders').closest('[role="option"]');
|
|
396
|
+
expect(ordersItem).not.toBeNull();
|
|
397
|
+
fireEvent.click(ordersItem!);
|
|
398
|
+
});
|
|
399
|
+
|
|
400
|
+
test('DB icon is rendered for each connection', () => {
|
|
401
|
+
const props = createDefaultProps({
|
|
402
|
+
connections: [mockPostgresConnection, mockMySQLConnection, mockSQLiteConnection, mockMongoDBConnection, mockRedisConnection, mockOracleConnection],
|
|
403
|
+
});
|
|
404
|
+
const { getAllByTestId } = render(<CommandPalette {...props} />);
|
|
405
|
+
|
|
406
|
+
// Open dialog
|
|
407
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
408
|
+
|
|
409
|
+
// Each connection should render a db-icon via the mocked getDBIcon
|
|
410
|
+
const dbIcons = getAllByTestId('db-icon');
|
|
411
|
+
expect(dbIcons.length).toBe(6);
|
|
412
|
+
});
|
|
413
|
+
|
|
414
|
+
test('"No results found" is shown in empty command state', () => {
|
|
415
|
+
const props = createDefaultProps({
|
|
416
|
+
connections: [],
|
|
417
|
+
schema: [],
|
|
418
|
+
activeConnection: null,
|
|
419
|
+
});
|
|
420
|
+
const { queryByText } = render(<CommandPalette {...props} />);
|
|
421
|
+
|
|
422
|
+
// Open dialog
|
|
423
|
+
fireEvent.keyDown(document, { key: 'k', metaKey: true });
|
|
424
|
+
|
|
425
|
+
// CommandEmpty renders "No results found." text
|
|
426
|
+
expect(queryByText('No results found.')).not.toBeNull();
|
|
427
|
+
});
|
|
428
|
+
});
|