@libredb/studio 0.9.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude/settings.local.json +127 -0
- package/.cursorrules +426 -0
- package/.devin/wiki.json +143 -0
- package/.dockerignore +80 -0
- package/.env.example +159 -0
- package/.github/ISSUE_TEMPLATE/bug_report.md +49 -0
- package/.github/ISSUE_TEMPLATE/feature_request.md +29 -0
- package/.github/PULL_REQUEST_TEMPLATE.md +57 -0
- package/.github/workflows/ci.yml +185 -0
- package/.github/workflows/codeql.yml +57 -0
- package/.github/workflows/docker-build-push.yml +118 -0
- package/.github/workflows/helm-release.yml +113 -0
- package/CLAUDE.md +265 -0
- package/CODE_OF_CONDUCT.md +124 -0
- package/CONTRIBUTING.md +154 -0
- package/Dockerfile +73 -0
- package/LICENSE +21 -0
- package/README.md +614 -0
- package/SECURITY.md +107 -0
- package/artifacthub-repo.yml +4 -0
- package/bun.lock +1714 -0
- package/bunfig.toml +3 -0
- package/charts/libredb-studio/.helmignore +11 -0
- package/charts/libredb-studio/Chart.lock +6 -0
- package/charts/libredb-studio/Chart.yaml +50 -0
- package/charts/libredb-studio/README.md +206 -0
- package/charts/libredb-studio/templates/NOTES.txt +59 -0
- package/charts/libredb-studio/templates/_helpers.tpl +135 -0
- package/charts/libredb-studio/templates/configmap.yaml +37 -0
- package/charts/libredb-studio/templates/deployment.yaml +184 -0
- package/charts/libredb-studio/templates/hpa.yaml +32 -0
- package/charts/libredb-studio/templates/ingress.yaml +41 -0
- package/charts/libredb-studio/templates/networkpolicy.yaml +50 -0
- package/charts/libredb-studio/templates/pdb.yaml +18 -0
- package/charts/libredb-studio/templates/pvc.yaml +23 -0
- package/charts/libredb-studio/templates/secret.yaml +30 -0
- package/charts/libredb-studio/templates/seed-configmap.yaml +11 -0
- package/charts/libredb-studio/templates/service.yaml +22 -0
- package/charts/libredb-studio/templates/serviceaccount.yaml +13 -0
- package/charts/libredb-studio/values.schema.json +246 -0
- package/charts/libredb-studio/values.yaml +286 -0
- package/components.json +22 -0
- package/conductor/code_styleguides/typescript.md +43 -0
- package/conductor/product-guidelines.md +43 -0
- package/conductor/product.md +3 -0
- package/conductor/setup_state.json +1 -0
- package/conductor/tech-stack.md +39 -0
- package/conductor/tracks/enhance_postgres_monitoring_20251227/metadata.json +8 -0
- package/conductor/tracks/enhance_postgres_monitoring_20251227/plan.md +44 -0
- package/conductor/tracks/enhance_postgres_monitoring_20251227/spec.md +31 -0
- package/conductor/tracks.md +8 -0
- package/conductor/workflow.md +333 -0
- package/database-compose.yml +55 -0
- package/docker/postgres-init/01-extensions.sql +10 -0
- package/docker/postgres-init/02-sample-data.sql +585 -0
- package/docker/postgres.yml +68 -0
- package/docker-compose.yml +38 -0
- package/docs/AI_PLAN.md +74 -0
- package/docs/API_DOCS.md +875 -0
- package/docs/ARCHITECTURE.md +218 -0
- package/docs/DATABASE_PROVIDERS.md +358 -0
- package/docs/FEATURES.md +116 -0
- package/docs/HELM_CHART.md +252 -0
- package/docs/LOGIN_PAGE.md +178 -0
- package/docs/MONACO_EDITOR_PERFORMANCE.md +315 -0
- package/docs/OIDC_ARCH.md +681 -0
- package/docs/OIDC_SETUP.md +322 -0
- package/docs/POSTGRES_METRICS.md +516 -0
- package/docs/QUERY_OPTIMIZATION.md +370 -0
- package/docs/SEED_CONNECTIONS.md +468 -0
- package/docs/SQL_ALIAS_COMPLETION.md +190 -0
- package/docs/STORAGE_ARCHITECTURE.md +565 -0
- package/docs/STORAGE_QUICK_SETUP.md +419 -0
- package/docs/TECHNICAL_PLAN.md +36 -0
- package/docs/THEMING.md +345 -0
- package/docs/adding-a-new-database-provider.md +642 -0
- package/docs/backlogs/000-PLATFORM_DATA_SYNC_DATABASE.md +360 -0
- package/docs/backlogs/001-INLINE_DATA_EDITING.md +118 -0
- package/docs/backlogs/002-DATA_IMPORT.md +215 -0
- package/docs/backlogs/003-QUERY_TIME_MACHINE.md +183 -0
- package/docs/backlogs/004-AI_DATA_STORYTELLER.md +292 -0
- package/docs/backlogs/005-QUERY_PLAYGROUND.md +352 -0
- package/docs/backlogs/006-DATA_MASKING.md +418 -0
- package/docs/enterprise-features.md +718 -0
- package/docs/kubernetes-helm-chart-artifacthub-plan.md +803 -0
- package/docs/medium-koyeb-article-en.md +215 -0
- package/docs/plans/test-plans.md +445 -0
- package/docs/releases/RELEASE.V0.3.0.md +22 -0
- package/docs/releases/RELEASE.V0.4.0.md +154 -0
- package/docs/releases/RELEASE.V0.5.0.md +252 -0
- package/docs/releases/RELEASE_v0.5.6.md +145 -0
- package/docs/releases/RELEASE_v0.6.1.md +303 -0
- package/docs/releases/RELEASE_v0.6.7.md +292 -0
- package/docs/releases/RELEASE_v0.7.0.md +332 -0
- package/docs/releases/RELEASE_v0.8.0.md +521 -0
- package/docs/sampledb/titanic.sql +1379 -0
- package/docs/superpowers/plans/2026-03-25-seed-connections.md +1362 -0
- package/docs/superpowers/specs/2026-03-25-seed-connections-design.md +590 -0
- package/e2e/admin-dashboard.spec.ts +64 -0
- package/e2e/connection-management.spec.ts +58 -0
- package/e2e/export.spec.ts +34 -0
- package/e2e/login.spec.ts +85 -0
- package/e2e/query-execution.spec.ts +35 -0
- package/e2e/tab-management.spec.ts +64 -0
- package/eslint.config.mjs +28 -0
- package/fly.toml +43 -0
- package/next.config.ts +32 -0
- package/package.json +130 -0
- package/playwright.config.ts +34 -0
- package/postcss.config.mjs +7 -0
- package/public/favicon-32x32.png +0 -0
- package/public/favicon.ico +0 -0
- package/public/file.svg +1 -0
- package/public/globe.svg +1 -0
- package/public/logo.svg +32 -0
- package/public/next.svg +1 -0
- package/public/screenshots/code-generator.png +0 -0
- package/public/screenshots/connection-modal.png +0 -0
- package/public/screenshots/data-profiler.png +0 -0
- package/public/screenshots/erd-diagram.png +0 -0
- package/public/screenshots/hero-editor.png +0 -0
- package/public/screenshots/nl2sql.png +0 -0
- package/public/vercel.svg +1 -0
- package/public/window.svg +1 -0
- package/render.yaml +58 -0
- package/scripts/merge-lcov.mjs +239 -0
- package/sonar-project.properties +16 -0
- package/src/app/admin/error.tsx +46 -0
- package/src/app/admin/page.tsx +10 -0
- package/src/app/api/admin/audit/route.ts +52 -0
- package/src/app/api/admin/fleet-health/route.ts +81 -0
- package/src/app/api/ai/autopilot/route.ts +105 -0
- package/src/app/api/ai/chat/route.ts +132 -0
- package/src/app/api/ai/describe-schema/route.ts +52 -0
- package/src/app/api/ai/explain/route.ts +86 -0
- package/src/app/api/ai/impact/route.ts +97 -0
- package/src/app/api/ai/index-advisor/route.ts +98 -0
- package/src/app/api/ai/nl2sql/route.ts +87 -0
- package/src/app/api/ai/query-safety/route.ts +87 -0
- package/src/app/api/auth/login/route.ts +62 -0
- package/src/app/api/auth/logout/route.ts +25 -0
- package/src/app/api/auth/me/route.ts +10 -0
- package/src/app/api/auth/oidc/callback/route.ts +82 -0
- package/src/app/api/auth/oidc/login/route.ts +43 -0
- package/src/app/api/connections/managed/route.ts +35 -0
- package/src/app/api/db/cancel/route.ts +42 -0
- package/src/app/api/db/disconnect/route.ts +28 -0
- package/src/app/api/db/health/route.ts +49 -0
- package/src/app/api/db/maintenance/route.ts +72 -0
- package/src/app/api/db/monitoring/route.ts +62 -0
- package/src/app/api/db/multi-query/route.ts +116 -0
- package/src/app/api/db/pool-stats/route.ts +37 -0
- package/src/app/api/db/profile/route.ts +144 -0
- package/src/app/api/db/provider-meta/route.ts +49 -0
- package/src/app/api/db/query/route.ts +50 -0
- package/src/app/api/db/schema/route.ts +47 -0
- package/src/app/api/db/schema-snapshot/route.ts +42 -0
- package/src/app/api/db/test-connection/route.ts +55 -0
- package/src/app/api/db/transaction/route.ts +111 -0
- package/src/app/api/storage/[collection]/route.ts +67 -0
- package/src/app/api/storage/config/route.ts +17 -0
- package/src/app/api/storage/migrate/route.ts +45 -0
- package/src/app/api/storage/route.ts +32 -0
- package/src/app/error.tsx +49 -0
- package/src/app/global-error.tsx +55 -0
- package/src/app/globals.css +146 -0
- package/src/app/icon.svg +42 -0
- package/src/app/layout.tsx +34 -0
- package/src/app/login/login-form.tsx +301 -0
- package/src/app/login/page.tsx +11 -0
- package/src/app/monitoring/page.tsx +8 -0
- package/src/app/not-found.tsx +29 -0
- package/src/app/page.tsx +5 -0
- package/src/components/AIAutopilotPanel.tsx +238 -0
- package/src/components/CodeGenerator.tsx +271 -0
- package/src/components/CommandPalette.tsx +227 -0
- package/src/components/ConnectionModal.tsx +759 -0
- package/src/components/CreateTableModal.tsx +281 -0
- package/src/components/DataCharts.tsx +962 -0
- package/src/components/DataImportModal.tsx +582 -0
- package/src/components/DataProfiler.tsx +335 -0
- package/src/components/DatabaseDocs.tsx +251 -0
- package/src/components/MaskingSettings.tsx +414 -0
- package/src/components/MobileNav.tsx +50 -0
- package/src/components/NL2SQLPanel.tsx +281 -0
- package/src/components/PivotTable.tsx +257 -0
- package/src/components/QueryEditor.tsx +760 -0
- package/src/components/QueryHistory.tsx +344 -0
- package/src/components/QuerySafetyDialog.tsx +290 -0
- package/src/components/ResultsGrid.tsx +644 -0
- package/src/components/SaveQueryModal.tsx +104 -0
- package/src/components/SavedQueries.tsx +128 -0
- package/src/components/SchemaDiagram.tsx +473 -0
- package/src/components/SchemaDiff.tsx +473 -0
- package/src/components/SnapshotTimeline.tsx +116 -0
- package/src/components/Studio.tsx +639 -0
- package/src/components/TestDataGenerator.tsx +261 -0
- package/src/components/VisualExplain.tsx +820 -0
- package/src/components/admin/AdminDashboard.tsx +163 -0
- package/src/components/admin/tabs/AuditTab.tsx +531 -0
- package/src/components/admin/tabs/MonitoringEmbed.tsx +11 -0
- package/src/components/admin/tabs/OperationsTab.tsx +646 -0
- package/src/components/admin/tabs/OverviewTab.tsx +1328 -0
- package/src/components/admin/tabs/SecurityTab.tsx +284 -0
- package/src/components/community-section.tsx +92 -0
- package/src/components/icons/db-icons.tsx +84 -0
- package/src/components/libredb-logo.tsx +61 -0
- package/src/components/monitoring/MonitoringDashboard.tsx +345 -0
- package/src/components/monitoring/tabs/MetricChart.tsx +82 -0
- package/src/components/monitoring/tabs/OverviewTab.tsx +263 -0
- package/src/components/monitoring/tabs/PerformanceTab.tsx +254 -0
- package/src/components/monitoring/tabs/PoolTab.tsx +174 -0
- package/src/components/monitoring/tabs/QueriesTab.tsx +287 -0
- package/src/components/monitoring/tabs/SessionsTab.tsx +316 -0
- package/src/components/monitoring/tabs/StorageTab.tsx +335 -0
- package/src/components/monitoring/tabs/TablesTab.tsx +300 -0
- package/src/components/results-grid/ResultCard.tsx +111 -0
- package/src/components/results-grid/RowDetailSheet.tsx +178 -0
- package/src/components/results-grid/StatsBar.tsx +201 -0
- package/src/components/results-grid/index.ts +1 -0
- package/src/components/results-grid/utils.ts +23 -0
- package/src/components/schema-explorer/ColumnList.tsx +53 -0
- package/src/components/schema-explorer/SchemaExplorer.tsx +182 -0
- package/src/components/schema-explorer/TableItem.tsx +210 -0
- package/src/components/schema-explorer/index.ts +1 -0
- package/src/components/sidebar/ConnectionItem.tsx +105 -0
- package/src/components/sidebar/ConnectionsList.tsx +62 -0
- package/src/components/sidebar/Sidebar.tsx +130 -0
- package/src/components/sidebar/index.ts +2 -0
- package/src/components/studio/BottomPanel.tsx +286 -0
- package/src/components/studio/QueryToolbar.tsx +180 -0
- package/src/components/studio/StudioDesktopHeader.tsx +114 -0
- package/src/components/studio/StudioMobileHeader.tsx +340 -0
- package/src/components/studio/StudioTabBar.tsx +82 -0
- package/src/components/studio/index.ts +5 -0
- package/src/components/ui/accordion.tsx +66 -0
- package/src/components/ui/alert-dialog.tsx +157 -0
- package/src/components/ui/alert.tsx +66 -0
- package/src/components/ui/aspect-ratio.tsx +11 -0
- package/src/components/ui/avatar.tsx +53 -0
- package/src/components/ui/badge.tsx +46 -0
- package/src/components/ui/breadcrumb.tsx +109 -0
- package/src/components/ui/button-group.tsx +83 -0
- package/src/components/ui/button.tsx +60 -0
- package/src/components/ui/calendar.tsx +216 -0
- package/src/components/ui/card.tsx +92 -0
- package/src/components/ui/carousel.tsx +241 -0
- package/src/components/ui/chart.tsx +357 -0
- package/src/components/ui/checkbox.tsx +32 -0
- package/src/components/ui/collapsible.tsx +33 -0
- package/src/components/ui/command.tsx +184 -0
- package/src/components/ui/context-menu.tsx +252 -0
- package/src/components/ui/dialog.tsx +143 -0
- package/src/components/ui/drawer.tsx +135 -0
- package/src/components/ui/dropdown-menu.tsx +257 -0
- package/src/components/ui/empty.tsx +104 -0
- package/src/components/ui/field.tsx +248 -0
- package/src/components/ui/form.tsx +167 -0
- package/src/components/ui/hover-card.tsx +44 -0
- package/src/components/ui/input-group.tsx +170 -0
- package/src/components/ui/input-otp.tsx +77 -0
- package/src/components/ui/input.tsx +21 -0
- package/src/components/ui/item.tsx +193 -0
- package/src/components/ui/kbd.tsx +28 -0
- package/src/components/ui/label.tsx +24 -0
- package/src/components/ui/menubar.tsx +276 -0
- package/src/components/ui/navigation-menu.tsx +168 -0
- package/src/components/ui/pagination.tsx +127 -0
- package/src/components/ui/popover.tsx +48 -0
- package/src/components/ui/progress.tsx +31 -0
- package/src/components/ui/radio-group.tsx +45 -0
- package/src/components/ui/resizable.tsx +56 -0
- package/src/components/ui/scroll-area.tsx +58 -0
- package/src/components/ui/select.tsx +187 -0
- package/src/components/ui/separator.tsx +28 -0
- package/src/components/ui/sheet.tsx +139 -0
- package/src/components/ui/sidebar.tsx +726 -0
- package/src/components/ui/skeleton.tsx +13 -0
- package/src/components/ui/slider.tsx +63 -0
- package/src/components/ui/sonner.tsx +40 -0
- package/src/components/ui/spinner.tsx +16 -0
- package/src/components/ui/switch.tsx +31 -0
- package/src/components/ui/table.tsx +116 -0
- package/src/components/ui/tabs.tsx +66 -0
- package/src/components/ui/textarea.tsx +18 -0
- package/src/components/ui/toggle-group.tsx +83 -0
- package/src/components/ui/toggle.tsx +47 -0
- package/src/components/ui/tooltip.tsx +61 -0
- package/src/exports/components.ts +15 -0
- package/src/exports/index.ts +4 -0
- package/src/exports/providers.ts +4 -0
- package/src/exports/types.ts +26 -0
- package/src/hooks/use-ai-chat.ts +182 -0
- package/src/hooks/use-all-connections.ts +66 -0
- package/src/hooks/use-api-call.ts +71 -0
- package/src/hooks/use-auth.ts +51 -0
- package/src/hooks/use-connection-form.ts +349 -0
- package/src/hooks/use-connection-manager.ts +169 -0
- package/src/hooks/use-connection-payload.ts +15 -0
- package/src/hooks/use-inline-editing.ts +109 -0
- package/src/hooks/use-mobile.ts +20 -0
- package/src/hooks/use-monitoring-data.ts +270 -0
- package/src/hooks/use-provider-metadata.ts +62 -0
- package/src/hooks/use-query-execution.ts +478 -0
- package/src/hooks/use-storage-sync.ts +259 -0
- package/src/hooks/use-tab-manager.ts +231 -0
- package/src/hooks/use-toast.ts +20 -0
- package/src/hooks/use-transaction-control.ts +64 -0
- package/src/lib/api/error-codes.ts +30 -0
- package/src/lib/api/errors.ts +236 -0
- package/src/lib/api/with-error-handler.ts +41 -0
- package/src/lib/audit.ts +105 -0
- package/src/lib/auth.ts +87 -0
- package/src/lib/connection-string-parser.ts +172 -0
- package/src/lib/data-masking.ts +385 -0
- package/src/lib/db/base-provider.ts +325 -0
- package/src/lib/db/errors.ts +317 -0
- package/src/lib/db/factory.ts +324 -0
- package/src/lib/db/index.ts +123 -0
- package/src/lib/db/providers/document/index.ts +6 -0
- package/src/lib/db/providers/document/mongodb.ts +992 -0
- package/src/lib/db/providers/keyvalue/redis.ts +554 -0
- package/src/lib/db/providers/sql/index.ts +11 -0
- package/src/lib/db/providers/sql/mssql.ts +1065 -0
- package/src/lib/db/providers/sql/mysql.ts +978 -0
- package/src/lib/db/providers/sql/oracle.ts +1044 -0
- package/src/lib/db/providers/sql/postgres.ts +1179 -0
- package/src/lib/db/providers/sql/sql-base.ts +174 -0
- package/src/lib/db/providers/sql/sqlite.ts +721 -0
- package/src/lib/db/types.ts +437 -0
- package/src/lib/db/utils/pool-manager.ts +287 -0
- package/src/lib/db/utils/query-limiter.ts +239 -0
- package/src/lib/db-ui-config.ts +86 -0
- package/src/lib/editor/mongodb-completions.ts +172 -0
- package/src/lib/editor/sql-completions.ts +280 -0
- package/src/lib/llm/base-provider.ts +117 -0
- package/src/lib/llm/factory.ts +102 -0
- package/src/lib/llm/index.ts +90 -0
- package/src/lib/llm/providers/custom.ts +181 -0
- package/src/lib/llm/providers/gemini.ts +126 -0
- package/src/lib/llm/providers/ollama.ts +154 -0
- package/src/lib/llm/providers/openai.ts +146 -0
- package/src/lib/llm/types.ts +173 -0
- package/src/lib/llm/utils/config.ts +187 -0
- package/src/lib/llm/utils/retry.ts +119 -0
- package/src/lib/llm/utils/streaming.ts +202 -0
- package/src/lib/logger.ts +127 -0
- package/src/lib/monitoring-thresholds.ts +44 -0
- package/src/lib/oidc.ts +262 -0
- package/src/lib/query-generators.ts +61 -0
- package/src/lib/schema-diff/diff-engine.ts +273 -0
- package/src/lib/schema-diff/migration-generator.ts +208 -0
- package/src/lib/schema-diff/types.ts +55 -0
- package/src/lib/seed/config-loader.ts +79 -0
- package/src/lib/seed/connection-filter.ts +49 -0
- package/src/lib/seed/credential-resolver.ts +62 -0
- package/src/lib/seed/index.ts +40 -0
- package/src/lib/seed/resolve-connection.ts +57 -0
- package/src/lib/seed/types.ts +69 -0
- package/src/lib/sql/alias-extractor.ts +267 -0
- package/src/lib/sql/index.ts +8 -0
- package/src/lib/sql/statement-splitter.ts +167 -0
- package/src/lib/sql/types.ts +40 -0
- package/src/lib/ssh/tunnel.ts +142 -0
- package/src/lib/storage/factory.ts +84 -0
- package/src/lib/storage/index.ts +14 -0
- package/src/lib/storage/local-storage.ts +99 -0
- package/src/lib/storage/providers/postgres.ts +225 -0
- package/src/lib/storage/providers/sqlite.ts +153 -0
- package/src/lib/storage/storage-facade.ts +272 -0
- package/src/lib/storage/types.ts +75 -0
- package/src/lib/time-series-buffer.ts +58 -0
- package/src/lib/types.ts +173 -0
- package/src/lib/utils.ts +6 -0
- package/src/proxy.ts +104 -0
- package/src/types/db-drivers.d.ts +23 -0
- package/src/types/html2canvas.d.ts +9 -0
- package/tests/api/admin/audit.test.ts +178 -0
- package/tests/api/admin/fleet-health.test.ts +183 -0
- package/tests/api/ai/autopilot.test.ts +174 -0
- package/tests/api/ai/chat.test.ts +250 -0
- package/tests/api/ai/describe-schema.test.ts +266 -0
- package/tests/api/ai/explain.test.ts +199 -0
- package/tests/api/ai/impact.test.ts +168 -0
- package/tests/api/ai/index-advisor.test.ts +171 -0
- package/tests/api/ai/nl2sql.test.ts +202 -0
- package/tests/api/ai/query-safety.test.ts +196 -0
- package/tests/api/auth/login.test.ts +170 -0
- package/tests/api/auth/logout.test.ts +140 -0
- package/tests/api/auth/me.test.ts +73 -0
- package/tests/api/auth/oidc-callback.test.ts +215 -0
- package/tests/api/auth/oidc-login.test.ts +127 -0
- package/tests/api/db/cancel.test.ts +198 -0
- package/tests/api/db/disconnect.test.ts +124 -0
- package/tests/api/db/health.test.ts +222 -0
- package/tests/api/db/maintenance.test.ts +263 -0
- package/tests/api/db/monitoring.test.ts +221 -0
- package/tests/api/db/multi-query.test.ts +316 -0
- package/tests/api/db/pool-stats.test.ts +135 -0
- package/tests/api/db/profile.test.ts +330 -0
- package/tests/api/db/provider-meta.test.ts +193 -0
- package/tests/api/db/query.test.ts +314 -0
- package/tests/api/db/schema-snapshot.test.ts +170 -0
- package/tests/api/db/schema.test.ts +191 -0
- package/tests/api/db/test-connection.test.ts +185 -0
- package/tests/api/db/transaction.test.ts +314 -0
- package/tests/api/proxy.test.ts +191 -0
- package/tests/api/seed/managed-route.test.ts +113 -0
- package/tests/api/storage/config.test.ts +42 -0
- package/tests/api/storage/storage-routes.test.ts +309 -0
- package/tests/components/AIAutopilotPanel.test.tsx +756 -0
- package/tests/components/AdminPage.test.tsx +33 -0
- package/tests/components/CodeGenerator.test.tsx +182 -0
- package/tests/components/CommandPalette.test.tsx +428 -0
- package/tests/components/CommunitySection.test.tsx +91 -0
- package/tests/components/ConnectionModal.mobile.test.tsx +284 -0
- package/tests/components/ConnectionModal.test.tsx +570 -0
- package/tests/components/CreateTableModal.test.tsx +383 -0
- package/tests/components/DataCharts.test.tsx +739 -0
- package/tests/components/DataImportModal.test.tsx +751 -0
- package/tests/components/DataProfiler.test.tsx +589 -0
- package/tests/components/DatabaseDocs.test.tsx +353 -0
- package/tests/components/LoginPage.test.tsx +163 -0
- package/tests/components/LoginPageOIDC.test.tsx +92 -0
- package/tests/components/MaskingSettings.test.tsx +498 -0
- package/tests/components/MobileNav.test.tsx +30 -0
- package/tests/components/MonitoringPage.test.tsx +32 -0
- package/tests/components/NL2SQLPanel.test.tsx +621 -0
- package/tests/components/Page.test.tsx +33 -0
- package/tests/components/PivotTable.test.tsx +350 -0
- package/tests/components/QueryEditor.test.tsx +1730 -0
- package/tests/components/QueryHistory.test.tsx +572 -0
- package/tests/components/QuerySafetyDialog.test.tsx +586 -0
- package/tests/components/ResultsGrid.test.tsx +804 -0
- package/tests/components/RootLayout.test.tsx +83 -0
- package/tests/components/SaveQueryModal.test.tsx +25 -0
- package/tests/components/SavedQueries.test.tsx +43 -0
- package/tests/components/SchemaDiagram.test.tsx +1034 -0
- package/tests/components/SchemaDiff.test.tsx +906 -0
- package/tests/components/SnapshotTimeline.test.tsx +174 -0
- package/tests/components/Studio.test.tsx +1030 -0
- package/tests/components/TestDataGenerator.test.tsx +291 -0
- package/tests/components/VisualExplain.test.tsx +704 -0
- package/tests/components/admin/AdminDashboard.test.tsx +205 -0
- package/tests/components/admin/AuditTab.test.tsx +220 -0
- package/tests/components/admin/MonitoringEmbed.test.tsx +58 -0
- package/tests/components/admin/OperationsTab.test.tsx +975 -0
- package/tests/components/admin/OverviewTab.test.tsx +254 -0
- package/tests/components/admin/SecurityTab.test.tsx +467 -0
- package/tests/components/monitoring/MetricChart.test.tsx +111 -0
- package/tests/components/monitoring/MonitoringDashboard.test.tsx +259 -0
- package/tests/components/monitoring/OverviewTab.test.tsx +78 -0
- package/tests/components/monitoring/PerformanceTab.test.tsx +87 -0
- package/tests/components/monitoring/PoolTab.test.tsx +42 -0
- package/tests/components/monitoring/QueriesTab.test.tsx +80 -0
- package/tests/components/monitoring/SessionsTab.test.tsx +154 -0
- package/tests/components/monitoring/StorageTab.test.tsx +127 -0
- package/tests/components/monitoring/TablesTab.test.tsx +153 -0
- package/tests/components/results-grid/ResultCard.test.tsx +105 -0
- package/tests/components/results-grid/RowDetailSheet.test.tsx +308 -0
- package/tests/components/results-grid/StatsBar.test.tsx +162 -0
- package/tests/components/schema-explorer/ColumnList.test.tsx +151 -0
- package/tests/components/schema-explorer/SchemaExplorer.test.tsx +461 -0
- package/tests/components/schema-explorer/TableItem.test.tsx +415 -0
- package/tests/components/sidebar/ConnectionItem.test.tsx +201 -0
- package/tests/components/sidebar/ConnectionsList.test.tsx +176 -0
- package/tests/components/sidebar/Sidebar.test.tsx +187 -0
- package/tests/components/studio/BottomPanel.test.tsx +383 -0
- package/tests/components/studio/QueryToolbar.test.tsx +321 -0
- package/tests/components/studio/StudioDesktopHeader.test.tsx +377 -0
- package/tests/components/studio/StudioMobileHeader.test.tsx +198 -0
- package/tests/components/studio/StudioTabBar.test.tsx +331 -0
- package/tests/fixtures/connections.ts +96 -0
- package/tests/fixtures/masking-configs.ts +86 -0
- package/tests/fixtures/query-results.ts +71 -0
- package/tests/fixtures/schemas.ts +64 -0
- package/tests/fixtures/seed-connections/invalid-config.yaml +7 -0
- package/tests/fixtures/seed-connections/minimal-config.yaml +8 -0
- package/tests/fixtures/seed-connections/mixed-credentials.yaml +23 -0
- package/tests/fixtures/seed-connections/multi-role-config.yaml +30 -0
- package/tests/fixtures/seed-connections/valid-config.json +15 -0
- package/tests/fixtures/seed-connections/valid-config.yaml +51 -0
- package/tests/helpers/mock-fetch.ts +59 -0
- package/tests/helpers/mock-monaco.ts +112 -0
- package/tests/helpers/mock-navigation.ts +28 -0
- package/tests/helpers/mock-next.ts +80 -0
- package/tests/helpers/mock-provider.ts +133 -0
- package/tests/helpers/mock-sonner.ts +29 -0
- package/tests/helpers/render-with-providers.tsx +19 -0
- package/tests/hooks/use-ai-chat.test.ts +600 -0
- package/tests/hooks/use-auth.test.ts +371 -0
- package/tests/hooks/use-connection-form.test.ts +743 -0
- package/tests/hooks/use-connection-manager.test.ts +466 -0
- package/tests/hooks/use-inline-editing.test.ts +321 -0
- package/tests/hooks/use-mobile.test.ts +177 -0
- package/tests/hooks/use-monitoring-data.test.ts +819 -0
- package/tests/hooks/use-provider-metadata.test.ts +228 -0
- package/tests/hooks/use-query-execution.test.ts +1212 -0
- package/tests/hooks/use-tab-manager.test.ts +756 -0
- package/tests/hooks/use-toast.test.ts +74 -0
- package/tests/hooks/use-transaction-control.test.ts +211 -0
- package/tests/integration/db/mongodb-provider.test.ts +698 -0
- package/tests/integration/db/mssql-provider.test.ts +840 -0
- package/tests/integration/db/mysql-provider.test.ts +872 -0
- package/tests/integration/db/oracle-provider.test.ts +843 -0
- package/tests/integration/db/postgres-provider.test.ts +1382 -0
- package/tests/integration/db/redis-provider.test.ts +526 -0
- package/tests/integration/db/sqlite-provider.test.ts +480 -0
- package/tests/integration/seed/seed-pipeline.test.ts +102 -0
- package/tests/isolated/factory-singleton.test.ts +150 -0
- package/tests/isolated/use-storage-sync.test.ts +389 -0
- package/tests/run-components.sh +196 -0
- package/tests/setup-dom.ts +58 -0
- package/tests/setup.ts +40 -0
- package/tests/unit/api-errors.test.ts +210 -0
- package/tests/unit/code-generator-functions.test.ts +271 -0
- package/tests/unit/components/column-list.test.tsx +190 -0
- package/tests/unit/components/data-import-modal.test.tsx +441 -0
- package/tests/unit/components/studio-mobile-header.test.tsx +327 -0
- package/tests/unit/data-charts-functions.test.ts +496 -0
- package/tests/unit/data-import-functions.test.ts +320 -0
- package/tests/unit/data-import-utils.test.ts +125 -0
- package/tests/unit/db/base-provider.test.ts +517 -0
- package/tests/unit/db/errors.test.ts +403 -0
- package/tests/unit/db/factory.test.ts +436 -0
- package/tests/unit/db/pool-manager.test.ts +440 -0
- package/tests/unit/db/query-limiter.test.ts +387 -0
- package/tests/unit/db/sql-base.test.ts +438 -0
- package/tests/unit/lib/api/error-codes.test.ts +39 -0
- package/tests/unit/lib/audit.test.ts +326 -0
- package/tests/unit/lib/auth.test.ts +146 -0
- package/tests/unit/lib/connection-string-parser.test.ts +424 -0
- package/tests/unit/lib/data-masking.test.ts +583 -0
- package/tests/unit/lib/db-icons.test.tsx +41 -0
- package/tests/unit/lib/monitoring-thresholds.test.ts +133 -0
- package/tests/unit/lib/oidc.test.ts +509 -0
- package/tests/unit/lib/query-generators.test.ts +127 -0
- package/tests/unit/lib/storage/factory.test.ts +71 -0
- package/tests/unit/lib/storage/local-storage.test.ts +114 -0
- package/tests/unit/lib/storage/providers/postgres.test.ts +312 -0
- package/tests/unit/lib/storage/providers/sqlite.test.ts +232 -0
- package/tests/unit/lib/storage/storage-facade-extended.test.ts +331 -0
- package/tests/unit/lib/storage/storage-facade.test.ts +184 -0
- package/tests/unit/lib/storage.test.ts +317 -0
- package/tests/unit/lib/time-series-buffer.test.ts +212 -0
- package/tests/unit/lib/utils.test.ts +24 -0
- package/tests/unit/llm/base-provider.test.ts +238 -0
- package/tests/unit/llm/config.test.ts +262 -0
- package/tests/unit/llm/custom-provider.test.ts +281 -0
- package/tests/unit/llm/gemini-provider.test.ts +248 -0
- package/tests/unit/llm/llm-factory.test.ts +155 -0
- package/tests/unit/llm/ollama-provider.test.ts +288 -0
- package/tests/unit/llm/openai-provider.test.ts +324 -0
- package/tests/unit/llm/retry.test.ts +180 -0
- package/tests/unit/llm/streaming.test.ts +355 -0
- package/tests/unit/logger.test.ts +198 -0
- package/tests/unit/mongodb-completions.test.ts +516 -0
- package/tests/unit/pivot-table-functions.test.ts +76 -0
- package/tests/unit/query-cancelled-error.test.ts +81 -0
- package/tests/unit/schema-diff/diff-engine.test.ts +367 -0
- package/tests/unit/schema-diff/migration-generator.test.ts +513 -0
- package/tests/unit/seed/config-loader.test.ts +73 -0
- package/tests/unit/seed/connection-filter.test.ts +91 -0
- package/tests/unit/seed/credential-resolver.test.ts +85 -0
- package/tests/unit/seed/index.test.ts +72 -0
- package/tests/unit/seed/resolve-connection.test.ts +74 -0
- package/tests/unit/seed/types.test.ts +129 -0
- package/tests/unit/sql/alias-extractor.test.ts +444 -0
- package/tests/unit/sql/statement-splitter.test.ts +348 -0
- package/tests/unit/sql-completions.test.ts +463 -0
- package/tests/unit/ssh-tunnel.test.ts +465 -0
- package/tsconfig.json +42 -0
|
@@ -0,0 +1,173 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* LLM Provider Types & Interfaces
|
|
3
|
+
* Strategy Pattern implementation for multi-provider LLM support
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
// ============================================================================
|
|
7
|
+
// Provider Types
|
|
8
|
+
// ============================================================================
|
|
9
|
+
|
|
10
|
+
export type LLMProviderType = 'gemini' | 'openai' | 'ollama' | 'custom';
|
|
11
|
+
|
|
12
|
+
// ============================================================================
|
|
13
|
+
// Configuration
|
|
14
|
+
// ============================================================================
|
|
15
|
+
|
|
16
|
+
export interface LLMConfig {
|
|
17
|
+
provider: LLMProviderType;
|
|
18
|
+
apiKey?: string;
|
|
19
|
+
model: string;
|
|
20
|
+
apiUrl?: string;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
// ============================================================================
|
|
24
|
+
// Messages
|
|
25
|
+
// ============================================================================
|
|
26
|
+
|
|
27
|
+
export type LLMMessageRole = 'system' | 'user' | 'assistant';
|
|
28
|
+
|
|
29
|
+
export interface LLMMessage {
|
|
30
|
+
role: LLMMessageRole;
|
|
31
|
+
content: string;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
// ============================================================================
|
|
35
|
+
// Stream Options
|
|
36
|
+
// ============================================================================
|
|
37
|
+
|
|
38
|
+
export interface LLMStreamOptions {
|
|
39
|
+
messages: LLMMessage[];
|
|
40
|
+
model?: string;
|
|
41
|
+
temperature?: number;
|
|
42
|
+
maxTokens?: number;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
// ============================================================================
|
|
46
|
+
// Provider Interface (Strategy Pattern)
|
|
47
|
+
// ============================================================================
|
|
48
|
+
|
|
49
|
+
export interface LLMProvider {
|
|
50
|
+
/** Provider identifier */
|
|
51
|
+
readonly name: LLMProviderType;
|
|
52
|
+
|
|
53
|
+
/** Current configuration */
|
|
54
|
+
readonly config: LLMConfig;
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Stream a completion response
|
|
58
|
+
* @param options - Stream configuration including messages
|
|
59
|
+
* @returns ReadableStream of UTF-8 encoded chunks
|
|
60
|
+
*/
|
|
61
|
+
stream(options: LLMStreamOptions): Promise<ReadableStream<Uint8Array>>;
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* Validate provider configuration
|
|
65
|
+
* @throws LLMConfigError if configuration is invalid
|
|
66
|
+
*/
|
|
67
|
+
validate(): void;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
// ============================================================================
|
|
71
|
+
// Error Classes
|
|
72
|
+
// ============================================================================
|
|
73
|
+
|
|
74
|
+
/**
|
|
75
|
+
* Base error class for LLM-related errors
|
|
76
|
+
*/
|
|
77
|
+
export class LLMError extends Error {
|
|
78
|
+
constructor(
|
|
79
|
+
message: string,
|
|
80
|
+
public readonly provider?: LLMProviderType,
|
|
81
|
+
public readonly statusCode?: number
|
|
82
|
+
) {
|
|
83
|
+
super(message);
|
|
84
|
+
this.name = 'LLMError';
|
|
85
|
+
Object.setPrototypeOf(this, LLMError.prototype);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Configuration error - missing or invalid config
|
|
91
|
+
*/
|
|
92
|
+
export class LLMConfigError extends LLMError {
|
|
93
|
+
constructor(message: string, provider?: LLMProviderType) {
|
|
94
|
+
super(message, provider);
|
|
95
|
+
this.name = 'LLMConfigError';
|
|
96
|
+
Object.setPrototypeOf(this, LLMConfigError.prototype);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/**
|
|
101
|
+
* Authentication error - invalid API key (401/403)
|
|
102
|
+
*/
|
|
103
|
+
export class LLMAuthError extends LLMError {
|
|
104
|
+
constructor(message: string, provider?: LLMProviderType) {
|
|
105
|
+
super(message, provider, 401);
|
|
106
|
+
this.name = 'LLMAuthError';
|
|
107
|
+
Object.setPrototypeOf(this, LLMAuthError.prototype);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Rate limit error - quota exceeded (429)
|
|
113
|
+
*/
|
|
114
|
+
export class LLMRateLimitError extends LLMError {
|
|
115
|
+
constructor(message: string, provider?: LLMProviderType) {
|
|
116
|
+
super(message, provider, 429);
|
|
117
|
+
this.name = 'LLMRateLimitError';
|
|
118
|
+
Object.setPrototypeOf(this, LLMRateLimitError.prototype);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
/**
|
|
123
|
+
* Safety filter error - content blocked
|
|
124
|
+
*/
|
|
125
|
+
export class LLMSafetyError extends LLMError {
|
|
126
|
+
constructor(message: string, provider?: LLMProviderType) {
|
|
127
|
+
super(message, provider, 400);
|
|
128
|
+
this.name = 'LLMSafetyError';
|
|
129
|
+
Object.setPrototypeOf(this, LLMSafetyError.prototype);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Streaming error - connection or parsing failure
|
|
135
|
+
*/
|
|
136
|
+
export class LLMStreamError extends LLMError {
|
|
137
|
+
constructor(message: string, provider?: LLMProviderType) {
|
|
138
|
+
super(message, provider);
|
|
139
|
+
this.name = 'LLMStreamError';
|
|
140
|
+
Object.setPrototypeOf(this, LLMStreamError.prototype);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// ============================================================================
|
|
145
|
+
// Type Guards
|
|
146
|
+
// ============================================================================
|
|
147
|
+
|
|
148
|
+
export function isLLMError(error: unknown): error is LLMError {
|
|
149
|
+
return error instanceof LLMError;
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
export function isRetryableError(error: unknown): boolean {
|
|
153
|
+
if (!isLLMError(error)) {
|
|
154
|
+
// Network errors are retryable
|
|
155
|
+
if (error instanceof TypeError && error.message.includes('fetch')) {
|
|
156
|
+
return true;
|
|
157
|
+
}
|
|
158
|
+
return false;
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
// Auth and safety errors are not retryable
|
|
162
|
+
if (error instanceof LLMAuthError || error instanceof LLMSafetyError) {
|
|
163
|
+
return false;
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
// Config errors are not retryable
|
|
167
|
+
if (error instanceof LLMConfigError) {
|
|
168
|
+
return false;
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
// Rate limit and stream errors may be retryable
|
|
172
|
+
return true;
|
|
173
|
+
}
|
|
@@ -0,0 +1,187 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* LLM Configuration Utilities
|
|
3
|
+
* Environment variable resolution and default configuration
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { type LLMConfig, type LLMProviderType, LLMConfigError } from '../types';
|
|
7
|
+
|
|
8
|
+
// ============================================================================
|
|
9
|
+
// Default Configuration
|
|
10
|
+
// ============================================================================
|
|
11
|
+
|
|
12
|
+
export const DEFAULT_PROVIDER: LLMProviderType = 'gemini';
|
|
13
|
+
|
|
14
|
+
export const DEFAULT_MODELS: Record<LLMProviderType, string> = {
|
|
15
|
+
gemini: 'gemini-2.5-flash',
|
|
16
|
+
openai: 'gpt-4o',
|
|
17
|
+
ollama: 'llama3.2',
|
|
18
|
+
custom: 'gpt-3.5-turbo',
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
export const DEFAULT_API_URLS: Record<string, string> = {
|
|
22
|
+
ollama: 'http://localhost:11434/v1',
|
|
23
|
+
openai: 'https://api.openai.com/v1',
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
// ============================================================================
|
|
27
|
+
// Environment Resolution
|
|
28
|
+
// ============================================================================
|
|
29
|
+
|
|
30
|
+
function getEnvVar(key: string): string | undefined {
|
|
31
|
+
return process.env[key];
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
function resolveProvider(): LLMProviderType {
|
|
35
|
+
const provider = getEnvVar('LLM_PROVIDER')?.toLowerCase();
|
|
36
|
+
|
|
37
|
+
if (!provider) {
|
|
38
|
+
return DEFAULT_PROVIDER;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
const validProviders: LLMProviderType[] = ['gemini', 'openai', 'ollama', 'custom'];
|
|
42
|
+
|
|
43
|
+
if (!validProviders.includes(provider as LLMProviderType)) {
|
|
44
|
+
console.error(`[LLM] Invalid provider "${provider}", falling back to "${DEFAULT_PROVIDER}"`);
|
|
45
|
+
return DEFAULT_PROVIDER;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
return provider as LLMProviderType;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function resolveApiKey(provider: LLMProviderType): string | undefined {
|
|
52
|
+
const apiKey = getEnvVar('LLM_API_KEY');
|
|
53
|
+
|
|
54
|
+
// Ollama doesn't require API key
|
|
55
|
+
if (!apiKey && provider === 'ollama') {
|
|
56
|
+
return undefined;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
return apiKey;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
function resolveModel(provider: LLMProviderType): string {
|
|
63
|
+
const model = getEnvVar('LLM_MODEL');
|
|
64
|
+
return model || DEFAULT_MODELS[provider];
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
function resolveApiUrl(provider: LLMProviderType): string | undefined {
|
|
68
|
+
// Primary: LLM_API_URL
|
|
69
|
+
const apiUrl = getEnvVar('LLM_API_URL');
|
|
70
|
+
if (apiUrl) {
|
|
71
|
+
return apiUrl;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// Default URLs for specific providers
|
|
75
|
+
if (provider === 'ollama') {
|
|
76
|
+
return DEFAULT_API_URLS.ollama;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
if (provider === 'openai') {
|
|
80
|
+
return DEFAULT_API_URLS.openai;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// Custom provider requires explicit URL
|
|
84
|
+
if (provider === 'custom') {
|
|
85
|
+
return undefined;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
return undefined;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
// ============================================================================
|
|
92
|
+
// Configuration Resolution
|
|
93
|
+
// ============================================================================
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* Resolve LLM configuration from environment variables with optional overrides
|
|
97
|
+
*/
|
|
98
|
+
export function resolveConfig(overrides?: Partial<LLMConfig>): LLMConfig {
|
|
99
|
+
const provider = overrides?.provider ?? resolveProvider();
|
|
100
|
+
|
|
101
|
+
return {
|
|
102
|
+
provider,
|
|
103
|
+
apiKey: overrides?.apiKey ?? resolveApiKey(provider),
|
|
104
|
+
model: overrides?.model ?? resolveModel(provider),
|
|
105
|
+
apiUrl: overrides?.apiUrl ?? resolveApiUrl(provider),
|
|
106
|
+
};
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// ============================================================================
|
|
110
|
+
// Configuration Validation
|
|
111
|
+
// ============================================================================
|
|
112
|
+
|
|
113
|
+
/**
|
|
114
|
+
* Validate LLM configuration
|
|
115
|
+
* @throws LLMConfigError if configuration is invalid
|
|
116
|
+
*/
|
|
117
|
+
export function validateConfig(config: LLMConfig): void {
|
|
118
|
+
// Validate provider
|
|
119
|
+
const validProviders: LLMProviderType[] = ['gemini', 'openai', 'ollama', 'custom'];
|
|
120
|
+
if (!validProviders.includes(config.provider)) {
|
|
121
|
+
throw new LLMConfigError(
|
|
122
|
+
`Invalid provider: ${config.provider}. Valid options: ${validProviders.join(', ')}`,
|
|
123
|
+
config.provider
|
|
124
|
+
);
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
// Validate API key requirements
|
|
128
|
+
if (config.provider === 'gemini' && !config.apiKey) {
|
|
129
|
+
throw new LLMConfigError(
|
|
130
|
+
'Gemini API key is required. Set LLM_API_KEY environment variable.',
|
|
131
|
+
'gemini'
|
|
132
|
+
);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
if (config.provider === 'openai' && !config.apiKey) {
|
|
136
|
+
throw new LLMConfigError(
|
|
137
|
+
'OpenAI API key is required. Set LLM_API_KEY environment variable.',
|
|
138
|
+
'openai'
|
|
139
|
+
);
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
// Validate API URL for custom provider
|
|
143
|
+
if (config.provider === 'custom' && !config.apiUrl) {
|
|
144
|
+
throw new LLMConfigError(
|
|
145
|
+
'Custom provider requires LLM_API_URL environment variable.',
|
|
146
|
+
'custom'
|
|
147
|
+
);
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
// Validate model
|
|
151
|
+
if (!config.model || config.model.trim() === '') {
|
|
152
|
+
throw new LLMConfigError(
|
|
153
|
+
`Model name is required for ${config.provider} provider.`,
|
|
154
|
+
config.provider
|
|
155
|
+
);
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
// ============================================================================
|
|
160
|
+
// Utility Functions
|
|
161
|
+
// ============================================================================
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* Check if a provider requires an API key
|
|
165
|
+
*/
|
|
166
|
+
export function requiresApiKey(provider: LLMProviderType): boolean {
|
|
167
|
+
return provider === 'gemini' || provider === 'openai';
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
/**
|
|
171
|
+
* Check if a provider requires a custom API URL
|
|
172
|
+
*/
|
|
173
|
+
export function requiresApiUrl(provider: LLMProviderType): boolean {
|
|
174
|
+
return provider === 'custom';
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
/**
|
|
178
|
+
* Get safe config for logging (API key masked)
|
|
179
|
+
*/
|
|
180
|
+
export function getSafeConfigForLogging(config: LLMConfig): Record<string, string | undefined> {
|
|
181
|
+
return {
|
|
182
|
+
provider: config.provider,
|
|
183
|
+
model: config.model,
|
|
184
|
+
apiUrl: config.apiUrl,
|
|
185
|
+
apiKey: config.apiKey ? '***' : undefined,
|
|
186
|
+
};
|
|
187
|
+
}
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Retry Utility with Exponential Backoff
|
|
3
|
+
* Handles transient failures in LLM API calls
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { isRetryableError, LLMError, type LLMProviderType } from '../types';
|
|
7
|
+
|
|
8
|
+
// ============================================================================
|
|
9
|
+
// Types
|
|
10
|
+
// ============================================================================
|
|
11
|
+
|
|
12
|
+
export interface RetryOptions {
|
|
13
|
+
/** Maximum number of retry attempts (default: 3) */
|
|
14
|
+
maxAttempts?: number;
|
|
15
|
+
/** Initial delay in milliseconds (default: 1000) */
|
|
16
|
+
initialDelay?: number;
|
|
17
|
+
/** Backoff multiplier (default: 2) */
|
|
18
|
+
backoffMultiplier?: number;
|
|
19
|
+
/** Maximum delay in milliseconds (default: 10000) */
|
|
20
|
+
maxDelay?: number;
|
|
21
|
+
/** Provider name for logging */
|
|
22
|
+
provider?: LLMProviderType;
|
|
23
|
+
/** Operation name for logging */
|
|
24
|
+
operation?: string;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
// ============================================================================
|
|
28
|
+
// Default Configuration
|
|
29
|
+
// ============================================================================
|
|
30
|
+
|
|
31
|
+
const DEFAULT_MAX_ATTEMPTS = 3;
|
|
32
|
+
const DEFAULT_INITIAL_DELAY = 1000;
|
|
33
|
+
const DEFAULT_BACKOFF_MULTIPLIER = 2;
|
|
34
|
+
const DEFAULT_MAX_DELAY = 10000;
|
|
35
|
+
|
|
36
|
+
// ============================================================================
|
|
37
|
+
// Retry Implementation
|
|
38
|
+
// ============================================================================
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* Execute a function with retry logic and exponential backoff
|
|
42
|
+
* @param fn - Async function to execute
|
|
43
|
+
* @param options - Retry configuration
|
|
44
|
+
* @returns Result of the function
|
|
45
|
+
* @throws Last error if all retries fail
|
|
46
|
+
*/
|
|
47
|
+
export async function withRetry<T>(
|
|
48
|
+
fn: () => Promise<T>,
|
|
49
|
+
options: RetryOptions = {}
|
|
50
|
+
): Promise<T> {
|
|
51
|
+
const {
|
|
52
|
+
maxAttempts = DEFAULT_MAX_ATTEMPTS,
|
|
53
|
+
initialDelay = DEFAULT_INITIAL_DELAY,
|
|
54
|
+
backoffMultiplier = DEFAULT_BACKOFF_MULTIPLIER,
|
|
55
|
+
maxDelay = DEFAULT_MAX_DELAY,
|
|
56
|
+
provider,
|
|
57
|
+
operation = 'LLM request',
|
|
58
|
+
} = options;
|
|
59
|
+
|
|
60
|
+
let lastError: Error | undefined;
|
|
61
|
+
let delay = initialDelay;
|
|
62
|
+
|
|
63
|
+
for (let attempt = 1; attempt <= maxAttempts; attempt++) {
|
|
64
|
+
try {
|
|
65
|
+
return await fn();
|
|
66
|
+
} catch (error) {
|
|
67
|
+
lastError = error instanceof Error ? error : new Error(String(error));
|
|
68
|
+
|
|
69
|
+
// Check if error is retryable
|
|
70
|
+
if (!isRetryableError(error)) {
|
|
71
|
+
throw error;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// Don't retry on last attempt
|
|
75
|
+
if (attempt === maxAttempts) {
|
|
76
|
+
break;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// Log retry attempt
|
|
80
|
+
console.error(
|
|
81
|
+
`[LLM${provider ? `:${provider}` : ''}] ${operation} failed (attempt ${attempt}/${maxAttempts}): ${lastError.message}. Retrying in ${delay}ms...`
|
|
82
|
+
);
|
|
83
|
+
|
|
84
|
+
// Wait before retrying
|
|
85
|
+
await sleep(delay);
|
|
86
|
+
|
|
87
|
+
// Increase delay with exponential backoff
|
|
88
|
+
delay = Math.min(delay * backoffMultiplier, maxDelay);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// All retries exhausted
|
|
93
|
+
console.error(
|
|
94
|
+
`[LLM${provider ? `:${provider}` : ''}] ${operation} failed after ${maxAttempts} attempts: ${lastError?.message}`
|
|
95
|
+
);
|
|
96
|
+
|
|
97
|
+
throw lastError ?? new LLMError('Unknown error during retry', provider);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/**
|
|
101
|
+
* Sleep for a specified duration
|
|
102
|
+
*/
|
|
103
|
+
function sleep(ms: number): Promise<void> {
|
|
104
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// ============================================================================
|
|
108
|
+
// Retry Decorators (for class methods)
|
|
109
|
+
// ============================================================================
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Create a retryable version of an async function
|
|
113
|
+
*/
|
|
114
|
+
export function makeRetryable<T extends unknown[], R>(
|
|
115
|
+
fn: (...args: T) => Promise<R>,
|
|
116
|
+
options: RetryOptions = {}
|
|
117
|
+
): (...args: T) => Promise<R> {
|
|
118
|
+
return (...args: T) => withRetry(() => fn(...args), options);
|
|
119
|
+
}
|
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Streaming Utilities for LLM Providers
|
|
3
|
+
* SSE parsing and stream transformation helpers
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { LLMStreamError, type LLMProviderType } from '../types';
|
|
7
|
+
import { logger } from '@/lib/logger';
|
|
8
|
+
|
|
9
|
+
// ============================================================================
|
|
10
|
+
// Text Encoding/Decoding
|
|
11
|
+
// ============================================================================
|
|
12
|
+
|
|
13
|
+
const textEncoder = new TextEncoder();
|
|
14
|
+
const textDecoder = new TextDecoder();
|
|
15
|
+
|
|
16
|
+
export function encodeText(text: string): Uint8Array {
|
|
17
|
+
return textEncoder.encode(text);
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export function decodeText(bytes: Uint8Array): string {
|
|
21
|
+
return textDecoder.decode(bytes);
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
// ============================================================================
|
|
25
|
+
// SSE (Server-Sent Events) Parser
|
|
26
|
+
// ============================================================================
|
|
27
|
+
|
|
28
|
+
export interface SSEChunk {
|
|
29
|
+
data: string;
|
|
30
|
+
event?: string;
|
|
31
|
+
id?: string;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Parse SSE formatted response for OpenAI-compatible APIs
|
|
36
|
+
* Handles chunked responses and extracts content from delta objects
|
|
37
|
+
*/
|
|
38
|
+
export function createSSEParser(): TransformStream<Uint8Array, Uint8Array> {
|
|
39
|
+
let buffer = '';
|
|
40
|
+
|
|
41
|
+
return new TransformStream<Uint8Array, Uint8Array>({
|
|
42
|
+
transform(chunk, controller) {
|
|
43
|
+
buffer += decodeText(chunk);
|
|
44
|
+
|
|
45
|
+
// Process complete lines
|
|
46
|
+
const lines = buffer.split('\n');
|
|
47
|
+
buffer = lines.pop() ?? ''; // Keep incomplete line in buffer
|
|
48
|
+
|
|
49
|
+
for (const line of lines) {
|
|
50
|
+
const trimmed = line.trim();
|
|
51
|
+
|
|
52
|
+
// Skip empty lines and comments
|
|
53
|
+
if (!trimmed || trimmed.startsWith(':')) {
|
|
54
|
+
continue;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
// Handle data lines
|
|
58
|
+
if (trimmed.startsWith('data: ')) {
|
|
59
|
+
const data = trimmed.slice(6);
|
|
60
|
+
|
|
61
|
+
// Handle stream end marker
|
|
62
|
+
if (data === '[DONE]') {
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
try {
|
|
67
|
+
const parsed = JSON.parse(data);
|
|
68
|
+
const content = extractContent(parsed);
|
|
69
|
+
|
|
70
|
+
if (content) {
|
|
71
|
+
controller.enqueue(encodeText(content));
|
|
72
|
+
}
|
|
73
|
+
} catch {
|
|
74
|
+
logger.debug('Skipping malformed JSON chunk in SSE stream');
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
},
|
|
79
|
+
|
|
80
|
+
flush(controller) {
|
|
81
|
+
// Process any remaining data in buffer
|
|
82
|
+
if (buffer.trim()) {
|
|
83
|
+
const trimmed = buffer.trim();
|
|
84
|
+
if (trimmed.startsWith('data: ') && trimmed.slice(6) !== '[DONE]') {
|
|
85
|
+
try {
|
|
86
|
+
const parsed = JSON.parse(trimmed.slice(6));
|
|
87
|
+
const content = extractContent(parsed);
|
|
88
|
+
if (content) {
|
|
89
|
+
controller.enqueue(encodeText(content));
|
|
90
|
+
}
|
|
91
|
+
} catch {
|
|
92
|
+
logger.debug('Skipping malformed JSON chunk in SSE stream flush');
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
},
|
|
97
|
+
});
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/**
|
|
101
|
+
* Extract content from parsed SSE data based on provider format
|
|
102
|
+
*/
|
|
103
|
+
function extractContent(data: unknown): string | null {
|
|
104
|
+
if (!data || typeof data !== 'object') {
|
|
105
|
+
return null;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
const obj = data as Record<string, unknown>;
|
|
109
|
+
|
|
110
|
+
// OpenAI format: choices[0].delta.content
|
|
111
|
+
if (Array.isArray(obj.choices) && obj.choices.length > 0) {
|
|
112
|
+
const choice = obj.choices[0] as Record<string, unknown>;
|
|
113
|
+
const delta = choice.delta as Record<string, unknown> | undefined;
|
|
114
|
+
|
|
115
|
+
if (delta && typeof delta.content === 'string') {
|
|
116
|
+
return delta.content;
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
return null;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// ============================================================================
|
|
124
|
+
// Stream Utilities
|
|
125
|
+
// ============================================================================
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* Create a ReadableStream from an async iterable
|
|
129
|
+
*/
|
|
130
|
+
export function streamFromAsyncIterable<T>(
|
|
131
|
+
iterable: AsyncIterable<T>,
|
|
132
|
+
transform: (item: T) => Uint8Array | null
|
|
133
|
+
): ReadableStream<Uint8Array> {
|
|
134
|
+
return new ReadableStream<Uint8Array>({
|
|
135
|
+
async start(controller) {
|
|
136
|
+
try {
|
|
137
|
+
for await (const item of iterable) {
|
|
138
|
+
const chunk = transform(item);
|
|
139
|
+
if (chunk) {
|
|
140
|
+
controller.enqueue(chunk);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
controller.close();
|
|
144
|
+
} catch (error) {
|
|
145
|
+
controller.error(error);
|
|
146
|
+
}
|
|
147
|
+
},
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* Create a ReadableStream from a fetch Response with SSE parsing
|
|
153
|
+
*/
|
|
154
|
+
export function createStreamFromSSEResponse(
|
|
155
|
+
response: Response,
|
|
156
|
+
provider: LLMProviderType
|
|
157
|
+
): ReadableStream<Uint8Array> {
|
|
158
|
+
const body = response.body;
|
|
159
|
+
|
|
160
|
+
if (!body) {
|
|
161
|
+
throw new LLMStreamError('Response body is empty', provider);
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
return body.pipeThrough(createSSEParser());
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
/**
|
|
168
|
+
* Merge multiple streams into one (useful for multi-part responses)
|
|
169
|
+
*/
|
|
170
|
+
export function mergeStreams(
|
|
171
|
+
streams: ReadableStream<Uint8Array>[]
|
|
172
|
+
): ReadableStream<Uint8Array> {
|
|
173
|
+
return new ReadableStream<Uint8Array>({
|
|
174
|
+
async start(controller) {
|
|
175
|
+
for (const stream of streams) {
|
|
176
|
+
const reader = stream.getReader();
|
|
177
|
+
try {
|
|
178
|
+
while (true) {
|
|
179
|
+
const { done, value } = await reader.read();
|
|
180
|
+
if (done) break;
|
|
181
|
+
controller.enqueue(value);
|
|
182
|
+
}
|
|
183
|
+
} finally {
|
|
184
|
+
reader.releaseLock();
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
controller.close();
|
|
188
|
+
},
|
|
189
|
+
});
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
/**
|
|
193
|
+
* Create an error stream that emits a single error message
|
|
194
|
+
*/
|
|
195
|
+
export function createErrorStream(message: string): ReadableStream<Uint8Array> {
|
|
196
|
+
return new ReadableStream<Uint8Array>({
|
|
197
|
+
start(controller) {
|
|
198
|
+
controller.enqueue(encodeText(`Error: ${message}`));
|
|
199
|
+
controller.close();
|
|
200
|
+
},
|
|
201
|
+
});
|
|
202
|
+
}
|