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,483 @@
1
+ import React, { useEffect, useRef, useState } from 'react';
2
+ import { Calendar, ChevronLeft, ChevronRight, Clock } from 'lucide-react';
3
+ import { Button } from '@/components/radix/Button';
4
+ import { cn } from '@/lib/utils/utils';
5
+ import { format, parse } from 'date-fns';
6
+ import { Popover, PopoverContent, PopoverTrigger } from '@/components/radix/Popover';
7
+ import type { DateCellEditorProps } from './types';
8
+ import { ColumnType } from '@insforge/shared-schemas';
9
+
10
+ type PickerMode = 'day' | 'month' | 'year';
11
+
12
+ interface TimeColumnProps {
13
+ label: string;
14
+ value: number;
15
+ range: number;
16
+ onChange: (value: number) => void;
17
+ scrollRef: React.RefObject<HTMLDivElement | null>;
18
+ }
19
+
20
+ const MONTHS = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'];
21
+ const WEEKDAYS = ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'];
22
+
23
+ // Reusable time column component
24
+ function TimeColumn({ label, value, range, onChange, scrollRef }: TimeColumnProps) {
25
+ return (
26
+ <div className="flex-1">
27
+ <div className="text-xs text-muted-foreground mb-1 text-center">{label}</div>
28
+ <div
29
+ ref={scrollRef}
30
+ className="h-60 overflow-y-auto border border-border-gray dark:border-neutral-600 rounded p-1 scrollbar-thin"
31
+ >
32
+ {Array.from({ length: range }, (_, i) => (
33
+ <button
34
+ key={i}
35
+ onClick={() => onChange(i)}
36
+ className={cn(
37
+ 'w-full px-1 py-1 text-sm rounded hover:bg-gray-100 dark:hover:text-zinc-950 text-center',
38
+ value === i && 'bg-primary text-primary-foreground hover:bg-primary'
39
+ )}
40
+ >
41
+ {i.toString().padStart(2, '0')}
42
+ </button>
43
+ ))}
44
+ </div>
45
+ </div>
46
+ );
47
+ }
48
+
49
+ export function DateCellEditor({
50
+ value,
51
+ type = ColumnType.DATETIME,
52
+ nullable,
53
+ onValueChange,
54
+ onCancel,
55
+ className,
56
+ }: DateCellEditorProps) {
57
+ const [open, setOpen] = useState(true);
58
+ const [pickerMode, setPickerMode] = useState<PickerMode>('day');
59
+ const [selectedDate, setSelectedDate] = useState<Date>(() => {
60
+ if (value && value !== 'null') {
61
+ if (type === ColumnType.DATE) {
62
+ return parse(value, 'yyyy-MM-dd', new Date());
63
+ } else {
64
+ return new Date(value);
65
+ }
66
+ }
67
+ return new Date();
68
+ });
69
+
70
+ const [selectedHour, setSelectedHour] = useState(() => {
71
+ if (value && value !== 'null' && type === ColumnType.DATETIME) {
72
+ return new Date(value).getHours();
73
+ }
74
+ return new Date().getHours();
75
+ });
76
+
77
+ const [selectedMinute, setSelectedMinute] = useState(() => {
78
+ if (value && value !== 'null' && type === ColumnType.DATETIME) {
79
+ return new Date(value).getMinutes();
80
+ }
81
+ return new Date().getMinutes();
82
+ });
83
+
84
+ const [displayMonth, setDisplayMonth] = useState(selectedDate.getMonth());
85
+ const [displayYear, setDisplayYear] = useState(selectedDate.getFullYear());
86
+
87
+ const hourScrollRef = useRef<HTMLDivElement>(null);
88
+ const minuteScrollRef = useRef<HTMLDivElement>(null);
89
+
90
+ useEffect(() => {
91
+ setOpen(true);
92
+ }, []);
93
+
94
+ useEffect(() => {
95
+ // Auto-scroll to selected time values when popover opens
96
+ if (open && type === ColumnType.DATETIME) {
97
+ setTimeout(() => {
98
+ if (hourScrollRef.current) {
99
+ const hourButton = hourScrollRef.current.querySelector(
100
+ `button:nth-child(${selectedHour + 1})`
101
+ );
102
+ if (hourButton) {
103
+ hourButton.scrollIntoView({ block: 'center', behavior: 'smooth' });
104
+ }
105
+ }
106
+ if (minuteScrollRef.current) {
107
+ const minuteButton = minuteScrollRef.current.querySelector(
108
+ `button:nth-child(${selectedMinute + 1})`
109
+ );
110
+ if (minuteButton) {
111
+ minuteButton.scrollIntoView({ block: 'center', behavior: 'smooth' });
112
+ }
113
+ }
114
+ }, 100);
115
+ }
116
+ }, [open, selectedHour, selectedMinute, type]);
117
+
118
+ const handleOpenChange = (isOpen: boolean) => {
119
+ if (!isOpen) {
120
+ onCancel();
121
+ }
122
+ setOpen(isOpen);
123
+ };
124
+
125
+ const getDaysInMonth = (year: number, month: number) => {
126
+ return new Date(year, month + 1, 0).getDate();
127
+ };
128
+
129
+ const getFirstDayOfMonth = (year: number, month: number) => {
130
+ return new Date(year, month, 1).getDay();
131
+ };
132
+
133
+ const handleDateClick = (day: number) => {
134
+ const newDate = new Date(displayYear, displayMonth, day);
135
+ setSelectedDate(newDate);
136
+
137
+ if (type === ColumnType.DATE) {
138
+ const dateString = format(newDate, 'yyyy-MM-dd');
139
+ onValueChange(dateString);
140
+ setOpen(false);
141
+ }
142
+ };
143
+
144
+ const handleMonthClick = (month: number) => {
145
+ setDisplayMonth(month);
146
+ setPickerMode('day');
147
+ };
148
+
149
+ const handleYearClick = (year: number) => {
150
+ setDisplayYear(year);
151
+ setPickerMode('month');
152
+ };
153
+
154
+ const handlePrevMonth = () => {
155
+ if (displayMonth === 0) {
156
+ setDisplayMonth(11);
157
+ setDisplayYear(displayYear - 1);
158
+ } else {
159
+ setDisplayMonth(displayMonth - 1);
160
+ }
161
+ };
162
+
163
+ const handleNextMonth = () => {
164
+ if (displayMonth === 11) {
165
+ setDisplayMonth(0);
166
+ setDisplayYear(displayYear + 1);
167
+ } else {
168
+ setDisplayMonth(displayMonth + 1);
169
+ }
170
+ };
171
+
172
+ const handlePrevYear = () => {
173
+ setDisplayYear(displayYear - 1);
174
+ };
175
+
176
+ const handleNextYear = () => {
177
+ setDisplayYear(displayYear + 1);
178
+ };
179
+
180
+ const handlePrevDecade = () => {
181
+ setDisplayYear(displayYear - 10);
182
+ };
183
+
184
+ const handleNextDecade = () => {
185
+ setDisplayYear(displayYear + 10);
186
+ };
187
+
188
+ const handleSave = () => {
189
+ if (type === ColumnType.DATETIME) {
190
+ const dateTime = new Date(selectedDate);
191
+ dateTime.setHours(selectedHour, selectedMinute, 0, 0);
192
+ // Format as local ISO string with timezone offset
193
+ const year = dateTime.getFullYear();
194
+ const month = String(dateTime.getMonth() + 1).padStart(2, '0');
195
+ const day = String(dateTime.getDate()).padStart(2, '0');
196
+ const hours = String(selectedHour).padStart(2, '0');
197
+ const minutes = String(selectedMinute).padStart(2, '0');
198
+
199
+ // Get timezone offset in format +/-HH:MM
200
+ const offset = dateTime.getTimezoneOffset();
201
+ const absOffset = Math.abs(offset);
202
+ const offsetHours = String(Math.floor(absOffset / 60)).padStart(2, '0');
203
+ const offsetMinutes = String(absOffset % 60).padStart(2, '0');
204
+ const offsetSign = offset <= 0 ? '+' : '-';
205
+ const timezoneOffset = `${offsetSign}${offsetHours}:${offsetMinutes}`;
206
+
207
+ const localISOString = `${year}-${month}-${day}T${hours}:${minutes}:00${timezoneOffset}`;
208
+ onValueChange(localISOString);
209
+ } else {
210
+ const dateString = format(selectedDate, 'yyyy-MM-dd');
211
+ onValueChange(dateString);
212
+ }
213
+ setOpen(false);
214
+ };
215
+
216
+ const handleClear = () => {
217
+ if (nullable) {
218
+ onValueChange(null);
219
+ setOpen(false);
220
+ }
221
+ };
222
+
223
+ const formatDisplayValue = () => {
224
+ if (!value || value === 'null') {
225
+ return 'Select date...';
226
+ }
227
+
228
+ if (type === ColumnType.DATETIME) {
229
+ const d = new Date(value);
230
+ return format(d, 'MMM dd, yyyy hh:mm a');
231
+ } else {
232
+ const date = parse(value, 'yyyy-MM-dd', new Date());
233
+ return format(date, 'MMM dd, yyyy');
234
+ }
235
+ };
236
+
237
+ const renderDayPicker = () => {
238
+ const daysInMonth = getDaysInMonth(displayYear, displayMonth);
239
+ const firstDay = getFirstDayOfMonth(displayYear, displayMonth);
240
+ const days = [];
241
+
242
+ // Empty cells for days before month starts
243
+ for (let i = 0; i < firstDay; i++) {
244
+ days.push(<div key={`empty-${i}`} className="h-8 w-8" />);
245
+ }
246
+
247
+ // Days of the month
248
+ for (let day = 1; day <= daysInMonth; day++) {
249
+ const isSelected =
250
+ selectedDate.getDate() === day &&
251
+ selectedDate.getMonth() === displayMonth &&
252
+ selectedDate.getFullYear() === displayYear;
253
+ const isToday =
254
+ new Date().getDate() === day &&
255
+ new Date().getMonth() === displayMonth &&
256
+ new Date().getFullYear() === displayYear;
257
+
258
+ days.push(
259
+ <button
260
+ key={day}
261
+ onClick={() => handleDateClick(day)}
262
+ className={cn(
263
+ 'h-8 w-8 text-sm rounded hover:bg-gray-100 dark:hover:text-zinc-950',
264
+ isSelected && 'bg-primary text-primary-foreground hover:bg-primary',
265
+ isToday && !isSelected && 'font-bold'
266
+ )}
267
+ >
268
+ {day}
269
+ </button>
270
+ );
271
+ }
272
+
273
+ return days;
274
+ };
275
+
276
+ const renderMonthPicker = () => {
277
+ return MONTHS.map((month, index) => {
278
+ const isSelected = selectedDate.getMonth() === index;
279
+ return (
280
+ <button
281
+ key={month}
282
+ onClick={() => handleMonthClick(index)}
283
+ className={cn(
284
+ 'h-12 w-20 text-sm rounded hover:bg-gray-100 dark:hover:text-zinc-950',
285
+ isSelected && 'bg-primary text-primary-foreground hover:bg-primary'
286
+ )}
287
+ >
288
+ {month}
289
+ </button>
290
+ );
291
+ });
292
+ };
293
+
294
+ const renderYearPicker = () => {
295
+ const startYear = Math.floor(displayYear / 10) * 10;
296
+ const years = [];
297
+
298
+ for (let i = 0; i < 12; i++) {
299
+ const year = startYear + i;
300
+ const isSelected = selectedDate.getFullYear() === year;
301
+ years.push(
302
+ <button
303
+ key={year}
304
+ onClick={() => handleYearClick(year)}
305
+ className={cn(
306
+ 'h-12 w-20 text-sm rounded hover:bg-gray-100',
307
+ isSelected && 'bg-primary text-primary-foreground hover:bg-primary'
308
+ )}
309
+ >
310
+ {year}
311
+ </button>
312
+ );
313
+ }
314
+
315
+ return years;
316
+ };
317
+
318
+ return (
319
+ <Popover open={open} onOpenChange={handleOpenChange}>
320
+ <PopoverTrigger asChild>
321
+ <Button
322
+ variant="ghost"
323
+ className={cn(
324
+ 'w-full justify-start text-left text-sm font-normal h-full border-0 p-0 hover:bg-transparent dark:text-white',
325
+ (!value || value === 'null') && 'text-muted-foreground',
326
+ className
327
+ )}
328
+ >
329
+ {type === ColumnType.DATETIME ? (
330
+ <Clock className="mr-2 h-4 w-4" />
331
+ ) : (
332
+ <Calendar className="mr-2 h-4 w-4" />
333
+ )}
334
+ {formatDisplayValue()}
335
+ </Button>
336
+ </PopoverTrigger>
337
+ <PopoverContent
338
+ className="overflow-hidden w-auto p-0 dark:bg-neutral-800 dark:border-neutral-700"
339
+ align="start"
340
+ side="bottom"
341
+ >
342
+ <div className={cn('flex', type === ColumnType.DATETIME && '')}>
343
+ <div className="p-3">
344
+ {/* Header */}
345
+ <div className="flex items-center justify-between mb-3">
346
+ <Button
347
+ variant="ghost"
348
+ size="icon"
349
+ className="h-7 w-7"
350
+ onClick={
351
+ pickerMode === 'day'
352
+ ? handlePrevMonth
353
+ : pickerMode === 'month'
354
+ ? handlePrevYear
355
+ : handlePrevDecade
356
+ }
357
+ >
358
+ <ChevronLeft className="h-4 w-4" />
359
+ </Button>
360
+
361
+ <button
362
+ className="text-sm font-medium px-2 py-1 rounded hover:bg-gray-100 dark:hover:text-zinc-950"
363
+ onClick={() => {
364
+ if (pickerMode === 'day') {
365
+ setPickerMode('month');
366
+ } else if (pickerMode === 'month') {
367
+ setPickerMode('year');
368
+ }
369
+ }}
370
+ >
371
+ {pickerMode === 'day' && `${MONTHS[displayMonth]} ${displayYear}`}
372
+ {pickerMode === 'month' && displayYear}
373
+ {pickerMode === 'year' &&
374
+ `${Math.floor(displayYear / 10) * 10}-${Math.floor(displayYear / 10) * 10 + 9}`}
375
+ </button>
376
+
377
+ <Button
378
+ variant="ghost"
379
+ size="icon"
380
+ className="h-7 w-7"
381
+ onClick={
382
+ pickerMode === 'day'
383
+ ? handleNextMonth
384
+ : pickerMode === 'month'
385
+ ? handleNextYear
386
+ : handleNextDecade
387
+ }
388
+ >
389
+ <ChevronRight className="h-4 w-4" />
390
+ </Button>
391
+ </div>
392
+
393
+ {/* Calendar Grid - Fixed dimensions */}
394
+ <div className="w-70 h-60">
395
+ {pickerMode === 'day' && (
396
+ <>
397
+ {/* Weekday headers */}
398
+ <div className="grid grid-cols-7 gap-1 mb-1">
399
+ {WEEKDAYS.map((day) => (
400
+ <div
401
+ key={day}
402
+ className="h-8 w-8 text-xs text-muted-foreground flex items-center justify-center"
403
+ >
404
+ {day}
405
+ </div>
406
+ ))}
407
+ </div>
408
+
409
+ {/* Days */}
410
+ <div className="grid grid-cols-7 gap-1">{renderDayPicker()}</div>
411
+ </>
412
+ )}
413
+
414
+ {pickerMode === 'month' && (
415
+ <div className="grid grid-cols-3 gap-2 pt-4">{renderMonthPicker()}</div>
416
+ )}
417
+
418
+ {pickerMode === 'year' && (
419
+ <div className="grid grid-cols-3 gap-2 pt-4">{renderYearPicker()}</div>
420
+ )}
421
+ </div>
422
+ </div>
423
+
424
+ {type === ColumnType.DATETIME && (
425
+ <div className="border-l border-border-gray dark:border-neutral-600 bg-muted/30 dark:bg-neutral-800 w-35">
426
+ <div className="p-3">
427
+ <div className="flex items-center justify-center gap-2 mb-3">
428
+ <Clock className="h-4 w-4 text-muted-foreground" />
429
+ <span className="text-sm font-medium">Time</span>
430
+ </div>
431
+
432
+ <div className="flex gap-2">
433
+ <TimeColumn
434
+ label="Hour"
435
+ value={selectedHour}
436
+ range={24}
437
+ onChange={setSelectedHour}
438
+ scrollRef={hourScrollRef}
439
+ />
440
+ <TimeColumn
441
+ label="Min"
442
+ value={selectedMinute}
443
+ range={60}
444
+ onChange={setSelectedMinute}
445
+ scrollRef={minuteScrollRef}
446
+ />
447
+ </div>
448
+ </div>
449
+ </div>
450
+ )}
451
+ </div>
452
+
453
+ {/* Action buttons */}
454
+ <div className="flex items-center gap-2 p-3 border-t border-border-gray dark:border-neutral-600">
455
+ {nullable && (
456
+ <Button
457
+ variant="outline"
458
+ size="sm"
459
+ onClick={handleClear}
460
+ className="flex-1 dark:bg-neutral-600 dark:text-white dark:hover:bg-neutral-700"
461
+ >
462
+ Null
463
+ </Button>
464
+ )}
465
+ <Button
466
+ variant="outline"
467
+ size="sm"
468
+ onClick={() => {
469
+ onCancel();
470
+ setOpen(false);
471
+ }}
472
+ className="flex-1 dark:bg-neutral-600 dark:text-white dark:hover:bg-neutral-700"
473
+ >
474
+ Cancel
475
+ </Button>
476
+ <Button size="sm" onClick={handleSave} className="flex-1">
477
+ Save
478
+ </Button>
479
+ </div>
480
+ </PopoverContent>
481
+ </Popover>
482
+ );
483
+ }