@libredb/studio 0.9.7 → 0.9.12
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/dist/chunk-34YQUUCM.mjs +319 -0
- package/dist/chunk-34YQUUCM.mjs.map +1 -0
- package/dist/chunk-4LVB3K53.mjs +37 -0
- package/dist/chunk-4LVB3K53.mjs.map +1 -0
- package/dist/chunk-6DRZXXNT.mjs +100 -0
- package/dist/chunk-6DRZXXNT.mjs.map +1 -0
- package/dist/chunk-CZVV3JJB.mjs +160 -0
- package/dist/chunk-CZVV3JJB.mjs.map +1 -0
- package/dist/chunk-D4WVWWWF.js +332 -0
- package/dist/chunk-D4WVWWWF.js.map +1 -0
- package/dist/chunk-DY3KXE44.mjs +3 -0
- package/dist/chunk-DY3KXE44.mjs.map +1 -0
- package/dist/chunk-DZ2UB3C6.mjs +6679 -0
- package/dist/chunk-DZ2UB3C6.mjs.map +1 -0
- package/dist/chunk-FYSE52VB.js +242 -0
- package/dist/chunk-FYSE52VB.js.map +1 -0
- package/dist/chunk-G4WYE6TI.js +4 -0
- package/dist/chunk-G4WYE6TI.js.map +1 -0
- package/dist/chunk-JOGLIOFO.js +1310 -0
- package/dist/chunk-JOGLIOFO.js.map +1 -0
- package/dist/chunk-JZO5KRZN.js +165 -0
- package/dist/chunk-JZO5KRZN.js.map +1 -0
- package/dist/chunk-KV356UXJ.js +253 -0
- package/dist/chunk-KV356UXJ.js.map +1 -0
- package/dist/chunk-PPODO6HX.mjs +237 -0
- package/dist/chunk-PPODO6HX.mjs.map +1 -0
- package/dist/chunk-PTIRB2JO.js +258 -0
- package/dist/chunk-PTIRB2JO.js.map +1 -0
- package/dist/chunk-Q6LRDBK7.js +42 -0
- package/dist/chunk-Q6LRDBK7.js.map +1 -0
- package/dist/chunk-QJP5FZRY.mjs +255 -0
- package/dist/chunk-QJP5FZRY.mjs.map +1 -0
- package/dist/chunk-R3POCJK6.mjs +248 -0
- package/dist/chunk-R3POCJK6.mjs.map +1 -0
- package/dist/chunk-RBVDMLFV.js +6747 -0
- package/dist/chunk-RBVDMLFV.js.map +1 -0
- package/dist/chunk-RCQB4FCE.js +186 -0
- package/dist/chunk-RCQB4FCE.js.map +1 -0
- package/dist/chunk-SR5DRGBX.mjs +174 -0
- package/dist/chunk-SR5DRGBX.mjs.map +1 -0
- package/dist/chunk-VLCRUZX7.js +102 -0
- package/dist/chunk-VLCRUZX7.js.map +1 -0
- package/dist/chunk-VWVRUCQO.mjs +1289 -0
- package/dist/chunk-VWVRUCQO.mjs.map +1 -0
- package/dist/components.d.mts +273 -0
- package/dist/components.d.ts +273 -0
- package/dist/components.js +59 -0
- package/dist/components.js.map +1 -0
- package/dist/components.mjs +6 -0
- package/dist/components.mjs.map +1 -0
- package/dist/custom-BNDOYC5P.js +134 -0
- package/dist/custom-BNDOYC5P.js.map +1 -0
- package/dist/custom-S2EKFMP3.mjs +132 -0
- package/dist/custom-S2EKFMP3.mjs.map +1 -0
- package/dist/gemini-4ASHNK4H.js +81 -0
- package/dist/gemini-4ASHNK4H.js.map +1 -0
- package/dist/gemini-C5RBLQEJ.mjs +79 -0
- package/dist/gemini-C5RBLQEJ.mjs.map +1 -0
- package/dist/index.d.mts +6 -0
- package/dist/index.d.ts +6 -0
- package/dist/index.js +95 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +10 -0
- package/dist/index.mjs.map +1 -0
- package/dist/mongodb-XMZEZA4A.mjs +748 -0
- package/dist/mongodb-XMZEZA4A.mjs.map +1 -0
- package/dist/mongodb-YQJJTLX3.js +750 -0
- package/dist/mongodb-YQJJTLX3.js.map +1 -0
- package/dist/mssql-PMOU4D36.js +916 -0
- package/dist/mssql-PMOU4D36.js.map +1 -0
- package/{src/lib/db/providers/sql/mssql.ts → dist/mssql-ZH5VP2C5.mjs} +268 -423
- package/dist/mssql-ZH5VP2C5.mjs.map +1 -0
- package/{src/lib/db/providers/sql/mysql.ts → dist/mysql-I3WJQXN2.mjs} +277 -428
- package/dist/mysql-I3WJQXN2.mjs.map +1 -0
- package/dist/mysql-Y3MSA5QY.js +833 -0
- package/dist/mysql-Y3MSA5QY.js.map +1 -0
- package/dist/ollama-26BYLVEV.mjs +115 -0
- package/dist/ollama-26BYLVEV.mjs.map +1 -0
- package/dist/ollama-HVWAGKQC.js +117 -0
- package/dist/ollama-HVWAGKQC.js.map +1 -0
- package/dist/openai-4U56KPG7.mjs +111 -0
- package/dist/openai-4U56KPG7.mjs.map +1 -0
- package/dist/openai-AK3R37BS.js +113 -0
- package/dist/openai-AK3R37BS.js.map +1 -0
- package/dist/oracle-L6VEAVXO.js +917 -0
- package/dist/oracle-L6VEAVXO.js.map +1 -0
- package/{src/lib/db/providers/sql/oracle.ts → dist/oracle-P2G7T4P4.mjs} +321 -454
- package/dist/oracle-P2G7T4P4.mjs.map +1 -0
- package/{src/lib/db/providers/sql/postgres.ts → dist/postgres-O5KOQUVP.mjs} +261 -471
- package/dist/postgres-O5KOQUVP.mjs.map +1 -0
- package/dist/postgres-RLCWNFFX.js +971 -0
- package/dist/postgres-RLCWNFFX.js.map +1 -0
- package/dist/providers.d.mts +149 -0
- package/dist/providers.d.ts +149 -0
- package/dist/providers.js +44 -0
- package/dist/providers.js.map +1 -0
- package/dist/providers.mjs +7 -0
- package/dist/providers.mjs.map +1 -0
- package/dist/redis-4WMQOVLX.mjs +435 -0
- package/dist/redis-4WMQOVLX.mjs.map +1 -0
- package/dist/redis-QVQ6YU62.js +441 -0
- package/dist/redis-QVQ6YU62.js.map +1 -0
- package/dist/sqlite-4I2P2OGQ.js +554 -0
- package/dist/sqlite-4I2P2OGQ.js.map +1 -0
- package/dist/sqlite-OA4YJX5S.mjs +531 -0
- package/dist/sqlite-OA4YJX5S.mjs.map +1 -0
- package/dist/types-BJvJfxSY.d.mts +141 -0
- package/dist/types-BJvJfxSY.d.ts +141 -0
- package/dist/types-ClAg_v5k.d.mts +343 -0
- package/dist/types-Der_X8E8.d.ts +343 -0
- package/dist/types.d.mts +2 -0
- package/dist/types.d.ts +2 -0
- package/dist/types.js +6 -0
- package/dist/types.js.map +1 -0
- package/dist/types.mjs +3 -0
- package/dist/types.mjs.map +1 -0
- package/dist/workspace.d.mts +80 -0
- package/dist/workspace.d.ts +80 -0
- package/dist/workspace.js +4182 -0
- package/dist/workspace.js.map +1 -0
- package/dist/workspace.mjs +4155 -0
- package/dist/workspace.mjs.map +1 -0
- package/package.json +60 -5
- package/.claude/settings.local.json +0 -127
- package/.cursorrules +0 -426
- package/.devin/wiki.json +0 -143
- package/.dockerignore +0 -80
- package/.env.example +0 -159
- package/.github/ISSUE_TEMPLATE/bug_report.md +0 -49
- package/.github/ISSUE_TEMPLATE/feature_request.md +0 -29
- package/.github/PULL_REQUEST_TEMPLATE.md +0 -57
- package/.github/workflows/ci.yml +0 -185
- package/.github/workflows/codeql.yml +0 -57
- package/.github/workflows/docker-build-push.yml +0 -118
- package/.github/workflows/helm-release.yml +0 -113
- package/CLAUDE.md +0 -265
- package/CODE_OF_CONDUCT.md +0 -124
- package/CONTRIBUTING.md +0 -154
- package/Dockerfile +0 -73
- package/SECURITY.md +0 -107
- package/artifacthub-repo.yml +0 -4
- package/bun.lock +0 -1714
- package/bunfig.toml +0 -3
- package/charts/libredb-studio/.helmignore +0 -11
- package/charts/libredb-studio/Chart.lock +0 -6
- package/charts/libredb-studio/Chart.yaml +0 -50
- package/charts/libredb-studio/README.md +0 -206
- package/charts/libredb-studio/templates/NOTES.txt +0 -59
- package/charts/libredb-studio/templates/_helpers.tpl +0 -135
- package/charts/libredb-studio/templates/configmap.yaml +0 -37
- package/charts/libredb-studio/templates/deployment.yaml +0 -184
- package/charts/libredb-studio/templates/hpa.yaml +0 -32
- package/charts/libredb-studio/templates/ingress.yaml +0 -41
- package/charts/libredb-studio/templates/networkpolicy.yaml +0 -50
- package/charts/libredb-studio/templates/pdb.yaml +0 -18
- package/charts/libredb-studio/templates/pvc.yaml +0 -23
- package/charts/libredb-studio/templates/secret.yaml +0 -30
- package/charts/libredb-studio/templates/seed-configmap.yaml +0 -11
- package/charts/libredb-studio/templates/service.yaml +0 -22
- package/charts/libredb-studio/templates/serviceaccount.yaml +0 -13
- package/charts/libredb-studio/values.schema.json +0 -246
- package/charts/libredb-studio/values.yaml +0 -286
- package/components.json +0 -22
- package/conductor/code_styleguides/typescript.md +0 -43
- package/conductor/product-guidelines.md +0 -43
- package/conductor/product.md +0 -3
- package/conductor/setup_state.json +0 -1
- package/conductor/tech-stack.md +0 -39
- package/conductor/tracks/enhance_postgres_monitoring_20251227/metadata.json +0 -8
- package/conductor/tracks/enhance_postgres_monitoring_20251227/plan.md +0 -44
- package/conductor/tracks/enhance_postgres_monitoring_20251227/spec.md +0 -31
- package/conductor/tracks.md +0 -8
- package/conductor/workflow.md +0 -333
- package/database-compose.yml +0 -55
- package/docker/postgres-init/01-extensions.sql +0 -10
- package/docker/postgres-init/02-sample-data.sql +0 -585
- package/docker/postgres.yml +0 -68
- package/docker-compose.yml +0 -38
- package/docs/AI_PLAN.md +0 -74
- package/docs/API_DOCS.md +0 -875
- package/docs/ARCHITECTURE.md +0 -218
- package/docs/DATABASE_PROVIDERS.md +0 -358
- package/docs/FEATURES.md +0 -116
- package/docs/HELM_CHART.md +0 -252
- package/docs/LOGIN_PAGE.md +0 -178
- package/docs/MONACO_EDITOR_PERFORMANCE.md +0 -315
- package/docs/OIDC_ARCH.md +0 -681
- package/docs/OIDC_SETUP.md +0 -322
- package/docs/POSTGRES_METRICS.md +0 -516
- package/docs/QUERY_OPTIMIZATION.md +0 -370
- package/docs/SEED_CONNECTIONS.md +0 -468
- package/docs/SQL_ALIAS_COMPLETION.md +0 -190
- package/docs/STORAGE_ARCHITECTURE.md +0 -565
- package/docs/STORAGE_QUICK_SETUP.md +0 -419
- package/docs/TECHNICAL_PLAN.md +0 -36
- package/docs/THEMING.md +0 -345
- package/docs/adding-a-new-database-provider.md +0 -642
- package/docs/backlogs/000-PLATFORM_DATA_SYNC_DATABASE.md +0 -360
- package/docs/backlogs/001-INLINE_DATA_EDITING.md +0 -118
- package/docs/backlogs/002-DATA_IMPORT.md +0 -215
- package/docs/backlogs/003-QUERY_TIME_MACHINE.md +0 -183
- package/docs/backlogs/004-AI_DATA_STORYTELLER.md +0 -292
- package/docs/backlogs/005-QUERY_PLAYGROUND.md +0 -352
- package/docs/backlogs/006-DATA_MASKING.md +0 -418
- package/docs/enterprise-features.md +0 -718
- package/docs/kubernetes-helm-chart-artifacthub-plan.md +0 -803
- package/docs/medium-koyeb-article-en.md +0 -215
- package/docs/plans/test-plans.md +0 -445
- package/docs/releases/RELEASE.V0.3.0.md +0 -22
- package/docs/releases/RELEASE.V0.4.0.md +0 -154
- package/docs/releases/RELEASE.V0.5.0.md +0 -252
- package/docs/releases/RELEASE_v0.5.6.md +0 -145
- package/docs/releases/RELEASE_v0.6.1.md +0 -303
- package/docs/releases/RELEASE_v0.6.7.md +0 -292
- package/docs/releases/RELEASE_v0.7.0.md +0 -332
- package/docs/releases/RELEASE_v0.8.0.md +0 -521
- package/docs/sampledb/titanic.sql +0 -1379
- package/docs/superpowers/plans/2026-03-25-seed-connections.md +0 -1362
- package/docs/superpowers/specs/2026-03-25-seed-connections-design.md +0 -590
- package/e2e/admin-dashboard.spec.ts +0 -64
- package/e2e/connection-management.spec.ts +0 -58
- package/e2e/export.spec.ts +0 -34
- package/e2e/login.spec.ts +0 -85
- package/e2e/query-execution.spec.ts +0 -35
- package/e2e/tab-management.spec.ts +0 -64
- package/eslint.config.mjs +0 -28
- package/fly.toml +0 -43
- package/next.config.ts +0 -32
- package/playwright.config.ts +0 -34
- package/postcss.config.mjs +0 -7
- package/public/favicon-32x32.png +0 -0
- package/public/favicon.ico +0 -0
- package/public/file.svg +0 -1
- package/public/globe.svg +0 -1
- package/public/logo.svg +0 -32
- package/public/next.svg +0 -1
- 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 +0 -1
- package/public/window.svg +0 -1
- package/render.yaml +0 -58
- package/scripts/merge-lcov.mjs +0 -239
- package/sonar-project.properties +0 -16
- package/src/app/admin/error.tsx +0 -46
- package/src/app/admin/page.tsx +0 -10
- package/src/app/api/admin/audit/route.ts +0 -52
- package/src/app/api/admin/fleet-health/route.ts +0 -81
- package/src/app/api/ai/autopilot/route.ts +0 -105
- package/src/app/api/ai/chat/route.ts +0 -132
- package/src/app/api/ai/describe-schema/route.ts +0 -52
- package/src/app/api/ai/explain/route.ts +0 -86
- package/src/app/api/ai/impact/route.ts +0 -97
- package/src/app/api/ai/index-advisor/route.ts +0 -98
- package/src/app/api/ai/nl2sql/route.ts +0 -87
- package/src/app/api/ai/query-safety/route.ts +0 -87
- package/src/app/api/auth/login/route.ts +0 -62
- package/src/app/api/auth/logout/route.ts +0 -25
- package/src/app/api/auth/me/route.ts +0 -10
- package/src/app/api/auth/oidc/callback/route.ts +0 -82
- package/src/app/api/auth/oidc/login/route.ts +0 -43
- package/src/app/api/connections/managed/route.ts +0 -35
- package/src/app/api/db/cancel/route.ts +0 -42
- package/src/app/api/db/disconnect/route.ts +0 -28
- package/src/app/api/db/health/route.ts +0 -49
- package/src/app/api/db/maintenance/route.ts +0 -72
- package/src/app/api/db/monitoring/route.ts +0 -62
- package/src/app/api/db/multi-query/route.ts +0 -116
- package/src/app/api/db/pool-stats/route.ts +0 -37
- package/src/app/api/db/profile/route.ts +0 -144
- package/src/app/api/db/provider-meta/route.ts +0 -49
- package/src/app/api/db/query/route.ts +0 -50
- package/src/app/api/db/schema/route.ts +0 -47
- package/src/app/api/db/schema-snapshot/route.ts +0 -42
- package/src/app/api/db/test-connection/route.ts +0 -55
- package/src/app/api/db/transaction/route.ts +0 -111
- package/src/app/api/storage/[collection]/route.ts +0 -67
- package/src/app/api/storage/config/route.ts +0 -17
- package/src/app/api/storage/migrate/route.ts +0 -45
- package/src/app/api/storage/route.ts +0 -32
- package/src/app/error.tsx +0 -49
- package/src/app/global-error.tsx +0 -55
- package/src/app/globals.css +0 -146
- package/src/app/icon.svg +0 -42
- package/src/app/layout.tsx +0 -34
- package/src/app/login/login-form.tsx +0 -301
- package/src/app/login/page.tsx +0 -11
- package/src/app/monitoring/page.tsx +0 -8
- package/src/app/not-found.tsx +0 -29
- package/src/app/page.tsx +0 -5
- package/src/components/AIAutopilotPanel.tsx +0 -238
- package/src/components/CodeGenerator.tsx +0 -271
- package/src/components/CommandPalette.tsx +0 -227
- package/src/components/ConnectionModal.tsx +0 -759
- package/src/components/CreateTableModal.tsx +0 -281
- package/src/components/DataCharts.tsx +0 -962
- package/src/components/DataImportModal.tsx +0 -582
- package/src/components/DataProfiler.tsx +0 -335
- package/src/components/DatabaseDocs.tsx +0 -251
- package/src/components/MaskingSettings.tsx +0 -414
- package/src/components/MobileNav.tsx +0 -50
- package/src/components/NL2SQLPanel.tsx +0 -281
- package/src/components/PivotTable.tsx +0 -257
- package/src/components/QueryEditor.tsx +0 -760
- package/src/components/QueryHistory.tsx +0 -344
- package/src/components/QuerySafetyDialog.tsx +0 -290
- package/src/components/ResultsGrid.tsx +0 -644
- package/src/components/SaveQueryModal.tsx +0 -104
- package/src/components/SavedQueries.tsx +0 -128
- package/src/components/SchemaDiagram.tsx +0 -473
- package/src/components/SchemaDiff.tsx +0 -473
- package/src/components/SnapshotTimeline.tsx +0 -116
- package/src/components/Studio.tsx +0 -639
- package/src/components/TestDataGenerator.tsx +0 -261
- package/src/components/VisualExplain.tsx +0 -820
- package/src/components/admin/AdminDashboard.tsx +0 -163
- package/src/components/admin/tabs/AuditTab.tsx +0 -531
- package/src/components/admin/tabs/MonitoringEmbed.tsx +0 -11
- package/src/components/admin/tabs/OperationsTab.tsx +0 -646
- package/src/components/admin/tabs/OverviewTab.tsx +0 -1328
- package/src/components/admin/tabs/SecurityTab.tsx +0 -284
- package/src/components/community-section.tsx +0 -92
- package/src/components/icons/db-icons.tsx +0 -84
- package/src/components/libredb-logo.tsx +0 -61
- package/src/components/monitoring/MonitoringDashboard.tsx +0 -345
- package/src/components/monitoring/tabs/MetricChart.tsx +0 -82
- package/src/components/monitoring/tabs/OverviewTab.tsx +0 -263
- package/src/components/monitoring/tabs/PerformanceTab.tsx +0 -254
- package/src/components/monitoring/tabs/PoolTab.tsx +0 -174
- package/src/components/monitoring/tabs/QueriesTab.tsx +0 -287
- package/src/components/monitoring/tabs/SessionsTab.tsx +0 -316
- package/src/components/monitoring/tabs/StorageTab.tsx +0 -335
- package/src/components/monitoring/tabs/TablesTab.tsx +0 -300
- package/src/components/results-grid/ResultCard.tsx +0 -111
- package/src/components/results-grid/RowDetailSheet.tsx +0 -178
- package/src/components/results-grid/StatsBar.tsx +0 -201
- package/src/components/results-grid/index.ts +0 -1
- package/src/components/results-grid/utils.ts +0 -23
- package/src/components/schema-explorer/ColumnList.tsx +0 -53
- package/src/components/schema-explorer/SchemaExplorer.tsx +0 -182
- package/src/components/schema-explorer/TableItem.tsx +0 -210
- package/src/components/schema-explorer/index.ts +0 -1
- package/src/components/sidebar/ConnectionItem.tsx +0 -105
- package/src/components/sidebar/ConnectionsList.tsx +0 -62
- package/src/components/sidebar/Sidebar.tsx +0 -130
- package/src/components/sidebar/index.ts +0 -2
- package/src/components/studio/BottomPanel.tsx +0 -286
- package/src/components/studio/QueryToolbar.tsx +0 -180
- package/src/components/studio/StudioDesktopHeader.tsx +0 -114
- package/src/components/studio/StudioMobileHeader.tsx +0 -340
- package/src/components/studio/StudioTabBar.tsx +0 -82
- package/src/components/studio/index.ts +0 -5
- package/src/components/ui/accordion.tsx +0 -66
- package/src/components/ui/alert-dialog.tsx +0 -157
- package/src/components/ui/alert.tsx +0 -66
- package/src/components/ui/aspect-ratio.tsx +0 -11
- package/src/components/ui/avatar.tsx +0 -53
- package/src/components/ui/badge.tsx +0 -46
- package/src/components/ui/breadcrumb.tsx +0 -109
- package/src/components/ui/button-group.tsx +0 -83
- package/src/components/ui/button.tsx +0 -60
- package/src/components/ui/calendar.tsx +0 -216
- package/src/components/ui/card.tsx +0 -92
- package/src/components/ui/carousel.tsx +0 -241
- package/src/components/ui/chart.tsx +0 -357
- package/src/components/ui/checkbox.tsx +0 -32
- package/src/components/ui/collapsible.tsx +0 -33
- package/src/components/ui/command.tsx +0 -184
- package/src/components/ui/context-menu.tsx +0 -252
- package/src/components/ui/dialog.tsx +0 -143
- package/src/components/ui/drawer.tsx +0 -135
- package/src/components/ui/dropdown-menu.tsx +0 -257
- package/src/components/ui/empty.tsx +0 -104
- package/src/components/ui/field.tsx +0 -248
- package/src/components/ui/form.tsx +0 -167
- package/src/components/ui/hover-card.tsx +0 -44
- package/src/components/ui/input-group.tsx +0 -170
- package/src/components/ui/input-otp.tsx +0 -77
- package/src/components/ui/input.tsx +0 -21
- package/src/components/ui/item.tsx +0 -193
- package/src/components/ui/kbd.tsx +0 -28
- package/src/components/ui/label.tsx +0 -24
- package/src/components/ui/menubar.tsx +0 -276
- package/src/components/ui/navigation-menu.tsx +0 -168
- package/src/components/ui/pagination.tsx +0 -127
- package/src/components/ui/popover.tsx +0 -48
- package/src/components/ui/progress.tsx +0 -31
- package/src/components/ui/radio-group.tsx +0 -45
- package/src/components/ui/resizable.tsx +0 -56
- package/src/components/ui/scroll-area.tsx +0 -58
- package/src/components/ui/select.tsx +0 -187
- package/src/components/ui/separator.tsx +0 -28
- package/src/components/ui/sheet.tsx +0 -139
- package/src/components/ui/sidebar.tsx +0 -726
- package/src/components/ui/skeleton.tsx +0 -13
- package/src/components/ui/slider.tsx +0 -63
- package/src/components/ui/sonner.tsx +0 -40
- package/src/components/ui/spinner.tsx +0 -16
- package/src/components/ui/switch.tsx +0 -31
- package/src/components/ui/table.tsx +0 -116
- package/src/components/ui/tabs.tsx +0 -66
- package/src/components/ui/textarea.tsx +0 -18
- package/src/components/ui/toggle-group.tsx +0 -83
- package/src/components/ui/toggle.tsx +0 -47
- package/src/components/ui/tooltip.tsx +0 -61
- package/src/exports/components.ts +0 -15
- package/src/exports/index.ts +0 -4
- package/src/exports/providers.ts +0 -4
- package/src/exports/types.ts +0 -26
- package/src/hooks/use-ai-chat.ts +0 -182
- package/src/hooks/use-all-connections.ts +0 -66
- package/src/hooks/use-api-call.ts +0 -71
- package/src/hooks/use-auth.ts +0 -51
- package/src/hooks/use-connection-form.ts +0 -349
- package/src/hooks/use-connection-manager.ts +0 -169
- package/src/hooks/use-connection-payload.ts +0 -15
- package/src/hooks/use-inline-editing.ts +0 -109
- package/src/hooks/use-mobile.ts +0 -20
- package/src/hooks/use-monitoring-data.ts +0 -270
- package/src/hooks/use-provider-metadata.ts +0 -62
- package/src/hooks/use-query-execution.ts +0 -478
- package/src/hooks/use-storage-sync.ts +0 -259
- package/src/hooks/use-tab-manager.ts +0 -231
- package/src/hooks/use-toast.ts +0 -20
- package/src/hooks/use-transaction-control.ts +0 -64
- package/src/lib/api/error-codes.ts +0 -30
- package/src/lib/api/errors.ts +0 -236
- package/src/lib/api/with-error-handler.ts +0 -41
- package/src/lib/audit.ts +0 -105
- package/src/lib/auth.ts +0 -87
- package/src/lib/connection-string-parser.ts +0 -172
- package/src/lib/data-masking.ts +0 -385
- package/src/lib/db/base-provider.ts +0 -325
- package/src/lib/db/errors.ts +0 -317
- package/src/lib/db/factory.ts +0 -324
- package/src/lib/db/index.ts +0 -123
- package/src/lib/db/providers/document/index.ts +0 -6
- package/src/lib/db/providers/document/mongodb.ts +0 -992
- package/src/lib/db/providers/keyvalue/redis.ts +0 -554
- package/src/lib/db/providers/sql/index.ts +0 -11
- package/src/lib/db/providers/sql/sql-base.ts +0 -174
- package/src/lib/db/providers/sql/sqlite.ts +0 -721
- package/src/lib/db/types.ts +0 -437
- package/src/lib/db/utils/pool-manager.ts +0 -287
- package/src/lib/db/utils/query-limiter.ts +0 -239
- package/src/lib/db-ui-config.ts +0 -86
- package/src/lib/editor/mongodb-completions.ts +0 -172
- package/src/lib/editor/sql-completions.ts +0 -280
- package/src/lib/llm/base-provider.ts +0 -117
- package/src/lib/llm/factory.ts +0 -102
- package/src/lib/llm/index.ts +0 -90
- package/src/lib/llm/providers/custom.ts +0 -181
- package/src/lib/llm/providers/gemini.ts +0 -126
- package/src/lib/llm/providers/ollama.ts +0 -154
- package/src/lib/llm/providers/openai.ts +0 -146
- package/src/lib/llm/types.ts +0 -173
- package/src/lib/llm/utils/config.ts +0 -187
- package/src/lib/llm/utils/retry.ts +0 -119
- package/src/lib/llm/utils/streaming.ts +0 -202
- package/src/lib/logger.ts +0 -127
- package/src/lib/monitoring-thresholds.ts +0 -44
- package/src/lib/oidc.ts +0 -262
- package/src/lib/query-generators.ts +0 -61
- package/src/lib/schema-diff/diff-engine.ts +0 -273
- package/src/lib/schema-diff/migration-generator.ts +0 -208
- package/src/lib/schema-diff/types.ts +0 -55
- package/src/lib/seed/config-loader.ts +0 -79
- package/src/lib/seed/connection-filter.ts +0 -49
- package/src/lib/seed/credential-resolver.ts +0 -62
- package/src/lib/seed/index.ts +0 -40
- package/src/lib/seed/resolve-connection.ts +0 -57
- package/src/lib/seed/types.ts +0 -69
- package/src/lib/sql/alias-extractor.ts +0 -267
- package/src/lib/sql/index.ts +0 -8
- package/src/lib/sql/statement-splitter.ts +0 -167
- package/src/lib/sql/types.ts +0 -40
- package/src/lib/ssh/tunnel.ts +0 -142
- package/src/lib/storage/factory.ts +0 -84
- package/src/lib/storage/index.ts +0 -14
- package/src/lib/storage/local-storage.ts +0 -99
- package/src/lib/storage/providers/postgres.ts +0 -225
- package/src/lib/storage/providers/sqlite.ts +0 -153
- package/src/lib/storage/storage-facade.ts +0 -272
- package/src/lib/storage/types.ts +0 -75
- package/src/lib/time-series-buffer.ts +0 -58
- package/src/lib/types.ts +0 -173
- package/src/lib/utils.ts +0 -6
- package/src/proxy.ts +0 -104
- package/src/types/db-drivers.d.ts +0 -23
- package/src/types/html2canvas.d.ts +0 -9
- package/tests/api/admin/audit.test.ts +0 -178
- package/tests/api/admin/fleet-health.test.ts +0 -183
- package/tests/api/ai/autopilot.test.ts +0 -174
- package/tests/api/ai/chat.test.ts +0 -250
- package/tests/api/ai/describe-schema.test.ts +0 -266
- package/tests/api/ai/explain.test.ts +0 -199
- package/tests/api/ai/impact.test.ts +0 -168
- package/tests/api/ai/index-advisor.test.ts +0 -171
- package/tests/api/ai/nl2sql.test.ts +0 -202
- package/tests/api/ai/query-safety.test.ts +0 -196
- package/tests/api/auth/login.test.ts +0 -170
- package/tests/api/auth/logout.test.ts +0 -140
- package/tests/api/auth/me.test.ts +0 -73
- package/tests/api/auth/oidc-callback.test.ts +0 -215
- package/tests/api/auth/oidc-login.test.ts +0 -127
- package/tests/api/db/cancel.test.ts +0 -198
- package/tests/api/db/disconnect.test.ts +0 -124
- package/tests/api/db/health.test.ts +0 -222
- package/tests/api/db/maintenance.test.ts +0 -263
- package/tests/api/db/monitoring.test.ts +0 -221
- package/tests/api/db/multi-query.test.ts +0 -316
- package/tests/api/db/pool-stats.test.ts +0 -135
- package/tests/api/db/profile.test.ts +0 -330
- package/tests/api/db/provider-meta.test.ts +0 -193
- package/tests/api/db/query.test.ts +0 -314
- package/tests/api/db/schema-snapshot.test.ts +0 -170
- package/tests/api/db/schema.test.ts +0 -191
- package/tests/api/db/test-connection.test.ts +0 -185
- package/tests/api/db/transaction.test.ts +0 -314
- package/tests/api/proxy.test.ts +0 -191
- package/tests/api/seed/managed-route.test.ts +0 -113
- package/tests/api/storage/config.test.ts +0 -42
- package/tests/api/storage/storage-routes.test.ts +0 -309
- package/tests/components/AIAutopilotPanel.test.tsx +0 -756
- package/tests/components/AdminPage.test.tsx +0 -33
- package/tests/components/CodeGenerator.test.tsx +0 -182
- package/tests/components/CommandPalette.test.tsx +0 -428
- package/tests/components/CommunitySection.test.tsx +0 -91
- package/tests/components/ConnectionModal.mobile.test.tsx +0 -284
- package/tests/components/ConnectionModal.test.tsx +0 -570
- package/tests/components/CreateTableModal.test.tsx +0 -383
- package/tests/components/DataCharts.test.tsx +0 -739
- package/tests/components/DataImportModal.test.tsx +0 -751
- package/tests/components/DataProfiler.test.tsx +0 -589
- package/tests/components/DatabaseDocs.test.tsx +0 -353
- package/tests/components/LoginPage.test.tsx +0 -163
- package/tests/components/LoginPageOIDC.test.tsx +0 -92
- package/tests/components/MaskingSettings.test.tsx +0 -498
- package/tests/components/MobileNav.test.tsx +0 -30
- package/tests/components/MonitoringPage.test.tsx +0 -32
- package/tests/components/NL2SQLPanel.test.tsx +0 -621
- package/tests/components/Page.test.tsx +0 -33
- package/tests/components/PivotTable.test.tsx +0 -350
- package/tests/components/QueryEditor.test.tsx +0 -1730
- package/tests/components/QueryHistory.test.tsx +0 -572
- package/tests/components/QuerySafetyDialog.test.tsx +0 -586
- package/tests/components/ResultsGrid.test.tsx +0 -804
- package/tests/components/RootLayout.test.tsx +0 -83
- package/tests/components/SaveQueryModal.test.tsx +0 -25
- package/tests/components/SavedQueries.test.tsx +0 -43
- package/tests/components/SchemaDiagram.test.tsx +0 -1034
- package/tests/components/SchemaDiff.test.tsx +0 -906
- package/tests/components/SnapshotTimeline.test.tsx +0 -174
- package/tests/components/Studio.test.tsx +0 -1030
- package/tests/components/TestDataGenerator.test.tsx +0 -291
- package/tests/components/VisualExplain.test.tsx +0 -704
- package/tests/components/admin/AdminDashboard.test.tsx +0 -205
- package/tests/components/admin/AuditTab.test.tsx +0 -220
- package/tests/components/admin/MonitoringEmbed.test.tsx +0 -58
- package/tests/components/admin/OperationsTab.test.tsx +0 -975
- package/tests/components/admin/OverviewTab.test.tsx +0 -254
- package/tests/components/admin/SecurityTab.test.tsx +0 -467
- package/tests/components/monitoring/MetricChart.test.tsx +0 -111
- package/tests/components/monitoring/MonitoringDashboard.test.tsx +0 -259
- package/tests/components/monitoring/OverviewTab.test.tsx +0 -78
- package/tests/components/monitoring/PerformanceTab.test.tsx +0 -87
- package/tests/components/monitoring/PoolTab.test.tsx +0 -42
- package/tests/components/monitoring/QueriesTab.test.tsx +0 -80
- package/tests/components/monitoring/SessionsTab.test.tsx +0 -154
- package/tests/components/monitoring/StorageTab.test.tsx +0 -127
- package/tests/components/monitoring/TablesTab.test.tsx +0 -153
- package/tests/components/results-grid/ResultCard.test.tsx +0 -105
- package/tests/components/results-grid/RowDetailSheet.test.tsx +0 -308
- package/tests/components/results-grid/StatsBar.test.tsx +0 -162
- package/tests/components/schema-explorer/ColumnList.test.tsx +0 -151
- package/tests/components/schema-explorer/SchemaExplorer.test.tsx +0 -461
- package/tests/components/schema-explorer/TableItem.test.tsx +0 -415
- package/tests/components/sidebar/ConnectionItem.test.tsx +0 -201
- package/tests/components/sidebar/ConnectionsList.test.tsx +0 -176
- package/tests/components/sidebar/Sidebar.test.tsx +0 -187
- package/tests/components/studio/BottomPanel.test.tsx +0 -383
- package/tests/components/studio/QueryToolbar.test.tsx +0 -321
- package/tests/components/studio/StudioDesktopHeader.test.tsx +0 -377
- package/tests/components/studio/StudioMobileHeader.test.tsx +0 -198
- package/tests/components/studio/StudioTabBar.test.tsx +0 -331
- package/tests/fixtures/connections.ts +0 -96
- package/tests/fixtures/masking-configs.ts +0 -86
- package/tests/fixtures/query-results.ts +0 -71
- package/tests/fixtures/schemas.ts +0 -64
- package/tests/fixtures/seed-connections/invalid-config.yaml +0 -7
- package/tests/fixtures/seed-connections/minimal-config.yaml +0 -8
- package/tests/fixtures/seed-connections/mixed-credentials.yaml +0 -23
- package/tests/fixtures/seed-connections/multi-role-config.yaml +0 -30
- package/tests/fixtures/seed-connections/valid-config.json +0 -15
- package/tests/fixtures/seed-connections/valid-config.yaml +0 -51
- package/tests/helpers/mock-fetch.ts +0 -59
- package/tests/helpers/mock-monaco.ts +0 -112
- package/tests/helpers/mock-navigation.ts +0 -28
- package/tests/helpers/mock-next.ts +0 -80
- package/tests/helpers/mock-provider.ts +0 -133
- package/tests/helpers/mock-sonner.ts +0 -29
- package/tests/helpers/render-with-providers.tsx +0 -19
- package/tests/hooks/use-ai-chat.test.ts +0 -600
- package/tests/hooks/use-auth.test.ts +0 -371
- package/tests/hooks/use-connection-form.test.ts +0 -743
- package/tests/hooks/use-connection-manager.test.ts +0 -466
- package/tests/hooks/use-inline-editing.test.ts +0 -321
- package/tests/hooks/use-mobile.test.ts +0 -177
- package/tests/hooks/use-monitoring-data.test.ts +0 -819
- package/tests/hooks/use-provider-metadata.test.ts +0 -228
- package/tests/hooks/use-query-execution.test.ts +0 -1212
- package/tests/hooks/use-tab-manager.test.ts +0 -756
- package/tests/hooks/use-toast.test.ts +0 -74
- package/tests/hooks/use-transaction-control.test.ts +0 -211
- package/tests/integration/db/mongodb-provider.test.ts +0 -698
- package/tests/integration/db/mssql-provider.test.ts +0 -840
- package/tests/integration/db/mysql-provider.test.ts +0 -872
- package/tests/integration/db/oracle-provider.test.ts +0 -843
- package/tests/integration/db/postgres-provider.test.ts +0 -1382
- package/tests/integration/db/redis-provider.test.ts +0 -526
- package/tests/integration/db/sqlite-provider.test.ts +0 -480
- package/tests/integration/seed/seed-pipeline.test.ts +0 -102
- package/tests/isolated/factory-singleton.test.ts +0 -150
- package/tests/isolated/use-storage-sync.test.ts +0 -389
- package/tests/run-components.sh +0 -196
- package/tests/setup-dom.ts +0 -58
- package/tests/setup.ts +0 -40
- package/tests/unit/api-errors.test.ts +0 -210
- package/tests/unit/code-generator-functions.test.ts +0 -271
- package/tests/unit/components/column-list.test.tsx +0 -190
- package/tests/unit/components/data-import-modal.test.tsx +0 -441
- package/tests/unit/components/studio-mobile-header.test.tsx +0 -327
- package/tests/unit/data-charts-functions.test.ts +0 -496
- package/tests/unit/data-import-functions.test.ts +0 -320
- package/tests/unit/data-import-utils.test.ts +0 -125
- package/tests/unit/db/base-provider.test.ts +0 -517
- package/tests/unit/db/errors.test.ts +0 -403
- package/tests/unit/db/factory.test.ts +0 -436
- package/tests/unit/db/pool-manager.test.ts +0 -440
- package/tests/unit/db/query-limiter.test.ts +0 -387
- package/tests/unit/db/sql-base.test.ts +0 -438
- package/tests/unit/lib/api/error-codes.test.ts +0 -39
- package/tests/unit/lib/audit.test.ts +0 -326
- package/tests/unit/lib/auth.test.ts +0 -146
- package/tests/unit/lib/connection-string-parser.test.ts +0 -424
- package/tests/unit/lib/data-masking.test.ts +0 -583
- package/tests/unit/lib/db-icons.test.tsx +0 -41
- package/tests/unit/lib/monitoring-thresholds.test.ts +0 -133
- package/tests/unit/lib/oidc.test.ts +0 -509
- package/tests/unit/lib/query-generators.test.ts +0 -127
- package/tests/unit/lib/storage/factory.test.ts +0 -71
- package/tests/unit/lib/storage/local-storage.test.ts +0 -114
- package/tests/unit/lib/storage/providers/postgres.test.ts +0 -312
- package/tests/unit/lib/storage/providers/sqlite.test.ts +0 -232
- package/tests/unit/lib/storage/storage-facade-extended.test.ts +0 -331
- package/tests/unit/lib/storage/storage-facade.test.ts +0 -184
- package/tests/unit/lib/storage.test.ts +0 -317
- package/tests/unit/lib/time-series-buffer.test.ts +0 -212
- package/tests/unit/lib/utils.test.ts +0 -24
- package/tests/unit/llm/base-provider.test.ts +0 -238
- package/tests/unit/llm/config.test.ts +0 -262
- package/tests/unit/llm/custom-provider.test.ts +0 -281
- package/tests/unit/llm/gemini-provider.test.ts +0 -248
- package/tests/unit/llm/llm-factory.test.ts +0 -155
- package/tests/unit/llm/ollama-provider.test.ts +0 -288
- package/tests/unit/llm/openai-provider.test.ts +0 -324
- package/tests/unit/llm/retry.test.ts +0 -180
- package/tests/unit/llm/streaming.test.ts +0 -355
- package/tests/unit/logger.test.ts +0 -198
- package/tests/unit/mongodb-completions.test.ts +0 -516
- package/tests/unit/pivot-table-functions.test.ts +0 -76
- package/tests/unit/query-cancelled-error.test.ts +0 -81
- package/tests/unit/schema-diff/diff-engine.test.ts +0 -367
- package/tests/unit/schema-diff/migration-generator.test.ts +0 -513
- package/tests/unit/seed/config-loader.test.ts +0 -73
- package/tests/unit/seed/connection-filter.test.ts +0 -91
- package/tests/unit/seed/credential-resolver.test.ts +0 -85
- package/tests/unit/seed/index.test.ts +0 -72
- package/tests/unit/seed/resolve-connection.test.ts +0 -74
- package/tests/unit/seed/types.test.ts +0 -129
- package/tests/unit/sql/alias-extractor.test.ts +0 -444
- package/tests/unit/sql/statement-splitter.test.ts +0 -348
- package/tests/unit/sql-completions.test.ts +0 -463
- package/tests/unit/ssh-tunnel.test.ts +0 -465
- package/tsconfig.json +0 -42
|
@@ -0,0 +1,186 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
// src/lib/llm/types.ts
|
|
4
|
+
var LLMError = class _LLMError extends Error {
|
|
5
|
+
constructor(message, provider, statusCode) {
|
|
6
|
+
super(message);
|
|
7
|
+
this.provider = provider;
|
|
8
|
+
this.statusCode = statusCode;
|
|
9
|
+
this.name = "LLMError";
|
|
10
|
+
Object.setPrototypeOf(this, _LLMError.prototype);
|
|
11
|
+
}
|
|
12
|
+
};
|
|
13
|
+
var LLMConfigError = class _LLMConfigError extends LLMError {
|
|
14
|
+
constructor(message, provider) {
|
|
15
|
+
super(message, provider);
|
|
16
|
+
this.name = "LLMConfigError";
|
|
17
|
+
Object.setPrototypeOf(this, _LLMConfigError.prototype);
|
|
18
|
+
}
|
|
19
|
+
};
|
|
20
|
+
var LLMAuthError = class _LLMAuthError extends LLMError {
|
|
21
|
+
constructor(message, provider) {
|
|
22
|
+
super(message, provider, 401);
|
|
23
|
+
this.name = "LLMAuthError";
|
|
24
|
+
Object.setPrototypeOf(this, _LLMAuthError.prototype);
|
|
25
|
+
}
|
|
26
|
+
};
|
|
27
|
+
var LLMRateLimitError = class _LLMRateLimitError extends LLMError {
|
|
28
|
+
constructor(message, provider) {
|
|
29
|
+
super(message, provider, 429);
|
|
30
|
+
this.name = "LLMRateLimitError";
|
|
31
|
+
Object.setPrototypeOf(this, _LLMRateLimitError.prototype);
|
|
32
|
+
}
|
|
33
|
+
};
|
|
34
|
+
var LLMSafetyError = class _LLMSafetyError extends LLMError {
|
|
35
|
+
constructor(message, provider) {
|
|
36
|
+
super(message, provider, 400);
|
|
37
|
+
this.name = "LLMSafetyError";
|
|
38
|
+
Object.setPrototypeOf(this, _LLMSafetyError.prototype);
|
|
39
|
+
}
|
|
40
|
+
};
|
|
41
|
+
var LLMStreamError = class _LLMStreamError extends LLMError {
|
|
42
|
+
constructor(message, provider) {
|
|
43
|
+
super(message, provider);
|
|
44
|
+
this.name = "LLMStreamError";
|
|
45
|
+
Object.setPrototypeOf(this, _LLMStreamError.prototype);
|
|
46
|
+
}
|
|
47
|
+
};
|
|
48
|
+
function isLLMError(error) {
|
|
49
|
+
return error instanceof LLMError;
|
|
50
|
+
}
|
|
51
|
+
function isRetryableError(error) {
|
|
52
|
+
if (!isLLMError(error)) {
|
|
53
|
+
if (error instanceof TypeError && error.message.includes("fetch")) {
|
|
54
|
+
return true;
|
|
55
|
+
}
|
|
56
|
+
return false;
|
|
57
|
+
}
|
|
58
|
+
if (error instanceof LLMAuthError || error instanceof LLMSafetyError) {
|
|
59
|
+
return false;
|
|
60
|
+
}
|
|
61
|
+
if (error instanceof LLMConfigError) {
|
|
62
|
+
return false;
|
|
63
|
+
}
|
|
64
|
+
return true;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// src/lib/llm/utils/config.ts
|
|
68
|
+
var DEFAULT_PROVIDER = "gemini";
|
|
69
|
+
var DEFAULT_MODELS = {
|
|
70
|
+
gemini: "gemini-2.5-flash",
|
|
71
|
+
openai: "gpt-4o",
|
|
72
|
+
ollama: "llama3.2",
|
|
73
|
+
custom: "gpt-3.5-turbo"
|
|
74
|
+
};
|
|
75
|
+
var DEFAULT_API_URLS = {
|
|
76
|
+
ollama: "http://localhost:11434/v1",
|
|
77
|
+
openai: "https://api.openai.com/v1"
|
|
78
|
+
};
|
|
79
|
+
function getEnvVar(key) {
|
|
80
|
+
return process.env[key];
|
|
81
|
+
}
|
|
82
|
+
function resolveProvider() {
|
|
83
|
+
var _a;
|
|
84
|
+
const provider = (_a = getEnvVar("LLM_PROVIDER")) == null ? void 0 : _a.toLowerCase();
|
|
85
|
+
if (!provider) {
|
|
86
|
+
return DEFAULT_PROVIDER;
|
|
87
|
+
}
|
|
88
|
+
const validProviders = ["gemini", "openai", "ollama", "custom"];
|
|
89
|
+
if (!validProviders.includes(provider)) {
|
|
90
|
+
console.error(`[LLM] Invalid provider "${provider}", falling back to "${DEFAULT_PROVIDER}"`);
|
|
91
|
+
return DEFAULT_PROVIDER;
|
|
92
|
+
}
|
|
93
|
+
return provider;
|
|
94
|
+
}
|
|
95
|
+
function resolveApiKey(provider) {
|
|
96
|
+
const apiKey = getEnvVar("LLM_API_KEY");
|
|
97
|
+
if (!apiKey && provider === "ollama") {
|
|
98
|
+
return void 0;
|
|
99
|
+
}
|
|
100
|
+
return apiKey;
|
|
101
|
+
}
|
|
102
|
+
function resolveModel(provider) {
|
|
103
|
+
const model = getEnvVar("LLM_MODEL");
|
|
104
|
+
return model || DEFAULT_MODELS[provider];
|
|
105
|
+
}
|
|
106
|
+
function resolveApiUrl(provider) {
|
|
107
|
+
const apiUrl = getEnvVar("LLM_API_URL");
|
|
108
|
+
if (apiUrl) {
|
|
109
|
+
return apiUrl;
|
|
110
|
+
}
|
|
111
|
+
if (provider === "ollama") {
|
|
112
|
+
return DEFAULT_API_URLS.ollama;
|
|
113
|
+
}
|
|
114
|
+
if (provider === "openai") {
|
|
115
|
+
return DEFAULT_API_URLS.openai;
|
|
116
|
+
}
|
|
117
|
+
if (provider === "custom") {
|
|
118
|
+
return void 0;
|
|
119
|
+
}
|
|
120
|
+
return void 0;
|
|
121
|
+
}
|
|
122
|
+
function resolveConfig(overrides) {
|
|
123
|
+
var _a, _b, _c, _d;
|
|
124
|
+
const provider = (_a = overrides == null ? void 0 : overrides.provider) != null ? _a : resolveProvider();
|
|
125
|
+
return {
|
|
126
|
+
provider,
|
|
127
|
+
apiKey: (_b = overrides == null ? void 0 : overrides.apiKey) != null ? _b : resolveApiKey(provider),
|
|
128
|
+
model: (_c = overrides == null ? void 0 : overrides.model) != null ? _c : resolveModel(provider),
|
|
129
|
+
apiUrl: (_d = overrides == null ? void 0 : overrides.apiUrl) != null ? _d : resolveApiUrl(provider)
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
function validateConfig(config) {
|
|
133
|
+
const validProviders = ["gemini", "openai", "ollama", "custom"];
|
|
134
|
+
if (!validProviders.includes(config.provider)) {
|
|
135
|
+
throw new LLMConfigError(
|
|
136
|
+
`Invalid provider: ${config.provider}. Valid options: ${validProviders.join(", ")}`,
|
|
137
|
+
config.provider
|
|
138
|
+
);
|
|
139
|
+
}
|
|
140
|
+
if (config.provider === "gemini" && !config.apiKey) {
|
|
141
|
+
throw new LLMConfigError(
|
|
142
|
+
"Gemini API key is required. Set LLM_API_KEY environment variable.",
|
|
143
|
+
"gemini"
|
|
144
|
+
);
|
|
145
|
+
}
|
|
146
|
+
if (config.provider === "openai" && !config.apiKey) {
|
|
147
|
+
throw new LLMConfigError(
|
|
148
|
+
"OpenAI API key is required. Set LLM_API_KEY environment variable.",
|
|
149
|
+
"openai"
|
|
150
|
+
);
|
|
151
|
+
}
|
|
152
|
+
if (config.provider === "custom" && !config.apiUrl) {
|
|
153
|
+
throw new LLMConfigError(
|
|
154
|
+
"Custom provider requires LLM_API_URL environment variable.",
|
|
155
|
+
"custom"
|
|
156
|
+
);
|
|
157
|
+
}
|
|
158
|
+
if (!config.model || config.model.trim() === "") {
|
|
159
|
+
throw new LLMConfigError(
|
|
160
|
+
`Model name is required for ${config.provider} provider.`,
|
|
161
|
+
config.provider
|
|
162
|
+
);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
function getSafeConfigForLogging(config) {
|
|
166
|
+
return {
|
|
167
|
+
provider: config.provider,
|
|
168
|
+
model: config.model,
|
|
169
|
+
apiUrl: config.apiUrl,
|
|
170
|
+
apiKey: config.apiKey ? "***" : void 0
|
|
171
|
+
};
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
exports.DEFAULT_API_URLS = DEFAULT_API_URLS;
|
|
175
|
+
exports.LLMAuthError = LLMAuthError;
|
|
176
|
+
exports.LLMConfigError = LLMConfigError;
|
|
177
|
+
exports.LLMError = LLMError;
|
|
178
|
+
exports.LLMRateLimitError = LLMRateLimitError;
|
|
179
|
+
exports.LLMSafetyError = LLMSafetyError;
|
|
180
|
+
exports.LLMStreamError = LLMStreamError;
|
|
181
|
+
exports.getSafeConfigForLogging = getSafeConfigForLogging;
|
|
182
|
+
exports.isRetryableError = isRetryableError;
|
|
183
|
+
exports.resolveConfig = resolveConfig;
|
|
184
|
+
exports.validateConfig = validateConfig;
|
|
185
|
+
//# sourceMappingURL=chunk-RCQB4FCE.js.map
|
|
186
|
+
//# sourceMappingURL=chunk-RCQB4FCE.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/lib/llm/types.ts","../src/lib/llm/utils/config.ts"],"names":[],"mappings":"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","file":"chunk-RCQB4FCE.js","sourcesContent":["/**\n * LLM Provider Types & Interfaces\n * Strategy Pattern implementation for multi-provider LLM support\n */\n\n// ============================================================================\n// Provider Types\n// ============================================================================\n\nexport type LLMProviderType = 'gemini' | 'openai' | 'ollama' | 'custom';\n\n// ============================================================================\n// Configuration\n// ============================================================================\n\nexport interface LLMConfig {\n provider: LLMProviderType;\n apiKey?: string;\n model: string;\n apiUrl?: string;\n}\n\n// ============================================================================\n// Messages\n// ============================================================================\n\nexport type LLMMessageRole = 'system' | 'user' | 'assistant';\n\nexport interface LLMMessage {\n role: LLMMessageRole;\n content: string;\n}\n\n// ============================================================================\n// Stream Options\n// ============================================================================\n\nexport interface LLMStreamOptions {\n messages: LLMMessage[];\n model?: string;\n temperature?: number;\n maxTokens?: number;\n}\n\n// ============================================================================\n// Provider Interface (Strategy Pattern)\n// ============================================================================\n\nexport interface LLMProvider {\n /** Provider identifier */\n readonly name: LLMProviderType;\n\n /** Current configuration */\n readonly config: LLMConfig;\n\n /**\n * Stream a completion response\n * @param options - Stream configuration including messages\n * @returns ReadableStream of UTF-8 encoded chunks\n */\n stream(options: LLMStreamOptions): Promise<ReadableStream<Uint8Array>>;\n\n /**\n * Validate provider configuration\n * @throws LLMConfigError if configuration is invalid\n */\n validate(): void;\n}\n\n// ============================================================================\n// Error Classes\n// ============================================================================\n\n/**\n * Base error class for LLM-related errors\n */\nexport class LLMError extends Error {\n constructor(\n message: string,\n public readonly provider?: LLMProviderType,\n public readonly statusCode?: number\n ) {\n super(message);\n this.name = 'LLMError';\n Object.setPrototypeOf(this, LLMError.prototype);\n }\n}\n\n/**\n * Configuration error - missing or invalid config\n */\nexport class LLMConfigError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider);\n this.name = 'LLMConfigError';\n Object.setPrototypeOf(this, LLMConfigError.prototype);\n }\n}\n\n/**\n * Authentication error - invalid API key (401/403)\n */\nexport class LLMAuthError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider, 401);\n this.name = 'LLMAuthError';\n Object.setPrototypeOf(this, LLMAuthError.prototype);\n }\n}\n\n/**\n * Rate limit error - quota exceeded (429)\n */\nexport class LLMRateLimitError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider, 429);\n this.name = 'LLMRateLimitError';\n Object.setPrototypeOf(this, LLMRateLimitError.prototype);\n }\n}\n\n/**\n * Safety filter error - content blocked\n */\nexport class LLMSafetyError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider, 400);\n this.name = 'LLMSafetyError';\n Object.setPrototypeOf(this, LLMSafetyError.prototype);\n }\n}\n\n/**\n * Streaming error - connection or parsing failure\n */\nexport class LLMStreamError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider);\n this.name = 'LLMStreamError';\n Object.setPrototypeOf(this, LLMStreamError.prototype);\n }\n}\n\n// ============================================================================\n// Type Guards\n// ============================================================================\n\nexport function isLLMError(error: unknown): error is LLMError {\n return error instanceof LLMError;\n}\n\nexport function isRetryableError(error: unknown): boolean {\n if (!isLLMError(error)) {\n // Network errors are retryable\n if (error instanceof TypeError && error.message.includes('fetch')) {\n return true;\n }\n return false;\n }\n\n // Auth and safety errors are not retryable\n if (error instanceof LLMAuthError || error instanceof LLMSafetyError) {\n return false;\n }\n\n // Config errors are not retryable\n if (error instanceof LLMConfigError) {\n return false;\n }\n\n // Rate limit and stream errors may be retryable\n return true;\n}\n","/**\n * LLM Configuration Utilities\n * Environment variable resolution and default configuration\n */\n\nimport { type LLMConfig, type LLMProviderType, LLMConfigError } from '../types';\n\n// ============================================================================\n// Default Configuration\n// ============================================================================\n\nexport const DEFAULT_PROVIDER: LLMProviderType = 'gemini';\n\nexport const DEFAULT_MODELS: Record<LLMProviderType, string> = {\n gemini: 'gemini-2.5-flash',\n openai: 'gpt-4o',\n ollama: 'llama3.2',\n custom: 'gpt-3.5-turbo',\n};\n\nexport const DEFAULT_API_URLS: Record<string, string> = {\n ollama: 'http://localhost:11434/v1',\n openai: 'https://api.openai.com/v1',\n};\n\n// ============================================================================\n// Environment Resolution\n// ============================================================================\n\nfunction getEnvVar(key: string): string | undefined {\n return process.env[key];\n}\n\nfunction resolveProvider(): LLMProviderType {\n const provider = getEnvVar('LLM_PROVIDER')?.toLowerCase();\n\n if (!provider) {\n return DEFAULT_PROVIDER;\n }\n\n const validProviders: LLMProviderType[] = ['gemini', 'openai', 'ollama', 'custom'];\n\n if (!validProviders.includes(provider as LLMProviderType)) {\n console.error(`[LLM] Invalid provider \"${provider}\", falling back to \"${DEFAULT_PROVIDER}\"`);\n return DEFAULT_PROVIDER;\n }\n\n return provider as LLMProviderType;\n}\n\nfunction resolveApiKey(provider: LLMProviderType): string | undefined {\n const apiKey = getEnvVar('LLM_API_KEY');\n\n // Ollama doesn't require API key\n if (!apiKey && provider === 'ollama') {\n return undefined;\n }\n\n return apiKey;\n}\n\nfunction resolveModel(provider: LLMProviderType): string {\n const model = getEnvVar('LLM_MODEL');\n return model || DEFAULT_MODELS[provider];\n}\n\nfunction resolveApiUrl(provider: LLMProviderType): string | undefined {\n // Primary: LLM_API_URL\n const apiUrl = getEnvVar('LLM_API_URL');\n if (apiUrl) {\n return apiUrl;\n }\n\n // Default URLs for specific providers\n if (provider === 'ollama') {\n return DEFAULT_API_URLS.ollama;\n }\n\n if (provider === 'openai') {\n return DEFAULT_API_URLS.openai;\n }\n\n // Custom provider requires explicit URL\n if (provider === 'custom') {\n return undefined;\n }\n\n return undefined;\n}\n\n// ============================================================================\n// Configuration Resolution\n// ============================================================================\n\n/**\n * Resolve LLM configuration from environment variables with optional overrides\n */\nexport function resolveConfig(overrides?: Partial<LLMConfig>): LLMConfig {\n const provider = overrides?.provider ?? resolveProvider();\n\n return {\n provider,\n apiKey: overrides?.apiKey ?? resolveApiKey(provider),\n model: overrides?.model ?? resolveModel(provider),\n apiUrl: overrides?.apiUrl ?? resolveApiUrl(provider),\n };\n}\n\n// ============================================================================\n// Configuration Validation\n// ============================================================================\n\n/**\n * Validate LLM configuration\n * @throws LLMConfigError if configuration is invalid\n */\nexport function validateConfig(config: LLMConfig): void {\n // Validate provider\n const validProviders: LLMProviderType[] = ['gemini', 'openai', 'ollama', 'custom'];\n if (!validProviders.includes(config.provider)) {\n throw new LLMConfigError(\n `Invalid provider: ${config.provider}. Valid options: ${validProviders.join(', ')}`,\n config.provider\n );\n }\n\n // Validate API key requirements\n if (config.provider === 'gemini' && !config.apiKey) {\n throw new LLMConfigError(\n 'Gemini API key is required. Set LLM_API_KEY environment variable.',\n 'gemini'\n );\n }\n\n if (config.provider === 'openai' && !config.apiKey) {\n throw new LLMConfigError(\n 'OpenAI API key is required. Set LLM_API_KEY environment variable.',\n 'openai'\n );\n }\n\n // Validate API URL for custom provider\n if (config.provider === 'custom' && !config.apiUrl) {\n throw new LLMConfigError(\n 'Custom provider requires LLM_API_URL environment variable.',\n 'custom'\n );\n }\n\n // Validate model\n if (!config.model || config.model.trim() === '') {\n throw new LLMConfigError(\n `Model name is required for ${config.provider} provider.`,\n config.provider\n );\n }\n}\n\n// ============================================================================\n// Utility Functions\n// ============================================================================\n\n/**\n * Check if a provider requires an API key\n */\nexport function requiresApiKey(provider: LLMProviderType): boolean {\n return provider === 'gemini' || provider === 'openai';\n}\n\n/**\n * Check if a provider requires a custom API URL\n */\nexport function requiresApiUrl(provider: LLMProviderType): boolean {\n return provider === 'custom';\n}\n\n/**\n * Get safe config for logging (API key masked)\n */\nexport function getSafeConfigForLogging(config: LLMConfig): Record<string, string | undefined> {\n return {\n provider: config.provider,\n model: config.model,\n apiUrl: config.apiUrl,\n apiKey: config.apiKey ? '***' : undefined,\n };\n}\n"]}
|
|
@@ -0,0 +1,174 @@
|
|
|
1
|
+
// src/lib/llm/types.ts
|
|
2
|
+
var LLMError = class _LLMError extends Error {
|
|
3
|
+
constructor(message, provider, statusCode) {
|
|
4
|
+
super(message);
|
|
5
|
+
this.provider = provider;
|
|
6
|
+
this.statusCode = statusCode;
|
|
7
|
+
this.name = "LLMError";
|
|
8
|
+
Object.setPrototypeOf(this, _LLMError.prototype);
|
|
9
|
+
}
|
|
10
|
+
};
|
|
11
|
+
var LLMConfigError = class _LLMConfigError extends LLMError {
|
|
12
|
+
constructor(message, provider) {
|
|
13
|
+
super(message, provider);
|
|
14
|
+
this.name = "LLMConfigError";
|
|
15
|
+
Object.setPrototypeOf(this, _LLMConfigError.prototype);
|
|
16
|
+
}
|
|
17
|
+
};
|
|
18
|
+
var LLMAuthError = class _LLMAuthError extends LLMError {
|
|
19
|
+
constructor(message, provider) {
|
|
20
|
+
super(message, provider, 401);
|
|
21
|
+
this.name = "LLMAuthError";
|
|
22
|
+
Object.setPrototypeOf(this, _LLMAuthError.prototype);
|
|
23
|
+
}
|
|
24
|
+
};
|
|
25
|
+
var LLMRateLimitError = class _LLMRateLimitError extends LLMError {
|
|
26
|
+
constructor(message, provider) {
|
|
27
|
+
super(message, provider, 429);
|
|
28
|
+
this.name = "LLMRateLimitError";
|
|
29
|
+
Object.setPrototypeOf(this, _LLMRateLimitError.prototype);
|
|
30
|
+
}
|
|
31
|
+
};
|
|
32
|
+
var LLMSafetyError = class _LLMSafetyError extends LLMError {
|
|
33
|
+
constructor(message, provider) {
|
|
34
|
+
super(message, provider, 400);
|
|
35
|
+
this.name = "LLMSafetyError";
|
|
36
|
+
Object.setPrototypeOf(this, _LLMSafetyError.prototype);
|
|
37
|
+
}
|
|
38
|
+
};
|
|
39
|
+
var LLMStreamError = class _LLMStreamError extends LLMError {
|
|
40
|
+
constructor(message, provider) {
|
|
41
|
+
super(message, provider);
|
|
42
|
+
this.name = "LLMStreamError";
|
|
43
|
+
Object.setPrototypeOf(this, _LLMStreamError.prototype);
|
|
44
|
+
}
|
|
45
|
+
};
|
|
46
|
+
function isLLMError(error) {
|
|
47
|
+
return error instanceof LLMError;
|
|
48
|
+
}
|
|
49
|
+
function isRetryableError(error) {
|
|
50
|
+
if (!isLLMError(error)) {
|
|
51
|
+
if (error instanceof TypeError && error.message.includes("fetch")) {
|
|
52
|
+
return true;
|
|
53
|
+
}
|
|
54
|
+
return false;
|
|
55
|
+
}
|
|
56
|
+
if (error instanceof LLMAuthError || error instanceof LLMSafetyError) {
|
|
57
|
+
return false;
|
|
58
|
+
}
|
|
59
|
+
if (error instanceof LLMConfigError) {
|
|
60
|
+
return false;
|
|
61
|
+
}
|
|
62
|
+
return true;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
// src/lib/llm/utils/config.ts
|
|
66
|
+
var DEFAULT_PROVIDER = "gemini";
|
|
67
|
+
var DEFAULT_MODELS = {
|
|
68
|
+
gemini: "gemini-2.5-flash",
|
|
69
|
+
openai: "gpt-4o",
|
|
70
|
+
ollama: "llama3.2",
|
|
71
|
+
custom: "gpt-3.5-turbo"
|
|
72
|
+
};
|
|
73
|
+
var DEFAULT_API_URLS = {
|
|
74
|
+
ollama: "http://localhost:11434/v1",
|
|
75
|
+
openai: "https://api.openai.com/v1"
|
|
76
|
+
};
|
|
77
|
+
function getEnvVar(key) {
|
|
78
|
+
return process.env[key];
|
|
79
|
+
}
|
|
80
|
+
function resolveProvider() {
|
|
81
|
+
var _a;
|
|
82
|
+
const provider = (_a = getEnvVar("LLM_PROVIDER")) == null ? void 0 : _a.toLowerCase();
|
|
83
|
+
if (!provider) {
|
|
84
|
+
return DEFAULT_PROVIDER;
|
|
85
|
+
}
|
|
86
|
+
const validProviders = ["gemini", "openai", "ollama", "custom"];
|
|
87
|
+
if (!validProviders.includes(provider)) {
|
|
88
|
+
console.error(`[LLM] Invalid provider "${provider}", falling back to "${DEFAULT_PROVIDER}"`);
|
|
89
|
+
return DEFAULT_PROVIDER;
|
|
90
|
+
}
|
|
91
|
+
return provider;
|
|
92
|
+
}
|
|
93
|
+
function resolveApiKey(provider) {
|
|
94
|
+
const apiKey = getEnvVar("LLM_API_KEY");
|
|
95
|
+
if (!apiKey && provider === "ollama") {
|
|
96
|
+
return void 0;
|
|
97
|
+
}
|
|
98
|
+
return apiKey;
|
|
99
|
+
}
|
|
100
|
+
function resolveModel(provider) {
|
|
101
|
+
const model = getEnvVar("LLM_MODEL");
|
|
102
|
+
return model || DEFAULT_MODELS[provider];
|
|
103
|
+
}
|
|
104
|
+
function resolveApiUrl(provider) {
|
|
105
|
+
const apiUrl = getEnvVar("LLM_API_URL");
|
|
106
|
+
if (apiUrl) {
|
|
107
|
+
return apiUrl;
|
|
108
|
+
}
|
|
109
|
+
if (provider === "ollama") {
|
|
110
|
+
return DEFAULT_API_URLS.ollama;
|
|
111
|
+
}
|
|
112
|
+
if (provider === "openai") {
|
|
113
|
+
return DEFAULT_API_URLS.openai;
|
|
114
|
+
}
|
|
115
|
+
if (provider === "custom") {
|
|
116
|
+
return void 0;
|
|
117
|
+
}
|
|
118
|
+
return void 0;
|
|
119
|
+
}
|
|
120
|
+
function resolveConfig(overrides) {
|
|
121
|
+
var _a, _b, _c, _d;
|
|
122
|
+
const provider = (_a = overrides == null ? void 0 : overrides.provider) != null ? _a : resolveProvider();
|
|
123
|
+
return {
|
|
124
|
+
provider,
|
|
125
|
+
apiKey: (_b = overrides == null ? void 0 : overrides.apiKey) != null ? _b : resolveApiKey(provider),
|
|
126
|
+
model: (_c = overrides == null ? void 0 : overrides.model) != null ? _c : resolveModel(provider),
|
|
127
|
+
apiUrl: (_d = overrides == null ? void 0 : overrides.apiUrl) != null ? _d : resolveApiUrl(provider)
|
|
128
|
+
};
|
|
129
|
+
}
|
|
130
|
+
function validateConfig(config) {
|
|
131
|
+
const validProviders = ["gemini", "openai", "ollama", "custom"];
|
|
132
|
+
if (!validProviders.includes(config.provider)) {
|
|
133
|
+
throw new LLMConfigError(
|
|
134
|
+
`Invalid provider: ${config.provider}. Valid options: ${validProviders.join(", ")}`,
|
|
135
|
+
config.provider
|
|
136
|
+
);
|
|
137
|
+
}
|
|
138
|
+
if (config.provider === "gemini" && !config.apiKey) {
|
|
139
|
+
throw new LLMConfigError(
|
|
140
|
+
"Gemini API key is required. Set LLM_API_KEY environment variable.",
|
|
141
|
+
"gemini"
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
if (config.provider === "openai" && !config.apiKey) {
|
|
145
|
+
throw new LLMConfigError(
|
|
146
|
+
"OpenAI API key is required. Set LLM_API_KEY environment variable.",
|
|
147
|
+
"openai"
|
|
148
|
+
);
|
|
149
|
+
}
|
|
150
|
+
if (config.provider === "custom" && !config.apiUrl) {
|
|
151
|
+
throw new LLMConfigError(
|
|
152
|
+
"Custom provider requires LLM_API_URL environment variable.",
|
|
153
|
+
"custom"
|
|
154
|
+
);
|
|
155
|
+
}
|
|
156
|
+
if (!config.model || config.model.trim() === "") {
|
|
157
|
+
throw new LLMConfigError(
|
|
158
|
+
`Model name is required for ${config.provider} provider.`,
|
|
159
|
+
config.provider
|
|
160
|
+
);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
function getSafeConfigForLogging(config) {
|
|
164
|
+
return {
|
|
165
|
+
provider: config.provider,
|
|
166
|
+
model: config.model,
|
|
167
|
+
apiUrl: config.apiUrl,
|
|
168
|
+
apiKey: config.apiKey ? "***" : void 0
|
|
169
|
+
};
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
export { DEFAULT_API_URLS, LLMAuthError, LLMConfigError, LLMError, LLMRateLimitError, LLMSafetyError, LLMStreamError, getSafeConfigForLogging, isRetryableError, resolveConfig, validateConfig };
|
|
173
|
+
//# sourceMappingURL=chunk-SR5DRGBX.mjs.map
|
|
174
|
+
//# sourceMappingURL=chunk-SR5DRGBX.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/lib/llm/types.ts","../src/lib/llm/utils/config.ts"],"names":[],"mappings":"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","file":"chunk-SR5DRGBX.mjs","sourcesContent":["/**\n * LLM Provider Types & Interfaces\n * Strategy Pattern implementation for multi-provider LLM support\n */\n\n// ============================================================================\n// Provider Types\n// ============================================================================\n\nexport type LLMProviderType = 'gemini' | 'openai' | 'ollama' | 'custom';\n\n// ============================================================================\n// Configuration\n// ============================================================================\n\nexport interface LLMConfig {\n provider: LLMProviderType;\n apiKey?: string;\n model: string;\n apiUrl?: string;\n}\n\n// ============================================================================\n// Messages\n// ============================================================================\n\nexport type LLMMessageRole = 'system' | 'user' | 'assistant';\n\nexport interface LLMMessage {\n role: LLMMessageRole;\n content: string;\n}\n\n// ============================================================================\n// Stream Options\n// ============================================================================\n\nexport interface LLMStreamOptions {\n messages: LLMMessage[];\n model?: string;\n temperature?: number;\n maxTokens?: number;\n}\n\n// ============================================================================\n// Provider Interface (Strategy Pattern)\n// ============================================================================\n\nexport interface LLMProvider {\n /** Provider identifier */\n readonly name: LLMProviderType;\n\n /** Current configuration */\n readonly config: LLMConfig;\n\n /**\n * Stream a completion response\n * @param options - Stream configuration including messages\n * @returns ReadableStream of UTF-8 encoded chunks\n */\n stream(options: LLMStreamOptions): Promise<ReadableStream<Uint8Array>>;\n\n /**\n * Validate provider configuration\n * @throws LLMConfigError if configuration is invalid\n */\n validate(): void;\n}\n\n// ============================================================================\n// Error Classes\n// ============================================================================\n\n/**\n * Base error class for LLM-related errors\n */\nexport class LLMError extends Error {\n constructor(\n message: string,\n public readonly provider?: LLMProviderType,\n public readonly statusCode?: number\n ) {\n super(message);\n this.name = 'LLMError';\n Object.setPrototypeOf(this, LLMError.prototype);\n }\n}\n\n/**\n * Configuration error - missing or invalid config\n */\nexport class LLMConfigError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider);\n this.name = 'LLMConfigError';\n Object.setPrototypeOf(this, LLMConfigError.prototype);\n }\n}\n\n/**\n * Authentication error - invalid API key (401/403)\n */\nexport class LLMAuthError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider, 401);\n this.name = 'LLMAuthError';\n Object.setPrototypeOf(this, LLMAuthError.prototype);\n }\n}\n\n/**\n * Rate limit error - quota exceeded (429)\n */\nexport class LLMRateLimitError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider, 429);\n this.name = 'LLMRateLimitError';\n Object.setPrototypeOf(this, LLMRateLimitError.prototype);\n }\n}\n\n/**\n * Safety filter error - content blocked\n */\nexport class LLMSafetyError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider, 400);\n this.name = 'LLMSafetyError';\n Object.setPrototypeOf(this, LLMSafetyError.prototype);\n }\n}\n\n/**\n * Streaming error - connection or parsing failure\n */\nexport class LLMStreamError extends LLMError {\n constructor(message: string, provider?: LLMProviderType) {\n super(message, provider);\n this.name = 'LLMStreamError';\n Object.setPrototypeOf(this, LLMStreamError.prototype);\n }\n}\n\n// ============================================================================\n// Type Guards\n// ============================================================================\n\nexport function isLLMError(error: unknown): error is LLMError {\n return error instanceof LLMError;\n}\n\nexport function isRetryableError(error: unknown): boolean {\n if (!isLLMError(error)) {\n // Network errors are retryable\n if (error instanceof TypeError && error.message.includes('fetch')) {\n return true;\n }\n return false;\n }\n\n // Auth and safety errors are not retryable\n if (error instanceof LLMAuthError || error instanceof LLMSafetyError) {\n return false;\n }\n\n // Config errors are not retryable\n if (error instanceof LLMConfigError) {\n return false;\n }\n\n // Rate limit and stream errors may be retryable\n return true;\n}\n","/**\n * LLM Configuration Utilities\n * Environment variable resolution and default configuration\n */\n\nimport { type LLMConfig, type LLMProviderType, LLMConfigError } from '../types';\n\n// ============================================================================\n// Default Configuration\n// ============================================================================\n\nexport const DEFAULT_PROVIDER: LLMProviderType = 'gemini';\n\nexport const DEFAULT_MODELS: Record<LLMProviderType, string> = {\n gemini: 'gemini-2.5-flash',\n openai: 'gpt-4o',\n ollama: 'llama3.2',\n custom: 'gpt-3.5-turbo',\n};\n\nexport const DEFAULT_API_URLS: Record<string, string> = {\n ollama: 'http://localhost:11434/v1',\n openai: 'https://api.openai.com/v1',\n};\n\n// ============================================================================\n// Environment Resolution\n// ============================================================================\n\nfunction getEnvVar(key: string): string | undefined {\n return process.env[key];\n}\n\nfunction resolveProvider(): LLMProviderType {\n const provider = getEnvVar('LLM_PROVIDER')?.toLowerCase();\n\n if (!provider) {\n return DEFAULT_PROVIDER;\n }\n\n const validProviders: LLMProviderType[] = ['gemini', 'openai', 'ollama', 'custom'];\n\n if (!validProviders.includes(provider as LLMProviderType)) {\n console.error(`[LLM] Invalid provider \"${provider}\", falling back to \"${DEFAULT_PROVIDER}\"`);\n return DEFAULT_PROVIDER;\n }\n\n return provider as LLMProviderType;\n}\n\nfunction resolveApiKey(provider: LLMProviderType): string | undefined {\n const apiKey = getEnvVar('LLM_API_KEY');\n\n // Ollama doesn't require API key\n if (!apiKey && provider === 'ollama') {\n return undefined;\n }\n\n return apiKey;\n}\n\nfunction resolveModel(provider: LLMProviderType): string {\n const model = getEnvVar('LLM_MODEL');\n return model || DEFAULT_MODELS[provider];\n}\n\nfunction resolveApiUrl(provider: LLMProviderType): string | undefined {\n // Primary: LLM_API_URL\n const apiUrl = getEnvVar('LLM_API_URL');\n if (apiUrl) {\n return apiUrl;\n }\n\n // Default URLs for specific providers\n if (provider === 'ollama') {\n return DEFAULT_API_URLS.ollama;\n }\n\n if (provider === 'openai') {\n return DEFAULT_API_URLS.openai;\n }\n\n // Custom provider requires explicit URL\n if (provider === 'custom') {\n return undefined;\n }\n\n return undefined;\n}\n\n// ============================================================================\n// Configuration Resolution\n// ============================================================================\n\n/**\n * Resolve LLM configuration from environment variables with optional overrides\n */\nexport function resolveConfig(overrides?: Partial<LLMConfig>): LLMConfig {\n const provider = overrides?.provider ?? resolveProvider();\n\n return {\n provider,\n apiKey: overrides?.apiKey ?? resolveApiKey(provider),\n model: overrides?.model ?? resolveModel(provider),\n apiUrl: overrides?.apiUrl ?? resolveApiUrl(provider),\n };\n}\n\n// ============================================================================\n// Configuration Validation\n// ============================================================================\n\n/**\n * Validate LLM configuration\n * @throws LLMConfigError if configuration is invalid\n */\nexport function validateConfig(config: LLMConfig): void {\n // Validate provider\n const validProviders: LLMProviderType[] = ['gemini', 'openai', 'ollama', 'custom'];\n if (!validProviders.includes(config.provider)) {\n throw new LLMConfigError(\n `Invalid provider: ${config.provider}. Valid options: ${validProviders.join(', ')}`,\n config.provider\n );\n }\n\n // Validate API key requirements\n if (config.provider === 'gemini' && !config.apiKey) {\n throw new LLMConfigError(\n 'Gemini API key is required. Set LLM_API_KEY environment variable.',\n 'gemini'\n );\n }\n\n if (config.provider === 'openai' && !config.apiKey) {\n throw new LLMConfigError(\n 'OpenAI API key is required. Set LLM_API_KEY environment variable.',\n 'openai'\n );\n }\n\n // Validate API URL for custom provider\n if (config.provider === 'custom' && !config.apiUrl) {\n throw new LLMConfigError(\n 'Custom provider requires LLM_API_URL environment variable.',\n 'custom'\n );\n }\n\n // Validate model\n if (!config.model || config.model.trim() === '') {\n throw new LLMConfigError(\n `Model name is required for ${config.provider} provider.`,\n config.provider\n );\n }\n}\n\n// ============================================================================\n// Utility Functions\n// ============================================================================\n\n/**\n * Check if a provider requires an API key\n */\nexport function requiresApiKey(provider: LLMProviderType): boolean {\n return provider === 'gemini' || provider === 'openai';\n}\n\n/**\n * Check if a provider requires a custom API URL\n */\nexport function requiresApiUrl(provider: LLMProviderType): boolean {\n return provider === 'custom';\n}\n\n/**\n * Get safe config for logging (API key masked)\n */\nexport function getSafeConfigForLogging(config: LLMConfig): Record<string, string | undefined> {\n return {\n provider: config.provider,\n model: config.model,\n apiUrl: config.apiUrl,\n apiKey: config.apiKey ? '***' : undefined,\n };\n}\n"]}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
// src/lib/logger.ts
|
|
4
|
+
var LEVEL_ORDER = {
|
|
5
|
+
debug: 0,
|
|
6
|
+
info: 1,
|
|
7
|
+
warn: 2,
|
|
8
|
+
error: 3
|
|
9
|
+
};
|
|
10
|
+
function getMinLevel() {
|
|
11
|
+
var _a, _b;
|
|
12
|
+
if (typeof process !== "undefined" && ((_a = process.env) == null ? void 0 : _a.LOG_LEVEL)) {
|
|
13
|
+
const env = process.env.LOG_LEVEL.toLowerCase();
|
|
14
|
+
if (env in LEVEL_ORDER) return env;
|
|
15
|
+
}
|
|
16
|
+
const isDev = typeof process !== "undefined" && ((_b = process.env) == null ? void 0 : _b.NODE_ENV) !== "production";
|
|
17
|
+
return isDev ? "debug" : "info";
|
|
18
|
+
}
|
|
19
|
+
function shouldLog(level) {
|
|
20
|
+
return LEVEL_ORDER[level] >= LEVEL_ORDER[getMinLevel()];
|
|
21
|
+
}
|
|
22
|
+
function formatContext(ctx) {
|
|
23
|
+
if (!ctx) return "";
|
|
24
|
+
const parts = [];
|
|
25
|
+
if (ctx.route) parts.push(`route=${ctx.route}`);
|
|
26
|
+
if (ctx.provider) parts.push(`provider=${ctx.provider}`);
|
|
27
|
+
if (ctx.connectionId) parts.push(`connId=${ctx.connectionId}`);
|
|
28
|
+
if (ctx.duration !== void 0) parts.push(`duration=${ctx.duration}ms`);
|
|
29
|
+
for (const [k, v] of Object.entries(ctx)) {
|
|
30
|
+
if (["route", "provider", "connectionId", "duration"].includes(k)) continue;
|
|
31
|
+
if (v !== void 0) parts.push(`${k}=${typeof v === "object" ? JSON.stringify(v) : v}`);
|
|
32
|
+
}
|
|
33
|
+
return parts.length ? ` {${parts.join(", ")}}` : "";
|
|
34
|
+
}
|
|
35
|
+
function extractError(error) {
|
|
36
|
+
var _a;
|
|
37
|
+
if (!error) return void 0;
|
|
38
|
+
if (error instanceof Error) {
|
|
39
|
+
const isDev = typeof process !== "undefined" && ((_a = process.env) == null ? void 0 : _a.NODE_ENV) !== "production";
|
|
40
|
+
return {
|
|
41
|
+
name: error.name,
|
|
42
|
+
message: error.message,
|
|
43
|
+
stack: isDev ? error.stack : void 0
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
return { name: "Unknown", message: String(error) };
|
|
47
|
+
}
|
|
48
|
+
function sanitizeLogValue(value) {
|
|
49
|
+
return value.replace(/[\r\n]/g, " ").replace(/[\x00-\x08\x0b\x0c\x0e-\x1f]/g, "");
|
|
50
|
+
}
|
|
51
|
+
function log(level, message, error, context) {
|
|
52
|
+
if (!shouldLog(level)) return;
|
|
53
|
+
const timestamp = (/* @__PURE__ */ new Date()).toISOString();
|
|
54
|
+
const tag = level.toUpperCase().padEnd(5);
|
|
55
|
+
const ctx = formatContext(context);
|
|
56
|
+
const errInfo = level === "error" ? extractError(error) : void 0;
|
|
57
|
+
const line = `[${tag}] [${timestamp}]${sanitizeLogValue(ctx)} ${sanitizeLogValue(message)}`;
|
|
58
|
+
if (errInfo) {
|
|
59
|
+
const errLine = ` | ${sanitizeLogValue(errInfo.name)}: ${sanitizeLogValue(errInfo.message)}`;
|
|
60
|
+
const full = line + errLine;
|
|
61
|
+
if (errInfo.stack) {
|
|
62
|
+
const safeStack = errInfo.stack.replace(/[\x00-\x08\x0b\x0c\x0e-\x1f]/g, "");
|
|
63
|
+
console.error(full, "\n", safeStack);
|
|
64
|
+
} else {
|
|
65
|
+
console.error(full);
|
|
66
|
+
}
|
|
67
|
+
return;
|
|
68
|
+
}
|
|
69
|
+
const safeLine = line;
|
|
70
|
+
switch (level) {
|
|
71
|
+
case "debug":
|
|
72
|
+
console.debug(safeLine);
|
|
73
|
+
break;
|
|
74
|
+
case "info":
|
|
75
|
+
console.info(safeLine);
|
|
76
|
+
break;
|
|
77
|
+
case "warn":
|
|
78
|
+
console.warn(safeLine);
|
|
79
|
+
break;
|
|
80
|
+
case "error":
|
|
81
|
+
console.error(safeLine);
|
|
82
|
+
break;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
var logger = {
|
|
86
|
+
debug(message, context) {
|
|
87
|
+
log("debug", message, void 0, context);
|
|
88
|
+
},
|
|
89
|
+
info(message, context) {
|
|
90
|
+
log("info", message, void 0, context);
|
|
91
|
+
},
|
|
92
|
+
warn(message, context) {
|
|
93
|
+
log("warn", message, void 0, context);
|
|
94
|
+
},
|
|
95
|
+
error(message, error, context) {
|
|
96
|
+
log("error", message, error, context);
|
|
97
|
+
}
|
|
98
|
+
};
|
|
99
|
+
|
|
100
|
+
exports.logger = logger;
|
|
101
|
+
//# sourceMappingURL=chunk-VLCRUZX7.js.map
|
|
102
|
+
//# sourceMappingURL=chunk-VLCRUZX7.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/lib/logger.ts"],"names":[],"mappings":"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","file":"chunk-VLCRUZX7.js","sourcesContent":["/**\n * Structured Logger\n * Isomorphic logger with level filtering and context support\n */\n\ntype LogLevel = 'debug' | 'info' | 'warn' | 'error';\n\nexport interface LogContext {\n route?: string;\n provider?: string;\n connectionId?: string;\n duration?: number;\n [key: string]: unknown;\n}\n\nconst LEVEL_ORDER: Record<LogLevel, number> = {\n debug: 0,\n info: 1,\n warn: 2,\n error: 3,\n};\n\nfunction getMinLevel(): LogLevel {\n if (typeof process !== 'undefined' && process.env?.LOG_LEVEL) {\n const env = process.env.LOG_LEVEL.toLowerCase();\n if (env in LEVEL_ORDER) return env as LogLevel;\n }\n const isDev =\n typeof process !== 'undefined' &&\n process.env?.NODE_ENV !== 'production';\n return isDev ? 'debug' : 'info';\n}\n\nfunction shouldLog(level: LogLevel): boolean {\n return LEVEL_ORDER[level] >= LEVEL_ORDER[getMinLevel()];\n}\n\nfunction formatContext(ctx?: LogContext): string {\n if (!ctx) return '';\n const parts: string[] = [];\n if (ctx.route) parts.push(`route=${ctx.route}`);\n if (ctx.provider) parts.push(`provider=${ctx.provider}`);\n if (ctx.connectionId) parts.push(`connId=${ctx.connectionId}`);\n if (ctx.duration !== undefined) parts.push(`duration=${ctx.duration}ms`);\n // Extra keys\n for (const [k, v] of Object.entries(ctx)) {\n if (['route', 'provider', 'connectionId', 'duration'].includes(k)) continue;\n if (v !== undefined) parts.push(`${k}=${typeof v === 'object' ? JSON.stringify(v) : v}`);\n }\n return parts.length ? ` {${parts.join(', ')}}` : '';\n}\n\nfunction extractError(error: unknown): { name: string; message: string; stack?: string } | undefined {\n if (!error) return undefined;\n if (error instanceof Error) {\n const isDev =\n typeof process !== 'undefined' &&\n process.env?.NODE_ENV !== 'production';\n return {\n name: error.name,\n message: error.message,\n stack: isDev ? error.stack : undefined,\n };\n }\n return { name: 'Unknown', message: String(error) };\n}\n\n/** Sanitize log message to prevent log injection (newlines, control chars) */\nfunction sanitizeLogValue(value: string): string {\n return value.replace(/[\\r\\n]/g, ' ').replace(/[\\x00-\\x08\\x0b\\x0c\\x0e-\\x1f]/g, '');\n}\n\nfunction log(level: LogLevel, message: string, error?: unknown, context?: LogContext): void {\n if (!shouldLog(level)) return;\n\n const timestamp = new Date().toISOString();\n const tag = level.toUpperCase().padEnd(5);\n const ctx = formatContext(context);\n const errInfo = level === 'error' ? extractError(error) : undefined;\n\n const line = `[${tag}] [${timestamp}]${sanitizeLogValue(ctx)} ${sanitizeLogValue(message)}`;\n\n if (errInfo) {\n const errLine = ` | ${sanitizeLogValue(errInfo.name)}: ${sanitizeLogValue(errInfo.message)}`;\n const full = line + errLine;\n if (errInfo.stack) {\n // Stack traces contain intentional newlines — sanitize control chars only\n const safeStack = errInfo.stack.replace(/[\\x00-\\x08\\x0b\\x0c\\x0e-\\x1f]/g, '');\n console.error(full, '\\n', safeStack);\n } else {\n console.error(full);\n }\n return;\n }\n\n // All values in `line` are already sanitized via sanitizeLogValue\n const safeLine = line;\n switch (level) {\n case 'debug':\n console.debug(safeLine);\n break;\n case 'info':\n console.info(safeLine);\n break;\n case 'warn':\n console.warn(safeLine);\n break;\n case 'error':\n console.error(safeLine);\n break;\n }\n}\n\nexport const logger = {\n debug(message: string, context?: LogContext): void {\n log('debug', message, undefined, context);\n },\n info(message: string, context?: LogContext): void {\n log('info', message, undefined, context);\n },\n warn(message: string, context?: LogContext): void {\n log('warn', message, undefined, context);\n },\n error(message: string, error?: unknown, context?: LogContext): void {\n log('error', message, error, context);\n },\n};\n"]}
|