@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,760 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import React, { useRef, useEffect, useState, useMemo, forwardRef, useImperativeHandle, useCallback } from 'react';
|
|
4
|
+
import Editor, { useMonaco } from '@monaco-editor/react';
|
|
5
|
+
import type * as Monaco from 'monaco-editor';
|
|
6
|
+
import { Zap, Sparkles, Send, X, Loader2, AlignLeft, Trash2, Copy, Play, Hash } from 'lucide-react';
|
|
7
|
+
import { cn } from '@/lib/utils';
|
|
8
|
+
import { motion, AnimatePresence } from 'framer-motion';
|
|
9
|
+
import { format } from 'sql-formatter';
|
|
10
|
+
import { registerSQLCompletionProvider } from '@/lib/editor/sql-completions';
|
|
11
|
+
import type { SchemaCompletionCache, SchemaColumnItem } from '@/lib/editor/sql-completions';
|
|
12
|
+
import { registerMongoDBCompletionProvider } from '@/lib/editor/mongodb-completions';
|
|
13
|
+
import { useAiChat } from '@/hooks/use-ai-chat';
|
|
14
|
+
|
|
15
|
+
export interface QueryEditorRef {
|
|
16
|
+
getSelectedText: () => string;
|
|
17
|
+
getEffectiveQuery: () => string;
|
|
18
|
+
getValue: () => string;
|
|
19
|
+
setValue: (value: string) => void;
|
|
20
|
+
focus: () => void;
|
|
21
|
+
format: () => void;
|
|
22
|
+
toggleAi: () => void;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
interface QueryEditorProps {
|
|
26
|
+
/** Initial value for the editor. Changes to this prop will update the editor content. */
|
|
27
|
+
value: string;
|
|
28
|
+
/** Optional callback for value changes. Only called on blur, execute, or explicit sync - NOT on every keystroke. */
|
|
29
|
+
onChange?: (val: string) => void;
|
|
30
|
+
/** Called when content changes in real-time. Use sparingly as it triggers on every keystroke. */
|
|
31
|
+
onContentChange?: (val: string) => void;
|
|
32
|
+
onExplain?: () => void;
|
|
33
|
+
language?: 'sql' | 'json';
|
|
34
|
+
tables?: string[];
|
|
35
|
+
databaseType?: string;
|
|
36
|
+
schemaContext?: string;
|
|
37
|
+
capabilities?: import('@/lib/db/types').ProviderCapabilities;
|
|
38
|
+
/** Optional API adapter: when provided, bypasses the built-in /api/ai/chat fetch. */
|
|
39
|
+
onAiChat?: (params: { prompt: string; schemaContext: string; history: { role: string; content: string }[] }) => Promise<string>;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
interface ParsedTable {
|
|
43
|
+
name: string;
|
|
44
|
+
rowCount?: number;
|
|
45
|
+
columns?: Array<{
|
|
46
|
+
name: string;
|
|
47
|
+
type: string;
|
|
48
|
+
isPrimary?: boolean;
|
|
49
|
+
}>;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
// Static editor options - defined outside component to prevent re-creation on every render
|
|
53
|
+
const getEditorOptions = (showLineNumbers: boolean) => ({
|
|
54
|
+
minimap: { enabled: false },
|
|
55
|
+
fontSize: 13,
|
|
56
|
+
fontFamily: '"JetBrains Mono", "Fira Code", Menlo, Monaco, Consolas, monospace',
|
|
57
|
+
lineNumbers: showLineNumbers ? ('on' as const) : ('off' as const),
|
|
58
|
+
roundedSelection: true,
|
|
59
|
+
scrollBeyondLastLine: false,
|
|
60
|
+
readOnly: false,
|
|
61
|
+
automaticLayout: true,
|
|
62
|
+
padding: { top: 12 },
|
|
63
|
+
cursorSmoothCaretAnimation: 'on' as const,
|
|
64
|
+
cursorBlinking: 'smooth' as const,
|
|
65
|
+
smoothScrolling: true,
|
|
66
|
+
contextmenu: true,
|
|
67
|
+
renderLineHighlight: 'all' as const,
|
|
68
|
+
bracketPairColorization: { enabled: true },
|
|
69
|
+
guides: { indentation: true },
|
|
70
|
+
scrollbar: {
|
|
71
|
+
vertical: 'visible' as const,
|
|
72
|
+
horizontal: 'visible' as const,
|
|
73
|
+
verticalScrollbarSize: 8,
|
|
74
|
+
horizontalScrollbarSize: 8,
|
|
75
|
+
},
|
|
76
|
+
fontLigatures: true,
|
|
77
|
+
suggestOnTriggerCharacters: true,
|
|
78
|
+
quickSuggestions: {
|
|
79
|
+
other: true,
|
|
80
|
+
comments: false,
|
|
81
|
+
strings: true
|
|
82
|
+
},
|
|
83
|
+
parameterHints: {
|
|
84
|
+
enabled: true
|
|
85
|
+
}
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
export const QueryEditor = forwardRef<QueryEditorRef, QueryEditorProps>(({
|
|
89
|
+
value,
|
|
90
|
+
onChange,
|
|
91
|
+
onContentChange,
|
|
92
|
+
onExplain,
|
|
93
|
+
language = 'sql',
|
|
94
|
+
tables = [],
|
|
95
|
+
databaseType,
|
|
96
|
+
schemaContext,
|
|
97
|
+
capabilities,
|
|
98
|
+
onAiChat,
|
|
99
|
+
}, ref) => {
|
|
100
|
+
const monaco = useMonaco();
|
|
101
|
+
const editorRef = useRef<Monaco.editor.IStandaloneCodeEditor | null>(null);
|
|
102
|
+
const [hasSelection, setHasSelection] = useState(false);
|
|
103
|
+
|
|
104
|
+
// Line numbers toggle state (persisted in localStorage)
|
|
105
|
+
const [showLineNumbers, setShowLineNumbers] = useState<boolean>(() => {
|
|
106
|
+
if (typeof window !== 'undefined') {
|
|
107
|
+
const saved = localStorage.getItem('editor-line-numbers');
|
|
108
|
+
return saved !== null ? saved === 'true' : true; // default: true
|
|
109
|
+
}
|
|
110
|
+
return true;
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
// Track last synced value to detect external changes
|
|
114
|
+
const lastSyncedValueRef = useRef<string>(value);
|
|
115
|
+
const isInternalChangeRef = useRef<boolean>(false);
|
|
116
|
+
|
|
117
|
+
// Sync editor content when value prop changes externally (e.g., tab switch)
|
|
118
|
+
useEffect(() => {
|
|
119
|
+
if (editorRef.current && value !== lastSyncedValueRef.current) {
|
|
120
|
+
const currentEditorValue = editorRef.current.getValue();
|
|
121
|
+
// Only update if the new value is different from current editor content
|
|
122
|
+
// This prevents unnecessary updates when we're the source of the change
|
|
123
|
+
if (value !== currentEditorValue) {
|
|
124
|
+
isInternalChangeRef.current = true;
|
|
125
|
+
editorRef.current.setValue(value);
|
|
126
|
+
lastSyncedValueRef.current = value;
|
|
127
|
+
isInternalChangeRef.current = false;
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
}, [value]);
|
|
131
|
+
|
|
132
|
+
// Update editor options when line numbers toggle changes
|
|
133
|
+
useEffect(() => {
|
|
134
|
+
if (editorRef.current) {
|
|
135
|
+
editorRef.current.updateOptions({ lineNumbers: showLineNumbers ? 'on' : 'off' });
|
|
136
|
+
}
|
|
137
|
+
}, [showLineNumbers]);
|
|
138
|
+
|
|
139
|
+
// Persist line numbers preference to localStorage
|
|
140
|
+
useEffect(() => {
|
|
141
|
+
if (typeof window !== 'undefined') {
|
|
142
|
+
localStorage.setItem('editor-line-numbers', String(showLineNumbers));
|
|
143
|
+
}
|
|
144
|
+
}, [showLineNumbers]);
|
|
145
|
+
|
|
146
|
+
const parsedSchema = useMemo((): ParsedTable[] => {
|
|
147
|
+
if (!schemaContext) return [];
|
|
148
|
+
try {
|
|
149
|
+
return JSON.parse(schemaContext);
|
|
150
|
+
} catch (e) {
|
|
151
|
+
console.error('Failed to parse schema context for editor:', e);
|
|
152
|
+
return [];
|
|
153
|
+
}
|
|
154
|
+
}, [schemaContext]);
|
|
155
|
+
|
|
156
|
+
// Pre-compute schema-based completion items for faster lookups
|
|
157
|
+
const schemaCompletionCache = useMemo((): SchemaCompletionCache => {
|
|
158
|
+
const tableItems: SchemaCompletionCache['tableItems'] = [];
|
|
159
|
+
const columnMap = new Map<string, SchemaColumnItem[]>();
|
|
160
|
+
const allColumns = new Map<string, SchemaColumnItem>();
|
|
161
|
+
|
|
162
|
+
parsedSchema.forEach((table) => {
|
|
163
|
+
const tableLower = table.name.toLowerCase();
|
|
164
|
+
tableItems.push({
|
|
165
|
+
label: table.name,
|
|
166
|
+
labelLower: tableLower,
|
|
167
|
+
rowCount: table.rowCount || 0,
|
|
168
|
+
columnNames: table.columns?.map((c) => c.name).join(', ') || ''
|
|
169
|
+
});
|
|
170
|
+
|
|
171
|
+
const tableColumns: SchemaColumnItem[] = [];
|
|
172
|
+
table.columns?.forEach((col) => {
|
|
173
|
+
const colItem: SchemaColumnItem = {
|
|
174
|
+
label: col.name,
|
|
175
|
+
labelLower: col.name.toLowerCase(),
|
|
176
|
+
type: col.type,
|
|
177
|
+
isPrimary: col.isPrimary || false,
|
|
178
|
+
tableName: table.name
|
|
179
|
+
};
|
|
180
|
+
tableColumns.push(colItem);
|
|
181
|
+
|
|
182
|
+
// Only store first occurrence for global column suggestions
|
|
183
|
+
if (!allColumns.has(col.name)) {
|
|
184
|
+
allColumns.set(col.name, colItem);
|
|
185
|
+
}
|
|
186
|
+
});
|
|
187
|
+
columnMap.set(tableLower, tableColumns);
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
return { tableItems, columnMap, allColumns };
|
|
191
|
+
}, [parsedSchema]);
|
|
192
|
+
|
|
193
|
+
const handleFormat = () => {
|
|
194
|
+
if (!editorRef.current) return;
|
|
195
|
+
const currentValue = editorRef.current.getValue();
|
|
196
|
+
if (!currentValue) return;
|
|
197
|
+
|
|
198
|
+
try {
|
|
199
|
+
let formatted: string;
|
|
200
|
+
if (language === 'json') {
|
|
201
|
+
// JSON formatting for MongoDB queries
|
|
202
|
+
const parsed = JSON.parse(currentValue);
|
|
203
|
+
formatted = JSON.stringify(parsed, null, 2);
|
|
204
|
+
} else if (language === 'sql') {
|
|
205
|
+
formatted = format(currentValue, {
|
|
206
|
+
language: 'postgresql',
|
|
207
|
+
keywordCase: 'upper',
|
|
208
|
+
dataTypeCase: 'upper',
|
|
209
|
+
indentStyle: 'tabularLeft',
|
|
210
|
+
logicalOperatorNewline: 'before',
|
|
211
|
+
expressionWidth: 100,
|
|
212
|
+
tabWidth: 2,
|
|
213
|
+
linesBetweenQueries: 2,
|
|
214
|
+
});
|
|
215
|
+
} else {
|
|
216
|
+
return;
|
|
217
|
+
}
|
|
218
|
+
editorRef.current.setValue(formatted);
|
|
219
|
+
lastSyncedValueRef.current = formatted;
|
|
220
|
+
onChange?.(formatted);
|
|
221
|
+
} catch (e) {
|
|
222
|
+
console.error('Formatting failed:', e);
|
|
223
|
+
}
|
|
224
|
+
};
|
|
225
|
+
|
|
226
|
+
const getSelectedText = () => {
|
|
227
|
+
if (!editorRef.current) return '';
|
|
228
|
+
const selection = editorRef.current.getSelection();
|
|
229
|
+
const model = editorRef.current.getModel();
|
|
230
|
+
if (!selection || !model) return '';
|
|
231
|
+
return model.getValueInRange(selection);
|
|
232
|
+
};
|
|
233
|
+
|
|
234
|
+
const getEffectiveQuery = () => {
|
|
235
|
+
const editorValue = editorRef.current?.getValue() || '';
|
|
236
|
+
if (!editorRef.current || !monaco) return { query: editorValue, range: null };
|
|
237
|
+
|
|
238
|
+
const model = editorRef.current.getModel();
|
|
239
|
+
if (!model) return { query: editorValue, range: null };
|
|
240
|
+
|
|
241
|
+
// 1. Check for explicit selection
|
|
242
|
+
const selection = editorRef.current.getSelection();
|
|
243
|
+
if (selection) {
|
|
244
|
+
const selectedText = model.getValueInRange(selection);
|
|
245
|
+
if (selectedText && selectedText.trim().length > 0) {
|
|
246
|
+
return { query: selectedText, range: selection };
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
// 2. If no selection, try to find the current statement (between semicolons)
|
|
251
|
+
if (language === 'sql') {
|
|
252
|
+
const position = editorRef.current.getPosition();
|
|
253
|
+
if (position) {
|
|
254
|
+
const fullText = model.getValue();
|
|
255
|
+
const cursorOffset = model.getOffsetAt(position);
|
|
256
|
+
|
|
257
|
+
// Find boundaries of the current statement
|
|
258
|
+
let startOffset = fullText.lastIndexOf(';', cursorOffset - 1);
|
|
259
|
+
let endOffset = fullText.indexOf(';', cursorOffset);
|
|
260
|
+
|
|
261
|
+
if (startOffset === -1) startOffset = 0;
|
|
262
|
+
else startOffset += 1; // skip the semicolon
|
|
263
|
+
|
|
264
|
+
if (endOffset === -1) endOffset = fullText.length;
|
|
265
|
+
|
|
266
|
+
const statement = fullText.substring(startOffset, endOffset).trim();
|
|
267
|
+
if (statement.length > 0) {
|
|
268
|
+
const startPos = model.getPositionAt(startOffset);
|
|
269
|
+
const endPos = model.getPositionAt(endOffset);
|
|
270
|
+
const range = new monaco.Range(startPos.lineNumber, startPos.column, endPos.lineNumber, endPos.column);
|
|
271
|
+
return { query: statement, range };
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
return { query: editorValue, range: null };
|
|
277
|
+
};
|
|
278
|
+
|
|
279
|
+
// Track active highlight timeout to prevent race conditions
|
|
280
|
+
const highlightTimeoutRef = useRef<NodeJS.Timeout | null>(null);
|
|
281
|
+
const activeDecorationsRef = useRef<string[]>([]);
|
|
282
|
+
|
|
283
|
+
const flashHighlight = (range: Monaco.Range | null) => {
|
|
284
|
+
if (!editorRef.current || !monaco || !range) return;
|
|
285
|
+
|
|
286
|
+
// Clear any existing highlight first
|
|
287
|
+
if (highlightTimeoutRef.current) {
|
|
288
|
+
clearTimeout(highlightTimeoutRef.current);
|
|
289
|
+
highlightTimeoutRef.current = null;
|
|
290
|
+
}
|
|
291
|
+
if (activeDecorationsRef.current.length > 0 && editorRef.current) {
|
|
292
|
+
editorRef.current.deltaDecorations(activeDecorationsRef.current, []);
|
|
293
|
+
activeDecorationsRef.current = [];
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
// Create new decoration
|
|
297
|
+
const decorations = editorRef.current.deltaDecorations([], [
|
|
298
|
+
{
|
|
299
|
+
range: range,
|
|
300
|
+
options: {
|
|
301
|
+
isWholeLine: false,
|
|
302
|
+
className: 'executed-query-highlight',
|
|
303
|
+
inlineClassName: 'executed-query-inline-highlight'
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
]);
|
|
307
|
+
activeDecorationsRef.current = decorations;
|
|
308
|
+
|
|
309
|
+
// Schedule removal with ref tracking for safe cleanup
|
|
310
|
+
highlightTimeoutRef.current = setTimeout(() => {
|
|
311
|
+
if (editorRef.current && activeDecorationsRef.current.length > 0) {
|
|
312
|
+
editorRef.current.deltaDecorations(activeDecorationsRef.current, []);
|
|
313
|
+
activeDecorationsRef.current = [];
|
|
314
|
+
}
|
|
315
|
+
highlightTimeoutRef.current = null;
|
|
316
|
+
}, 1000);
|
|
317
|
+
};
|
|
318
|
+
|
|
319
|
+
// Cleanup highlight timeout on unmount
|
|
320
|
+
useEffect(() => {
|
|
321
|
+
return () => {
|
|
322
|
+
if (highlightTimeoutRef.current) {
|
|
323
|
+
clearTimeout(highlightTimeoutRef.current);
|
|
324
|
+
}
|
|
325
|
+
};
|
|
326
|
+
}, []);
|
|
327
|
+
|
|
328
|
+
// AI Chat hook (must be before useImperativeHandle that references showAi/setShowAi)
|
|
329
|
+
const getEditorValue = useCallback(() => editorRef.current?.getValue() || '', []);
|
|
330
|
+
const setEditorValueForAi = useCallback((val: string) => {
|
|
331
|
+
if (editorRef.current) {
|
|
332
|
+
editorRef.current.setValue(val);
|
|
333
|
+
lastSyncedValueRef.current = val;
|
|
334
|
+
}
|
|
335
|
+
}, []);
|
|
336
|
+
|
|
337
|
+
const {
|
|
338
|
+
showAi,
|
|
339
|
+
setShowAi,
|
|
340
|
+
aiPrompt,
|
|
341
|
+
setAiPrompt,
|
|
342
|
+
isAiLoading,
|
|
343
|
+
aiError,
|
|
344
|
+
setAiError,
|
|
345
|
+
aiConversationHistory,
|
|
346
|
+
setAiConversationHistory,
|
|
347
|
+
handleAiSubmit,
|
|
348
|
+
} = useAiChat({
|
|
349
|
+
parsedSchema,
|
|
350
|
+
schemaContext,
|
|
351
|
+
databaseType,
|
|
352
|
+
getEditorValue,
|
|
353
|
+
setEditorValue: setEditorValueForAi,
|
|
354
|
+
onChange,
|
|
355
|
+
onAiChat,
|
|
356
|
+
});
|
|
357
|
+
|
|
358
|
+
useImperativeHandle(ref, () => ({
|
|
359
|
+
getSelectedText,
|
|
360
|
+
getEffectiveQuery: () => getEffectiveQuery().query,
|
|
361
|
+
getValue: () => editorRef.current?.getValue() || '',
|
|
362
|
+
setValue: (newValue: string) => {
|
|
363
|
+
if (editorRef.current) {
|
|
364
|
+
editorRef.current.setValue(newValue);
|
|
365
|
+
lastSyncedValueRef.current = newValue;
|
|
366
|
+
}
|
|
367
|
+
},
|
|
368
|
+
focus: () => editorRef.current?.focus(),
|
|
369
|
+
format: handleFormat,
|
|
370
|
+
toggleAi: () => setShowAi(!showAi),
|
|
371
|
+
}));
|
|
372
|
+
|
|
373
|
+
const handleCopy = () => {
|
|
374
|
+
const textToCopy = getSelectedText() || editorRef.current?.getValue() || '';
|
|
375
|
+
navigator.clipboard.writeText(textToCopy);
|
|
376
|
+
};
|
|
377
|
+
|
|
378
|
+
const handleClear = () => {
|
|
379
|
+
if (editorRef.current) {
|
|
380
|
+
editorRef.current.setValue('');
|
|
381
|
+
lastSyncedValueRef.current = '';
|
|
382
|
+
onChange?.('');
|
|
383
|
+
}
|
|
384
|
+
};
|
|
385
|
+
|
|
386
|
+
// Store original console.error for cleanup
|
|
387
|
+
const originalConsoleErrorRef = useRef<typeof console.error | null>(null);
|
|
388
|
+
|
|
389
|
+
// Cleanup console.error override on unmount
|
|
390
|
+
useEffect(() => {
|
|
391
|
+
return () => {
|
|
392
|
+
if (originalConsoleErrorRef.current) {
|
|
393
|
+
console.error = originalConsoleErrorRef.current;
|
|
394
|
+
originalConsoleErrorRef.current = null;
|
|
395
|
+
}
|
|
396
|
+
};
|
|
397
|
+
}, []);
|
|
398
|
+
|
|
399
|
+
const handleBeforeMount = (monacoInstance: typeof Monaco) => {
|
|
400
|
+
// Suppress Monaco's "Canceled" errors in console (with cleanup tracking)
|
|
401
|
+
if (!originalConsoleErrorRef.current) {
|
|
402
|
+
originalConsoleErrorRef.current = console.error;
|
|
403
|
+
const originalConsoleError = console.error;
|
|
404
|
+
console.error = (...args: unknown[]) => {
|
|
405
|
+
const message = args[0]?.toString?.() || '';
|
|
406
|
+
if (message.includes('Canceled') || message.includes('ERR Canceled')) {
|
|
407
|
+
return; // Suppress Monaco cancellation errors
|
|
408
|
+
}
|
|
409
|
+
originalConsoleError.apply(console, args as Parameters<typeof console.error>);
|
|
410
|
+
};
|
|
411
|
+
}
|
|
412
|
+
|
|
413
|
+
monacoInstance.editor.defineTheme('db-dark', {
|
|
414
|
+
base: 'vs-dark',
|
|
415
|
+
inherit: true,
|
|
416
|
+
rules: [
|
|
417
|
+
{ token: 'keyword', foreground: '569cd6', fontStyle: 'bold' },
|
|
418
|
+
{ token: 'function', foreground: 'dcdcaa' },
|
|
419
|
+
{ token: 'string', foreground: 'ce9178' },
|
|
420
|
+
{ token: 'number', foreground: 'b5cea8' },
|
|
421
|
+
{ token: 'comment', foreground: '6a9955' },
|
|
422
|
+
{ token: 'operator', foreground: 'd4d4d4' },
|
|
423
|
+
{ token: 'identifier', foreground: '9cdcfe' },
|
|
424
|
+
],
|
|
425
|
+
colors: {
|
|
426
|
+
'editor.background': '#050505',
|
|
427
|
+
'editor.foreground': '#d4d4d4',
|
|
428
|
+
'editorCursor.foreground': '#569cd6',
|
|
429
|
+
'editor.lineHighlightBackground': '#111111',
|
|
430
|
+
'editorLineNumber.foreground': '#333333',
|
|
431
|
+
'editorLineNumber.activeForeground': '#666666',
|
|
432
|
+
'editor.selectionBackground': '#264f78',
|
|
433
|
+
'editor.inactiveSelectionBackground': '#3a3d41',
|
|
434
|
+
'editorIndentGuide.background': '#1a1a1a',
|
|
435
|
+
'editorIndentGuide.activeBackground': '#333333',
|
|
436
|
+
}
|
|
437
|
+
});
|
|
438
|
+
};
|
|
439
|
+
|
|
440
|
+
// SQL completion provider
|
|
441
|
+
useEffect(() => {
|
|
442
|
+
if (monaco && language === 'sql') {
|
|
443
|
+
const disposable = registerSQLCompletionProvider(monaco, schemaCompletionCache);
|
|
444
|
+
return () => disposable.dispose();
|
|
445
|
+
}
|
|
446
|
+
}, [monaco, language, schemaCompletionCache]);
|
|
447
|
+
|
|
448
|
+
// MongoDB JSON completion provider
|
|
449
|
+
useEffect(() => {
|
|
450
|
+
if (monaco && language === 'json') {
|
|
451
|
+
const disposable = registerMongoDBCompletionProvider(monaco, schemaCompletionCache);
|
|
452
|
+
return () => disposable.dispose();
|
|
453
|
+
}
|
|
454
|
+
}, [monaco, language, schemaCompletionCache]);
|
|
455
|
+
|
|
456
|
+
const handleEditorChange = (val: string | undefined) => {
|
|
457
|
+
const newValue = val || '';
|
|
458
|
+
// Only call onContentChange if provided (for real-time sync scenarios)
|
|
459
|
+
// This avoids the performance hit of updating parent state on every keystroke
|
|
460
|
+
onContentChange?.(newValue);
|
|
461
|
+
};
|
|
462
|
+
|
|
463
|
+
// Sync to parent on blur (when user leaves the editor)
|
|
464
|
+
const handleEditorBlur = () => {
|
|
465
|
+
if (editorRef.current) {
|
|
466
|
+
const currentValue = editorRef.current.getValue();
|
|
467
|
+
lastSyncedValueRef.current = currentValue;
|
|
468
|
+
onChange?.(currentValue);
|
|
469
|
+
}
|
|
470
|
+
};
|
|
471
|
+
|
|
472
|
+
const handleExecute = () => {
|
|
473
|
+
// Sync current content to parent before executing
|
|
474
|
+
if (editorRef.current) {
|
|
475
|
+
const currentValue = editorRef.current.getValue();
|
|
476
|
+
lastSyncedValueRef.current = currentValue;
|
|
477
|
+
onChange?.(currentValue);
|
|
478
|
+
}
|
|
479
|
+
|
|
480
|
+
const { query, range } = getEffectiveQuery();
|
|
481
|
+
flashHighlight(range);
|
|
482
|
+
const event = new CustomEvent('execute-query', { detail: { query } });
|
|
483
|
+
window.dispatchEvent(event);
|
|
484
|
+
};
|
|
485
|
+
|
|
486
|
+
|
|
487
|
+
return (
|
|
488
|
+
<div className="h-full w-full flex flex-col bg-[#050505] relative overflow-hidden group">
|
|
489
|
+
{/* Dynamic Pro Toolbar - Hidden on mobile */}
|
|
490
|
+
<div className="hidden md:flex items-center gap-1.5 px-3 py-1.5 bg-[#0a0a0a] border-b border-white/5 overflow-x-auto no-scrollbar scroll-smooth">
|
|
491
|
+
<div className="flex items-center gap-1 mr-2 px-1.5 py-1 rounded bg-white/5 border border-white/5">
|
|
492
|
+
<span className="text-[9px] font-black text-zinc-500 uppercase tracking-[0.2em]">Quick Actions</span>
|
|
493
|
+
</div>
|
|
494
|
+
|
|
495
|
+
{hasSelection && (
|
|
496
|
+
<button
|
|
497
|
+
onClick={handleExecute}
|
|
498
|
+
className="px-2.5 py-1.5 rounded bg-blue-600 hover:bg-blue-500 text-white text-[10px] font-bold transition-all border border-blue-400/30 active:scale-95 flex items-center gap-1.5 shadow-[0_0_15px_rgba(37,99,235,0.3)] animate-in fade-in zoom-in duration-200"
|
|
499
|
+
>
|
|
500
|
+
<Play className="w-3 h-3 fill-current" />
|
|
501
|
+
RUN SELECTION
|
|
502
|
+
</button>
|
|
503
|
+
)}
|
|
504
|
+
|
|
505
|
+
<button
|
|
506
|
+
onClick={handleFormat}
|
|
507
|
+
title={language === 'json' ? "Format JSON (Shift+Alt+F)" : "Format SQL (Shift+Alt+F)"}
|
|
508
|
+
className="px-2.5 py-1.5 rounded bg-[#111] hover:bg-zinc-800 text-zinc-500 hover:text-zinc-200 text-[10px] font-mono transition-all border border-white/5 active:scale-95 flex items-center gap-1.5"
|
|
509
|
+
>
|
|
510
|
+
<AlignLeft className="w-3 h-3" />
|
|
511
|
+
FORMAT
|
|
512
|
+
</button>
|
|
513
|
+
|
|
514
|
+
<button
|
|
515
|
+
onClick={handleCopy}
|
|
516
|
+
className="px-2.5 py-1.5 rounded bg-[#111] hover:bg-zinc-800 text-zinc-500 hover:text-zinc-200 text-[10px] font-mono transition-all border border-white/5 active:scale-95 flex items-center gap-1.5"
|
|
517
|
+
>
|
|
518
|
+
<Copy className="w-3 h-3" />
|
|
519
|
+
{hasSelection ? 'COPY SELECTION' : 'COPY'}
|
|
520
|
+
</button>
|
|
521
|
+
|
|
522
|
+
<button
|
|
523
|
+
onClick={handleClear}
|
|
524
|
+
className="px-2.5 py-1.5 rounded bg-[#111] hover:bg-zinc-800 text-zinc-500 hover:text-red-400 text-[10px] font-mono transition-all border border-white/5 active:scale-95 flex items-center gap-1.5"
|
|
525
|
+
>
|
|
526
|
+
<Trash2 className="w-3 h-3" />
|
|
527
|
+
CLEAR
|
|
528
|
+
</button>
|
|
529
|
+
|
|
530
|
+
<div className="w-px h-4 bg-white/5 mx-1" />
|
|
531
|
+
|
|
532
|
+
<button
|
|
533
|
+
onClick={() => setShowLineNumbers(!showLineNumbers)}
|
|
534
|
+
title={showLineNumbers ? "Hide line numbers" : "Show line numbers"}
|
|
535
|
+
className={cn(
|
|
536
|
+
"px-2.5 py-1.5 rounded text-[10px] font-mono transition-all border active:scale-95 flex items-center gap-1.5",
|
|
537
|
+
showLineNumbers
|
|
538
|
+
? "bg-zinc-800 border-white/10 text-zinc-300"
|
|
539
|
+
: "bg-[#111] border-white/5 text-zinc-500 hover:text-zinc-300"
|
|
540
|
+
)}
|
|
541
|
+
>
|
|
542
|
+
<Hash className="w-3 h-3" />
|
|
543
|
+
LINES
|
|
544
|
+
</button>
|
|
545
|
+
|
|
546
|
+
<button
|
|
547
|
+
onClick={() => setShowAi(!showAi)}
|
|
548
|
+
className={cn(
|
|
549
|
+
"px-2.5 py-1.5 rounded text-[10px] font-bold transition-all border active:scale-95 flex items-center gap-1.5",
|
|
550
|
+
showAi
|
|
551
|
+
? "bg-blue-600 border-blue-500 text-white shadow-[0_0_10px_rgba(37,99,235,0.4)]"
|
|
552
|
+
: "bg-zinc-900 border-white/5 text-zinc-400 hover:text-blue-400 hover:border-blue-500/30"
|
|
553
|
+
)}
|
|
554
|
+
>
|
|
555
|
+
<Sparkles className={cn("w-3.5 h-3.5", showAi && "animate-pulse")} />
|
|
556
|
+
AI ASSISTANT
|
|
557
|
+
</button>
|
|
558
|
+
|
|
559
|
+
<div className="flex-1" />
|
|
560
|
+
|
|
561
|
+
<div className="flex items-center gap-1.5 opacity-50 hover:opacity-100 transition-opacity">
|
|
562
|
+
{onExplain && capabilities?.supportsExplain && (
|
|
563
|
+
<button
|
|
564
|
+
onClick={onExplain}
|
|
565
|
+
className="px-2.5 py-1.5 rounded bg-zinc-900 hover:bg-zinc-800 text-amber-500 hover:text-amber-400 text-[10px] font-bold transition-all border border-amber-500/10 active:scale-95 flex items-center gap-1.5 mr-2"
|
|
566
|
+
>
|
|
567
|
+
<Zap className="w-3 h-3" />
|
|
568
|
+
EXPLAIN
|
|
569
|
+
</button>
|
|
570
|
+
)}
|
|
571
|
+
<kbd className="px-2 py-1 rounded bg-zinc-900 border border-white/5 text-[9px] text-zinc-500 font-mono">
|
|
572
|
+
⌘ + ENTER TO RUN
|
|
573
|
+
</kbd>
|
|
574
|
+
</div>
|
|
575
|
+
</div>
|
|
576
|
+
|
|
577
|
+
{/* Floating AI Input */}
|
|
578
|
+
<AnimatePresence>
|
|
579
|
+
{showAi && (
|
|
580
|
+
<motion.div
|
|
581
|
+
initial={{ opacity: 0, y: -10, scale: 0.95 }}
|
|
582
|
+
animate={{ opacity: 1, y: 0, scale: 1 }}
|
|
583
|
+
exit={{ opacity: 0, y: -10, scale: 0.95 }}
|
|
584
|
+
className="absolute top-2 md:top-12 left-1/2 -translate-x-1/2 w-full max-w-2xl z-50 px-2 md:px-4"
|
|
585
|
+
>
|
|
586
|
+
<form
|
|
587
|
+
onSubmit={handleAiSubmit}
|
|
588
|
+
className="bg-[#0f0f0f]/95 backdrop-blur-xl border border-blue-500/40 rounded-2xl shadow-[0_0_50px_rgba(37,99,235,0.25)] overflow-hidden flex flex-col p-1.5"
|
|
589
|
+
>
|
|
590
|
+
<div className="flex items-center justify-between px-3 py-1.5 border-b border-white/5 mb-1.5">
|
|
591
|
+
<div className="flex items-center gap-2">
|
|
592
|
+
<div className="p-1 rounded-md bg-blue-500/10">
|
|
593
|
+
<Sparkles className="w-3.5 h-3.5 text-blue-400" />
|
|
594
|
+
</div>
|
|
595
|
+
<span className="text-[10px] font-black text-blue-400 uppercase tracking-[0.2em]">Expert DBA Mode</span>
|
|
596
|
+
</div>
|
|
597
|
+
<div className="flex items-center gap-2">
|
|
598
|
+
{aiConversationHistory.length > 0 && (
|
|
599
|
+
<button
|
|
600
|
+
type="button"
|
|
601
|
+
onClick={() => setAiConversationHistory([])}
|
|
602
|
+
className="text-[9px] text-zinc-500 hover:text-zinc-300 font-medium px-1.5 py-0.5 rounded bg-white/5 hover:bg-white/10 transition-colors"
|
|
603
|
+
title="Clear conversation history"
|
|
604
|
+
>
|
|
605
|
+
{aiConversationHistory.length / 2} turns - Clear
|
|
606
|
+
</button>
|
|
607
|
+
)}
|
|
608
|
+
<span className="text-[9px] text-zinc-500 font-medium">Context: {tables.length} tables</span>
|
|
609
|
+
<div className="w-1 h-1 rounded-full bg-emerald-500 animate-pulse" />
|
|
610
|
+
</div>
|
|
611
|
+
</div>
|
|
612
|
+
|
|
613
|
+
<AnimatePresence>
|
|
614
|
+
{aiError && (
|
|
615
|
+
<motion.div
|
|
616
|
+
initial={{ height: 0, opacity: 0 }}
|
|
617
|
+
animate={{ height: 'auto', opacity: 1 }}
|
|
618
|
+
exit={{ height: 0, opacity: 0 }}
|
|
619
|
+
className="px-3 pb-2"
|
|
620
|
+
>
|
|
621
|
+
<div className="bg-red-500/10 border border-red-500/20 rounded-lg p-2.5 flex items-start gap-2.5">
|
|
622
|
+
<div className="p-1 rounded bg-red-500/20 mt-0.5">
|
|
623
|
+
<X className="w-3 h-3 text-red-400" />
|
|
624
|
+
</div>
|
|
625
|
+
<div className="flex-1">
|
|
626
|
+
<p className="text-[11px] font-bold text-red-400 uppercase tracking-tight mb-0.5">AI Error</p>
|
|
627
|
+
<p className="text-[12px] text-red-300/90 leading-relaxed">{aiError}</p>
|
|
628
|
+
</div>
|
|
629
|
+
<button
|
|
630
|
+
type="button"
|
|
631
|
+
onClick={() => setAiError(null)}
|
|
632
|
+
className="text-red-400/50 hover:text-red-400 transition-colors"
|
|
633
|
+
>
|
|
634
|
+
<X className="w-3.5 h-3.5" />
|
|
635
|
+
</button>
|
|
636
|
+
</div>
|
|
637
|
+
</motion.div>
|
|
638
|
+
)}
|
|
639
|
+
</AnimatePresence>
|
|
640
|
+
|
|
641
|
+
<div className="flex items-center gap-2 px-3 pb-1.5">
|
|
642
|
+
|
|
643
|
+
<input
|
|
644
|
+
autoFocus
|
|
645
|
+
value={aiPrompt}
|
|
646
|
+
onChange={(e) => setAiPrompt(e.target.value)}
|
|
647
|
+
placeholder="Describe the data you need in plain English... (e.g. 'Show me the revenue growth per month')"
|
|
648
|
+
className="bg-transparent border-none outline-none text-[13px] text-zinc-100 w-full h-12 placeholder:text-zinc-600 font-medium"
|
|
649
|
+
/>
|
|
650
|
+
<div className="flex items-center gap-1.5">
|
|
651
|
+
<button
|
|
652
|
+
type="button"
|
|
653
|
+
onClick={() => setShowAi(false)}
|
|
654
|
+
className="p-2.5 rounded-xl hover:bg-white/5 text-zinc-500 transition-colors"
|
|
655
|
+
>
|
|
656
|
+
<X className="w-4.5 h-4.5" />
|
|
657
|
+
</button>
|
|
658
|
+
<button
|
|
659
|
+
type="submit"
|
|
660
|
+
disabled={isAiLoading || !aiPrompt.trim()}
|
|
661
|
+
className="bg-blue-600 hover:bg-blue-500 disabled:opacity-50 disabled:hover:bg-blue-600 px-5 py-2.5 rounded-xl text-white text-xs font-bold transition-all shadow-lg shadow-blue-600/30 flex items-center gap-2"
|
|
662
|
+
>
|
|
663
|
+
{isAiLoading ? (
|
|
664
|
+
<>
|
|
665
|
+
<Loader2 className="w-3.5 h-3.5 animate-spin" />
|
|
666
|
+
<span>Thinking...</span>
|
|
667
|
+
</>
|
|
668
|
+
) : (
|
|
669
|
+
<>
|
|
670
|
+
<span>Generate</span>
|
|
671
|
+
<Send className="w-3.5 h-3.5" />
|
|
672
|
+
</>
|
|
673
|
+
)}
|
|
674
|
+
</button>
|
|
675
|
+
</div>
|
|
676
|
+
</div>
|
|
677
|
+
</form>
|
|
678
|
+
</motion.div>
|
|
679
|
+
)}
|
|
680
|
+
</AnimatePresence>
|
|
681
|
+
|
|
682
|
+
<div className="flex-1 relative">
|
|
683
|
+
<Editor
|
|
684
|
+
height="100%"
|
|
685
|
+
language={language}
|
|
686
|
+
theme="db-dark"
|
|
687
|
+
value={value}
|
|
688
|
+
beforeMount={handleBeforeMount}
|
|
689
|
+
onChange={handleEditorChange}
|
|
690
|
+
loading={<div className="h-full w-full bg-[#050505] flex items-center justify-center"><Loader2 className="w-6 h-6 animate-spin text-zinc-800" /></div>}
|
|
691
|
+
onMount={(editor, monaco) => {
|
|
692
|
+
editorRef.current = editor;
|
|
693
|
+
|
|
694
|
+
// Sync to parent when editor loses focus
|
|
695
|
+
editor.onDidBlurEditorText(() => {
|
|
696
|
+
handleEditorBlur();
|
|
697
|
+
});
|
|
698
|
+
|
|
699
|
+
editor.onDidChangeCursorSelection(() => {
|
|
700
|
+
const selection = editor.getSelection();
|
|
701
|
+
setHasSelection(selection ? !selection.isEmpty() : false);
|
|
702
|
+
});
|
|
703
|
+
|
|
704
|
+
// Add custom keyboard shortcut
|
|
705
|
+
editor.addCommand(monaco.KeyMod.CtrlCmd | monaco.KeyCode.Enter, () => {
|
|
706
|
+
handleExecute();
|
|
707
|
+
});
|
|
708
|
+
|
|
709
|
+
// Add format shortcut
|
|
710
|
+
editor.addCommand(monaco.KeyMod.Alt | monaco.KeyMod.Shift | monaco.KeyCode.KeyF, () => {
|
|
711
|
+
handleFormat();
|
|
712
|
+
});
|
|
713
|
+
|
|
714
|
+
// Context Menu Actions
|
|
715
|
+
editor.addAction({
|
|
716
|
+
id: 'run-query',
|
|
717
|
+
label: 'Run Query',
|
|
718
|
+
keybindings: [monaco.KeyMod.CtrlCmd | monaco.KeyCode.Enter],
|
|
719
|
+
contextMenuGroupId: 'navigation',
|
|
720
|
+
contextMenuOrder: 1,
|
|
721
|
+
run: () => handleExecute()
|
|
722
|
+
});
|
|
723
|
+
|
|
724
|
+
if (onExplain) {
|
|
725
|
+
editor.addAction({
|
|
726
|
+
id: 'explain-query',
|
|
727
|
+
label: 'Explain Plan',
|
|
728
|
+
contextMenuGroupId: 'navigation',
|
|
729
|
+
contextMenuOrder: 2,
|
|
730
|
+
run: () => onExplain()
|
|
731
|
+
});
|
|
732
|
+
}
|
|
733
|
+
|
|
734
|
+
editor.addAction({
|
|
735
|
+
id: 'format-sql',
|
|
736
|
+
label: 'Format SQL',
|
|
737
|
+
keybindings: [monaco.KeyMod.Alt | monaco.KeyMod.Shift | monaco.KeyCode.KeyF],
|
|
738
|
+
contextMenuGroupId: 'modification',
|
|
739
|
+
contextMenuOrder: 1,
|
|
740
|
+
run: () => handleFormat()
|
|
741
|
+
});
|
|
742
|
+
}}
|
|
743
|
+
options={getEditorOptions(showLineNumbers)}
|
|
744
|
+
/>
|
|
745
|
+
|
|
746
|
+
{/* Connection Type Badge */}
|
|
747
|
+
<div className="absolute top-3 right-6 pointer-events-none select-none z-10">
|
|
748
|
+
<div className="flex items-center gap-2 px-3 py-1 rounded-md bg-zinc-900/90 border border-white/10 backdrop-blur-md shadow-2xl">
|
|
749
|
+
<div className="w-1.5 h-1.5 rounded-full bg-blue-500 shadow-[0_0_8px_rgba(59,130,246,0.5)]" />
|
|
750
|
+
<span className="text-[10px] font-bold text-zinc-400 uppercase tracking-widest">
|
|
751
|
+
{language} Engine
|
|
752
|
+
</span>
|
|
753
|
+
</div>
|
|
754
|
+
</div>
|
|
755
|
+
</div>
|
|
756
|
+
</div>
|
|
757
|
+
);
|
|
758
|
+
});
|
|
759
|
+
|
|
760
|
+
QueryEditor.displayName = 'QueryEditor';
|