@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,263 @@
|
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
import React from 'react';
|
|
4
|
+
import {
|
|
5
|
+
Database,
|
|
6
|
+
Zap,
|
|
7
|
+
Activity,
|
|
8
|
+
Clock,
|
|
9
|
+
Table2,
|
|
10
|
+
Hash,
|
|
11
|
+
Server,
|
|
12
|
+
} from 'lucide-react';
|
|
13
|
+
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
|
|
14
|
+
import { Badge } from '@/components/ui/badge';
|
|
15
|
+
import { Progress } from '@/components/ui/progress';
|
|
16
|
+
import { Skeleton } from '@/components/ui/skeleton';
|
|
17
|
+
import type { MonitoringData } from '@/lib/db/types';
|
|
18
|
+
import type { TimeSeriesPoint } from '@/lib/time-series-buffer';
|
|
19
|
+
import { evaluateThreshold, getThresholdColor, DEFAULT_THRESHOLDS } from '@/lib/monitoring-thresholds';
|
|
20
|
+
import { MetricChart } from './MetricChart';
|
|
21
|
+
|
|
22
|
+
interface OverviewTabProps {
|
|
23
|
+
data: MonitoringData | null;
|
|
24
|
+
loading: boolean;
|
|
25
|
+
history?: TimeSeriesPoint<MonitoringData>[];
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export function OverviewTab({ data, loading, history = [] }: OverviewTabProps) {
|
|
29
|
+
if (loading && !data) {
|
|
30
|
+
return <OverviewSkeleton />;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
const overview = data?.overview;
|
|
34
|
+
const performance = data?.performance;
|
|
35
|
+
|
|
36
|
+
const connectionPercent = overview
|
|
37
|
+
? Math.round((overview.activeConnections / overview.maxConnections) * 100)
|
|
38
|
+
: 0;
|
|
39
|
+
|
|
40
|
+
// Evaluate thresholds
|
|
41
|
+
const connThreshold = evaluateThreshold(connectionPercent, DEFAULT_THRESHOLDS.find(t => t.metric === 'connectionPercent')!);
|
|
42
|
+
const cacheThreshold = evaluateThreshold(performance?.cacheHitRatio ?? 100, DEFAULT_THRESHOLDS.find(t => t.metric === 'cacheHitRatio')!);
|
|
43
|
+
|
|
44
|
+
// Build chart data from history
|
|
45
|
+
const connectionHistory = history.map(h => ({
|
|
46
|
+
timestamp: h.timestamp,
|
|
47
|
+
value: h.data.overview?.activeConnections ?? 0,
|
|
48
|
+
}));
|
|
49
|
+
|
|
50
|
+
return (
|
|
51
|
+
<div className="p-3 sm:p-6 space-y-4 sm:space-y-6">
|
|
52
|
+
{/* Version & Status */}
|
|
53
|
+
<div className="flex items-center gap-2 sm:gap-4 flex-wrap">
|
|
54
|
+
<Badge variant="outline" className="gap-1.5 sm:gap-2 py-1 sm:py-1.5 px-2 sm:px-3 text-xs">
|
|
55
|
+
<Server className="h-3 w-3 sm:h-4 sm:w-4" />
|
|
56
|
+
<span className="truncate max-w-[120px] sm:max-w-none">{overview?.version || 'Unknown'}</span>
|
|
57
|
+
</Badge>
|
|
58
|
+
<Badge variant="secondary" className="gap-1.5 sm:gap-2 py-1 sm:py-1.5 px-2 sm:px-3 text-xs">
|
|
59
|
+
<Clock className="h-3 w-3 sm:h-4 sm:w-4" />
|
|
60
|
+
{overview?.uptime || 'N/A'}
|
|
61
|
+
</Badge>
|
|
62
|
+
{data?.timestamp && (
|
|
63
|
+
<span className="text-[10px] sm:text-xs text-muted-foreground">
|
|
64
|
+
{new Date(data.timestamp).toLocaleTimeString()}
|
|
65
|
+
</span>
|
|
66
|
+
)}
|
|
67
|
+
</div>
|
|
68
|
+
|
|
69
|
+
{/* Main Stats Grid */}
|
|
70
|
+
<div className="grid grid-cols-2 lg:grid-cols-4 gap-2 sm:gap-4">
|
|
71
|
+
{/* Active Connections */}
|
|
72
|
+
<Card className={`p-0 border-2 transition-colors ${getThresholdColor(connThreshold)}`}>
|
|
73
|
+
<CardHeader className="flex flex-row items-center justify-between space-y-0 p-3 sm:p-4 pb-1 sm:pb-2">
|
|
74
|
+
<CardTitle className="text-xs sm:text-sm font-medium text-muted-foreground">
|
|
75
|
+
Connections
|
|
76
|
+
</CardTitle>
|
|
77
|
+
<Zap className="h-3 w-3 sm:h-4 sm:w-4 text-yellow-500" />
|
|
78
|
+
</CardHeader>
|
|
79
|
+
<CardContent className="p-3 sm:p-4 pt-0">
|
|
80
|
+
<div className="text-lg sm:text-2xl font-bold">
|
|
81
|
+
{overview?.activeConnections ?? 0}
|
|
82
|
+
<span className="text-xs sm:text-sm font-normal text-muted-foreground">
|
|
83
|
+
/{overview?.maxConnections ?? 0}
|
|
84
|
+
</span>
|
|
85
|
+
</div>
|
|
86
|
+
<Progress value={connectionPercent} className="h-1 mt-1 sm:mt-2" />
|
|
87
|
+
<p className="text-[10px] sm:text-xs text-muted-foreground mt-1">
|
|
88
|
+
{connectionPercent}% used
|
|
89
|
+
</p>
|
|
90
|
+
</CardContent>
|
|
91
|
+
</Card>
|
|
92
|
+
|
|
93
|
+
{/* Database Size */}
|
|
94
|
+
<Card className="p-0">
|
|
95
|
+
<CardHeader className="flex flex-row items-center justify-between space-y-0 p-3 sm:p-4 pb-1 sm:pb-2">
|
|
96
|
+
<CardTitle className="text-xs sm:text-sm font-medium text-muted-foreground">
|
|
97
|
+
DB Size
|
|
98
|
+
</CardTitle>
|
|
99
|
+
<Database className="h-3 w-3 sm:h-4 sm:w-4 text-blue-500" />
|
|
100
|
+
</CardHeader>
|
|
101
|
+
<CardContent className="p-3 sm:p-4 pt-0">
|
|
102
|
+
<div className="text-lg sm:text-2xl font-bold">{overview?.databaseSize || 'N/A'}</div>
|
|
103
|
+
<p className="text-[10px] sm:text-xs text-muted-foreground mt-1">
|
|
104
|
+
Total storage
|
|
105
|
+
</p>
|
|
106
|
+
</CardContent>
|
|
107
|
+
</Card>
|
|
108
|
+
|
|
109
|
+
{/* Cache Hit Ratio */}
|
|
110
|
+
<Card className={`p-0 border-2 transition-colors ${getThresholdColor(cacheThreshold)}`}>
|
|
111
|
+
<CardHeader className="flex flex-row items-center justify-between space-y-0 p-3 sm:p-4 pb-1 sm:pb-2">
|
|
112
|
+
<CardTitle className="text-xs sm:text-sm font-medium text-muted-foreground">
|
|
113
|
+
Cache Hit
|
|
114
|
+
</CardTitle>
|
|
115
|
+
<Activity className="h-3 w-3 sm:h-4 sm:w-4 text-green-500" />
|
|
116
|
+
</CardHeader>
|
|
117
|
+
<CardContent className="p-3 sm:p-4 pt-0">
|
|
118
|
+
<div className="text-lg sm:text-2xl font-bold">
|
|
119
|
+
{performance?.cacheHitRatio?.toFixed(1) ?? 0}%
|
|
120
|
+
</div>
|
|
121
|
+
<Progress
|
|
122
|
+
value={performance?.cacheHitRatio ?? 0}
|
|
123
|
+
className="h-1 mt-1 sm:mt-2"
|
|
124
|
+
/>
|
|
125
|
+
<p className="text-[10px] sm:text-xs text-muted-foreground mt-1 truncate">
|
|
126
|
+
{(performance?.cacheHitRatio ?? 0) >= 90
|
|
127
|
+
? 'Excellent'
|
|
128
|
+
: (performance?.cacheHitRatio ?? 0) >= 80
|
|
129
|
+
? 'Good'
|
|
130
|
+
: 'Needs tuning'}
|
|
131
|
+
</p>
|
|
132
|
+
</CardContent>
|
|
133
|
+
</Card>
|
|
134
|
+
|
|
135
|
+
{/* Tables & Indexes */}
|
|
136
|
+
<Card className="p-0">
|
|
137
|
+
<CardHeader className="flex flex-row items-center justify-between space-y-0 p-3 sm:p-4 pb-1 sm:pb-2">
|
|
138
|
+
<CardTitle className="text-xs sm:text-sm font-medium text-muted-foreground">
|
|
139
|
+
Tables
|
|
140
|
+
</CardTitle>
|
|
141
|
+
<Table2 className="h-3 w-3 sm:h-4 sm:w-4 text-purple-500" />
|
|
142
|
+
</CardHeader>
|
|
143
|
+
<CardContent className="p-3 sm:p-4 pt-0">
|
|
144
|
+
<div className="text-lg sm:text-2xl font-bold">
|
|
145
|
+
{overview?.tableCount ?? 0}
|
|
146
|
+
</div>
|
|
147
|
+
<p className="text-[10px] sm:text-xs text-muted-foreground mt-1">
|
|
148
|
+
{overview?.indexCount ?? 0} indexes
|
|
149
|
+
</p>
|
|
150
|
+
</CardContent>
|
|
151
|
+
</Card>
|
|
152
|
+
</div>
|
|
153
|
+
|
|
154
|
+
{/* Connection Trend Chart */}
|
|
155
|
+
{connectionHistory.length >= 2 && (
|
|
156
|
+
<Card className="p-0">
|
|
157
|
+
<CardHeader className="p-3 sm:p-4 pb-1">
|
|
158
|
+
<CardTitle className="text-xs sm:text-sm font-medium flex items-center gap-2">
|
|
159
|
+
<Activity className="h-3 w-3 sm:h-4 sm:w-4" />
|
|
160
|
+
Connection Trend
|
|
161
|
+
</CardTitle>
|
|
162
|
+
</CardHeader>
|
|
163
|
+
<CardContent className="p-3 sm:p-4 pt-0">
|
|
164
|
+
<MetricChart data={connectionHistory} color="#eab308" title="Connections" />
|
|
165
|
+
</CardContent>
|
|
166
|
+
</Card>
|
|
167
|
+
)}
|
|
168
|
+
|
|
169
|
+
{/* Secondary Stats */}
|
|
170
|
+
<div className="grid grid-cols-1 sm:grid-cols-2 gap-2 sm:gap-4">
|
|
171
|
+
{/* Performance Metrics */}
|
|
172
|
+
<Card className="p-0">
|
|
173
|
+
<CardHeader className="p-3 sm:p-4 pb-2">
|
|
174
|
+
<CardTitle className="text-xs sm:text-sm font-medium flex items-center gap-2">
|
|
175
|
+
<Activity className="h-3 w-3 sm:h-4 sm:w-4" />
|
|
176
|
+
Performance
|
|
177
|
+
</CardTitle>
|
|
178
|
+
</CardHeader>
|
|
179
|
+
<CardContent className="p-3 sm:p-4 pt-0 space-y-2 sm:space-y-3">
|
|
180
|
+
<div className="flex justify-between items-center gap-2">
|
|
181
|
+
<span className="text-xs sm:text-sm text-muted-foreground">Buffer Pool</span>
|
|
182
|
+
<div className="flex items-center gap-1 sm:gap-2">
|
|
183
|
+
<Progress
|
|
184
|
+
value={performance?.bufferPoolUsage ?? 0}
|
|
185
|
+
className="w-16 sm:w-24 h-1.5 sm:h-2"
|
|
186
|
+
/>
|
|
187
|
+
<span className="text-xs sm:text-sm font-medium w-8 sm:w-12 text-right">
|
|
188
|
+
{performance?.bufferPoolUsage?.toFixed(0) ?? 0}%
|
|
189
|
+
</span>
|
|
190
|
+
</div>
|
|
191
|
+
</div>
|
|
192
|
+
<div className="flex justify-between items-center">
|
|
193
|
+
<span className="text-xs sm:text-sm text-muted-foreground">Deadlocks</span>
|
|
194
|
+
<Badge variant={performance?.deadlocks ? 'destructive' : 'secondary'} className="text-xs">
|
|
195
|
+
{performance?.deadlocks ?? 0}
|
|
196
|
+
</Badge>
|
|
197
|
+
</div>
|
|
198
|
+
<div className="flex justify-between items-center">
|
|
199
|
+
<span className="text-xs sm:text-sm text-muted-foreground">Checkpoint</span>
|
|
200
|
+
<span className="text-xs sm:text-sm font-mono truncate max-w-[100px] sm:max-w-none">
|
|
201
|
+
{performance?.checkpointWriteTime || 'N/A'}
|
|
202
|
+
</span>
|
|
203
|
+
</div>
|
|
204
|
+
</CardContent>
|
|
205
|
+
</Card>
|
|
206
|
+
|
|
207
|
+
{/* Quick Stats */}
|
|
208
|
+
<Card className="p-0">
|
|
209
|
+
<CardHeader className="p-3 sm:p-4 pb-2">
|
|
210
|
+
<CardTitle className="text-xs sm:text-sm font-medium flex items-center gap-2">
|
|
211
|
+
<Hash className="h-3 w-3 sm:h-4 sm:w-4" />
|
|
212
|
+
Quick Stats
|
|
213
|
+
</CardTitle>
|
|
214
|
+
</CardHeader>
|
|
215
|
+
<CardContent className="p-3 sm:p-4 pt-0 space-y-2 sm:space-y-3">
|
|
216
|
+
<div className="flex justify-between items-center">
|
|
217
|
+
<span className="text-xs sm:text-sm text-muted-foreground">Slow Queries</span>
|
|
218
|
+
<Badge variant={data?.slowQueries?.length ? 'outline' : 'secondary'} className="text-xs">
|
|
219
|
+
{data?.slowQueries?.length ?? 0}
|
|
220
|
+
</Badge>
|
|
221
|
+
</div>
|
|
222
|
+
<div className="flex justify-between items-center">
|
|
223
|
+
<span className="text-xs sm:text-sm text-muted-foreground">Active</span>
|
|
224
|
+
<Badge variant="secondary" className="text-xs">
|
|
225
|
+
{data?.activeSessions?.filter((s) => s.state === 'active').length ?? 0}
|
|
226
|
+
</Badge>
|
|
227
|
+
</div>
|
|
228
|
+
<div className="flex justify-between items-center">
|
|
229
|
+
<span className="text-xs sm:text-sm text-muted-foreground">Idle</span>
|
|
230
|
+
<Badge variant="secondary" className="text-xs">
|
|
231
|
+
{data?.activeSessions?.filter((s) => s.state === 'idle').length ?? 0}
|
|
232
|
+
</Badge>
|
|
233
|
+
</div>
|
|
234
|
+
</CardContent>
|
|
235
|
+
</Card>
|
|
236
|
+
</div>
|
|
237
|
+
</div>
|
|
238
|
+
);
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
function OverviewSkeleton() {
|
|
242
|
+
return (
|
|
243
|
+
<div className="p-3 sm:p-6 space-y-4 sm:space-y-6">
|
|
244
|
+
<div className="flex items-center gap-2 sm:gap-4">
|
|
245
|
+
<Skeleton className="h-6 sm:h-8 w-32 sm:w-48" />
|
|
246
|
+
<Skeleton className="h-6 sm:h-8 w-20 sm:w-32" />
|
|
247
|
+
</div>
|
|
248
|
+
<div className="grid grid-cols-2 lg:grid-cols-4 gap-2 sm:gap-4">
|
|
249
|
+
{[...Array(4)].map((_, i) => (
|
|
250
|
+
<Card key={i} className="p-0">
|
|
251
|
+
<CardHeader className="p-3 sm:p-4 pb-1 sm:pb-2">
|
|
252
|
+
<Skeleton className="h-3 sm:h-4 w-16 sm:w-24" />
|
|
253
|
+
</CardHeader>
|
|
254
|
+
<CardContent className="p-3 sm:p-4 pt-0">
|
|
255
|
+
<Skeleton className="h-5 sm:h-8 w-12 sm:w-20" />
|
|
256
|
+
<Skeleton className="h-1 w-full mt-2" />
|
|
257
|
+
</CardContent>
|
|
258
|
+
</Card>
|
|
259
|
+
))}
|
|
260
|
+
</div>
|
|
261
|
+
</div>
|
|
262
|
+
);
|
|
263
|
+
}
|
|
@@ -0,0 +1,254 @@
|
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
import React from 'react';
|
|
4
|
+
import { Activity, Gauge, Zap, AlertTriangle } from 'lucide-react';
|
|
5
|
+
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
|
|
6
|
+
import { Progress } from '@/components/ui/progress';
|
|
7
|
+
import { Skeleton } from '@/components/ui/skeleton';
|
|
8
|
+
import { Badge } from '@/components/ui/badge';
|
|
9
|
+
import type { MonitoringData } from '@/lib/db/types';
|
|
10
|
+
import type { TimeSeriesPoint } from '@/lib/time-series-buffer';
|
|
11
|
+
import { evaluateThreshold, getThresholdColor, DEFAULT_THRESHOLDS } from '@/lib/monitoring-thresholds';
|
|
12
|
+
import { MetricChart } from './MetricChart';
|
|
13
|
+
|
|
14
|
+
interface PerformanceTabProps {
|
|
15
|
+
data: MonitoringData | null;
|
|
16
|
+
loading: boolean;
|
|
17
|
+
history?: TimeSeriesPoint<MonitoringData>[];
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export function PerformanceTab({ data, loading, history = [] }: PerformanceTabProps) {
|
|
21
|
+
if (loading && !data) {
|
|
22
|
+
return <PerformanceSkeleton />;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
const performance = data?.performance;
|
|
26
|
+
|
|
27
|
+
const getHealthStatus = (ratio: number) => {
|
|
28
|
+
if (ratio >= 95) return { label: 'Excellent', color: 'text-green-500', bg: 'bg-green-500' };
|
|
29
|
+
if (ratio >= 90) return { label: 'Good', color: 'text-blue-500', bg: 'bg-blue-500' };
|
|
30
|
+
if (ratio >= 80) return { label: 'Fair', color: 'text-yellow-500', bg: 'bg-yellow-500' };
|
|
31
|
+
return { label: 'Poor', color: 'text-red-500', bg: 'bg-red-500' };
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
const cacheStatus = getHealthStatus(performance?.cacheHitRatio ?? 0);
|
|
35
|
+
const bufferStatus = getHealthStatus(performance?.bufferPoolUsage ?? 0);
|
|
36
|
+
|
|
37
|
+
// Threshold evaluations
|
|
38
|
+
const cacheThreshold = evaluateThreshold(performance?.cacheHitRatio ?? 100, DEFAULT_THRESHOLDS.find(t => t.metric === 'cacheHitRatio')!);
|
|
39
|
+
const bufferThreshold = evaluateThreshold(performance?.bufferPoolUsage ?? 0, DEFAULT_THRESHOLDS.find(t => t.metric === 'bufferPoolUsage')!);
|
|
40
|
+
const deadlockThreshold = evaluateThreshold(performance?.deadlocks ?? 0, DEFAULT_THRESHOLDS.find(t => t.metric === 'deadlocks')!);
|
|
41
|
+
|
|
42
|
+
// Build trend data from history
|
|
43
|
+
const cacheHistory = history.map(h => ({ timestamp: h.timestamp, value: h.data.performance?.cacheHitRatio ?? 0 }));
|
|
44
|
+
const bufferHistory = history.map(h => ({ timestamp: h.timestamp, value: h.data.performance?.bufferPoolUsage ?? 0 }));
|
|
45
|
+
const deadlockHistory = history.map(h => ({ timestamp: h.timestamp, value: h.data.performance?.deadlocks ?? 0 }));
|
|
46
|
+
|
|
47
|
+
return (
|
|
48
|
+
<div className="p-3 sm:p-6 space-y-4 sm:space-y-6">
|
|
49
|
+
{/* Key Metrics */}
|
|
50
|
+
<div className="grid grid-cols-3 gap-2 sm:gap-4">
|
|
51
|
+
{/* Cache Hit Ratio */}
|
|
52
|
+
<Card className={`p-0 border-2 transition-colors ${getThresholdColor(cacheThreshold)}`}>
|
|
53
|
+
<CardHeader className="flex flex-row items-center justify-between space-y-0 p-2 sm:p-4 pb-1 sm:pb-2">
|
|
54
|
+
<CardTitle className="text-[10px] sm:text-sm font-medium text-muted-foreground">
|
|
55
|
+
Cache Hit
|
|
56
|
+
</CardTitle>
|
|
57
|
+
<Activity className={`h-3 w-3 sm:h-4 sm:w-4 ${cacheStatus.color}`} />
|
|
58
|
+
</CardHeader>
|
|
59
|
+
<CardContent className="p-2 sm:p-4 pt-0">
|
|
60
|
+
<div className="flex items-end gap-1">
|
|
61
|
+
<span className="text-lg sm:text-3xl font-bold">
|
|
62
|
+
{performance?.cacheHitRatio?.toFixed(1) ?? 0}
|
|
63
|
+
</span>
|
|
64
|
+
<span className="text-sm sm:text-xl text-muted-foreground">%</span>
|
|
65
|
+
</div>
|
|
66
|
+
<Progress
|
|
67
|
+
value={performance?.cacheHitRatio ?? 0}
|
|
68
|
+
className="h-1 sm:h-2 mt-1 sm:mt-3"
|
|
69
|
+
/>
|
|
70
|
+
<div className="flex items-center justify-between mt-1 sm:mt-2">
|
|
71
|
+
<Badge variant="outline" className={`${cacheStatus.color} text-[10px] sm:text-xs`}>
|
|
72
|
+
{cacheStatus.label}
|
|
73
|
+
</Badge>
|
|
74
|
+
<span className="text-[10px] sm:text-xs text-muted-foreground hidden sm:inline">95%+</span>
|
|
75
|
+
</div>
|
|
76
|
+
</CardContent>
|
|
77
|
+
</Card>
|
|
78
|
+
|
|
79
|
+
{/* Buffer Pool Usage */}
|
|
80
|
+
<Card className={`p-0 border-2 transition-colors ${getThresholdColor(bufferThreshold)}`}>
|
|
81
|
+
<CardHeader className="flex flex-row items-center justify-between space-y-0 p-2 sm:p-4 pb-1 sm:pb-2">
|
|
82
|
+
<CardTitle className="text-[10px] sm:text-sm font-medium text-muted-foreground">
|
|
83
|
+
Buffer
|
|
84
|
+
</CardTitle>
|
|
85
|
+
<Gauge className={`h-3 w-3 sm:h-4 sm:w-4 ${bufferStatus.color}`} />
|
|
86
|
+
</CardHeader>
|
|
87
|
+
<CardContent className="p-2 sm:p-4 pt-0">
|
|
88
|
+
<div className="flex items-end gap-1">
|
|
89
|
+
<span className="text-lg sm:text-3xl font-bold">
|
|
90
|
+
{performance?.bufferPoolUsage?.toFixed(0) ?? 0}
|
|
91
|
+
</span>
|
|
92
|
+
<span className="text-sm sm:text-xl text-muted-foreground">%</span>
|
|
93
|
+
</div>
|
|
94
|
+
<Progress
|
|
95
|
+
value={performance?.bufferPoolUsage ?? 0}
|
|
96
|
+
className="h-1 sm:h-2 mt-1 sm:mt-3"
|
|
97
|
+
/>
|
|
98
|
+
<div className="flex items-center justify-between mt-1 sm:mt-2">
|
|
99
|
+
<Badge variant="outline" className={`${bufferStatus.color} text-[10px] sm:text-xs`}>
|
|
100
|
+
{bufferStatus.label}
|
|
101
|
+
</Badge>
|
|
102
|
+
<span className="text-[10px] sm:text-xs text-muted-foreground hidden sm:inline">Cache</span>
|
|
103
|
+
</div>
|
|
104
|
+
</CardContent>
|
|
105
|
+
</Card>
|
|
106
|
+
|
|
107
|
+
{/* Deadlocks */}
|
|
108
|
+
<Card className={`p-0 border-2 transition-colors ${getThresholdColor(deadlockThreshold)}`}>
|
|
109
|
+
<CardHeader className="flex flex-row items-center justify-between space-y-0 p-2 sm:p-4 pb-1 sm:pb-2">
|
|
110
|
+
<CardTitle className="text-[10px] sm:text-sm font-medium text-muted-foreground">
|
|
111
|
+
Deadlocks
|
|
112
|
+
</CardTitle>
|
|
113
|
+
<AlertTriangle
|
|
114
|
+
className={`h-3 w-3 sm:h-4 sm:w-4 ${performance?.deadlocks ? 'text-red-500' : 'text-green-500'}`}
|
|
115
|
+
/>
|
|
116
|
+
</CardHeader>
|
|
117
|
+
<CardContent className="p-2 sm:p-4 pt-0">
|
|
118
|
+
<div className="flex items-end gap-1">
|
|
119
|
+
<span className="text-lg sm:text-3xl font-bold">
|
|
120
|
+
{performance?.deadlocks ?? 0}
|
|
121
|
+
</span>
|
|
122
|
+
</div>
|
|
123
|
+
<p className="text-[10px] sm:text-xs text-muted-foreground mt-1 sm:mt-3 hidden sm:block">
|
|
124
|
+
{performance?.deadlocks
|
|
125
|
+
? 'Review queries'
|
|
126
|
+
: 'None detected'}
|
|
127
|
+
</p>
|
|
128
|
+
<Badge
|
|
129
|
+
variant={performance?.deadlocks ? 'destructive' : 'secondary'}
|
|
130
|
+
className="mt-1 sm:mt-2 text-[10px] sm:text-xs"
|
|
131
|
+
>
|
|
132
|
+
{performance?.deadlocks ? 'Attention' : 'Healthy'}
|
|
133
|
+
</Badge>
|
|
134
|
+
</CardContent>
|
|
135
|
+
</Card>
|
|
136
|
+
</div>
|
|
137
|
+
|
|
138
|
+
{/* Trend Charts */}
|
|
139
|
+
{history.length >= 2 && (
|
|
140
|
+
<div className="grid grid-cols-1 sm:grid-cols-3 gap-2 sm:gap-4">
|
|
141
|
+
<Card className="p-0">
|
|
142
|
+
<CardHeader className="p-2 sm:p-3 pb-0">
|
|
143
|
+
<CardTitle className="text-[10px] sm:text-xs font-medium text-muted-foreground">Cache Hit Trend</CardTitle>
|
|
144
|
+
</CardHeader>
|
|
145
|
+
<CardContent className="p-2 sm:p-3 pt-0">
|
|
146
|
+
<MetricChart data={cacheHistory} color="#22c55e" title="Cache Hit" unit="%" />
|
|
147
|
+
</CardContent>
|
|
148
|
+
</Card>
|
|
149
|
+
<Card className="p-0">
|
|
150
|
+
<CardHeader className="p-2 sm:p-3 pb-0">
|
|
151
|
+
<CardTitle className="text-[10px] sm:text-xs font-medium text-muted-foreground">Buffer Pool Trend</CardTitle>
|
|
152
|
+
</CardHeader>
|
|
153
|
+
<CardContent className="p-2 sm:p-3 pt-0">
|
|
154
|
+
<MetricChart data={bufferHistory} color="#3b82f6" title="Buffer Pool" unit="%" />
|
|
155
|
+
</CardContent>
|
|
156
|
+
</Card>
|
|
157
|
+
<Card className="p-0">
|
|
158
|
+
<CardHeader className="p-2 sm:p-3 pb-0">
|
|
159
|
+
<CardTitle className="text-[10px] sm:text-xs font-medium text-muted-foreground">Deadlock Trend</CardTitle>
|
|
160
|
+
</CardHeader>
|
|
161
|
+
<CardContent className="p-2 sm:p-3 pt-0">
|
|
162
|
+
<MetricChart data={deadlockHistory} color="#ef4444" title="Deadlocks" />
|
|
163
|
+
</CardContent>
|
|
164
|
+
</Card>
|
|
165
|
+
</div>
|
|
166
|
+
)}
|
|
167
|
+
|
|
168
|
+
{/* Additional Metrics */}
|
|
169
|
+
<div className="grid grid-cols-1 sm:grid-cols-2 gap-2 sm:gap-4">
|
|
170
|
+
{/* Checkpoint Stats */}
|
|
171
|
+
<Card className="p-0">
|
|
172
|
+
<CardHeader className="p-3 sm:p-4 pb-2">
|
|
173
|
+
<CardTitle className="text-xs sm:text-sm font-medium flex items-center gap-2">
|
|
174
|
+
<Zap className="h-3 w-3 sm:h-4 sm:w-4" />
|
|
175
|
+
Checkpoint Stats
|
|
176
|
+
</CardTitle>
|
|
177
|
+
</CardHeader>
|
|
178
|
+
<CardContent className="p-3 sm:p-4 pt-0 space-y-2 sm:space-y-4">
|
|
179
|
+
<div className="p-2 sm:p-4 bg-muted/30 rounded-lg">
|
|
180
|
+
<p className="text-[10px] sm:text-sm text-muted-foreground">Write & Sync</p>
|
|
181
|
+
<p className="text-sm sm:text-lg font-mono mt-1 truncate">
|
|
182
|
+
{performance?.checkpointWriteTime || 'N/A'}
|
|
183
|
+
</p>
|
|
184
|
+
</div>
|
|
185
|
+
<p className="text-[10px] sm:text-xs text-muted-foreground hidden sm:block">
|
|
186
|
+
Checkpoint write time affects database performance during heavy writes.
|
|
187
|
+
</p>
|
|
188
|
+
</CardContent>
|
|
189
|
+
</Card>
|
|
190
|
+
|
|
191
|
+
{/* Performance Tips */}
|
|
192
|
+
<Card className="p-0">
|
|
193
|
+
<CardHeader className="p-3 sm:p-4 pb-2">
|
|
194
|
+
<CardTitle className="text-xs sm:text-sm font-medium flex items-center gap-2">
|
|
195
|
+
<Activity className="h-3 w-3 sm:h-4 sm:w-4" />
|
|
196
|
+
Tips
|
|
197
|
+
</CardTitle>
|
|
198
|
+
</CardHeader>
|
|
199
|
+
<CardContent className="p-3 sm:p-4 pt-0 space-y-2 sm:space-y-3">
|
|
200
|
+
{(performance?.cacheHitRatio ?? 100) < 90 && (
|
|
201
|
+
<div className="flex items-start gap-2 p-2 bg-yellow-500/10 rounded-md">
|
|
202
|
+
<AlertTriangle className="h-3 w-3 sm:h-4 sm:w-4 text-yellow-500 mt-0.5 flex-shrink-0" />
|
|
203
|
+
<div>
|
|
204
|
+
<p className="text-xs sm:text-sm font-medium">Low Cache Hit</p>
|
|
205
|
+
<p className="text-[10px] sm:text-xs text-muted-foreground hidden sm:block">
|
|
206
|
+
Increase shared_buffers
|
|
207
|
+
</p>
|
|
208
|
+
</div>
|
|
209
|
+
</div>
|
|
210
|
+
)}
|
|
211
|
+
{(performance?.deadlocks ?? 0) > 0 && (
|
|
212
|
+
<div className="flex items-start gap-2 p-2 bg-red-500/10 rounded-md">
|
|
213
|
+
<AlertTriangle className="h-3 w-3 sm:h-4 sm:w-4 text-red-500 mt-0.5 flex-shrink-0" />
|
|
214
|
+
<div>
|
|
215
|
+
<p className="text-xs sm:text-sm font-medium">Deadlocks</p>
|
|
216
|
+
<p className="text-[10px] sm:text-xs text-muted-foreground hidden sm:block">
|
|
217
|
+
Review lock ordering
|
|
218
|
+
</p>
|
|
219
|
+
</div>
|
|
220
|
+
</div>
|
|
221
|
+
)}
|
|
222
|
+
{(performance?.cacheHitRatio ?? 0) >= 90 && !(performance?.deadlocks) && (
|
|
223
|
+
<div className="flex items-center gap-2 p-2 bg-green-500/10 rounded-md">
|
|
224
|
+
<Activity className="h-3 w-3 sm:h-4 sm:w-4 text-green-500 flex-shrink-0" />
|
|
225
|
+
<p className="text-xs sm:text-sm">Performing well!</p>
|
|
226
|
+
</div>
|
|
227
|
+
)}
|
|
228
|
+
</CardContent>
|
|
229
|
+
</Card>
|
|
230
|
+
</div>
|
|
231
|
+
</div>
|
|
232
|
+
);
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
function PerformanceSkeleton() {
|
|
236
|
+
return (
|
|
237
|
+
<div className="p-3 sm:p-6 space-y-4 sm:space-y-6">
|
|
238
|
+
<div className="grid grid-cols-3 gap-2 sm:gap-4">
|
|
239
|
+
{[...Array(3)].map((_, i) => (
|
|
240
|
+
<Card key={i} className="p-0">
|
|
241
|
+
<CardHeader className="p-2 sm:p-4 pb-1 sm:pb-2">
|
|
242
|
+
<Skeleton className="h-3 sm:h-4 w-12 sm:w-24" />
|
|
243
|
+
</CardHeader>
|
|
244
|
+
<CardContent className="p-2 sm:p-4 pt-0">
|
|
245
|
+
<Skeleton className="h-5 sm:h-10 w-10 sm:w-24" />
|
|
246
|
+
<Skeleton className="h-1 sm:h-2 w-full mt-1 sm:mt-3" />
|
|
247
|
+
<Skeleton className="h-4 sm:h-6 w-12 sm:w-20 mt-1 sm:mt-2" />
|
|
248
|
+
</CardContent>
|
|
249
|
+
</Card>
|
|
250
|
+
))}
|
|
251
|
+
</div>
|
|
252
|
+
</div>
|
|
253
|
+
);
|
|
254
|
+
}
|