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,183 @@
1
+ import { useMemo } from 'react';
2
+ import {
3
+ DataGrid,
4
+ createDefaultCellRenderer,
5
+ type DataGridProps,
6
+ type DataGridColumn,
7
+ type RenderCellProps,
8
+ ConvertedValue,
9
+ } from '@/components/datagrid';
10
+ import { Badge } from '@/components/radix/Badge';
11
+ import { cn } from '@/lib/utils/utils';
12
+ import type { UserSchema } from '@insforge/shared-schemas';
13
+
14
+ // Create a type that makes UserSchema compatible with DataGrid requirements
15
+ // We bypass the strict DatabaseRecord constraint since UserSchema has its own structure
16
+ type UserDataGridRow = UserSchema & {
17
+ [key: string]: ConvertedValue | { [key: string]: string }[];
18
+ };
19
+
20
+ // Provider icon component
21
+ const ProviderIcon = ({ provider }: { provider: string }) => {
22
+ const getProviderInfo = (provider: string) => {
23
+ switch (provider.toLowerCase()) {
24
+ case 'google':
25
+ return {
26
+ label: 'Google',
27
+ color:
28
+ 'bg-red-100 text-red-700 dark:bg-neutral-800 dark:text-red-300 dark:border-red-500',
29
+ };
30
+ case 'github':
31
+ return {
32
+ label: 'GitHub',
33
+ color:
34
+ 'bg-gray-100 text-gray-700 dark:bg-neutral-800 dark:text-zinc-300 dark:border-gray-500',
35
+ };
36
+ case 'email':
37
+ return {
38
+ label: 'Email',
39
+ color:
40
+ 'bg-green-100 text-green-700 dark:bg-neutral-800 dark:text-green-300 dark:border-green-500',
41
+ };
42
+ default:
43
+ return {
44
+ label: provider,
45
+ color:
46
+ 'bg-gray-100 text-gray-700 dark:bg-neutral-800 dark:text-zinc-300 dark:border-gray-500',
47
+ };
48
+ }
49
+ };
50
+
51
+ const { label, color } = getProviderInfo(provider);
52
+
53
+ return (
54
+ <Badge
55
+ variant="secondary"
56
+ className={cn('text-xs font-medium px-2 py-1 border border-transparent', color)}
57
+ >
58
+ {label}
59
+ </Badge>
60
+ );
61
+ };
62
+
63
+ const IdentitiesCellRenderer = ({ row }: RenderCellProps<UserDataGridRow>) => {
64
+ const identities = row.identities;
65
+
66
+ if (!identities || !Array.isArray(identities) || identities.length === 0) {
67
+ return <span className="text-sm text-black dark:text-zinc-300">null</span>;
68
+ }
69
+
70
+ // Get unique providers to avoid duplicates
71
+ const uniqueProviders = [
72
+ ...new Set(identities.map((identity: { provider: string }) => identity.provider)),
73
+ ];
74
+
75
+ return (
76
+ <div
77
+ className="flex flex-wrap gap-1"
78
+ title={identities.map((identity: { provider: string }) => identity.provider).join(', ')}
79
+ >
80
+ {uniqueProviders.slice(0, 2).map((provider: string, index: number) => (
81
+ <ProviderIcon key={index} provider={provider} />
82
+ ))}
83
+ {uniqueProviders.length > 2 && (
84
+ <Badge
85
+ variant="secondary"
86
+ className="text-xs bg-gray-100 text-gray-600 dark:bg-neutral-800 dark:text-zinc-300 dark:border-neutral-700"
87
+ >
88
+ +{uniqueProviders.length - 2}
89
+ </Badge>
90
+ )}
91
+ </div>
92
+ );
93
+ };
94
+
95
+ // Convert users data to DataGrid columns
96
+ export function createUsersColumns(): DataGridColumn<UserDataGridRow>[] {
97
+ const cellRenderers = createDefaultCellRenderer<UserDataGridRow>();
98
+
99
+ return [
100
+ {
101
+ key: 'id',
102
+ name: 'ID',
103
+ width: '1fr',
104
+ resizable: true,
105
+ sortable: true,
106
+ renderCell: cellRenderers.id,
107
+ },
108
+ {
109
+ key: 'email',
110
+ name: 'Email',
111
+ width: '1fr',
112
+ resizable: true,
113
+ sortable: true,
114
+ renderCell: cellRenderers.email,
115
+ },
116
+ {
117
+ key: 'name',
118
+ name: 'Name',
119
+ width: '1fr',
120
+ resizable: true,
121
+ sortable: true,
122
+ renderCell: cellRenderers.text,
123
+ },
124
+ {
125
+ key: 'identities',
126
+ name: 'Identities',
127
+ width: '1fr',
128
+ resizable: true,
129
+ sortable: true,
130
+ renderCell: IdentitiesCellRenderer,
131
+ },
132
+ {
133
+ key: 'providerType',
134
+ name: 'Provider Type',
135
+ width: '1fr',
136
+ resizable: true,
137
+ sortable: true,
138
+ renderCell: cellRenderers.text,
139
+ },
140
+ {
141
+ key: 'createdAt',
142
+ name: 'Created',
143
+ width: '1fr',
144
+ resizable: true,
145
+ sortable: true,
146
+ renderCell: cellRenderers.datetime,
147
+ },
148
+ {
149
+ key: 'updatedAt',
150
+ name: 'Updated',
151
+ width: '1fr',
152
+ resizable: true,
153
+ sortable: true,
154
+ renderCell: cellRenderers.datetime,
155
+ },
156
+ ];
157
+ }
158
+
159
+ // Users-specific DataGrid props
160
+ export type UsersDataGridProps = Omit<DataGridProps<UserDataGridRow>, 'columns'>;
161
+
162
+ // Specialized DataGrid for users
163
+ export function UsersDataGrid({
164
+ emptyStateTitle = 'No users available',
165
+ emptyStateActionText,
166
+ onEmptyStateAction,
167
+ ...props
168
+ }: UsersDataGridProps) {
169
+ const columns = useMemo(() => createUsersColumns(), []);
170
+
171
+ return (
172
+ <DataGrid<UserDataGridRow>
173
+ {...props}
174
+ columns={columns}
175
+ emptyStateTitle={emptyStateTitle}
176
+ emptyStateActionText={emptyStateActionText}
177
+ onEmptyStateAction={onEmptyStateAction}
178
+ showSelection={true}
179
+ showPagination={true}
180
+ showTypeBadge={false}
181
+ />
182
+ );
183
+ }
@@ -0,0 +1,114 @@
1
+ import { useEffect, useMemo, useState } from 'react';
2
+ import { useUsers } from '@/features/auth/hooks/useUsers';
3
+ import { UsersDataGrid } from './UsersDataGrid';
4
+ import { SortColumn } from 'react-data-grid';
5
+ import { UserSchema } from '@insforge/shared-schemas';
6
+
7
+ interface UsersTabProps {
8
+ searchQuery?: string;
9
+ selectedRows?: Set<string>;
10
+ onSelectedRowsChange?: (selectedRows: Set<string>) => void;
11
+ }
12
+
13
+ export function UsersTab({
14
+ searchQuery: externalSearchQuery = '',
15
+ selectedRows: externalSelectedRows,
16
+ onSelectedRowsChange: externalOnSelectedRowsChange,
17
+ }: UsersTabProps) {
18
+ // Default page size of 20 records per page
19
+ const pageSize = 20;
20
+ const { users, totalUsers, isLoading, currentPage, setCurrentPage, totalPages, refetch } =
21
+ useUsers({ searchQuery: externalSearchQuery, pageSize });
22
+
23
+ // Multi-select state - use external state if provided, otherwise internal
24
+ const [internalSelectedRows, setInternalSelectedRows] = useState<Set<string>>(new Set());
25
+ const selectedRows = externalSelectedRows || internalSelectedRows;
26
+ const setSelectedRows = externalOnSelectedRowsChange || setInternalSelectedRows;
27
+
28
+ // Sorting state
29
+ const [sortColumns, setSortColumns] = useState<SortColumn[]>([]);
30
+
31
+ // Listen for refresh events
32
+ useEffect(() => {
33
+ const handleRefresh = () => {
34
+ // Reset sorting columns
35
+ setSortColumns([]);
36
+ // Reset selected rows
37
+ setSelectedRows(new Set());
38
+ // Refetch data
39
+ void refetch();
40
+ };
41
+ window.addEventListener('refreshUsers', handleRefresh);
42
+ return () => window.removeEventListener('refreshUsers', handleRefresh);
43
+ }, [refetch, setSelectedRows]);
44
+
45
+ // Clear selection when page changes or search changes
46
+ useEffect(() => {
47
+ setSelectedRows(new Set());
48
+ }, [currentPage, externalSearchQuery, setSelectedRows]);
49
+
50
+ // Apply sorting to users data
51
+ const sortedUsers = useMemo(() => {
52
+ if (!sortColumns.length) {
53
+ return users;
54
+ }
55
+
56
+ return [...users].sort((a, b) => {
57
+ for (const sort of sortColumns) {
58
+ const { columnKey, direction } = sort;
59
+ let aVal = a[columnKey as keyof UserSchema];
60
+ let bVal = b[columnKey as keyof UserSchema];
61
+
62
+ // Handle null/undefined values
63
+ if ((aVal === null || aVal === undefined) && (bVal === null || bVal === undefined)) {
64
+ continue;
65
+ }
66
+ if (aVal === null || aVal === undefined) {
67
+ return direction === 'ASC' ? -1 : 1;
68
+ }
69
+ if (bVal === null || bVal === undefined) {
70
+ return direction === 'ASC' ? 1 : -1;
71
+ }
72
+
73
+ // Convert to comparable values
74
+ if (typeof aVal === 'string') {
75
+ aVal = aVal.toLowerCase();
76
+ }
77
+ if (typeof bVal === 'string') {
78
+ bVal = bVal.toLowerCase();
79
+ }
80
+
81
+ if (aVal < bVal) {
82
+ return direction === 'ASC' ? -1 : 1;
83
+ }
84
+ if (aVal > bVal) {
85
+ return direction === 'ASC' ? 1 : -1;
86
+ }
87
+ }
88
+ return 0;
89
+ });
90
+ }, [users, sortColumns]);
91
+
92
+ return (
93
+ <div className="relative flex-1 flex flex-col overflow-hidden">
94
+ <div className="flex-1 flex flex-col overflow-hidden">
95
+ <UsersDataGrid
96
+ data={sortedUsers}
97
+ loading={isLoading}
98
+ selectedRows={selectedRows}
99
+ onSelectedRowsChange={setSelectedRows}
100
+ sortColumns={sortColumns}
101
+ onSortColumnsChange={setSortColumns}
102
+ currentPage={currentPage}
103
+ totalPages={totalPages}
104
+ pageSize={pageSize}
105
+ totalRecords={totalUsers}
106
+ onPageChange={setCurrentPage}
107
+ emptyStateTitle={
108
+ externalSearchQuery ? 'No users match your search criteria' : 'No users found'
109
+ }
110
+ />
111
+ </div>
112
+ </div>
113
+ );
114
+ }
@@ -0,0 +1,129 @@
1
+ import { useState, useCallback } from 'react';
2
+ import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query';
3
+ import {
4
+ OAuthConfigSchema,
5
+ CreateOAuthConfigRequest,
6
+ UpdateOAuthConfigRequest,
7
+ ListOAuthConfigsResponse,
8
+ } from '@insforge/shared-schemas';
9
+ import { oauthConfigService } from '../services/oauth.service';
10
+ import { useToast } from '@/lib/hooks/useToast';
11
+
12
+ export function useOAuthConfig() {
13
+ const queryClient = useQueryClient();
14
+ const { showToast } = useToast();
15
+ const [selectedProvider, setSelectedProvider] = useState<string | null>(null);
16
+
17
+ // Query to fetch all OAuth configurations
18
+ const {
19
+ data: configs,
20
+ isLoading: isLoadingConfigs,
21
+ error: configsError,
22
+ refetch: refetchConfigs,
23
+ } = useQuery<ListOAuthConfigsResponse>({
24
+ queryKey: ['oauth-configs'],
25
+ queryFn: () => oauthConfigService.getAllConfigs(),
26
+ });
27
+
28
+ // Query to fetch specific provider config
29
+ const {
30
+ data: providerConfig,
31
+ isLoading: isLoadingProvider,
32
+ error: providerError,
33
+ refetch: refetchProvider,
34
+ } = useQuery<OAuthConfigSchema & { clientSecret?: string }>({
35
+ queryKey: ['oauth-config', selectedProvider],
36
+ queryFn: () => oauthConfigService.getConfigByProvider(selectedProvider ?? ''),
37
+ enabled: configs && configs.data.some((config) => config.provider === selectedProvider),
38
+ });
39
+
40
+ // Mutation to create OAuth configuration
41
+ const createConfigMutation = useMutation({
42
+ mutationFn: (config: CreateOAuthConfigRequest) => oauthConfigService.createConfig(config),
43
+ onSuccess: (data) => {
44
+ void queryClient.invalidateQueries({ queryKey: ['oauth-configs'] });
45
+ void queryClient.invalidateQueries({ queryKey: ['oauth-config', data.provider] });
46
+ showToast(`OAuth configuration for ${data.provider} created successfully`, 'success');
47
+ },
48
+ onError: (error: Error) => {
49
+ showToast(error.message || 'Failed to create OAuth configuration', 'error');
50
+ },
51
+ });
52
+
53
+ // Mutation to update OAuth configuration
54
+ const updateConfigMutation = useMutation({
55
+ mutationFn: ({ provider, config }: { provider: string; config: UpdateOAuthConfigRequest }) =>
56
+ oauthConfigService.updateConfig(provider, config),
57
+ onSuccess: (data) => {
58
+ void queryClient.invalidateQueries({ queryKey: ['oauth-configs'] });
59
+ void queryClient.invalidateQueries({ queryKey: ['oauth-config', data.provider] });
60
+ showToast(`OAuth configuration for ${data.provider} updated successfully`, 'success');
61
+ },
62
+ onError: (error: Error) => {
63
+ showToast(error.message || 'Failed to update OAuth configuration', 'error');
64
+ },
65
+ });
66
+
67
+ // Mutation to delete OAuth configuration
68
+ const deleteConfigMutation = useMutation({
69
+ mutationFn: (provider: string) => oauthConfigService.deleteConfig(provider),
70
+ onSuccess: (_, provider) => {
71
+ queryClient.removeQueries({ queryKey: ['oauth-configs'] });
72
+ queryClient.removeQueries({ queryKey: ['oauth-config', provider] });
73
+ showToast(`OAuth configuration for ${provider} deleted successfully`, 'success');
74
+ if (selectedProvider === provider) {
75
+ setSelectedProvider(null);
76
+ }
77
+ },
78
+ onError: (error: Error) => {
79
+ showToast(error.message || 'Failed to delete OAuth configuration', 'error');
80
+ },
81
+ });
82
+
83
+ // Helper to check if a provider is configured
84
+ const isProviderConfigured = useCallback(
85
+ (provider: string): boolean => {
86
+ return configs?.data?.some((config) => config.provider === provider) ?? false;
87
+ },
88
+ [configs]
89
+ );
90
+
91
+ // Helper to get config for a specific provider from the list
92
+ const getProviderConfig = useCallback(
93
+ (provider: string): OAuthConfigSchema | undefined => {
94
+ return configs?.data?.find((config) => config.provider === provider);
95
+ },
96
+ [configs]
97
+ );
98
+
99
+ return {
100
+ // State
101
+ configs: configs?.data ?? [],
102
+ configsCount: configs?.count ?? 0,
103
+ selectedProvider,
104
+ setSelectedProvider,
105
+ providerConfig,
106
+
107
+ // Loading states
108
+ isLoadingConfigs,
109
+ isLoadingProvider,
110
+ isCreating: createConfigMutation.isPending,
111
+ isUpdating: updateConfigMutation.isPending,
112
+ isDeleting: deleteConfigMutation.isPending,
113
+
114
+ // Errors
115
+ configsError,
116
+ providerError,
117
+
118
+ // Actions
119
+ createConfig: createConfigMutation.mutate,
120
+ updateConfig: updateConfigMutation.mutate,
121
+ deleteConfig: deleteConfigMutation.mutate,
122
+ refetchConfigs,
123
+ refetchProvider,
124
+
125
+ // Helpers
126
+ isProviderConfigured,
127
+ getProviderConfig,
128
+ };
129
+ }
@@ -0,0 +1,57 @@
1
+ import { useState } from 'react';
2
+ import { useQuery } from '@tanstack/react-query';
3
+ import { authService } from '@/features/auth/services/auth.service';
4
+
5
+ interface UseUsersOptions {
6
+ pageSize?: number;
7
+ enabled?: boolean;
8
+ searchQuery?: string;
9
+ }
10
+
11
+ export function useUsers(options: UseUsersOptions = {}) {
12
+ const { pageSize = 20, enabled = true, searchQuery = '' } = options;
13
+ const [currentPage, setCurrentPage] = useState(1);
14
+
15
+ // Fetch users data
16
+ const {
17
+ data: usersData,
18
+ isLoading,
19
+ error,
20
+ refetch,
21
+ } = useQuery({
22
+ queryKey: ['users', currentPage, searchQuery],
23
+ queryFn: () => {
24
+ const params = new URLSearchParams({
25
+ limit: pageSize.toString(),
26
+ offset: ((currentPage - 1) * pageSize).toString(),
27
+ });
28
+ // Use the auth service to get users with search, backend handles filtering
29
+ return authService.getUsers(params.toString(), searchQuery);
30
+ },
31
+ enabled: enabled,
32
+ placeholderData: (previousData) => previousData, // Keep previous data while loading
33
+ });
34
+
35
+ // Pagination calculations
36
+ const totalPages = Math.ceil((usersData?.pagination.total || 0) / pageSize);
37
+
38
+ return {
39
+ // Data
40
+ users: usersData?.users || [],
41
+ totalUsers: usersData?.pagination.total || 0,
42
+ isLoading,
43
+ error,
44
+
45
+ // Pagination
46
+ currentPage,
47
+ setCurrentPage,
48
+ totalPages,
49
+ pageSize,
50
+
51
+ // Search
52
+ searchQuery,
53
+
54
+ // Operations
55
+ refetch,
56
+ };
57
+ }
@@ -0,0 +1,9 @@
1
+ // Components
2
+ export { UserFormDialog } from './components/UserFormDialog';
3
+ export { UsersTab } from './components/UsersTab';
4
+
5
+ // Hooks
6
+ export { useUsers } from './hooks/useUsers';
7
+
8
+ // Services
9
+ export { authService } from './services/auth.service';
@@ -0,0 +1,169 @@
1
+ import { useState } from 'react';
2
+ import { useLocation } from 'react-router-dom';
3
+ import { UserPlus, Users, Key } from 'lucide-react';
4
+ import { Button, SearchInput, SelectionClearButton, DeleteActionButton } from '@/components';
5
+ import { UsersTab } from '@/features/auth/components/UsersTab';
6
+ import { Tooltip, TooltipContent, TooltipProvider } from '@/components/radix/Tooltip';
7
+ import { UserFormDialog } from '@/features/auth/components/UserFormDialog';
8
+ import { AuthMethodTab } from '@/features/auth/components/AuthMethodTab';
9
+ import { ConfirmDialog } from '@/components/ConfirmDialog';
10
+ import { authService } from '@/features/auth/services/auth.service';
11
+ import { useToast } from '@/lib/hooks/useToast';
12
+ import { cn } from '@/lib/utils/utils';
13
+ import { useUsers } from '@/features/auth/hooks/useUsers';
14
+
15
+ export default function AuthenticationPage() {
16
+ const location = useLocation();
17
+ const [selectedSection, setSelectedSection] = useState<string>(
18
+ location.state?.initialTab || 'users'
19
+ );
20
+ const [searchQuery, setSearchQuery] = useState('');
21
+ const [addDialogOpen, setAddDialogOpen] = useState(false);
22
+ const [confirmDeleteOpen, setConfirmDeleteOpen] = useState(false);
23
+ const [selectedRows, setSelectedRows] = useState<Set<string>>(new Set());
24
+
25
+ const { showToast } = useToast();
26
+ const { refetch } = useUsers();
27
+
28
+ const handleBulkDelete = async () => {
29
+ if (selectedRows.size === 0) {
30
+ return;
31
+ }
32
+
33
+ try {
34
+ const userIds = Array.from(selectedRows);
35
+ await authService.deleteUsers(userIds);
36
+ void refetch();
37
+ setSelectedRows(new Set());
38
+ showToast(
39
+ `${userIds.length} user${userIds.length > 1 ? 's' : ''} deleted successfully`,
40
+ 'success'
41
+ );
42
+ } catch (error) {
43
+ showToast(error instanceof Error ? error.message : 'Failed to delete users', 'error');
44
+ }
45
+ };
46
+
47
+ const authSections = [
48
+ {
49
+ id: 'users',
50
+ name: 'Users',
51
+ icon: Users,
52
+ description: 'Manage user accounts',
53
+ },
54
+ {
55
+ id: 'auth-methods',
56
+ name: 'Auth Methods',
57
+ icon: Key,
58
+ description: 'Configure authentication',
59
+ },
60
+ ];
61
+
62
+ return (
63
+ <div className="h-full bg-slate-50 dark:bg-neutral-800 flex flex-col overflow-hidden">
64
+ {/* Tab Navigation */}
65
+ <div className="h-12 flex items-center gap-6 px-6 border-b border-border-gray dark:border-neutral-700 relative">
66
+ {authSections.map((section) => (
67
+ <button
68
+ key={section.id}
69
+ onClick={() => setSelectedSection(section.id)}
70
+ className={cn(
71
+ 'flex h-full items-center gap-2 px-0 text-base font-semibold transition-colors relative',
72
+ selectedSection === section.id
73
+ ? 'text-black dark:text-white'
74
+ : 'text-zinc-500 hover:text-zinc-700 dark:text-zinc-400 dark:hover:text-zinc-300'
75
+ )}
76
+ >
77
+ {section.name}
78
+ {selectedSection === section.id && (
79
+ <div className="absolute bottom-0 left-0 right-0 h-0.5 bg-black dark:bg-white" />
80
+ )}
81
+ </button>
82
+ ))}
83
+ </div>
84
+
85
+ {/* Content Area */}
86
+ <div className="flex-1 flex flex-col overflow-hidden">
87
+ {/* Users Section Header */}
88
+ {selectedSection === 'users' && (
89
+ <div className="px-3 py-4 dark:bg-neutral-800">
90
+ <div className="flex items-center justify-between">
91
+ {selectedRows.size > 0 ? (
92
+ <div className="flex items-center gap-3">
93
+ <SelectionClearButton
94
+ selectedCount={selectedRows.size}
95
+ itemType="user"
96
+ onClear={() => setSelectedRows(new Set())}
97
+ />
98
+ <DeleteActionButton
99
+ selectedCount={selectedRows.size}
100
+ itemType="user"
101
+ onDelete={() => setConfirmDeleteOpen(true)}
102
+ />
103
+ </div>
104
+ ) : (
105
+ <SearchInput
106
+ value={searchQuery}
107
+ onChange={setSearchQuery}
108
+ placeholder="Search Users by Name or Email"
109
+ className="flex-1 max-w-80 dark:bg-neutral-800 dark:text-white dark:border-neutral-700"
110
+ debounceTime={300}
111
+ />
112
+ )}
113
+ <div className="flex items-center gap-2">
114
+ {selectedRows.size === 0 && (
115
+ <>
116
+ <TooltipProvider>
117
+ <Tooltip>
118
+ <TooltipContent>
119
+ <p>Refresh</p>
120
+ </TooltipContent>
121
+ </Tooltip>
122
+ </TooltipProvider>
123
+ <Button
124
+ className="h-10 px-4 font-medium dark:bg-emerald-300 dark:text-black"
125
+ onClick={() => setAddDialogOpen(true)}
126
+ >
127
+ <UserPlus className="h-4 w-4 mr-2" />
128
+ New User
129
+ </Button>
130
+ </>
131
+ )}
132
+ </div>
133
+ </div>
134
+ </div>
135
+ )}
136
+
137
+ {/* Main Content */}
138
+
139
+ {selectedSection === 'users' && (
140
+ <UsersTab
141
+ searchQuery={searchQuery}
142
+ selectedRows={selectedRows}
143
+ onSelectedRowsChange={setSelectedRows}
144
+ />
145
+ )}
146
+
147
+ {selectedSection === 'auth-methods' && <AuthMethodTab />}
148
+ </div>
149
+
150
+ <UserFormDialog open={addDialogOpen} onOpenChange={setAddDialogOpen} />
151
+
152
+ <ConfirmDialog
153
+ open={confirmDeleteOpen}
154
+ onOpenChange={setConfirmDeleteOpen}
155
+ title={`Delete ${selectedRows.size} ${selectedRows.size === 1 ? 'User' : 'Users'}`}
156
+ description={
157
+ <span>
158
+ Are you sure to <strong>permanently delete {selectedRows.size}</strong>{' '}
159
+ {selectedRows.size === 1 ? 'user' : 'users'}? This action cannot be undone.
160
+ </span>
161
+ }
162
+ confirmText="Delete"
163
+ cancelText="Cancel"
164
+ destructive
165
+ onConfirm={handleBulkDelete}
166
+ />
167
+ </div>
168
+ );
169
+ }