insforge 0.3.1
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/.dockerignore +58 -0
- package/.env.example +49 -0
- package/.github/ISSUE_TEMPLATE/bug_report.yml +83 -0
- package/.github/ISSUE_TEMPLATE/config.yml +11 -0
- package/.github/ISSUE_TEMPLATE/feature_request.yml +79 -0
- package/.github/copilot-instructions.md +147 -0
- package/.github/workflows/build-image.yml +65 -0
- package/.github/workflows/ci-premerge-check.yml +24 -0
- package/.github/workflows/deploy-aws.yml +130 -0
- package/.github/workflows/lint-and-format.yml +33 -0
- package/.prettierignore +65 -0
- package/.prettierrc +9 -0
- package/CHANGELOG.md +3 -0
- package/CONTRIBUTING.md +126 -0
- package/Dockerfile +27 -0
- package/GITHUB_OAUTH_SETUP.md +49 -0
- package/GOOGLE_OAUTH_SETUP.md +148 -0
- package/LICENSE +201 -0
- package/README.md +134 -0
- package/assets/Dark.svg +23 -0
- package/assets/archDiagram.png +0 -0
- package/assets/banner.png +0 -0
- package/assets/mcpInstallv2.png +0 -0
- package/assets/sampleResponse.png +0 -0
- package/assets/signin.png +0 -0
- package/assets/userflow.png +0 -0
- package/backend/migrations/000_create-base-tables.sql +142 -0
- package/backend/migrations/001_create-helper-functions.sql +41 -0
- package/backend/migrations/002_rename-auth-tables.sql +30 -0
- package/backend/migrations/003_create-users-table.sql +56 -0
- package/backend/migrations/004_add-reload-postgrest-func.sql +24 -0
- package/backend/migrations/005_enable-project-admin-modify-users.sql +30 -0
- package/backend/migrations/006_modify-ai-usage-table.sql +25 -0
- package/backend/migrations/007_drop-metadata-table.sql +2 -0
- package/backend/migrations/008_add-system-tables.sql +77 -0
- package/backend/migrations/009_add-function-secrets.sql +24 -0
- package/backend/migrations/010_modify-ai-config-modalities.sql +93 -0
- package/backend/migrations/011_refactor-secrets-table.sql +15 -0
- package/backend/migrations/012_add-storage-uploaded-by.sql +8 -0
- package/backend/package.json +75 -0
- package/backend/src/api/middleware/auth.ts +240 -0
- package/backend/src/api/middleware/error.ts +231 -0
- package/backend/src/api/middleware/upload.ts +59 -0
- package/backend/src/api/routes/agent.ts +29 -0
- package/backend/src/api/routes/ai.ts +472 -0
- package/backend/src/api/routes/auth.oauth.ts +482 -0
- package/backend/src/api/routes/auth.ts +386 -0
- package/backend/src/api/routes/database.advance.ts +275 -0
- package/backend/src/api/routes/database.records.ts +246 -0
- package/backend/src/api/routes/database.tables.ts +161 -0
- package/backend/src/api/routes/docs.ts +66 -0
- package/backend/src/api/routes/functions.ts +183 -0
- package/backend/src/api/routes/logs.ts +150 -0
- package/backend/src/api/routes/metadata.ts +160 -0
- package/backend/src/api/routes/openapi.ts +82 -0
- package/backend/src/api/routes/secrets.ts +199 -0
- package/backend/src/api/routes/storage.ts +547 -0
- package/backend/src/api/routes/usage.ts +96 -0
- package/backend/src/core/ai/chat.ts +207 -0
- package/backend/src/core/ai/client.ts +242 -0
- package/backend/src/core/ai/config.ts +187 -0
- package/backend/src/core/ai/image.ts +156 -0
- package/backend/src/core/ai/model.ts +117 -0
- package/backend/src/core/ai/usage.ts +290 -0
- package/backend/src/core/auth/auth.ts +781 -0
- package/backend/src/core/auth/oauth.ts +398 -0
- package/backend/src/core/database/advance.ts +1074 -0
- package/backend/src/core/database/manager.ts +178 -0
- package/backend/src/core/database/table.ts +772 -0
- package/backend/src/core/documentation/agent.ts +689 -0
- package/backend/src/core/documentation/openapi.ts +856 -0
- package/backend/src/core/functions/functions.ts +310 -0
- package/backend/src/core/logs/analytics.ts +76 -0
- package/backend/src/core/logs/audit.ts +255 -0
- package/backend/src/core/logs/providers/base.provider.ts +83 -0
- package/backend/src/core/logs/providers/cloudwatch.provider.ts +510 -0
- package/backend/src/core/logs/providers/localdb.provider.ts +246 -0
- package/backend/src/core/secrets/encryption.ts +58 -0
- package/backend/src/core/secrets/secrets.ts +410 -0
- package/backend/src/core/socket/socket.ts +388 -0
- package/backend/src/core/socket/types.ts +79 -0
- package/backend/src/core/storage/storage.ts +923 -0
- package/backend/src/server.ts +288 -0
- package/backend/src/types/ai.ts +46 -0
- package/backend/src/types/auth.ts +90 -0
- package/backend/src/types/database.ts +136 -0
- package/backend/src/types/error-constants.ts +86 -0
- package/backend/src/types/logs.ts +47 -0
- package/backend/src/types/profile.ts +55 -0
- package/backend/src/types/storage.ts +23 -0
- package/backend/src/utils/cloud-token.ts +39 -0
- package/backend/src/utils/constants.ts +1 -0
- package/backend/src/utils/environment.ts +35 -0
- package/backend/src/utils/helpers.ts +49 -0
- package/backend/src/utils/logger.ts +13 -0
- package/backend/src/utils/response.ts +62 -0
- package/backend/src/utils/seed.ts +205 -0
- package/backend/src/utils/sql-parser.ts +63 -0
- package/backend/src/utils/uuid.ts +9 -0
- package/backend/src/utils/validations.ts +129 -0
- package/backend/tests/README.md +134 -0
- package/backend/tests/cleanup-all-test-data.sh +231 -0
- package/backend/tests/cloud/test-s3-multitenant.sh +132 -0
- package/backend/tests/local/comprehensive-curl-tests.sh +156 -0
- package/backend/tests/local/test-auth-router.sh +144 -0
- package/backend/tests/local/test-database-router.sh +222 -0
- package/backend/tests/local/test-e2e.sh +241 -0
- package/backend/tests/local/test-fk-errors.sh +97 -0
- package/backend/tests/local/test-id-field.sh +201 -0
- package/backend/tests/local/test-public-bucket.sh +265 -0
- package/backend/tests/local/test-secrets.sh +248 -0
- package/backend/tests/local/test-serverless-functions.sh.disabled +325 -0
- package/backend/tests/local/test-traditional-rest.sh +209 -0
- package/backend/tests/manual/README.md +51 -0
- package/backend/tests/manual/create-large-table-simple.sql +11 -0
- package/backend/tests/manual/seed-large-table.sql +101 -0
- package/backend/tests/manual/setup-large-table-extras.sql +34 -0
- package/backend/tests/manual/test-better-auth.sh +303 -0
- package/backend/tests/manual/test-bulk-upsert.sh +410 -0
- package/backend/tests/manual/test-database-advance.sh +297 -0
- package/backend/tests/manual/test-postgrest-stability.sh +192 -0
- package/backend/tests/manual/test-rawsql-export-import.sh +412 -0
- package/backend/tests/manual/test-universal-storage.sh +264 -0
- package/backend/tests/manual/test-users.sql +18 -0
- package/backend/tests/run-all-tests.sh +140 -0
- package/backend/tests/setup.ts +22 -0
- package/backend/tests/test-config.sh +303 -0
- package/backend/tsconfig.json +23 -0
- package/backend/tsup.config.ts +18 -0
- package/backend/vitest.config.ts +22 -0
- package/docker-compose.prod.yml +145 -0
- package/docker-compose.yml +167 -0
- package/docker-init/db/db-init.sql +125 -0
- package/docker-init/db/jwt.sql +5 -0
- package/docker-init/db/logs.sql +9 -0
- package/docker-init/db/postgresql.conf +17 -0
- package/docs/deprecated/insforge-auth-api.md +215 -0
- package/docs/deprecated/insforge-auth-sdk.md +100 -0
- package/docs/deprecated/insforge-db-api.md +359 -0
- package/docs/deprecated/insforge-db-sdk.md +140 -0
- package/docs/deprecated/insforge-debug-sdk.md +157 -0
- package/docs/deprecated/insforge-debug.md +65 -0
- package/docs/deprecated/insforge-instructions.md +124 -0
- package/docs/deprecated/insforge-project.md +118 -0
- package/docs/deprecated/insforge-storage-api.md +279 -0
- package/docs/deprecated/insforge-storage-sdk.md +159 -0
- package/docs/insforge-instructions-sdk.md +407 -0
- package/eslint.config.js +317 -0
- package/examples/oauth/frontend-oauth-example.html +251 -0
- package/examples/response-examples.md +444 -0
- package/frontend/README.md +112 -0
- package/frontend/components.json +17 -0
- package/frontend/index.html +13 -0
- package/frontend/package.json +63 -0
- package/frontend/public/favicon.ico +0 -0
- package/frontend/src/App.tsx +106 -0
- package/frontend/src/assets/icons/checkbox_checked.svg +6 -0
- package/frontend/src/assets/icons/checkbox_undetermined.svg +6 -0
- package/frontend/src/assets/icons/checked.svg +3 -0
- package/frontend/src/assets/icons/error.svg +3 -0
- package/frontend/src/assets/icons/pencil.svg +4 -0
- package/frontend/src/assets/icons/refresh.svg +4 -0
- package/frontend/src/assets/icons/step_active.svg +3 -0
- package/frontend/src/assets/icons/step_inactive.svg +11 -0
- package/frontend/src/assets/icons/warning.svg +3 -0
- package/frontend/src/assets/logos/amazon.svg +1 -0
- package/frontend/src/assets/logos/claude_code.svg +3 -0
- package/frontend/src/assets/logos/cline.svg +6 -0
- package/frontend/src/assets/logos/cursor.svg +20 -0
- package/frontend/src/assets/logos/discord.svg +9 -0
- package/frontend/src/assets/logos/gemini.svg +19 -0
- package/frontend/src/assets/logos/github.svg +5 -0
- package/frontend/src/assets/logos/google.svg +13 -0
- package/frontend/src/assets/logos/grok.svg +10 -0
- package/frontend/src/assets/logos/insforge_dark.svg +15 -0
- package/frontend/src/assets/logos/insforge_light.svg +15 -0
- package/frontend/src/assets/logos/openai.svg +10 -0
- package/frontend/src/assets/logos/roo_code.svg +9 -0
- package/frontend/src/assets/logos/trae.svg +3 -0
- package/frontend/src/assets/logos/windsurf.svg +10 -0
- package/frontend/src/components/ButtonWithLoading.tsx +27 -0
- package/frontend/src/components/Checkbox.tsx +61 -0
- package/frontend/src/components/CodeBlock.tsx +32 -0
- package/frontend/src/components/ConfirmDialog.tsx +96 -0
- package/frontend/src/components/CopyButton.tsx +69 -0
- package/frontend/src/components/DeleteActionButton.tsx +42 -0
- package/frontend/src/components/EmptyState.tsx +41 -0
- package/frontend/src/components/ErrorState.tsx +35 -0
- package/frontend/src/components/FeatureSidebar.tsx +126 -0
- package/frontend/src/components/FeatureSidebarItem.tsx +101 -0
- package/frontend/src/components/JsonHighlight.tsx +61 -0
- package/frontend/src/components/LoadingState.tsx +16 -0
- package/frontend/src/components/PaginationControls.tsx +54 -0
- package/frontend/src/components/PromptDialog.tsx +68 -0
- package/frontend/src/components/SearchInput.tsx +90 -0
- package/frontend/src/components/SelectionClearButton.tsx +26 -0
- package/frontend/src/components/Stepper.tsx +139 -0
- package/frontend/src/components/ThemeToggle.tsx +58 -0
- package/frontend/src/components/TypeBadge.tsx +20 -0
- package/frontend/src/components/datagrid/DataGrid.tsx +264 -0
- package/frontend/src/components/datagrid/DefaultCellRenderer.tsx +114 -0
- package/frontend/src/components/datagrid/IdCell.tsx +44 -0
- package/frontend/src/components/datagrid/SortableHeader.tsx +74 -0
- package/frontend/src/components/datagrid/cell-editors/BooleanCellEditor.tsx +54 -0
- package/frontend/src/components/datagrid/cell-editors/DateCellEditor.tsx +483 -0
- package/frontend/src/components/datagrid/cell-editors/JsonCellEditor.tsx +362 -0
- package/frontend/src/components/datagrid/cell-editors/TextCellEditor.tsx +38 -0
- package/frontend/src/components/datagrid/cell-editors/index.ts +14 -0
- package/frontend/src/components/datagrid/cell-editors/types.ts +43 -0
- package/frontend/src/components/datagrid/datagridTypes.tsx +72 -0
- package/frontend/src/components/datagrid/index.tsx +20 -0
- package/frontend/src/components/index.ts +39 -0
- package/frontend/src/components/layout/AppHeader.tsx +146 -0
- package/frontend/src/components/layout/AppSidebar.tsx +190 -0
- package/frontend/src/components/layout/CloudLayout.tsx +95 -0
- package/frontend/src/components/layout/Layout.tsx +43 -0
- package/frontend/src/components/radix/Alert.tsx +45 -0
- package/frontend/src/components/radix/AlertDialog.tsx +115 -0
- package/frontend/src/components/radix/Avatar.tsx +45 -0
- package/frontend/src/components/radix/Badge.tsx +33 -0
- package/frontend/src/components/radix/Button.tsx +50 -0
- package/frontend/src/components/radix/Card.tsx +58 -0
- package/frontend/src/components/radix/Dialog.tsx +98 -0
- package/frontend/src/components/radix/DropdownMenu.tsx +185 -0
- package/frontend/src/components/radix/Form.tsx +167 -0
- package/frontend/src/components/radix/Input.tsx +22 -0
- package/frontend/src/components/radix/Label.tsx +19 -0
- package/frontend/src/components/radix/Popover.tsx +29 -0
- package/frontend/src/components/radix/ScrollArea.tsx +44 -0
- package/frontend/src/components/radix/Select.tsx +151 -0
- package/frontend/src/components/radix/Separator.tsx +26 -0
- package/frontend/src/components/radix/Sheet.tsx +119 -0
- package/frontend/src/components/radix/Skeleton.tsx +7 -0
- package/frontend/src/components/radix/Switch.tsx +29 -0
- package/frontend/src/components/radix/Tabs.tsx +50 -0
- package/frontend/src/components/radix/Textarea.tsx +21 -0
- package/frontend/src/components/radix/Tooltip.tsx +28 -0
- package/frontend/src/features/ai/components/AIConfigCard.tsx +154 -0
- package/frontend/src/features/ai/components/AIConfigDialog.tsx +76 -0
- package/frontend/src/features/ai/components/AIConfigForm.tsx +222 -0
- package/frontend/src/features/ai/components/AIEmptyState.tsx +18 -0
- package/frontend/src/features/ai/components/fields/ModalityField.tsx +87 -0
- package/frontend/src/features/ai/components/fields/ModelSelectionField.tsx +134 -0
- package/frontend/src/features/ai/components/fields/SystemPromptField.tsx +33 -0
- package/frontend/src/features/ai/helpers.ts +155 -0
- package/frontend/src/features/ai/hooks/useAIConfigs.ts +221 -0
- package/frontend/src/features/ai/hooks/useAIUsage.ts +77 -0
- package/frontend/src/features/ai/page/AIPage.tsx +178 -0
- package/frontend/src/features/ai/services/ai.service.ts +148 -0
- package/frontend/src/features/auth/components/AddOAuthDialog.tsx +106 -0
- package/frontend/src/features/auth/components/AuthMethodTab.tsx +238 -0
- package/frontend/src/features/auth/components/OAuthConfigDialog.tsx +303 -0
- package/frontend/src/features/auth/components/OAuthEmptyState.tsx +15 -0
- package/frontend/src/features/auth/components/UserFormDialog.tsx +248 -0
- package/frontend/src/features/auth/components/UsersDataGrid.tsx +183 -0
- package/frontend/src/features/auth/components/UsersTab.tsx +114 -0
- package/frontend/src/features/auth/hooks/useOAuthConfig.ts +129 -0
- package/frontend/src/features/auth/hooks/useUsers.ts +57 -0
- package/frontend/src/features/auth/index.ts +9 -0
- package/frontend/src/features/auth/page/AuthenticationPage.tsx +169 -0
- package/frontend/src/features/auth/services/auth.service.ts +112 -0
- package/frontend/src/features/auth/services/oauth.service.ts +49 -0
- package/frontend/src/features/dashboard/page/DashboardPage.tsx +194 -0
- package/frontend/src/features/database/components/ColumnTypeSelect.tsx +64 -0
- package/frontend/src/features/database/components/DatabaseDataGrid.tsx +282 -0
- package/frontend/src/features/database/components/ForeignKeyCell.tsx +187 -0
- package/frontend/src/features/database/components/ForeignKeyPopover.tsx +378 -0
- package/frontend/src/features/database/components/LinkRecordModal.tsx +288 -0
- package/frontend/src/features/database/components/RecordFormDialog.tsx +164 -0
- package/frontend/src/features/database/components/RecordFormField.tsx +568 -0
- package/frontend/src/features/database/components/TableEmptyState.tsx +21 -0
- package/frontend/src/features/database/components/TableForm.tsx +656 -0
- package/frontend/src/features/database/components/TableFormColumn.tsx +137 -0
- package/frontend/src/features/database/components/TableListSkeleton.tsx +9 -0
- package/frontend/src/features/database/components/TableSidebar.tsx +47 -0
- package/frontend/src/features/database/constants.ts +26 -0
- package/frontend/src/features/database/helpers.ts +125 -0
- package/frontend/src/features/database/hooks/UseLinkModal.tsx +78 -0
- package/frontend/src/features/database/index.ts +12 -0
- package/frontend/src/features/database/page/DatabasePage.tsx +626 -0
- package/frontend/src/features/database/schema.ts +25 -0
- package/frontend/src/features/database/services/database.service.ts +216 -0
- package/frontend/src/features/functions/components/FunctionEmptyState.tsx +15 -0
- package/frontend/src/features/functions/components/FunctionRow.tsx +71 -0
- package/frontend/src/features/functions/components/FunctionViewer.tsx +46 -0
- package/frontend/src/features/functions/components/FunctionsContent.tsx +88 -0
- package/frontend/src/features/functions/components/FunctionsSidebar.tsx +56 -0
- package/frontend/src/features/functions/components/SecretEmptyState.tsx +23 -0
- package/frontend/src/features/functions/components/SecretRow.tsx +68 -0
- package/frontend/src/features/functions/components/SecretsContent.tsx +120 -0
- package/frontend/src/features/functions/hooks/useFunctions.ts +106 -0
- package/frontend/src/features/functions/page/FunctionsPage.tsx +28 -0
- package/frontend/src/features/functions/services/functions.service.ts +48 -0
- package/frontend/src/features/login/components/AuthErrorBoundary.tsx +87 -0
- package/frontend/src/features/login/components/PrivateRoute.tsx +24 -0
- package/frontend/src/features/login/page/CloudLoginPage.tsx +93 -0
- package/frontend/src/features/login/page/LoginPage.tsx +174 -0
- package/frontend/src/features/logs/components/AnalyticsLogsTable.tsx +313 -0
- package/frontend/src/features/logs/components/LogsTable.tsx +199 -0
- package/frontend/src/features/logs/hooks/useAuditLogs.ts +39 -0
- package/frontend/src/features/logs/index.ts +5 -0
- package/frontend/src/features/logs/page/AnalyticsLogsPage.tsx +530 -0
- package/frontend/src/features/logs/page/AuditsPage.tsx +192 -0
- package/frontend/src/features/logs/services/log.service.ts +171 -0
- package/frontend/src/features/metadata/hooks/useMetadata.ts +53 -0
- package/frontend/src/features/metadata/index.ts +0 -0
- package/frontend/src/features/metadata/page/MetadataPage.tsx +136 -0
- package/frontend/src/features/metadata/services/metadata.service.ts +17 -0
- package/frontend/src/features/onboard/components/CompletionCard.tsx +41 -0
- package/frontend/src/features/onboard/components/OnboardButton.tsx +84 -0
- package/frontend/src/features/onboard/components/StepContent.tsx +91 -0
- package/frontend/src/features/onboard/components/TestConnectionStep.tsx +53 -0
- package/frontend/src/features/onboard/components/mcp/CursorDeeplinkGenerator.tsx +35 -0
- package/frontend/src/features/onboard/components/mcp/McpInstallation.tsx +144 -0
- package/frontend/src/features/onboard/components/mcp/index.ts +4 -0
- package/frontend/src/features/onboard/components/mcp/mcp-helper.tsx +98 -0
- package/frontend/src/features/onboard/index.ts +3 -0
- package/frontend/src/features/onboard/page/OnBoardPage.tsx +104 -0
- package/frontend/src/features/onboard/types.ts +8 -0
- package/frontend/src/features/secrets/hooks/useSecrets.ts +139 -0
- package/frontend/src/features/secrets/services/secrets.service.ts +57 -0
- package/frontend/src/features/storage/components/BucketEmptyState.tsx +19 -0
- package/frontend/src/features/storage/components/BucketFormDialog.tsx +194 -0
- package/frontend/src/features/storage/components/BucketListSkeleton.tsx +17 -0
- package/frontend/src/features/storage/components/FilePreviewDialog.tsx +287 -0
- package/frontend/src/features/storage/components/StorageDataGrid.tsx +239 -0
- package/frontend/src/features/storage/components/StorageManager.tsx +236 -0
- package/frontend/src/features/storage/components/StorageSidebar.tsx +44 -0
- package/frontend/src/features/storage/components/UploadToast.tsx +46 -0
- package/frontend/src/features/storage/index.ts +3 -0
- package/frontend/src/features/storage/page/StoragePage.tsx +553 -0
- package/frontend/src/features/storage/services/storage.service.ts +144 -0
- package/frontend/src/features/visualizer/components/AuthNode.tsx +107 -0
- package/frontend/src/features/visualizer/components/BucketNode.tsx +34 -0
- package/frontend/src/features/visualizer/components/SchemaVisualizer.tsx +359 -0
- package/frontend/src/features/visualizer/components/TableNode.tsx +152 -0
- package/frontend/src/features/visualizer/components/VisualizerSkeleton.tsx +24 -0
- package/frontend/src/features/visualizer/components/index.ts +5 -0
- package/frontend/src/features/visualizer/page/VisualizerPage.tsx +127 -0
- package/frontend/src/index.css +248 -0
- package/frontend/src/lib/api/client.ts +163 -0
- package/frontend/src/lib/contexts/AuthContext.tsx +157 -0
- package/frontend/src/lib/contexts/OnboardStepContext.tsx +68 -0
- package/frontend/src/lib/contexts/SocketContext.tsx +303 -0
- package/frontend/src/lib/contexts/ThemeContext.tsx +125 -0
- package/frontend/src/lib/hooks/useAuth.ts +4 -0
- package/frontend/src/lib/hooks/useConfirm.ts +55 -0
- package/frontend/src/lib/hooks/useInterval.ts +27 -0
- package/frontend/src/lib/hooks/useMediaQuery.ts +59 -0
- package/frontend/src/lib/hooks/useOnboardingCompletion.ts +29 -0
- package/frontend/src/lib/hooks/usePagination.ts +27 -0
- package/frontend/src/lib/hooks/useTimeout.ts +27 -0
- package/frontend/src/lib/hooks/useToast.tsx +229 -0
- package/frontend/src/lib/utils/constants.ts +38 -0
- package/frontend/src/lib/utils/utils.ts +165 -0
- package/frontend/src/lib/utils/validation-schemas.ts +126 -0
- package/frontend/src/main.tsx +16 -0
- package/frontend/src/rdg.css +194 -0
- package/frontend/src/vite-env.d.ts +12 -0
- package/frontend/tailwind.config.js +97 -0
- package/frontend/tsconfig.json +26 -0
- package/frontend/tsconfig.node.json +10 -0
- package/frontend/vite.config.ts +37 -0
- package/frontend/vitest.config.ts +36 -0
- package/functions/deno.json +25 -0
- package/functions/server.ts +290 -0
- package/functions/worker-template.js +126 -0
- package/openapi/ai.yaml +689 -0
- package/openapi/auth.yaml +563 -0
- package/openapi/functions.yaml +476 -0
- package/openapi/health.yaml +30 -0
- package/openapi/logs.yaml +224 -0
- package/openapi/metadata.yaml +178 -0
- package/openapi/records.yaml +382 -0
- package/openapi/secrets.yaml +371 -0
- package/openapi/storage.yaml +876 -0
- package/openapi/tables.yaml +464 -0
- package/package.json +88 -0
- package/shared-schemas/package.json +31 -0
- package/shared-schemas/src/ai-api.schema.ts +167 -0
- package/shared-schemas/src/ai.schema.ts +54 -0
- package/shared-schemas/src/auth-api.schema.ts +193 -0
- package/shared-schemas/src/auth.schema.ts +94 -0
- package/shared-schemas/src/database-api.schema.ts +259 -0
- package/shared-schemas/src/database.schema.ts +69 -0
- package/shared-schemas/src/functions-api.schema.ts +25 -0
- package/shared-schemas/src/functions.schema.ts +16 -0
- package/shared-schemas/src/index.ts +13 -0
- package/shared-schemas/src/logs-api.schema.ts +49 -0
- package/shared-schemas/src/logs.schema.ts +14 -0
- package/shared-schemas/src/metadata.schema.ts +56 -0
- package/shared-schemas/src/storage-api.schema.ts +65 -0
- package/shared-schemas/src/storage.schema.ts +19 -0
- package/shared-schemas/tsconfig.json +21 -0
- package/tsconfig.json +8 -0
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { X } from 'lucide-react';
|
|
3
|
+
import {
|
|
4
|
+
AlertDialog,
|
|
5
|
+
AlertDialogAction,
|
|
6
|
+
AlertDialogCancel,
|
|
7
|
+
AlertDialogContent,
|
|
8
|
+
AlertDialogDescription,
|
|
9
|
+
AlertDialogFooter,
|
|
10
|
+
AlertDialogTitle,
|
|
11
|
+
} from '@/components';
|
|
12
|
+
|
|
13
|
+
interface ConfirmDialogProps {
|
|
14
|
+
open: boolean;
|
|
15
|
+
onOpenChange: (open: boolean) => void;
|
|
16
|
+
title: string;
|
|
17
|
+
description: string | React.ReactNode;
|
|
18
|
+
confirmText?: string;
|
|
19
|
+
cancelText?: string;
|
|
20
|
+
onConfirm: () => void | Promise<void>;
|
|
21
|
+
destructive?: boolean;
|
|
22
|
+
isLoading?: boolean;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export function ConfirmDialog({
|
|
26
|
+
open,
|
|
27
|
+
onOpenChange,
|
|
28
|
+
title,
|
|
29
|
+
description,
|
|
30
|
+
confirmText = 'Confirm',
|
|
31
|
+
cancelText = 'Cancel',
|
|
32
|
+
onConfirm,
|
|
33
|
+
destructive = false,
|
|
34
|
+
isLoading = false,
|
|
35
|
+
}: ConfirmDialogProps) {
|
|
36
|
+
const handleConfirm = async () => {
|
|
37
|
+
await onConfirm();
|
|
38
|
+
onOpenChange(false);
|
|
39
|
+
};
|
|
40
|
+
|
|
41
|
+
return (
|
|
42
|
+
<AlertDialog open={open} onOpenChange={onOpenChange}>
|
|
43
|
+
<AlertDialogContent className="max-w-[480px] p-0 gap-0 bg-white border border-zinc-200 shadow-[0px_1px_3px_0px_rgba(0,0,0,0.1)] dark:bg-neutral-800 dark:border-neutral-700 dark:text-white">
|
|
44
|
+
{/* Header with close button */}
|
|
45
|
+
<div className="flex items-center justify-between px-6 py-3 border-b border-zinc-200 dark:border-neutral-700">
|
|
46
|
+
<AlertDialogTitle className="text-lg font-semibold text-zinc-950 dark:text-white">
|
|
47
|
+
{title}
|
|
48
|
+
</AlertDialogTitle>
|
|
49
|
+
<button
|
|
50
|
+
className="absolute right-4 top-4 rounded-sm text-zinc-500 transition-colors hover:text-zinc-700 focus:outline-none disabled:pointer-events-none dark:text-zinc-400 dark:hover:text-zinc-300"
|
|
51
|
+
onClick={() => onOpenChange(false)}
|
|
52
|
+
>
|
|
53
|
+
<X className="h-5 w-5" />
|
|
54
|
+
<span className="sr-only">Close</span>
|
|
55
|
+
</button>
|
|
56
|
+
</div>
|
|
57
|
+
|
|
58
|
+
{/* Description */}
|
|
59
|
+
<div className="p-6">
|
|
60
|
+
{typeof description === 'string' ? (
|
|
61
|
+
<AlertDialogDescription className="text-zinc-500 text-sm leading-5 dark:text-neutral-400">
|
|
62
|
+
{description}
|
|
63
|
+
</AlertDialogDescription>
|
|
64
|
+
) : (
|
|
65
|
+
<div className="text-zinc-500 text-sm leading-5 dark:text-neutral-400">
|
|
66
|
+
{description}
|
|
67
|
+
</div>
|
|
68
|
+
)}
|
|
69
|
+
</div>
|
|
70
|
+
|
|
71
|
+
{/* Footer */}
|
|
72
|
+
<AlertDialogFooter className="p-6 flex-row justify-end gap-3 border-t border-zinc-200 dark:border-neutral-700">
|
|
73
|
+
<AlertDialogCancel
|
|
74
|
+
disabled={isLoading}
|
|
75
|
+
className="h-9 w-30 px-3 py-2 text-sm font-medium bg-white border border-zinc-200 shadow-[0px_1px_2px_0px_rgba(0,0,0,0.1)] rounded hover:bg-zinc-50 dark:bg-neutral-600 dark:border-neutral-600 dark:text-white dark:hover:bg-neutral-700"
|
|
76
|
+
>
|
|
77
|
+
{cancelText}
|
|
78
|
+
</AlertDialogCancel>
|
|
79
|
+
<AlertDialogAction
|
|
80
|
+
onClick={() => {
|
|
81
|
+
void handleConfirm();
|
|
82
|
+
}}
|
|
83
|
+
disabled={isLoading}
|
|
84
|
+
className={`h-9 w-30 px-3 py-2 text-sm font-medium rounded shadow-[0px_1px_2px_0px_rgba(0,0,0,0.1)] dark:shadow-[0px_1px_2px_0px_rgba(0,0,0,0.1)] ${
|
|
85
|
+
destructive
|
|
86
|
+
? 'bg-red-600 hover:bg-red-700 text-white dark:bg-red-200 dark:hover:bg-red-300 dark:text-red-700'
|
|
87
|
+
: 'bg-zinc-950 hover:bg-zinc-900 text-white dark:bg-emerald-300 dark:text-black dark:hover:bg-emerald-400'
|
|
88
|
+
}`}
|
|
89
|
+
>
|
|
90
|
+
{isLoading ? 'Processing...' : confirmText}
|
|
91
|
+
</AlertDialogAction>
|
|
92
|
+
</AlertDialogFooter>
|
|
93
|
+
</AlertDialogContent>
|
|
94
|
+
</AlertDialog>
|
|
95
|
+
);
|
|
96
|
+
}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { useState } from 'react';
|
|
2
|
+
import { Copy } from 'lucide-react';
|
|
3
|
+
import { Button } from '@/components/radix/Button';
|
|
4
|
+
import { cn } from '@/lib/utils/utils';
|
|
5
|
+
import CheckedIcon from '@/assets/icons/checked.svg?react';
|
|
6
|
+
|
|
7
|
+
interface CopyButtonProps {
|
|
8
|
+
text: string;
|
|
9
|
+
onCopy?: (text: string) => void;
|
|
10
|
+
className?: string;
|
|
11
|
+
variant?: 'ghost' | 'outline' | 'default';
|
|
12
|
+
size?: 'sm' | 'default' | 'lg' | 'icon';
|
|
13
|
+
showText?: boolean;
|
|
14
|
+
copiedText?: string;
|
|
15
|
+
copyText?: string;
|
|
16
|
+
disabled?: boolean;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function CopyButton({
|
|
20
|
+
text,
|
|
21
|
+
onCopy,
|
|
22
|
+
className,
|
|
23
|
+
variant = 'ghost',
|
|
24
|
+
size = 'sm',
|
|
25
|
+
showText = true,
|
|
26
|
+
copiedText = 'Copied',
|
|
27
|
+
copyText = 'Copy',
|
|
28
|
+
disabled = false,
|
|
29
|
+
}: CopyButtonProps) {
|
|
30
|
+
const [copied, setCopied] = useState(false);
|
|
31
|
+
|
|
32
|
+
const handleCopy = async (e: React.MouseEvent<HTMLButtonElement>) => {
|
|
33
|
+
e.stopPropagation();
|
|
34
|
+
if (disabled) {
|
|
35
|
+
return;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
try {
|
|
39
|
+
await navigator.clipboard.writeText(text);
|
|
40
|
+
setCopied(true);
|
|
41
|
+
setTimeout(() => setCopied(false), 2000);
|
|
42
|
+
|
|
43
|
+
if (onCopy) {
|
|
44
|
+
onCopy(text);
|
|
45
|
+
}
|
|
46
|
+
} catch (error) {
|
|
47
|
+
// Failed to copy text
|
|
48
|
+
console.error(error);
|
|
49
|
+
}
|
|
50
|
+
};
|
|
51
|
+
|
|
52
|
+
return (
|
|
53
|
+
<Button
|
|
54
|
+
variant={variant}
|
|
55
|
+
size={size}
|
|
56
|
+
onClick={(e) => void handleCopy(e)}
|
|
57
|
+
disabled={disabled}
|
|
58
|
+
data-copied={copied}
|
|
59
|
+
className={cn(
|
|
60
|
+
'px-3 w-fit h-8 rounded-md bg-zinc-50 dark:bg-neutral-700 hover:bg-bg-gray-hover dark:hover:bg-neutral-700 border-border-gray dark:border-neutral-700 border text-zinc-950 dark:text-white shadow gap-1.5 transition-all duration-200',
|
|
61
|
+
'data-[copied=true]:bg-transparent data-[copied=true]:cursor-default data-[copied=true]:shadow-none data-[copied=true]:border-none data-[copied=true]:hover:bg-transparent',
|
|
62
|
+
className
|
|
63
|
+
)}
|
|
64
|
+
>
|
|
65
|
+
{copied ? <CheckedIcon className="w-4 h-4" /> : <Copy className="w-4 h-4 " />}
|
|
66
|
+
{showText && <span className="font-medium text-sm">{copied ? copiedText : copyText}</span>}
|
|
67
|
+
</Button>
|
|
68
|
+
);
|
|
69
|
+
}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { Button } from '@/components/radix/Button';
|
|
2
|
+
import { useTheme } from '@/lib/contexts/ThemeContext';
|
|
3
|
+
|
|
4
|
+
interface DeleteActionButtonProps {
|
|
5
|
+
selectedCount: number;
|
|
6
|
+
itemType: string;
|
|
7
|
+
onDelete: () => void;
|
|
8
|
+
className?: string;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export function DeleteActionButton({
|
|
12
|
+
selectedCount,
|
|
13
|
+
itemType,
|
|
14
|
+
onDelete,
|
|
15
|
+
className = '',
|
|
16
|
+
}: DeleteActionButtonProps) {
|
|
17
|
+
const { resolvedTheme } = useTheme();
|
|
18
|
+
|
|
19
|
+
const getItemLabel = (count: number, type: string) => {
|
|
20
|
+
const singular = type.charAt(0).toUpperCase() + type.slice(1);
|
|
21
|
+
const plural =
|
|
22
|
+
type === 'user'
|
|
23
|
+
? 'Users'
|
|
24
|
+
: type === 'record'
|
|
25
|
+
? 'Records'
|
|
26
|
+
: type === 'file'
|
|
27
|
+
? 'Files'
|
|
28
|
+
: `${singular}s`;
|
|
29
|
+
|
|
30
|
+
return count === 1 ? singular : plural;
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
return (
|
|
34
|
+
<Button
|
|
35
|
+
variant={resolvedTheme === 'light' ? 'outline' : 'default'}
|
|
36
|
+
className={`h-10 px-3 text-sm text-red-600 hover:text-red-400 hover:bg-zinc-50 border border-border-gray shadow-0 dark:bg-red-200 dark:text-red-600 dark:border-transparent dark:hover:bg-red-300 ${className}`}
|
|
37
|
+
onClick={onDelete}
|
|
38
|
+
>
|
|
39
|
+
Delete {selectedCount} {getItemLabel(selectedCount, itemType)}
|
|
40
|
+
</Button>
|
|
41
|
+
);
|
|
42
|
+
}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { LucideIcon } from 'lucide-react';
|
|
2
|
+
import { Button } from '@/components/radix/Button';
|
|
3
|
+
import { cn } from '@/lib/utils/utils';
|
|
4
|
+
|
|
5
|
+
interface EmptyStateProps {
|
|
6
|
+
icon?: LucideIcon;
|
|
7
|
+
image?: string;
|
|
8
|
+
title: string;
|
|
9
|
+
description?: string;
|
|
10
|
+
action?: {
|
|
11
|
+
label: string;
|
|
12
|
+
onClick: () => void;
|
|
13
|
+
};
|
|
14
|
+
className?: string;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export function EmptyState({
|
|
18
|
+
icon: Icon,
|
|
19
|
+
image,
|
|
20
|
+
title,
|
|
21
|
+
description,
|
|
22
|
+
action,
|
|
23
|
+
className,
|
|
24
|
+
}: EmptyStateProps) {
|
|
25
|
+
return (
|
|
26
|
+
<div
|
|
27
|
+
className={cn(
|
|
28
|
+
'text-center flex flex-col items-center justify-center text-zinc-500',
|
|
29
|
+
className
|
|
30
|
+
)}
|
|
31
|
+
>
|
|
32
|
+
{Icon && <Icon className="mx-auto h-50 w-50 text-muted-foreground" />}
|
|
33
|
+
{image && !Icon && (
|
|
34
|
+
<img src={image} alt={title} className="mx-auto h-50 w-50 object-contain" />
|
|
35
|
+
)}
|
|
36
|
+
<h3 className="text-sm font-medium">{title}</h3>
|
|
37
|
+
{description && <p className="text-xs max-w-sm">{description}</p>}
|
|
38
|
+
{action && <Button onClick={action.onClick}>{action.label}</Button>}
|
|
39
|
+
</div>
|
|
40
|
+
);
|
|
41
|
+
}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { AlertCircle } from 'lucide-react';
|
|
2
|
+
import { Button } from '@/components/radix/Button';
|
|
3
|
+
import { Alert, AlertDescription, AlertTitle } from '@/components/radix/Alert';
|
|
4
|
+
import { cn } from '@/lib/utils/utils';
|
|
5
|
+
|
|
6
|
+
interface ErrorStateProps {
|
|
7
|
+
error: Error | string;
|
|
8
|
+
title?: string;
|
|
9
|
+
onRetry?: () => void;
|
|
10
|
+
className?: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export function ErrorState({
|
|
14
|
+
error,
|
|
15
|
+
title = 'Something went wrong',
|
|
16
|
+
onRetry,
|
|
17
|
+
className,
|
|
18
|
+
}: ErrorStateProps) {
|
|
19
|
+
const errorMessage = error instanceof Error ? error.message : error;
|
|
20
|
+
|
|
21
|
+
return (
|
|
22
|
+
<Alert variant="destructive" className={cn('', className)}>
|
|
23
|
+
<AlertCircle className="h-4 w-4" />
|
|
24
|
+
<AlertTitle>{title}</AlertTitle>
|
|
25
|
+
<AlertDescription className="mt-2">
|
|
26
|
+
<p>{errorMessage}</p>
|
|
27
|
+
{onRetry && (
|
|
28
|
+
<Button variant="outline" size="sm" onClick={onRetry} className="mt-4">
|
|
29
|
+
Try again
|
|
30
|
+
</Button>
|
|
31
|
+
)}
|
|
32
|
+
</AlertDescription>
|
|
33
|
+
</Alert>
|
|
34
|
+
);
|
|
35
|
+
}
|
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
import { useState, ReactNode } from 'react';
|
|
2
|
+
import { Plus, LucideIcon } from 'lucide-react';
|
|
3
|
+
import { ScrollArea } from '@/components/radix/ScrollArea';
|
|
4
|
+
import { Button } from '@/components/radix/Button';
|
|
5
|
+
import { SearchInput } from '@/components/SearchInput';
|
|
6
|
+
import { FeatureSidebarItem } from '@/components/FeatureSidebarItem';
|
|
7
|
+
import {
|
|
8
|
+
Tooltip,
|
|
9
|
+
TooltipContent,
|
|
10
|
+
TooltipProvider,
|
|
11
|
+
TooltipTrigger,
|
|
12
|
+
} from '@/components/radix/Tooltip';
|
|
13
|
+
|
|
14
|
+
interface FeatureSidebarProps {
|
|
15
|
+
title: string;
|
|
16
|
+
items: string[];
|
|
17
|
+
selectedItem?: string;
|
|
18
|
+
onItemSelect: (itemName: string) => void;
|
|
19
|
+
loading?: boolean;
|
|
20
|
+
onNewItem?: () => void;
|
|
21
|
+
onEditItem?: (itemName: string) => void;
|
|
22
|
+
onDeleteItem?: (itemName: string) => void;
|
|
23
|
+
searchPlaceholder?: string;
|
|
24
|
+
newItemTooltip?: string;
|
|
25
|
+
editLabel?: string;
|
|
26
|
+
deleteLabel?: string;
|
|
27
|
+
icon: LucideIcon;
|
|
28
|
+
filterItems?: (itemNames: string[]) => string[];
|
|
29
|
+
renderSkeleton: () => ReactNode;
|
|
30
|
+
renderEmptyState: (searchTerm: string) => ReactNode;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export function FeatureSidebar({
|
|
34
|
+
title,
|
|
35
|
+
items,
|
|
36
|
+
selectedItem,
|
|
37
|
+
onItemSelect,
|
|
38
|
+
loading,
|
|
39
|
+
onNewItem,
|
|
40
|
+
onEditItem,
|
|
41
|
+
onDeleteItem,
|
|
42
|
+
searchPlaceholder = 'Search...',
|
|
43
|
+
newItemTooltip = 'Add new',
|
|
44
|
+
editLabel = 'Edit',
|
|
45
|
+
deleteLabel = 'Delete',
|
|
46
|
+
icon,
|
|
47
|
+
filterItems,
|
|
48
|
+
renderSkeleton,
|
|
49
|
+
renderEmptyState,
|
|
50
|
+
}: FeatureSidebarProps) {
|
|
51
|
+
const [searchTerm, setSearchTerm] = useState('');
|
|
52
|
+
|
|
53
|
+
let itemNames = items;
|
|
54
|
+
if (filterItems) {
|
|
55
|
+
itemNames = filterItems(itemNames);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
const normalizedSearch = searchTerm.toLowerCase().replace(/\s+/g, '');
|
|
59
|
+
const filteredItems = itemNames.filter((name) =>
|
|
60
|
+
name.toLowerCase().replace(/\s+/g, '').includes(normalizedSearch)
|
|
61
|
+
);
|
|
62
|
+
|
|
63
|
+
return (
|
|
64
|
+
<div className="w-70 flex flex-col h-full bg-white dark:bg-neutral-800 border-r border-border-gray dark:border-neutral-700">
|
|
65
|
+
{/* Header */}
|
|
66
|
+
<div className="flex flex-row justify-between items-center pl-4 pr-1.5 py-1.5 bg-white dark:bg-neutral-800">
|
|
67
|
+
<h2 className="text-base font-bold text-black dark:text-white">{title}</h2>
|
|
68
|
+
{onNewItem && (
|
|
69
|
+
<TooltipProvider>
|
|
70
|
+
<Tooltip>
|
|
71
|
+
<TooltipTrigger asChild>
|
|
72
|
+
<Button
|
|
73
|
+
onClick={onNewItem}
|
|
74
|
+
variant="ghost"
|
|
75
|
+
size="icon"
|
|
76
|
+
className="h-9 w-9 dark:bg-transparent dark:text-neutral-400 dark:hover:bg-neutral-700"
|
|
77
|
+
>
|
|
78
|
+
<Plus className="w-5 h-5" />
|
|
79
|
+
</Button>
|
|
80
|
+
</TooltipTrigger>
|
|
81
|
+
<TooltipContent>
|
|
82
|
+
<p>{newItemTooltip}</p>
|
|
83
|
+
</TooltipContent>
|
|
84
|
+
</Tooltip>
|
|
85
|
+
</TooltipProvider>
|
|
86
|
+
)}
|
|
87
|
+
</div>
|
|
88
|
+
|
|
89
|
+
{/* Add Button and Search */}
|
|
90
|
+
<div className="py-2 px-3 bg-white dark:bg-neutral-800">
|
|
91
|
+
<SearchInput
|
|
92
|
+
value={searchTerm}
|
|
93
|
+
onChange={setSearchTerm}
|
|
94
|
+
placeholder={searchPlaceholder}
|
|
95
|
+
className="w-full dark:text-white"
|
|
96
|
+
debounceTime={200}
|
|
97
|
+
/>
|
|
98
|
+
</div>
|
|
99
|
+
|
|
100
|
+
{/* Item List */}
|
|
101
|
+
<ScrollArea className="flex-1 px-3 pb-3 dark:bg-neutral-800">
|
|
102
|
+
{loading ? (
|
|
103
|
+
renderSkeleton()
|
|
104
|
+
) : filteredItems.length === 0 ? (
|
|
105
|
+
renderEmptyState(searchTerm)
|
|
106
|
+
) : (
|
|
107
|
+
<div className="space-y-1 dark:text-zinc-300">
|
|
108
|
+
{filteredItems.map((itemName) => (
|
|
109
|
+
<FeatureSidebarItem
|
|
110
|
+
key={itemName}
|
|
111
|
+
name={itemName}
|
|
112
|
+
icon={icon}
|
|
113
|
+
isSelected={selectedItem === itemName}
|
|
114
|
+
onClick={() => onItemSelect(itemName)}
|
|
115
|
+
onEdit={onEditItem ? () => onEditItem(itemName) : undefined}
|
|
116
|
+
onDelete={onDeleteItem ? () => onDeleteItem(itemName) : undefined}
|
|
117
|
+
editLabel={editLabel}
|
|
118
|
+
deleteLabel={deleteLabel}
|
|
119
|
+
/>
|
|
120
|
+
))}
|
|
121
|
+
</div>
|
|
122
|
+
)}
|
|
123
|
+
</ScrollArea>
|
|
124
|
+
</div>
|
|
125
|
+
);
|
|
126
|
+
}
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
import { useState } from 'react';
|
|
2
|
+
import { Pencil, Trash2, MoreVertical, LucideIcon } from 'lucide-react';
|
|
3
|
+
import { Button } from '@/components/radix/Button';
|
|
4
|
+
import {
|
|
5
|
+
DropdownMenu,
|
|
6
|
+
DropdownMenuContent,
|
|
7
|
+
DropdownMenuItem,
|
|
8
|
+
DropdownMenuTrigger,
|
|
9
|
+
} from '@/components/radix/DropdownMenu';
|
|
10
|
+
import { cn } from '@/lib/utils/utils';
|
|
11
|
+
|
|
12
|
+
interface FeatureSidebarItemProps {
|
|
13
|
+
name: string;
|
|
14
|
+
icon: LucideIcon;
|
|
15
|
+
isSelected: boolean;
|
|
16
|
+
onClick: () => void;
|
|
17
|
+
onEdit?: () => void;
|
|
18
|
+
onDelete?: () => void;
|
|
19
|
+
editLabel?: string;
|
|
20
|
+
deleteLabel?: string;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export function FeatureSidebarItem({
|
|
24
|
+
name,
|
|
25
|
+
icon: Icon,
|
|
26
|
+
isSelected,
|
|
27
|
+
onClick,
|
|
28
|
+
onEdit,
|
|
29
|
+
onDelete,
|
|
30
|
+
editLabel = 'Edit',
|
|
31
|
+
deleteLabel = 'Delete',
|
|
32
|
+
}: FeatureSidebarItemProps) {
|
|
33
|
+
const [dropdownOpen, setDropdownOpen] = useState(false);
|
|
34
|
+
|
|
35
|
+
return (
|
|
36
|
+
<div className="relative h-12 group">
|
|
37
|
+
<Button
|
|
38
|
+
variant="ghost"
|
|
39
|
+
className={cn(
|
|
40
|
+
'w-full h-full flex items-center justify-start px-[14px] py-[14px]',
|
|
41
|
+
!isSelected && 'hover:bg-[#F8FAFC] dark:hover:bg-neutral-700',
|
|
42
|
+
isSelected && 'bg-[#F1F5F9] dark:bg-neutral-600'
|
|
43
|
+
)}
|
|
44
|
+
onClick={onClick}
|
|
45
|
+
>
|
|
46
|
+
<Icon className="mr-2 h-4 w-4" />
|
|
47
|
+
<span
|
|
48
|
+
title={name}
|
|
49
|
+
className={cn(
|
|
50
|
+
'flex-1 min-w-0 truncate text-left',
|
|
51
|
+
isSelected ? 'font-bold dark:text-zinc-300' : 'font-medium dark:text-zinc-300'
|
|
52
|
+
)}
|
|
53
|
+
>
|
|
54
|
+
{name}
|
|
55
|
+
</span>
|
|
56
|
+
{(onEdit || onDelete) && (
|
|
57
|
+
<div className="ml-2 flex items-center">
|
|
58
|
+
<DropdownMenu open={dropdownOpen} onOpenChange={setDropdownOpen}>
|
|
59
|
+
<DropdownMenuTrigger asChild onClick={(e) => e.stopPropagation()}>
|
|
60
|
+
<div
|
|
61
|
+
className={cn(
|
|
62
|
+
'h-6 w-6 p-0 flex items-center justify-center rounded cursor-pointer opacity-100',
|
|
63
|
+
`bg-transparent hover:bg-[#E2E8F0] dark:hover:bg-neutral-700`
|
|
64
|
+
)}
|
|
65
|
+
>
|
|
66
|
+
<MoreVertical className="h-5 w-5 text-[#71717A] dark:text-zinc-300" />
|
|
67
|
+
</div>
|
|
68
|
+
</DropdownMenuTrigger>
|
|
69
|
+
<DropdownMenuContent align="start" className="w-48">
|
|
70
|
+
{onEdit && (
|
|
71
|
+
<DropdownMenuItem
|
|
72
|
+
onClick={(e) => {
|
|
73
|
+
e.stopPropagation();
|
|
74
|
+
onEdit();
|
|
75
|
+
}}
|
|
76
|
+
className="cursor-pointer"
|
|
77
|
+
>
|
|
78
|
+
<Pencil className="mr-2 h-5 w-5 text-[#71717A] dark:text-zinc-300" />
|
|
79
|
+
{editLabel}
|
|
80
|
+
</DropdownMenuItem>
|
|
81
|
+
)}
|
|
82
|
+
{onDelete && (
|
|
83
|
+
<DropdownMenuItem
|
|
84
|
+
onClick={(e) => {
|
|
85
|
+
e.stopPropagation();
|
|
86
|
+
onDelete();
|
|
87
|
+
}}
|
|
88
|
+
className="text-red-600 focus:text-red-600 dark:text-red-400 dark:focus:text-red-400 cursor-pointer"
|
|
89
|
+
>
|
|
90
|
+
<Trash2 className="mr-2 h-5 w-5 dark:text-red-400" />
|
|
91
|
+
{deleteLabel}
|
|
92
|
+
</DropdownMenuItem>
|
|
93
|
+
)}
|
|
94
|
+
</DropdownMenuContent>
|
|
95
|
+
</DropdownMenu>
|
|
96
|
+
</div>
|
|
97
|
+
)}
|
|
98
|
+
</Button>
|
|
99
|
+
</div>
|
|
100
|
+
);
|
|
101
|
+
}
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { CopyButton } from './CopyButton';
|
|
2
|
+
|
|
3
|
+
interface JsonHighlightProps {
|
|
4
|
+
json: string;
|
|
5
|
+
}
|
|
6
|
+
|
|
7
|
+
export function JsonHighlight({ json }: JsonHighlightProps) {
|
|
8
|
+
const highlightJson = (str: string) => {
|
|
9
|
+
// Tokenize JSON string
|
|
10
|
+
const tokens = str.split(/("(?:[^"\\]|\\.)*")|(\s+)|([:,{}[\]])/g).filter(Boolean);
|
|
11
|
+
|
|
12
|
+
return tokens.map((token, index) => {
|
|
13
|
+
// String values (including keys)
|
|
14
|
+
if (token.startsWith('"') && token.endsWith('"')) {
|
|
15
|
+
// Check if this is a key (next non-whitespace token is ':')
|
|
16
|
+
let isKey = false;
|
|
17
|
+
for (let i = index + 1; i < tokens.length; i++) {
|
|
18
|
+
if (tokens[i].trim()) {
|
|
19
|
+
isKey = tokens[i] === ':';
|
|
20
|
+
break;
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
if (isKey) {
|
|
25
|
+
return (
|
|
26
|
+
<span key={index} className="text-blue-600 dark:text-blue-400">
|
|
27
|
+
{token}
|
|
28
|
+
</span>
|
|
29
|
+
);
|
|
30
|
+
} else {
|
|
31
|
+
return (
|
|
32
|
+
<span key={index} className="text-green-600 dark:text-green-400">
|
|
33
|
+
{token}
|
|
34
|
+
</span>
|
|
35
|
+
);
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
// Punctuation
|
|
40
|
+
if (/^[:,{}[\]]$/.test(token)) {
|
|
41
|
+
return (
|
|
42
|
+
<span key={index} className="text-gray-600 dark:text-gray-400">
|
|
43
|
+
{token}
|
|
44
|
+
</span>
|
|
45
|
+
);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
// Whitespace and other
|
|
49
|
+
return <span key={index}>{token}</span>;
|
|
50
|
+
});
|
|
51
|
+
};
|
|
52
|
+
|
|
53
|
+
return (
|
|
54
|
+
<div className="relative">
|
|
55
|
+
<pre className="font-mono text-sm leading-6 whitespace-pre overflow-x-auto bg-gray-50 dark:bg-neutral-900 dark:text-white rounded-md py-4 px-6 pr-16">
|
|
56
|
+
{highlightJson(json)}
|
|
57
|
+
</pre>
|
|
58
|
+
<CopyButton text={json} className="absolute top-4 right-4" />
|
|
59
|
+
</div>
|
|
60
|
+
);
|
|
61
|
+
}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { Loader2 } from 'lucide-react';
|
|
2
|
+
import { cn } from '@/lib/utils/utils';
|
|
3
|
+
|
|
4
|
+
interface LoadingStateProps {
|
|
5
|
+
message?: string;
|
|
6
|
+
className?: string;
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export function LoadingState({ message = 'Loading...', className }: LoadingStateProps) {
|
|
10
|
+
return (
|
|
11
|
+
<div className={cn('flex flex-col items-center justify-center py-12', className)}>
|
|
12
|
+
<Loader2 className="h-8 w-8 animate-spin text-muted-foreground mb-4" />
|
|
13
|
+
<p className="text-sm text-muted-foreground">{message}</p>
|
|
14
|
+
</div>
|
|
15
|
+
);
|
|
16
|
+
}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import { Button } from '@/components/radix/Button';
|
|
2
|
+
import { ChevronLeft, ChevronRight } from 'lucide-react';
|
|
3
|
+
|
|
4
|
+
export interface PaginationControlsProps {
|
|
5
|
+
currentPage?: number;
|
|
6
|
+
totalPages?: number;
|
|
7
|
+
onPageChange?: (page: number) => void;
|
|
8
|
+
totalRecords?: number;
|
|
9
|
+
pageSize?: number;
|
|
10
|
+
recordLabel?: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export function PaginationControls({
|
|
14
|
+
currentPage = 1,
|
|
15
|
+
totalPages = 1,
|
|
16
|
+
onPageChange,
|
|
17
|
+
totalRecords = 0,
|
|
18
|
+
pageSize = 50,
|
|
19
|
+
recordLabel = 'results',
|
|
20
|
+
}: PaginationControlsProps) {
|
|
21
|
+
const startRecord = totalRecords === 0 ? 0 : (currentPage - 1) * pageSize + 1;
|
|
22
|
+
const endRecord = Math.min(currentPage * pageSize, totalRecords);
|
|
23
|
+
|
|
24
|
+
return (
|
|
25
|
+
<div className="flex items-center justify-between px-4 py-3 border-t border-gray-200 bg-gray-50 dark:bg-neutral-800 dark:border-neutral-700">
|
|
26
|
+
<div className="text-sm text-zinc-500 dark:text-zinc-400">
|
|
27
|
+
Showing {startRecord} to {endRecord} of {totalRecords} {recordLabel}
|
|
28
|
+
</div>
|
|
29
|
+
<div className="flex items-center gap-2">
|
|
30
|
+
<Button
|
|
31
|
+
variant="outline"
|
|
32
|
+
size="sm"
|
|
33
|
+
onClick={() => onPageChange?.(currentPage - 1)}
|
|
34
|
+
disabled={currentPage <= 1}
|
|
35
|
+
className="dark:bg-neutral-800 dark:border-neutral-700 dark:text-white"
|
|
36
|
+
>
|
|
37
|
+
<ChevronLeft className="h-4 w-4" />
|
|
38
|
+
</Button>
|
|
39
|
+
<span className="text-sm text-gray-700 dark:text-zinc-300">
|
|
40
|
+
Page {totalRecords === 0 ? 0 : currentPage} of {totalPages}
|
|
41
|
+
</span>
|
|
42
|
+
<Button
|
|
43
|
+
variant="outline"
|
|
44
|
+
size="sm"
|
|
45
|
+
onClick={() => onPageChange?.(currentPage + 1)}
|
|
46
|
+
disabled={currentPage >= totalPages}
|
|
47
|
+
className="dark:bg-neutral-800 dark:border-neutral-700 dark:text-white"
|
|
48
|
+
>
|
|
49
|
+
<ChevronRight className="h-4 w-4" />
|
|
50
|
+
</Button>
|
|
51
|
+
</div>
|
|
52
|
+
</div>
|
|
53
|
+
);
|
|
54
|
+
}
|