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.
Files changed (395) hide show
  1. package/.dockerignore +58 -0
  2. package/.env.example +49 -0
  3. package/.github/ISSUE_TEMPLATE/bug_report.yml +83 -0
  4. package/.github/ISSUE_TEMPLATE/config.yml +11 -0
  5. package/.github/ISSUE_TEMPLATE/feature_request.yml +79 -0
  6. package/.github/copilot-instructions.md +147 -0
  7. package/.github/workflows/build-image.yml +65 -0
  8. package/.github/workflows/ci-premerge-check.yml +24 -0
  9. package/.github/workflows/deploy-aws.yml +130 -0
  10. package/.github/workflows/lint-and-format.yml +33 -0
  11. package/.prettierignore +65 -0
  12. package/.prettierrc +9 -0
  13. package/CHANGELOG.md +3 -0
  14. package/CONTRIBUTING.md +126 -0
  15. package/Dockerfile +27 -0
  16. package/GITHUB_OAUTH_SETUP.md +49 -0
  17. package/GOOGLE_OAUTH_SETUP.md +148 -0
  18. package/LICENSE +201 -0
  19. package/README.md +134 -0
  20. package/assets/Dark.svg +23 -0
  21. package/assets/archDiagram.png +0 -0
  22. package/assets/banner.png +0 -0
  23. package/assets/mcpInstallv2.png +0 -0
  24. package/assets/sampleResponse.png +0 -0
  25. package/assets/signin.png +0 -0
  26. package/assets/userflow.png +0 -0
  27. package/backend/migrations/000_create-base-tables.sql +142 -0
  28. package/backend/migrations/001_create-helper-functions.sql +41 -0
  29. package/backend/migrations/002_rename-auth-tables.sql +30 -0
  30. package/backend/migrations/003_create-users-table.sql +56 -0
  31. package/backend/migrations/004_add-reload-postgrest-func.sql +24 -0
  32. package/backend/migrations/005_enable-project-admin-modify-users.sql +30 -0
  33. package/backend/migrations/006_modify-ai-usage-table.sql +25 -0
  34. package/backend/migrations/007_drop-metadata-table.sql +2 -0
  35. package/backend/migrations/008_add-system-tables.sql +77 -0
  36. package/backend/migrations/009_add-function-secrets.sql +24 -0
  37. package/backend/migrations/010_modify-ai-config-modalities.sql +93 -0
  38. package/backend/migrations/011_refactor-secrets-table.sql +15 -0
  39. package/backend/migrations/012_add-storage-uploaded-by.sql +8 -0
  40. package/backend/package.json +75 -0
  41. package/backend/src/api/middleware/auth.ts +240 -0
  42. package/backend/src/api/middleware/error.ts +231 -0
  43. package/backend/src/api/middleware/upload.ts +59 -0
  44. package/backend/src/api/routes/agent.ts +29 -0
  45. package/backend/src/api/routes/ai.ts +472 -0
  46. package/backend/src/api/routes/auth.oauth.ts +482 -0
  47. package/backend/src/api/routes/auth.ts +386 -0
  48. package/backend/src/api/routes/database.advance.ts +275 -0
  49. package/backend/src/api/routes/database.records.ts +246 -0
  50. package/backend/src/api/routes/database.tables.ts +161 -0
  51. package/backend/src/api/routes/docs.ts +66 -0
  52. package/backend/src/api/routes/functions.ts +183 -0
  53. package/backend/src/api/routes/logs.ts +150 -0
  54. package/backend/src/api/routes/metadata.ts +160 -0
  55. package/backend/src/api/routes/openapi.ts +82 -0
  56. package/backend/src/api/routes/secrets.ts +199 -0
  57. package/backend/src/api/routes/storage.ts +547 -0
  58. package/backend/src/api/routes/usage.ts +96 -0
  59. package/backend/src/core/ai/chat.ts +207 -0
  60. package/backend/src/core/ai/client.ts +242 -0
  61. package/backend/src/core/ai/config.ts +187 -0
  62. package/backend/src/core/ai/image.ts +156 -0
  63. package/backend/src/core/ai/model.ts +117 -0
  64. package/backend/src/core/ai/usage.ts +290 -0
  65. package/backend/src/core/auth/auth.ts +781 -0
  66. package/backend/src/core/auth/oauth.ts +398 -0
  67. package/backend/src/core/database/advance.ts +1074 -0
  68. package/backend/src/core/database/manager.ts +178 -0
  69. package/backend/src/core/database/table.ts +772 -0
  70. package/backend/src/core/documentation/agent.ts +689 -0
  71. package/backend/src/core/documentation/openapi.ts +856 -0
  72. package/backend/src/core/functions/functions.ts +310 -0
  73. package/backend/src/core/logs/analytics.ts +76 -0
  74. package/backend/src/core/logs/audit.ts +255 -0
  75. package/backend/src/core/logs/providers/base.provider.ts +83 -0
  76. package/backend/src/core/logs/providers/cloudwatch.provider.ts +510 -0
  77. package/backend/src/core/logs/providers/localdb.provider.ts +246 -0
  78. package/backend/src/core/secrets/encryption.ts +58 -0
  79. package/backend/src/core/secrets/secrets.ts +410 -0
  80. package/backend/src/core/socket/socket.ts +388 -0
  81. package/backend/src/core/socket/types.ts +79 -0
  82. package/backend/src/core/storage/storage.ts +923 -0
  83. package/backend/src/server.ts +288 -0
  84. package/backend/src/types/ai.ts +46 -0
  85. package/backend/src/types/auth.ts +90 -0
  86. package/backend/src/types/database.ts +136 -0
  87. package/backend/src/types/error-constants.ts +86 -0
  88. package/backend/src/types/logs.ts +47 -0
  89. package/backend/src/types/profile.ts +55 -0
  90. package/backend/src/types/storage.ts +23 -0
  91. package/backend/src/utils/cloud-token.ts +39 -0
  92. package/backend/src/utils/constants.ts +1 -0
  93. package/backend/src/utils/environment.ts +35 -0
  94. package/backend/src/utils/helpers.ts +49 -0
  95. package/backend/src/utils/logger.ts +13 -0
  96. package/backend/src/utils/response.ts +62 -0
  97. package/backend/src/utils/seed.ts +205 -0
  98. package/backend/src/utils/sql-parser.ts +63 -0
  99. package/backend/src/utils/uuid.ts +9 -0
  100. package/backend/src/utils/validations.ts +129 -0
  101. package/backend/tests/README.md +134 -0
  102. package/backend/tests/cleanup-all-test-data.sh +231 -0
  103. package/backend/tests/cloud/test-s3-multitenant.sh +132 -0
  104. package/backend/tests/local/comprehensive-curl-tests.sh +156 -0
  105. package/backend/tests/local/test-auth-router.sh +144 -0
  106. package/backend/tests/local/test-database-router.sh +222 -0
  107. package/backend/tests/local/test-e2e.sh +241 -0
  108. package/backend/tests/local/test-fk-errors.sh +97 -0
  109. package/backend/tests/local/test-id-field.sh +201 -0
  110. package/backend/tests/local/test-public-bucket.sh +265 -0
  111. package/backend/tests/local/test-secrets.sh +248 -0
  112. package/backend/tests/local/test-serverless-functions.sh.disabled +325 -0
  113. package/backend/tests/local/test-traditional-rest.sh +209 -0
  114. package/backend/tests/manual/README.md +51 -0
  115. package/backend/tests/manual/create-large-table-simple.sql +11 -0
  116. package/backend/tests/manual/seed-large-table.sql +101 -0
  117. package/backend/tests/manual/setup-large-table-extras.sql +34 -0
  118. package/backend/tests/manual/test-better-auth.sh +303 -0
  119. package/backend/tests/manual/test-bulk-upsert.sh +410 -0
  120. package/backend/tests/manual/test-database-advance.sh +297 -0
  121. package/backend/tests/manual/test-postgrest-stability.sh +192 -0
  122. package/backend/tests/manual/test-rawsql-export-import.sh +412 -0
  123. package/backend/tests/manual/test-universal-storage.sh +264 -0
  124. package/backend/tests/manual/test-users.sql +18 -0
  125. package/backend/tests/run-all-tests.sh +140 -0
  126. package/backend/tests/setup.ts +22 -0
  127. package/backend/tests/test-config.sh +303 -0
  128. package/backend/tsconfig.json +23 -0
  129. package/backend/tsup.config.ts +18 -0
  130. package/backend/vitest.config.ts +22 -0
  131. package/docker-compose.prod.yml +145 -0
  132. package/docker-compose.yml +167 -0
  133. package/docker-init/db/db-init.sql +125 -0
  134. package/docker-init/db/jwt.sql +5 -0
  135. package/docker-init/db/logs.sql +9 -0
  136. package/docker-init/db/postgresql.conf +17 -0
  137. package/docs/deprecated/insforge-auth-api.md +215 -0
  138. package/docs/deprecated/insforge-auth-sdk.md +100 -0
  139. package/docs/deprecated/insforge-db-api.md +359 -0
  140. package/docs/deprecated/insforge-db-sdk.md +140 -0
  141. package/docs/deprecated/insforge-debug-sdk.md +157 -0
  142. package/docs/deprecated/insforge-debug.md +65 -0
  143. package/docs/deprecated/insforge-instructions.md +124 -0
  144. package/docs/deprecated/insforge-project.md +118 -0
  145. package/docs/deprecated/insforge-storage-api.md +279 -0
  146. package/docs/deprecated/insforge-storage-sdk.md +159 -0
  147. package/docs/insforge-instructions-sdk.md +407 -0
  148. package/eslint.config.js +317 -0
  149. package/examples/oauth/frontend-oauth-example.html +251 -0
  150. package/examples/response-examples.md +444 -0
  151. package/frontend/README.md +112 -0
  152. package/frontend/components.json +17 -0
  153. package/frontend/index.html +13 -0
  154. package/frontend/package.json +63 -0
  155. package/frontend/public/favicon.ico +0 -0
  156. package/frontend/src/App.tsx +106 -0
  157. package/frontend/src/assets/icons/checkbox_checked.svg +6 -0
  158. package/frontend/src/assets/icons/checkbox_undetermined.svg +6 -0
  159. package/frontend/src/assets/icons/checked.svg +3 -0
  160. package/frontend/src/assets/icons/error.svg +3 -0
  161. package/frontend/src/assets/icons/pencil.svg +4 -0
  162. package/frontend/src/assets/icons/refresh.svg +4 -0
  163. package/frontend/src/assets/icons/step_active.svg +3 -0
  164. package/frontend/src/assets/icons/step_inactive.svg +11 -0
  165. package/frontend/src/assets/icons/warning.svg +3 -0
  166. package/frontend/src/assets/logos/amazon.svg +1 -0
  167. package/frontend/src/assets/logos/claude_code.svg +3 -0
  168. package/frontend/src/assets/logos/cline.svg +6 -0
  169. package/frontend/src/assets/logos/cursor.svg +20 -0
  170. package/frontend/src/assets/logos/discord.svg +9 -0
  171. package/frontend/src/assets/logos/gemini.svg +19 -0
  172. package/frontend/src/assets/logos/github.svg +5 -0
  173. package/frontend/src/assets/logos/google.svg +13 -0
  174. package/frontend/src/assets/logos/grok.svg +10 -0
  175. package/frontend/src/assets/logos/insforge_dark.svg +15 -0
  176. package/frontend/src/assets/logos/insforge_light.svg +15 -0
  177. package/frontend/src/assets/logos/openai.svg +10 -0
  178. package/frontend/src/assets/logos/roo_code.svg +9 -0
  179. package/frontend/src/assets/logos/trae.svg +3 -0
  180. package/frontend/src/assets/logos/windsurf.svg +10 -0
  181. package/frontend/src/components/ButtonWithLoading.tsx +27 -0
  182. package/frontend/src/components/Checkbox.tsx +61 -0
  183. package/frontend/src/components/CodeBlock.tsx +32 -0
  184. package/frontend/src/components/ConfirmDialog.tsx +96 -0
  185. package/frontend/src/components/CopyButton.tsx +69 -0
  186. package/frontend/src/components/DeleteActionButton.tsx +42 -0
  187. package/frontend/src/components/EmptyState.tsx +41 -0
  188. package/frontend/src/components/ErrorState.tsx +35 -0
  189. package/frontend/src/components/FeatureSidebar.tsx +126 -0
  190. package/frontend/src/components/FeatureSidebarItem.tsx +101 -0
  191. package/frontend/src/components/JsonHighlight.tsx +61 -0
  192. package/frontend/src/components/LoadingState.tsx +16 -0
  193. package/frontend/src/components/PaginationControls.tsx +54 -0
  194. package/frontend/src/components/PromptDialog.tsx +68 -0
  195. package/frontend/src/components/SearchInput.tsx +90 -0
  196. package/frontend/src/components/SelectionClearButton.tsx +26 -0
  197. package/frontend/src/components/Stepper.tsx +139 -0
  198. package/frontend/src/components/ThemeToggle.tsx +58 -0
  199. package/frontend/src/components/TypeBadge.tsx +20 -0
  200. package/frontend/src/components/datagrid/DataGrid.tsx +264 -0
  201. package/frontend/src/components/datagrid/DefaultCellRenderer.tsx +114 -0
  202. package/frontend/src/components/datagrid/IdCell.tsx +44 -0
  203. package/frontend/src/components/datagrid/SortableHeader.tsx +74 -0
  204. package/frontend/src/components/datagrid/cell-editors/BooleanCellEditor.tsx +54 -0
  205. package/frontend/src/components/datagrid/cell-editors/DateCellEditor.tsx +483 -0
  206. package/frontend/src/components/datagrid/cell-editors/JsonCellEditor.tsx +362 -0
  207. package/frontend/src/components/datagrid/cell-editors/TextCellEditor.tsx +38 -0
  208. package/frontend/src/components/datagrid/cell-editors/index.ts +14 -0
  209. package/frontend/src/components/datagrid/cell-editors/types.ts +43 -0
  210. package/frontend/src/components/datagrid/datagridTypes.tsx +72 -0
  211. package/frontend/src/components/datagrid/index.tsx +20 -0
  212. package/frontend/src/components/index.ts +39 -0
  213. package/frontend/src/components/layout/AppHeader.tsx +146 -0
  214. package/frontend/src/components/layout/AppSidebar.tsx +190 -0
  215. package/frontend/src/components/layout/CloudLayout.tsx +95 -0
  216. package/frontend/src/components/layout/Layout.tsx +43 -0
  217. package/frontend/src/components/radix/Alert.tsx +45 -0
  218. package/frontend/src/components/radix/AlertDialog.tsx +115 -0
  219. package/frontend/src/components/radix/Avatar.tsx +45 -0
  220. package/frontend/src/components/radix/Badge.tsx +33 -0
  221. package/frontend/src/components/radix/Button.tsx +50 -0
  222. package/frontend/src/components/radix/Card.tsx +58 -0
  223. package/frontend/src/components/radix/Dialog.tsx +98 -0
  224. package/frontend/src/components/radix/DropdownMenu.tsx +185 -0
  225. package/frontend/src/components/radix/Form.tsx +167 -0
  226. package/frontend/src/components/radix/Input.tsx +22 -0
  227. package/frontend/src/components/radix/Label.tsx +19 -0
  228. package/frontend/src/components/radix/Popover.tsx +29 -0
  229. package/frontend/src/components/radix/ScrollArea.tsx +44 -0
  230. package/frontend/src/components/radix/Select.tsx +151 -0
  231. package/frontend/src/components/radix/Separator.tsx +26 -0
  232. package/frontend/src/components/radix/Sheet.tsx +119 -0
  233. package/frontend/src/components/radix/Skeleton.tsx +7 -0
  234. package/frontend/src/components/radix/Switch.tsx +29 -0
  235. package/frontend/src/components/radix/Tabs.tsx +50 -0
  236. package/frontend/src/components/radix/Textarea.tsx +21 -0
  237. package/frontend/src/components/radix/Tooltip.tsx +28 -0
  238. package/frontend/src/features/ai/components/AIConfigCard.tsx +154 -0
  239. package/frontend/src/features/ai/components/AIConfigDialog.tsx +76 -0
  240. package/frontend/src/features/ai/components/AIConfigForm.tsx +222 -0
  241. package/frontend/src/features/ai/components/AIEmptyState.tsx +18 -0
  242. package/frontend/src/features/ai/components/fields/ModalityField.tsx +87 -0
  243. package/frontend/src/features/ai/components/fields/ModelSelectionField.tsx +134 -0
  244. package/frontend/src/features/ai/components/fields/SystemPromptField.tsx +33 -0
  245. package/frontend/src/features/ai/helpers.ts +155 -0
  246. package/frontend/src/features/ai/hooks/useAIConfigs.ts +221 -0
  247. package/frontend/src/features/ai/hooks/useAIUsage.ts +77 -0
  248. package/frontend/src/features/ai/page/AIPage.tsx +178 -0
  249. package/frontend/src/features/ai/services/ai.service.ts +148 -0
  250. package/frontend/src/features/auth/components/AddOAuthDialog.tsx +106 -0
  251. package/frontend/src/features/auth/components/AuthMethodTab.tsx +238 -0
  252. package/frontend/src/features/auth/components/OAuthConfigDialog.tsx +303 -0
  253. package/frontend/src/features/auth/components/OAuthEmptyState.tsx +15 -0
  254. package/frontend/src/features/auth/components/UserFormDialog.tsx +248 -0
  255. package/frontend/src/features/auth/components/UsersDataGrid.tsx +183 -0
  256. package/frontend/src/features/auth/components/UsersTab.tsx +114 -0
  257. package/frontend/src/features/auth/hooks/useOAuthConfig.ts +129 -0
  258. package/frontend/src/features/auth/hooks/useUsers.ts +57 -0
  259. package/frontend/src/features/auth/index.ts +9 -0
  260. package/frontend/src/features/auth/page/AuthenticationPage.tsx +169 -0
  261. package/frontend/src/features/auth/services/auth.service.ts +112 -0
  262. package/frontend/src/features/auth/services/oauth.service.ts +49 -0
  263. package/frontend/src/features/dashboard/page/DashboardPage.tsx +194 -0
  264. package/frontend/src/features/database/components/ColumnTypeSelect.tsx +64 -0
  265. package/frontend/src/features/database/components/DatabaseDataGrid.tsx +282 -0
  266. package/frontend/src/features/database/components/ForeignKeyCell.tsx +187 -0
  267. package/frontend/src/features/database/components/ForeignKeyPopover.tsx +378 -0
  268. package/frontend/src/features/database/components/LinkRecordModal.tsx +288 -0
  269. package/frontend/src/features/database/components/RecordFormDialog.tsx +164 -0
  270. package/frontend/src/features/database/components/RecordFormField.tsx +568 -0
  271. package/frontend/src/features/database/components/TableEmptyState.tsx +21 -0
  272. package/frontend/src/features/database/components/TableForm.tsx +656 -0
  273. package/frontend/src/features/database/components/TableFormColumn.tsx +137 -0
  274. package/frontend/src/features/database/components/TableListSkeleton.tsx +9 -0
  275. package/frontend/src/features/database/components/TableSidebar.tsx +47 -0
  276. package/frontend/src/features/database/constants.ts +26 -0
  277. package/frontend/src/features/database/helpers.ts +125 -0
  278. package/frontend/src/features/database/hooks/UseLinkModal.tsx +78 -0
  279. package/frontend/src/features/database/index.ts +12 -0
  280. package/frontend/src/features/database/page/DatabasePage.tsx +626 -0
  281. package/frontend/src/features/database/schema.ts +25 -0
  282. package/frontend/src/features/database/services/database.service.ts +216 -0
  283. package/frontend/src/features/functions/components/FunctionEmptyState.tsx +15 -0
  284. package/frontend/src/features/functions/components/FunctionRow.tsx +71 -0
  285. package/frontend/src/features/functions/components/FunctionViewer.tsx +46 -0
  286. package/frontend/src/features/functions/components/FunctionsContent.tsx +88 -0
  287. package/frontend/src/features/functions/components/FunctionsSidebar.tsx +56 -0
  288. package/frontend/src/features/functions/components/SecretEmptyState.tsx +23 -0
  289. package/frontend/src/features/functions/components/SecretRow.tsx +68 -0
  290. package/frontend/src/features/functions/components/SecretsContent.tsx +120 -0
  291. package/frontend/src/features/functions/hooks/useFunctions.ts +106 -0
  292. package/frontend/src/features/functions/page/FunctionsPage.tsx +28 -0
  293. package/frontend/src/features/functions/services/functions.service.ts +48 -0
  294. package/frontend/src/features/login/components/AuthErrorBoundary.tsx +87 -0
  295. package/frontend/src/features/login/components/PrivateRoute.tsx +24 -0
  296. package/frontend/src/features/login/page/CloudLoginPage.tsx +93 -0
  297. package/frontend/src/features/login/page/LoginPage.tsx +174 -0
  298. package/frontend/src/features/logs/components/AnalyticsLogsTable.tsx +313 -0
  299. package/frontend/src/features/logs/components/LogsTable.tsx +199 -0
  300. package/frontend/src/features/logs/hooks/useAuditLogs.ts +39 -0
  301. package/frontend/src/features/logs/index.ts +5 -0
  302. package/frontend/src/features/logs/page/AnalyticsLogsPage.tsx +530 -0
  303. package/frontend/src/features/logs/page/AuditsPage.tsx +192 -0
  304. package/frontend/src/features/logs/services/log.service.ts +171 -0
  305. package/frontend/src/features/metadata/hooks/useMetadata.ts +53 -0
  306. package/frontend/src/features/metadata/index.ts +0 -0
  307. package/frontend/src/features/metadata/page/MetadataPage.tsx +136 -0
  308. package/frontend/src/features/metadata/services/metadata.service.ts +17 -0
  309. package/frontend/src/features/onboard/components/CompletionCard.tsx +41 -0
  310. package/frontend/src/features/onboard/components/OnboardButton.tsx +84 -0
  311. package/frontend/src/features/onboard/components/StepContent.tsx +91 -0
  312. package/frontend/src/features/onboard/components/TestConnectionStep.tsx +53 -0
  313. package/frontend/src/features/onboard/components/mcp/CursorDeeplinkGenerator.tsx +35 -0
  314. package/frontend/src/features/onboard/components/mcp/McpInstallation.tsx +144 -0
  315. package/frontend/src/features/onboard/components/mcp/index.ts +4 -0
  316. package/frontend/src/features/onboard/components/mcp/mcp-helper.tsx +98 -0
  317. package/frontend/src/features/onboard/index.ts +3 -0
  318. package/frontend/src/features/onboard/page/OnBoardPage.tsx +104 -0
  319. package/frontend/src/features/onboard/types.ts +8 -0
  320. package/frontend/src/features/secrets/hooks/useSecrets.ts +139 -0
  321. package/frontend/src/features/secrets/services/secrets.service.ts +57 -0
  322. package/frontend/src/features/storage/components/BucketEmptyState.tsx +19 -0
  323. package/frontend/src/features/storage/components/BucketFormDialog.tsx +194 -0
  324. package/frontend/src/features/storage/components/BucketListSkeleton.tsx +17 -0
  325. package/frontend/src/features/storage/components/FilePreviewDialog.tsx +287 -0
  326. package/frontend/src/features/storage/components/StorageDataGrid.tsx +239 -0
  327. package/frontend/src/features/storage/components/StorageManager.tsx +236 -0
  328. package/frontend/src/features/storage/components/StorageSidebar.tsx +44 -0
  329. package/frontend/src/features/storage/components/UploadToast.tsx +46 -0
  330. package/frontend/src/features/storage/index.ts +3 -0
  331. package/frontend/src/features/storage/page/StoragePage.tsx +553 -0
  332. package/frontend/src/features/storage/services/storage.service.ts +144 -0
  333. package/frontend/src/features/visualizer/components/AuthNode.tsx +107 -0
  334. package/frontend/src/features/visualizer/components/BucketNode.tsx +34 -0
  335. package/frontend/src/features/visualizer/components/SchemaVisualizer.tsx +359 -0
  336. package/frontend/src/features/visualizer/components/TableNode.tsx +152 -0
  337. package/frontend/src/features/visualizer/components/VisualizerSkeleton.tsx +24 -0
  338. package/frontend/src/features/visualizer/components/index.ts +5 -0
  339. package/frontend/src/features/visualizer/page/VisualizerPage.tsx +127 -0
  340. package/frontend/src/index.css +248 -0
  341. package/frontend/src/lib/api/client.ts +163 -0
  342. package/frontend/src/lib/contexts/AuthContext.tsx +157 -0
  343. package/frontend/src/lib/contexts/OnboardStepContext.tsx +68 -0
  344. package/frontend/src/lib/contexts/SocketContext.tsx +303 -0
  345. package/frontend/src/lib/contexts/ThemeContext.tsx +125 -0
  346. package/frontend/src/lib/hooks/useAuth.ts +4 -0
  347. package/frontend/src/lib/hooks/useConfirm.ts +55 -0
  348. package/frontend/src/lib/hooks/useInterval.ts +27 -0
  349. package/frontend/src/lib/hooks/useMediaQuery.ts +59 -0
  350. package/frontend/src/lib/hooks/useOnboardingCompletion.ts +29 -0
  351. package/frontend/src/lib/hooks/usePagination.ts +27 -0
  352. package/frontend/src/lib/hooks/useTimeout.ts +27 -0
  353. package/frontend/src/lib/hooks/useToast.tsx +229 -0
  354. package/frontend/src/lib/utils/constants.ts +38 -0
  355. package/frontend/src/lib/utils/utils.ts +165 -0
  356. package/frontend/src/lib/utils/validation-schemas.ts +126 -0
  357. package/frontend/src/main.tsx +16 -0
  358. package/frontend/src/rdg.css +194 -0
  359. package/frontend/src/vite-env.d.ts +12 -0
  360. package/frontend/tailwind.config.js +97 -0
  361. package/frontend/tsconfig.json +26 -0
  362. package/frontend/tsconfig.node.json +10 -0
  363. package/frontend/vite.config.ts +37 -0
  364. package/frontend/vitest.config.ts +36 -0
  365. package/functions/deno.json +25 -0
  366. package/functions/server.ts +290 -0
  367. package/functions/worker-template.js +126 -0
  368. package/openapi/ai.yaml +689 -0
  369. package/openapi/auth.yaml +563 -0
  370. package/openapi/functions.yaml +476 -0
  371. package/openapi/health.yaml +30 -0
  372. package/openapi/logs.yaml +224 -0
  373. package/openapi/metadata.yaml +178 -0
  374. package/openapi/records.yaml +382 -0
  375. package/openapi/secrets.yaml +371 -0
  376. package/openapi/storage.yaml +876 -0
  377. package/openapi/tables.yaml +464 -0
  378. package/package.json +88 -0
  379. package/shared-schemas/package.json +31 -0
  380. package/shared-schemas/src/ai-api.schema.ts +167 -0
  381. package/shared-schemas/src/ai.schema.ts +54 -0
  382. package/shared-schemas/src/auth-api.schema.ts +193 -0
  383. package/shared-schemas/src/auth.schema.ts +94 -0
  384. package/shared-schemas/src/database-api.schema.ts +259 -0
  385. package/shared-schemas/src/database.schema.ts +69 -0
  386. package/shared-schemas/src/functions-api.schema.ts +25 -0
  387. package/shared-schemas/src/functions.schema.ts +16 -0
  388. package/shared-schemas/src/index.ts +13 -0
  389. package/shared-schemas/src/logs-api.schema.ts +49 -0
  390. package/shared-schemas/src/logs.schema.ts +14 -0
  391. package/shared-schemas/src/metadata.schema.ts +56 -0
  392. package/shared-schemas/src/storage-api.schema.ts +65 -0
  393. package/shared-schemas/src/storage.schema.ts +19 -0
  394. package/shared-schemas/tsconfig.json +21 -0
  395. 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
+ }