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