@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.
Files changed (572) hide show
  1. package/.claude/settings.local.json +127 -0
  2. package/.cursorrules +426 -0
  3. package/.devin/wiki.json +143 -0
  4. package/.dockerignore +80 -0
  5. package/.env.example +159 -0
  6. package/.github/ISSUE_TEMPLATE/bug_report.md +49 -0
  7. package/.github/ISSUE_TEMPLATE/feature_request.md +29 -0
  8. package/.github/PULL_REQUEST_TEMPLATE.md +57 -0
  9. package/.github/workflows/ci.yml +185 -0
  10. package/.github/workflows/codeql.yml +57 -0
  11. package/.github/workflows/docker-build-push.yml +118 -0
  12. package/.github/workflows/helm-release.yml +113 -0
  13. package/CLAUDE.md +265 -0
  14. package/CODE_OF_CONDUCT.md +124 -0
  15. package/CONTRIBUTING.md +154 -0
  16. package/Dockerfile +73 -0
  17. package/LICENSE +21 -0
  18. package/README.md +614 -0
  19. package/SECURITY.md +107 -0
  20. package/artifacthub-repo.yml +4 -0
  21. package/bun.lock +1714 -0
  22. package/bunfig.toml +3 -0
  23. package/charts/libredb-studio/.helmignore +11 -0
  24. package/charts/libredb-studio/Chart.lock +6 -0
  25. package/charts/libredb-studio/Chart.yaml +50 -0
  26. package/charts/libredb-studio/README.md +206 -0
  27. package/charts/libredb-studio/templates/NOTES.txt +59 -0
  28. package/charts/libredb-studio/templates/_helpers.tpl +135 -0
  29. package/charts/libredb-studio/templates/configmap.yaml +37 -0
  30. package/charts/libredb-studio/templates/deployment.yaml +184 -0
  31. package/charts/libredb-studio/templates/hpa.yaml +32 -0
  32. package/charts/libredb-studio/templates/ingress.yaml +41 -0
  33. package/charts/libredb-studio/templates/networkpolicy.yaml +50 -0
  34. package/charts/libredb-studio/templates/pdb.yaml +18 -0
  35. package/charts/libredb-studio/templates/pvc.yaml +23 -0
  36. package/charts/libredb-studio/templates/secret.yaml +30 -0
  37. package/charts/libredb-studio/templates/seed-configmap.yaml +11 -0
  38. package/charts/libredb-studio/templates/service.yaml +22 -0
  39. package/charts/libredb-studio/templates/serviceaccount.yaml +13 -0
  40. package/charts/libredb-studio/values.schema.json +246 -0
  41. package/charts/libredb-studio/values.yaml +286 -0
  42. package/components.json +22 -0
  43. package/conductor/code_styleguides/typescript.md +43 -0
  44. package/conductor/product-guidelines.md +43 -0
  45. package/conductor/product.md +3 -0
  46. package/conductor/setup_state.json +1 -0
  47. package/conductor/tech-stack.md +39 -0
  48. package/conductor/tracks/enhance_postgres_monitoring_20251227/metadata.json +8 -0
  49. package/conductor/tracks/enhance_postgres_monitoring_20251227/plan.md +44 -0
  50. package/conductor/tracks/enhance_postgres_monitoring_20251227/spec.md +31 -0
  51. package/conductor/tracks.md +8 -0
  52. package/conductor/workflow.md +333 -0
  53. package/database-compose.yml +55 -0
  54. package/docker/postgres-init/01-extensions.sql +10 -0
  55. package/docker/postgres-init/02-sample-data.sql +585 -0
  56. package/docker/postgres.yml +68 -0
  57. package/docker-compose.yml +38 -0
  58. package/docs/AI_PLAN.md +74 -0
  59. package/docs/API_DOCS.md +875 -0
  60. package/docs/ARCHITECTURE.md +218 -0
  61. package/docs/DATABASE_PROVIDERS.md +358 -0
  62. package/docs/FEATURES.md +116 -0
  63. package/docs/HELM_CHART.md +252 -0
  64. package/docs/LOGIN_PAGE.md +178 -0
  65. package/docs/MONACO_EDITOR_PERFORMANCE.md +315 -0
  66. package/docs/OIDC_ARCH.md +681 -0
  67. package/docs/OIDC_SETUP.md +322 -0
  68. package/docs/POSTGRES_METRICS.md +516 -0
  69. package/docs/QUERY_OPTIMIZATION.md +370 -0
  70. package/docs/SEED_CONNECTIONS.md +468 -0
  71. package/docs/SQL_ALIAS_COMPLETION.md +190 -0
  72. package/docs/STORAGE_ARCHITECTURE.md +565 -0
  73. package/docs/STORAGE_QUICK_SETUP.md +419 -0
  74. package/docs/TECHNICAL_PLAN.md +36 -0
  75. package/docs/THEMING.md +345 -0
  76. package/docs/adding-a-new-database-provider.md +642 -0
  77. package/docs/backlogs/000-PLATFORM_DATA_SYNC_DATABASE.md +360 -0
  78. package/docs/backlogs/001-INLINE_DATA_EDITING.md +118 -0
  79. package/docs/backlogs/002-DATA_IMPORT.md +215 -0
  80. package/docs/backlogs/003-QUERY_TIME_MACHINE.md +183 -0
  81. package/docs/backlogs/004-AI_DATA_STORYTELLER.md +292 -0
  82. package/docs/backlogs/005-QUERY_PLAYGROUND.md +352 -0
  83. package/docs/backlogs/006-DATA_MASKING.md +418 -0
  84. package/docs/enterprise-features.md +718 -0
  85. package/docs/kubernetes-helm-chart-artifacthub-plan.md +803 -0
  86. package/docs/medium-koyeb-article-en.md +215 -0
  87. package/docs/plans/test-plans.md +445 -0
  88. package/docs/releases/RELEASE.V0.3.0.md +22 -0
  89. package/docs/releases/RELEASE.V0.4.0.md +154 -0
  90. package/docs/releases/RELEASE.V0.5.0.md +252 -0
  91. package/docs/releases/RELEASE_v0.5.6.md +145 -0
  92. package/docs/releases/RELEASE_v0.6.1.md +303 -0
  93. package/docs/releases/RELEASE_v0.6.7.md +292 -0
  94. package/docs/releases/RELEASE_v0.7.0.md +332 -0
  95. package/docs/releases/RELEASE_v0.8.0.md +521 -0
  96. package/docs/sampledb/titanic.sql +1379 -0
  97. package/docs/superpowers/plans/2026-03-25-seed-connections.md +1362 -0
  98. package/docs/superpowers/specs/2026-03-25-seed-connections-design.md +590 -0
  99. package/e2e/admin-dashboard.spec.ts +64 -0
  100. package/e2e/connection-management.spec.ts +58 -0
  101. package/e2e/export.spec.ts +34 -0
  102. package/e2e/login.spec.ts +85 -0
  103. package/e2e/query-execution.spec.ts +35 -0
  104. package/e2e/tab-management.spec.ts +64 -0
  105. package/eslint.config.mjs +28 -0
  106. package/fly.toml +43 -0
  107. package/next.config.ts +32 -0
  108. package/package.json +130 -0
  109. package/playwright.config.ts +34 -0
  110. package/postcss.config.mjs +7 -0
  111. package/public/favicon-32x32.png +0 -0
  112. package/public/favicon.ico +0 -0
  113. package/public/file.svg +1 -0
  114. package/public/globe.svg +1 -0
  115. package/public/logo.svg +32 -0
  116. package/public/next.svg +1 -0
  117. package/public/screenshots/code-generator.png +0 -0
  118. package/public/screenshots/connection-modal.png +0 -0
  119. package/public/screenshots/data-profiler.png +0 -0
  120. package/public/screenshots/erd-diagram.png +0 -0
  121. package/public/screenshots/hero-editor.png +0 -0
  122. package/public/screenshots/nl2sql.png +0 -0
  123. package/public/vercel.svg +1 -0
  124. package/public/window.svg +1 -0
  125. package/render.yaml +58 -0
  126. package/scripts/merge-lcov.mjs +239 -0
  127. package/sonar-project.properties +16 -0
  128. package/src/app/admin/error.tsx +46 -0
  129. package/src/app/admin/page.tsx +10 -0
  130. package/src/app/api/admin/audit/route.ts +52 -0
  131. package/src/app/api/admin/fleet-health/route.ts +81 -0
  132. package/src/app/api/ai/autopilot/route.ts +105 -0
  133. package/src/app/api/ai/chat/route.ts +132 -0
  134. package/src/app/api/ai/describe-schema/route.ts +52 -0
  135. package/src/app/api/ai/explain/route.ts +86 -0
  136. package/src/app/api/ai/impact/route.ts +97 -0
  137. package/src/app/api/ai/index-advisor/route.ts +98 -0
  138. package/src/app/api/ai/nl2sql/route.ts +87 -0
  139. package/src/app/api/ai/query-safety/route.ts +87 -0
  140. package/src/app/api/auth/login/route.ts +62 -0
  141. package/src/app/api/auth/logout/route.ts +25 -0
  142. package/src/app/api/auth/me/route.ts +10 -0
  143. package/src/app/api/auth/oidc/callback/route.ts +82 -0
  144. package/src/app/api/auth/oidc/login/route.ts +43 -0
  145. package/src/app/api/connections/managed/route.ts +35 -0
  146. package/src/app/api/db/cancel/route.ts +42 -0
  147. package/src/app/api/db/disconnect/route.ts +28 -0
  148. package/src/app/api/db/health/route.ts +49 -0
  149. package/src/app/api/db/maintenance/route.ts +72 -0
  150. package/src/app/api/db/monitoring/route.ts +62 -0
  151. package/src/app/api/db/multi-query/route.ts +116 -0
  152. package/src/app/api/db/pool-stats/route.ts +37 -0
  153. package/src/app/api/db/profile/route.ts +144 -0
  154. package/src/app/api/db/provider-meta/route.ts +49 -0
  155. package/src/app/api/db/query/route.ts +50 -0
  156. package/src/app/api/db/schema/route.ts +47 -0
  157. package/src/app/api/db/schema-snapshot/route.ts +42 -0
  158. package/src/app/api/db/test-connection/route.ts +55 -0
  159. package/src/app/api/db/transaction/route.ts +111 -0
  160. package/src/app/api/storage/[collection]/route.ts +67 -0
  161. package/src/app/api/storage/config/route.ts +17 -0
  162. package/src/app/api/storage/migrate/route.ts +45 -0
  163. package/src/app/api/storage/route.ts +32 -0
  164. package/src/app/error.tsx +49 -0
  165. package/src/app/global-error.tsx +55 -0
  166. package/src/app/globals.css +146 -0
  167. package/src/app/icon.svg +42 -0
  168. package/src/app/layout.tsx +34 -0
  169. package/src/app/login/login-form.tsx +301 -0
  170. package/src/app/login/page.tsx +11 -0
  171. package/src/app/monitoring/page.tsx +8 -0
  172. package/src/app/not-found.tsx +29 -0
  173. package/src/app/page.tsx +5 -0
  174. package/src/components/AIAutopilotPanel.tsx +238 -0
  175. package/src/components/CodeGenerator.tsx +271 -0
  176. package/src/components/CommandPalette.tsx +227 -0
  177. package/src/components/ConnectionModal.tsx +759 -0
  178. package/src/components/CreateTableModal.tsx +281 -0
  179. package/src/components/DataCharts.tsx +962 -0
  180. package/src/components/DataImportModal.tsx +582 -0
  181. package/src/components/DataProfiler.tsx +335 -0
  182. package/src/components/DatabaseDocs.tsx +251 -0
  183. package/src/components/MaskingSettings.tsx +414 -0
  184. package/src/components/MobileNav.tsx +50 -0
  185. package/src/components/NL2SQLPanel.tsx +281 -0
  186. package/src/components/PivotTable.tsx +257 -0
  187. package/src/components/QueryEditor.tsx +760 -0
  188. package/src/components/QueryHistory.tsx +344 -0
  189. package/src/components/QuerySafetyDialog.tsx +290 -0
  190. package/src/components/ResultsGrid.tsx +644 -0
  191. package/src/components/SaveQueryModal.tsx +104 -0
  192. package/src/components/SavedQueries.tsx +128 -0
  193. package/src/components/SchemaDiagram.tsx +473 -0
  194. package/src/components/SchemaDiff.tsx +473 -0
  195. package/src/components/SnapshotTimeline.tsx +116 -0
  196. package/src/components/Studio.tsx +639 -0
  197. package/src/components/TestDataGenerator.tsx +261 -0
  198. package/src/components/VisualExplain.tsx +820 -0
  199. package/src/components/admin/AdminDashboard.tsx +163 -0
  200. package/src/components/admin/tabs/AuditTab.tsx +531 -0
  201. package/src/components/admin/tabs/MonitoringEmbed.tsx +11 -0
  202. package/src/components/admin/tabs/OperationsTab.tsx +646 -0
  203. package/src/components/admin/tabs/OverviewTab.tsx +1328 -0
  204. package/src/components/admin/tabs/SecurityTab.tsx +284 -0
  205. package/src/components/community-section.tsx +92 -0
  206. package/src/components/icons/db-icons.tsx +84 -0
  207. package/src/components/libredb-logo.tsx +61 -0
  208. package/src/components/monitoring/MonitoringDashboard.tsx +345 -0
  209. package/src/components/monitoring/tabs/MetricChart.tsx +82 -0
  210. package/src/components/monitoring/tabs/OverviewTab.tsx +263 -0
  211. package/src/components/monitoring/tabs/PerformanceTab.tsx +254 -0
  212. package/src/components/monitoring/tabs/PoolTab.tsx +174 -0
  213. package/src/components/monitoring/tabs/QueriesTab.tsx +287 -0
  214. package/src/components/monitoring/tabs/SessionsTab.tsx +316 -0
  215. package/src/components/monitoring/tabs/StorageTab.tsx +335 -0
  216. package/src/components/monitoring/tabs/TablesTab.tsx +300 -0
  217. package/src/components/results-grid/ResultCard.tsx +111 -0
  218. package/src/components/results-grid/RowDetailSheet.tsx +178 -0
  219. package/src/components/results-grid/StatsBar.tsx +201 -0
  220. package/src/components/results-grid/index.ts +1 -0
  221. package/src/components/results-grid/utils.ts +23 -0
  222. package/src/components/schema-explorer/ColumnList.tsx +53 -0
  223. package/src/components/schema-explorer/SchemaExplorer.tsx +182 -0
  224. package/src/components/schema-explorer/TableItem.tsx +210 -0
  225. package/src/components/schema-explorer/index.ts +1 -0
  226. package/src/components/sidebar/ConnectionItem.tsx +105 -0
  227. package/src/components/sidebar/ConnectionsList.tsx +62 -0
  228. package/src/components/sidebar/Sidebar.tsx +130 -0
  229. package/src/components/sidebar/index.ts +2 -0
  230. package/src/components/studio/BottomPanel.tsx +286 -0
  231. package/src/components/studio/QueryToolbar.tsx +180 -0
  232. package/src/components/studio/StudioDesktopHeader.tsx +114 -0
  233. package/src/components/studio/StudioMobileHeader.tsx +340 -0
  234. package/src/components/studio/StudioTabBar.tsx +82 -0
  235. package/src/components/studio/index.ts +5 -0
  236. package/src/components/ui/accordion.tsx +66 -0
  237. package/src/components/ui/alert-dialog.tsx +157 -0
  238. package/src/components/ui/alert.tsx +66 -0
  239. package/src/components/ui/aspect-ratio.tsx +11 -0
  240. package/src/components/ui/avatar.tsx +53 -0
  241. package/src/components/ui/badge.tsx +46 -0
  242. package/src/components/ui/breadcrumb.tsx +109 -0
  243. package/src/components/ui/button-group.tsx +83 -0
  244. package/src/components/ui/button.tsx +60 -0
  245. package/src/components/ui/calendar.tsx +216 -0
  246. package/src/components/ui/card.tsx +92 -0
  247. package/src/components/ui/carousel.tsx +241 -0
  248. package/src/components/ui/chart.tsx +357 -0
  249. package/src/components/ui/checkbox.tsx +32 -0
  250. package/src/components/ui/collapsible.tsx +33 -0
  251. package/src/components/ui/command.tsx +184 -0
  252. package/src/components/ui/context-menu.tsx +252 -0
  253. package/src/components/ui/dialog.tsx +143 -0
  254. package/src/components/ui/drawer.tsx +135 -0
  255. package/src/components/ui/dropdown-menu.tsx +257 -0
  256. package/src/components/ui/empty.tsx +104 -0
  257. package/src/components/ui/field.tsx +248 -0
  258. package/src/components/ui/form.tsx +167 -0
  259. package/src/components/ui/hover-card.tsx +44 -0
  260. package/src/components/ui/input-group.tsx +170 -0
  261. package/src/components/ui/input-otp.tsx +77 -0
  262. package/src/components/ui/input.tsx +21 -0
  263. package/src/components/ui/item.tsx +193 -0
  264. package/src/components/ui/kbd.tsx +28 -0
  265. package/src/components/ui/label.tsx +24 -0
  266. package/src/components/ui/menubar.tsx +276 -0
  267. package/src/components/ui/navigation-menu.tsx +168 -0
  268. package/src/components/ui/pagination.tsx +127 -0
  269. package/src/components/ui/popover.tsx +48 -0
  270. package/src/components/ui/progress.tsx +31 -0
  271. package/src/components/ui/radio-group.tsx +45 -0
  272. package/src/components/ui/resizable.tsx +56 -0
  273. package/src/components/ui/scroll-area.tsx +58 -0
  274. package/src/components/ui/select.tsx +187 -0
  275. package/src/components/ui/separator.tsx +28 -0
  276. package/src/components/ui/sheet.tsx +139 -0
  277. package/src/components/ui/sidebar.tsx +726 -0
  278. package/src/components/ui/skeleton.tsx +13 -0
  279. package/src/components/ui/slider.tsx +63 -0
  280. package/src/components/ui/sonner.tsx +40 -0
  281. package/src/components/ui/spinner.tsx +16 -0
  282. package/src/components/ui/switch.tsx +31 -0
  283. package/src/components/ui/table.tsx +116 -0
  284. package/src/components/ui/tabs.tsx +66 -0
  285. package/src/components/ui/textarea.tsx +18 -0
  286. package/src/components/ui/toggle-group.tsx +83 -0
  287. package/src/components/ui/toggle.tsx +47 -0
  288. package/src/components/ui/tooltip.tsx +61 -0
  289. package/src/exports/components.ts +15 -0
  290. package/src/exports/index.ts +4 -0
  291. package/src/exports/providers.ts +4 -0
  292. package/src/exports/types.ts +26 -0
  293. package/src/hooks/use-ai-chat.ts +182 -0
  294. package/src/hooks/use-all-connections.ts +66 -0
  295. package/src/hooks/use-api-call.ts +71 -0
  296. package/src/hooks/use-auth.ts +51 -0
  297. package/src/hooks/use-connection-form.ts +349 -0
  298. package/src/hooks/use-connection-manager.ts +169 -0
  299. package/src/hooks/use-connection-payload.ts +15 -0
  300. package/src/hooks/use-inline-editing.ts +109 -0
  301. package/src/hooks/use-mobile.ts +20 -0
  302. package/src/hooks/use-monitoring-data.ts +270 -0
  303. package/src/hooks/use-provider-metadata.ts +62 -0
  304. package/src/hooks/use-query-execution.ts +478 -0
  305. package/src/hooks/use-storage-sync.ts +259 -0
  306. package/src/hooks/use-tab-manager.ts +231 -0
  307. package/src/hooks/use-toast.ts +20 -0
  308. package/src/hooks/use-transaction-control.ts +64 -0
  309. package/src/lib/api/error-codes.ts +30 -0
  310. package/src/lib/api/errors.ts +236 -0
  311. package/src/lib/api/with-error-handler.ts +41 -0
  312. package/src/lib/audit.ts +105 -0
  313. package/src/lib/auth.ts +87 -0
  314. package/src/lib/connection-string-parser.ts +172 -0
  315. package/src/lib/data-masking.ts +385 -0
  316. package/src/lib/db/base-provider.ts +325 -0
  317. package/src/lib/db/errors.ts +317 -0
  318. package/src/lib/db/factory.ts +324 -0
  319. package/src/lib/db/index.ts +123 -0
  320. package/src/lib/db/providers/document/index.ts +6 -0
  321. package/src/lib/db/providers/document/mongodb.ts +992 -0
  322. package/src/lib/db/providers/keyvalue/redis.ts +554 -0
  323. package/src/lib/db/providers/sql/index.ts +11 -0
  324. package/src/lib/db/providers/sql/mssql.ts +1065 -0
  325. package/src/lib/db/providers/sql/mysql.ts +978 -0
  326. package/src/lib/db/providers/sql/oracle.ts +1044 -0
  327. package/src/lib/db/providers/sql/postgres.ts +1179 -0
  328. package/src/lib/db/providers/sql/sql-base.ts +174 -0
  329. package/src/lib/db/providers/sql/sqlite.ts +721 -0
  330. package/src/lib/db/types.ts +437 -0
  331. package/src/lib/db/utils/pool-manager.ts +287 -0
  332. package/src/lib/db/utils/query-limiter.ts +239 -0
  333. package/src/lib/db-ui-config.ts +86 -0
  334. package/src/lib/editor/mongodb-completions.ts +172 -0
  335. package/src/lib/editor/sql-completions.ts +280 -0
  336. package/src/lib/llm/base-provider.ts +117 -0
  337. package/src/lib/llm/factory.ts +102 -0
  338. package/src/lib/llm/index.ts +90 -0
  339. package/src/lib/llm/providers/custom.ts +181 -0
  340. package/src/lib/llm/providers/gemini.ts +126 -0
  341. package/src/lib/llm/providers/ollama.ts +154 -0
  342. package/src/lib/llm/providers/openai.ts +146 -0
  343. package/src/lib/llm/types.ts +173 -0
  344. package/src/lib/llm/utils/config.ts +187 -0
  345. package/src/lib/llm/utils/retry.ts +119 -0
  346. package/src/lib/llm/utils/streaming.ts +202 -0
  347. package/src/lib/logger.ts +127 -0
  348. package/src/lib/monitoring-thresholds.ts +44 -0
  349. package/src/lib/oidc.ts +262 -0
  350. package/src/lib/query-generators.ts +61 -0
  351. package/src/lib/schema-diff/diff-engine.ts +273 -0
  352. package/src/lib/schema-diff/migration-generator.ts +208 -0
  353. package/src/lib/schema-diff/types.ts +55 -0
  354. package/src/lib/seed/config-loader.ts +79 -0
  355. package/src/lib/seed/connection-filter.ts +49 -0
  356. package/src/lib/seed/credential-resolver.ts +62 -0
  357. package/src/lib/seed/index.ts +40 -0
  358. package/src/lib/seed/resolve-connection.ts +57 -0
  359. package/src/lib/seed/types.ts +69 -0
  360. package/src/lib/sql/alias-extractor.ts +267 -0
  361. package/src/lib/sql/index.ts +8 -0
  362. package/src/lib/sql/statement-splitter.ts +167 -0
  363. package/src/lib/sql/types.ts +40 -0
  364. package/src/lib/ssh/tunnel.ts +142 -0
  365. package/src/lib/storage/factory.ts +84 -0
  366. package/src/lib/storage/index.ts +14 -0
  367. package/src/lib/storage/local-storage.ts +99 -0
  368. package/src/lib/storage/providers/postgres.ts +225 -0
  369. package/src/lib/storage/providers/sqlite.ts +153 -0
  370. package/src/lib/storage/storage-facade.ts +272 -0
  371. package/src/lib/storage/types.ts +75 -0
  372. package/src/lib/time-series-buffer.ts +58 -0
  373. package/src/lib/types.ts +173 -0
  374. package/src/lib/utils.ts +6 -0
  375. package/src/proxy.ts +104 -0
  376. package/src/types/db-drivers.d.ts +23 -0
  377. package/src/types/html2canvas.d.ts +9 -0
  378. package/tests/api/admin/audit.test.ts +178 -0
  379. package/tests/api/admin/fleet-health.test.ts +183 -0
  380. package/tests/api/ai/autopilot.test.ts +174 -0
  381. package/tests/api/ai/chat.test.ts +250 -0
  382. package/tests/api/ai/describe-schema.test.ts +266 -0
  383. package/tests/api/ai/explain.test.ts +199 -0
  384. package/tests/api/ai/impact.test.ts +168 -0
  385. package/tests/api/ai/index-advisor.test.ts +171 -0
  386. package/tests/api/ai/nl2sql.test.ts +202 -0
  387. package/tests/api/ai/query-safety.test.ts +196 -0
  388. package/tests/api/auth/login.test.ts +170 -0
  389. package/tests/api/auth/logout.test.ts +140 -0
  390. package/tests/api/auth/me.test.ts +73 -0
  391. package/tests/api/auth/oidc-callback.test.ts +215 -0
  392. package/tests/api/auth/oidc-login.test.ts +127 -0
  393. package/tests/api/db/cancel.test.ts +198 -0
  394. package/tests/api/db/disconnect.test.ts +124 -0
  395. package/tests/api/db/health.test.ts +222 -0
  396. package/tests/api/db/maintenance.test.ts +263 -0
  397. package/tests/api/db/monitoring.test.ts +221 -0
  398. package/tests/api/db/multi-query.test.ts +316 -0
  399. package/tests/api/db/pool-stats.test.ts +135 -0
  400. package/tests/api/db/profile.test.ts +330 -0
  401. package/tests/api/db/provider-meta.test.ts +193 -0
  402. package/tests/api/db/query.test.ts +314 -0
  403. package/tests/api/db/schema-snapshot.test.ts +170 -0
  404. package/tests/api/db/schema.test.ts +191 -0
  405. package/tests/api/db/test-connection.test.ts +185 -0
  406. package/tests/api/db/transaction.test.ts +314 -0
  407. package/tests/api/proxy.test.ts +191 -0
  408. package/tests/api/seed/managed-route.test.ts +113 -0
  409. package/tests/api/storage/config.test.ts +42 -0
  410. package/tests/api/storage/storage-routes.test.ts +309 -0
  411. package/tests/components/AIAutopilotPanel.test.tsx +756 -0
  412. package/tests/components/AdminPage.test.tsx +33 -0
  413. package/tests/components/CodeGenerator.test.tsx +182 -0
  414. package/tests/components/CommandPalette.test.tsx +428 -0
  415. package/tests/components/CommunitySection.test.tsx +91 -0
  416. package/tests/components/ConnectionModal.mobile.test.tsx +284 -0
  417. package/tests/components/ConnectionModal.test.tsx +570 -0
  418. package/tests/components/CreateTableModal.test.tsx +383 -0
  419. package/tests/components/DataCharts.test.tsx +739 -0
  420. package/tests/components/DataImportModal.test.tsx +751 -0
  421. package/tests/components/DataProfiler.test.tsx +589 -0
  422. package/tests/components/DatabaseDocs.test.tsx +353 -0
  423. package/tests/components/LoginPage.test.tsx +163 -0
  424. package/tests/components/LoginPageOIDC.test.tsx +92 -0
  425. package/tests/components/MaskingSettings.test.tsx +498 -0
  426. package/tests/components/MobileNav.test.tsx +30 -0
  427. package/tests/components/MonitoringPage.test.tsx +32 -0
  428. package/tests/components/NL2SQLPanel.test.tsx +621 -0
  429. package/tests/components/Page.test.tsx +33 -0
  430. package/tests/components/PivotTable.test.tsx +350 -0
  431. package/tests/components/QueryEditor.test.tsx +1730 -0
  432. package/tests/components/QueryHistory.test.tsx +572 -0
  433. package/tests/components/QuerySafetyDialog.test.tsx +586 -0
  434. package/tests/components/ResultsGrid.test.tsx +804 -0
  435. package/tests/components/RootLayout.test.tsx +83 -0
  436. package/tests/components/SaveQueryModal.test.tsx +25 -0
  437. package/tests/components/SavedQueries.test.tsx +43 -0
  438. package/tests/components/SchemaDiagram.test.tsx +1034 -0
  439. package/tests/components/SchemaDiff.test.tsx +906 -0
  440. package/tests/components/SnapshotTimeline.test.tsx +174 -0
  441. package/tests/components/Studio.test.tsx +1030 -0
  442. package/tests/components/TestDataGenerator.test.tsx +291 -0
  443. package/tests/components/VisualExplain.test.tsx +704 -0
  444. package/tests/components/admin/AdminDashboard.test.tsx +205 -0
  445. package/tests/components/admin/AuditTab.test.tsx +220 -0
  446. package/tests/components/admin/MonitoringEmbed.test.tsx +58 -0
  447. package/tests/components/admin/OperationsTab.test.tsx +975 -0
  448. package/tests/components/admin/OverviewTab.test.tsx +254 -0
  449. package/tests/components/admin/SecurityTab.test.tsx +467 -0
  450. package/tests/components/monitoring/MetricChart.test.tsx +111 -0
  451. package/tests/components/monitoring/MonitoringDashboard.test.tsx +259 -0
  452. package/tests/components/monitoring/OverviewTab.test.tsx +78 -0
  453. package/tests/components/monitoring/PerformanceTab.test.tsx +87 -0
  454. package/tests/components/monitoring/PoolTab.test.tsx +42 -0
  455. package/tests/components/monitoring/QueriesTab.test.tsx +80 -0
  456. package/tests/components/monitoring/SessionsTab.test.tsx +154 -0
  457. package/tests/components/monitoring/StorageTab.test.tsx +127 -0
  458. package/tests/components/monitoring/TablesTab.test.tsx +153 -0
  459. package/tests/components/results-grid/ResultCard.test.tsx +105 -0
  460. package/tests/components/results-grid/RowDetailSheet.test.tsx +308 -0
  461. package/tests/components/results-grid/StatsBar.test.tsx +162 -0
  462. package/tests/components/schema-explorer/ColumnList.test.tsx +151 -0
  463. package/tests/components/schema-explorer/SchemaExplorer.test.tsx +461 -0
  464. package/tests/components/schema-explorer/TableItem.test.tsx +415 -0
  465. package/tests/components/sidebar/ConnectionItem.test.tsx +201 -0
  466. package/tests/components/sidebar/ConnectionsList.test.tsx +176 -0
  467. package/tests/components/sidebar/Sidebar.test.tsx +187 -0
  468. package/tests/components/studio/BottomPanel.test.tsx +383 -0
  469. package/tests/components/studio/QueryToolbar.test.tsx +321 -0
  470. package/tests/components/studio/StudioDesktopHeader.test.tsx +377 -0
  471. package/tests/components/studio/StudioMobileHeader.test.tsx +198 -0
  472. package/tests/components/studio/StudioTabBar.test.tsx +331 -0
  473. package/tests/fixtures/connections.ts +96 -0
  474. package/tests/fixtures/masking-configs.ts +86 -0
  475. package/tests/fixtures/query-results.ts +71 -0
  476. package/tests/fixtures/schemas.ts +64 -0
  477. package/tests/fixtures/seed-connections/invalid-config.yaml +7 -0
  478. package/tests/fixtures/seed-connections/minimal-config.yaml +8 -0
  479. package/tests/fixtures/seed-connections/mixed-credentials.yaml +23 -0
  480. package/tests/fixtures/seed-connections/multi-role-config.yaml +30 -0
  481. package/tests/fixtures/seed-connections/valid-config.json +15 -0
  482. package/tests/fixtures/seed-connections/valid-config.yaml +51 -0
  483. package/tests/helpers/mock-fetch.ts +59 -0
  484. package/tests/helpers/mock-monaco.ts +112 -0
  485. package/tests/helpers/mock-navigation.ts +28 -0
  486. package/tests/helpers/mock-next.ts +80 -0
  487. package/tests/helpers/mock-provider.ts +133 -0
  488. package/tests/helpers/mock-sonner.ts +29 -0
  489. package/tests/helpers/render-with-providers.tsx +19 -0
  490. package/tests/hooks/use-ai-chat.test.ts +600 -0
  491. package/tests/hooks/use-auth.test.ts +371 -0
  492. package/tests/hooks/use-connection-form.test.ts +743 -0
  493. package/tests/hooks/use-connection-manager.test.ts +466 -0
  494. package/tests/hooks/use-inline-editing.test.ts +321 -0
  495. package/tests/hooks/use-mobile.test.ts +177 -0
  496. package/tests/hooks/use-monitoring-data.test.ts +819 -0
  497. package/tests/hooks/use-provider-metadata.test.ts +228 -0
  498. package/tests/hooks/use-query-execution.test.ts +1212 -0
  499. package/tests/hooks/use-tab-manager.test.ts +756 -0
  500. package/tests/hooks/use-toast.test.ts +74 -0
  501. package/tests/hooks/use-transaction-control.test.ts +211 -0
  502. package/tests/integration/db/mongodb-provider.test.ts +698 -0
  503. package/tests/integration/db/mssql-provider.test.ts +840 -0
  504. package/tests/integration/db/mysql-provider.test.ts +872 -0
  505. package/tests/integration/db/oracle-provider.test.ts +843 -0
  506. package/tests/integration/db/postgres-provider.test.ts +1382 -0
  507. package/tests/integration/db/redis-provider.test.ts +526 -0
  508. package/tests/integration/db/sqlite-provider.test.ts +480 -0
  509. package/tests/integration/seed/seed-pipeline.test.ts +102 -0
  510. package/tests/isolated/factory-singleton.test.ts +150 -0
  511. package/tests/isolated/use-storage-sync.test.ts +389 -0
  512. package/tests/run-components.sh +196 -0
  513. package/tests/setup-dom.ts +58 -0
  514. package/tests/setup.ts +40 -0
  515. package/tests/unit/api-errors.test.ts +210 -0
  516. package/tests/unit/code-generator-functions.test.ts +271 -0
  517. package/tests/unit/components/column-list.test.tsx +190 -0
  518. package/tests/unit/components/data-import-modal.test.tsx +441 -0
  519. package/tests/unit/components/studio-mobile-header.test.tsx +327 -0
  520. package/tests/unit/data-charts-functions.test.ts +496 -0
  521. package/tests/unit/data-import-functions.test.ts +320 -0
  522. package/tests/unit/data-import-utils.test.ts +125 -0
  523. package/tests/unit/db/base-provider.test.ts +517 -0
  524. package/tests/unit/db/errors.test.ts +403 -0
  525. package/tests/unit/db/factory.test.ts +436 -0
  526. package/tests/unit/db/pool-manager.test.ts +440 -0
  527. package/tests/unit/db/query-limiter.test.ts +387 -0
  528. package/tests/unit/db/sql-base.test.ts +438 -0
  529. package/tests/unit/lib/api/error-codes.test.ts +39 -0
  530. package/tests/unit/lib/audit.test.ts +326 -0
  531. package/tests/unit/lib/auth.test.ts +146 -0
  532. package/tests/unit/lib/connection-string-parser.test.ts +424 -0
  533. package/tests/unit/lib/data-masking.test.ts +583 -0
  534. package/tests/unit/lib/db-icons.test.tsx +41 -0
  535. package/tests/unit/lib/monitoring-thresholds.test.ts +133 -0
  536. package/tests/unit/lib/oidc.test.ts +509 -0
  537. package/tests/unit/lib/query-generators.test.ts +127 -0
  538. package/tests/unit/lib/storage/factory.test.ts +71 -0
  539. package/tests/unit/lib/storage/local-storage.test.ts +114 -0
  540. package/tests/unit/lib/storage/providers/postgres.test.ts +312 -0
  541. package/tests/unit/lib/storage/providers/sqlite.test.ts +232 -0
  542. package/tests/unit/lib/storage/storage-facade-extended.test.ts +331 -0
  543. package/tests/unit/lib/storage/storage-facade.test.ts +184 -0
  544. package/tests/unit/lib/storage.test.ts +317 -0
  545. package/tests/unit/lib/time-series-buffer.test.ts +212 -0
  546. package/tests/unit/lib/utils.test.ts +24 -0
  547. package/tests/unit/llm/base-provider.test.ts +238 -0
  548. package/tests/unit/llm/config.test.ts +262 -0
  549. package/tests/unit/llm/custom-provider.test.ts +281 -0
  550. package/tests/unit/llm/gemini-provider.test.ts +248 -0
  551. package/tests/unit/llm/llm-factory.test.ts +155 -0
  552. package/tests/unit/llm/ollama-provider.test.ts +288 -0
  553. package/tests/unit/llm/openai-provider.test.ts +324 -0
  554. package/tests/unit/llm/retry.test.ts +180 -0
  555. package/tests/unit/llm/streaming.test.ts +355 -0
  556. package/tests/unit/logger.test.ts +198 -0
  557. package/tests/unit/mongodb-completions.test.ts +516 -0
  558. package/tests/unit/pivot-table-functions.test.ts +76 -0
  559. package/tests/unit/query-cancelled-error.test.ts +81 -0
  560. package/tests/unit/schema-diff/diff-engine.test.ts +367 -0
  561. package/tests/unit/schema-diff/migration-generator.test.ts +513 -0
  562. package/tests/unit/seed/config-loader.test.ts +73 -0
  563. package/tests/unit/seed/connection-filter.test.ts +91 -0
  564. package/tests/unit/seed/credential-resolver.test.ts +85 -0
  565. package/tests/unit/seed/index.test.ts +72 -0
  566. package/tests/unit/seed/resolve-connection.test.ts +74 -0
  567. package/tests/unit/seed/types.test.ts +129 -0
  568. package/tests/unit/sql/alias-extractor.test.ts +444 -0
  569. package/tests/unit/sql/statement-splitter.test.ts +348 -0
  570. package/tests/unit/sql-completions.test.ts +463 -0
  571. package/tests/unit/ssh-tunnel.test.ts +465 -0
  572. 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
+ }