alepha 0.15.1 → 0.15.3

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 (523) hide show
  1. package/README.md +68 -80
  2. package/dist/api/audits/index.d.ts +10 -33
  3. package/dist/api/audits/index.d.ts.map +1 -1
  4. package/dist/api/audits/index.js +10 -33
  5. package/dist/api/audits/index.js.map +1 -1
  6. package/dist/api/files/index.d.ts +10 -3
  7. package/dist/api/files/index.d.ts.map +1 -1
  8. package/dist/api/files/index.js +10 -3
  9. package/dist/api/files/index.js.map +1 -1
  10. package/dist/api/jobs/index.d.ts +162 -155
  11. package/dist/api/jobs/index.d.ts.map +1 -1
  12. package/dist/api/jobs/index.js +10 -3
  13. package/dist/api/jobs/index.js.map +1 -1
  14. package/dist/api/keys/index.d.ts +413 -0
  15. package/dist/api/keys/index.d.ts.map +1 -0
  16. package/dist/api/keys/index.js +476 -0
  17. package/dist/api/keys/index.js.map +1 -0
  18. package/dist/api/notifications/index.d.ts +10 -4
  19. package/dist/api/notifications/index.d.ts.map +1 -1
  20. package/dist/api/notifications/index.js +10 -4
  21. package/dist/api/notifications/index.js.map +1 -1
  22. package/dist/api/parameters/index.d.ts +43 -50
  23. package/dist/api/parameters/index.d.ts.map +1 -1
  24. package/dist/api/parameters/index.js +30 -37
  25. package/dist/api/parameters/index.js.map +1 -1
  26. package/dist/api/users/index.d.ts +1081 -760
  27. package/dist/api/users/index.d.ts.map +1 -1
  28. package/dist/api/users/index.js +2539 -218
  29. package/dist/api/users/index.js.map +1 -1
  30. package/dist/api/verifications/index.d.ts +138 -132
  31. package/dist/api/verifications/index.d.ts.map +1 -1
  32. package/dist/api/verifications/index.js +12 -4
  33. package/dist/api/verifications/index.js.map +1 -1
  34. package/dist/batch/index.d.ts +20 -40
  35. package/dist/batch/index.d.ts.map +1 -1
  36. package/dist/batch/index.js +31 -44
  37. package/dist/batch/index.js.map +1 -1
  38. package/dist/bucket/index.d.ts +440 -8
  39. package/dist/bucket/index.d.ts.map +1 -1
  40. package/dist/bucket/index.js +1861 -12
  41. package/dist/bucket/index.js.map +1 -1
  42. package/dist/cache/core/index.d.ts +179 -7
  43. package/dist/cache/core/index.d.ts.map +1 -1
  44. package/dist/cache/core/index.js +213 -7
  45. package/dist/cache/core/index.js.map +1 -1
  46. package/dist/cache/redis/index.d.ts +1 -0
  47. package/dist/cache/redis/index.d.ts.map +1 -1
  48. package/dist/cache/redis/index.js +4 -0
  49. package/dist/cache/redis/index.js.map +1 -1
  50. package/dist/cli/index.d.ts +638 -5645
  51. package/dist/cli/index.d.ts.map +1 -1
  52. package/dist/cli/index.js +2550 -368
  53. package/dist/cli/index.js.map +1 -1
  54. package/dist/command/index.d.ts +203 -45
  55. package/dist/command/index.d.ts.map +1 -1
  56. package/dist/command/index.js +2060 -71
  57. package/dist/command/index.js.map +1 -1
  58. package/dist/core/index.browser.js +70 -40
  59. package/dist/core/index.browser.js.map +1 -1
  60. package/dist/core/index.d.ts +34 -13
  61. package/dist/core/index.d.ts.map +1 -1
  62. package/dist/core/index.js +90 -40
  63. package/dist/core/index.js.map +1 -1
  64. package/dist/core/index.native.js +70 -40
  65. package/dist/core/index.native.js.map +1 -1
  66. package/dist/datetime/index.d.ts +15 -0
  67. package/dist/datetime/index.d.ts.map +1 -1
  68. package/dist/datetime/index.js +15 -0
  69. package/dist/datetime/index.js.map +1 -1
  70. package/dist/email/index.d.ts +323 -20
  71. package/dist/email/index.d.ts.map +1 -1
  72. package/dist/email/index.js +1857 -7
  73. package/dist/email/index.js.map +1 -1
  74. package/dist/fake/index.d.ts +90 -8
  75. package/dist/fake/index.d.ts.map +1 -1
  76. package/dist/fake/index.js +91 -20
  77. package/dist/fake/index.js.map +1 -1
  78. package/dist/lock/core/index.d.ts +11 -4
  79. package/dist/lock/core/index.d.ts.map +1 -1
  80. package/dist/lock/core/index.js +11 -4
  81. package/dist/lock/core/index.js.map +1 -1
  82. package/dist/logger/index.d.ts +17 -66
  83. package/dist/logger/index.d.ts.map +1 -1
  84. package/dist/logger/index.js +14 -63
  85. package/dist/logger/index.js.map +1 -1
  86. package/dist/mcp/index.d.ts +10 -30
  87. package/dist/mcp/index.d.ts.map +1 -1
  88. package/dist/mcp/index.js +12 -35
  89. package/dist/mcp/index.js.map +1 -1
  90. package/dist/orm/index.browser.js +3 -3
  91. package/dist/orm/index.browser.js.map +1 -1
  92. package/dist/orm/index.bun.js +39 -20
  93. package/dist/orm/index.bun.js.map +1 -1
  94. package/dist/orm/index.d.ts +517 -540
  95. package/dist/orm/index.d.ts.map +1 -1
  96. package/dist/orm/index.js +58 -71
  97. package/dist/orm/index.js.map +1 -1
  98. package/dist/queue/core/index.d.ts +18 -10
  99. package/dist/queue/core/index.d.ts.map +1 -1
  100. package/dist/queue/core/index.js +14 -6
  101. package/dist/queue/core/index.js.map +1 -1
  102. package/dist/react/auth/index.browser.js +108 -0
  103. package/dist/react/auth/index.browser.js.map +1 -0
  104. package/dist/react/auth/index.d.ts +100 -0
  105. package/dist/react/auth/index.d.ts.map +1 -0
  106. package/dist/react/auth/index.js +145 -0
  107. package/dist/react/auth/index.js.map +1 -0
  108. package/dist/react/core/index.d.ts +469 -0
  109. package/dist/react/core/index.d.ts.map +1 -0
  110. package/dist/react/core/index.js +464 -0
  111. package/dist/react/core/index.js.map +1 -0
  112. package/dist/react/form/index.d.ts +232 -0
  113. package/dist/react/form/index.d.ts.map +1 -0
  114. package/dist/react/form/index.js +432 -0
  115. package/dist/react/form/index.js.map +1 -0
  116. package/dist/react/head/index.browser.js +423 -0
  117. package/dist/react/head/index.browser.js.map +1 -0
  118. package/dist/react/head/index.d.ts +288 -0
  119. package/dist/react/head/index.d.ts.map +1 -0
  120. package/dist/react/head/index.js +465 -0
  121. package/dist/react/head/index.js.map +1 -0
  122. package/dist/react/i18n/index.d.ts +175 -0
  123. package/dist/react/i18n/index.d.ts.map +1 -0
  124. package/dist/react/i18n/index.js +224 -0
  125. package/dist/react/i18n/index.js.map +1 -0
  126. package/dist/react/router/index.browser.js +1974 -0
  127. package/dist/react/router/index.browser.js.map +1 -0
  128. package/dist/react/router/index.d.ts +1956 -0
  129. package/dist/react/router/index.d.ts.map +1 -0
  130. package/dist/react/router/index.js +4722 -0
  131. package/dist/react/router/index.js.map +1 -0
  132. package/dist/react/websocket/index.d.ts +117 -0
  133. package/dist/react/websocket/index.d.ts.map +1 -0
  134. package/dist/react/websocket/index.js +107 -0
  135. package/dist/react/websocket/index.js.map +1 -0
  136. package/dist/redis/index.bun.js +4 -0
  137. package/dist/redis/index.bun.js.map +1 -1
  138. package/dist/redis/index.d.ts +41 -44
  139. package/dist/redis/index.d.ts.map +1 -1
  140. package/dist/redis/index.js +16 -25
  141. package/dist/redis/index.js.map +1 -1
  142. package/dist/retry/index.d.ts +11 -2
  143. package/dist/retry/index.d.ts.map +1 -1
  144. package/dist/retry/index.js +11 -2
  145. package/dist/retry/index.js.map +1 -1
  146. package/dist/scheduler/index.d.ts +11 -2
  147. package/dist/scheduler/index.d.ts.map +1 -1
  148. package/dist/scheduler/index.js +11 -2
  149. package/dist/scheduler/index.js.map +1 -1
  150. package/dist/security/index.d.ts +140 -49
  151. package/dist/security/index.d.ts.map +1 -1
  152. package/dist/security/index.js +164 -32
  153. package/dist/security/index.js.map +1 -1
  154. package/dist/server/auth/index.d.ts +12 -7
  155. package/dist/server/auth/index.d.ts.map +1 -1
  156. package/dist/server/auth/index.js +12 -7
  157. package/dist/server/auth/index.js.map +1 -1
  158. package/dist/server/cache/index.d.ts +7 -22
  159. package/dist/server/cache/index.d.ts.map +1 -1
  160. package/dist/server/cache/index.js +7 -22
  161. package/dist/server/cache/index.js.map +1 -1
  162. package/dist/server/compress/index.d.ts +10 -2
  163. package/dist/server/compress/index.d.ts.map +1 -1
  164. package/dist/server/compress/index.js +10 -2
  165. package/dist/server/compress/index.js.map +1 -1
  166. package/dist/server/cookies/index.d.ts +40 -16
  167. package/dist/server/cookies/index.d.ts.map +1 -1
  168. package/dist/server/cookies/index.js +7 -5
  169. package/dist/server/cookies/index.js.map +1 -1
  170. package/dist/server/core/index.d.ts +124 -23
  171. package/dist/server/core/index.d.ts.map +1 -1
  172. package/dist/server/core/index.js +231 -14
  173. package/dist/server/core/index.js.map +1 -1
  174. package/dist/server/cors/index.d.ts +13 -23
  175. package/dist/server/cors/index.d.ts.map +1 -1
  176. package/dist/server/cors/index.js +7 -21
  177. package/dist/server/cors/index.js.map +1 -1
  178. package/dist/server/health/index.d.ts +8 -2
  179. package/dist/server/health/index.d.ts.map +1 -1
  180. package/dist/server/health/index.js +8 -2
  181. package/dist/server/health/index.js.map +1 -1
  182. package/dist/server/helmet/index.d.ts +11 -3
  183. package/dist/server/helmet/index.d.ts.map +1 -1
  184. package/dist/server/helmet/index.js +11 -3
  185. package/dist/server/helmet/index.js.map +1 -1
  186. package/dist/server/links/index.d.ts +11 -6
  187. package/dist/server/links/index.d.ts.map +1 -1
  188. package/dist/server/links/index.js +11 -6
  189. package/dist/server/links/index.js.map +1 -1
  190. package/dist/server/metrics/index.d.ts +10 -3
  191. package/dist/server/metrics/index.d.ts.map +1 -1
  192. package/dist/server/metrics/index.js +10 -3
  193. package/dist/server/metrics/index.js.map +1 -1
  194. package/dist/server/multipart/index.d.ts +9 -3
  195. package/dist/server/multipart/index.d.ts.map +1 -1
  196. package/dist/server/multipart/index.js +9 -3
  197. package/dist/server/multipart/index.js.map +1 -1
  198. package/dist/server/proxy/index.d.ts +8 -2
  199. package/dist/server/proxy/index.d.ts.map +1 -1
  200. package/dist/server/proxy/index.js +8 -2
  201. package/dist/server/proxy/index.js.map +1 -1
  202. package/dist/server/rate-limit/index.d.ts +30 -35
  203. package/dist/server/rate-limit/index.d.ts.map +1 -1
  204. package/dist/server/rate-limit/index.js +18 -55
  205. package/dist/server/rate-limit/index.js.map +1 -1
  206. package/dist/server/static/index.d.ts +137 -4
  207. package/dist/server/static/index.d.ts.map +1 -1
  208. package/dist/server/static/index.js +1853 -5
  209. package/dist/server/static/index.js.map +1 -1
  210. package/dist/server/swagger/index.d.ts +309 -6
  211. package/dist/server/swagger/index.d.ts.map +1 -1
  212. package/dist/server/swagger/index.js +1854 -6
  213. package/dist/server/swagger/index.js.map +1 -1
  214. package/dist/sms/index.d.ts +309 -7
  215. package/dist/sms/index.d.ts.map +1 -1
  216. package/dist/sms/index.js +1856 -7
  217. package/dist/sms/index.js.map +1 -1
  218. package/dist/system/index.browser.js +1218 -0
  219. package/dist/system/index.browser.js.map +1 -0
  220. package/dist/{file → system}/index.d.ts +343 -16
  221. package/dist/system/index.d.ts.map +1 -0
  222. package/dist/{file → system}/index.js +419 -22
  223. package/dist/system/index.js.map +1 -0
  224. package/dist/thread/index.d.ts +11 -2
  225. package/dist/thread/index.d.ts.map +1 -1
  226. package/dist/thread/index.js +11 -2
  227. package/dist/thread/index.js.map +1 -1
  228. package/dist/topic/core/index.d.ts +12 -5
  229. package/dist/topic/core/index.d.ts.map +1 -1
  230. package/dist/topic/core/index.js +12 -5
  231. package/dist/topic/core/index.js.map +1 -1
  232. package/dist/vite/index.d.ts +5 -6272
  233. package/dist/vite/index.d.ts.map +1 -1
  234. package/dist/vite/index.js +23 -10
  235. package/dist/vite/index.js.map +1 -1
  236. package/dist/websocket/index.d.ts +12 -8
  237. package/dist/websocket/index.d.ts.map +1 -1
  238. package/dist/websocket/index.js +12 -8
  239. package/dist/websocket/index.js.map +1 -1
  240. package/package.json +82 -11
  241. package/src/api/audits/index.ts +10 -33
  242. package/src/api/files/__tests__/$bucket.spec.ts +1 -1
  243. package/src/api/files/controllers/AdminFileStatsController.spec.ts +1 -1
  244. package/src/api/files/controllers/FileController.spec.ts +1 -1
  245. package/src/api/files/index.ts +10 -3
  246. package/src/api/files/jobs/FileJobs.spec.ts +1 -1
  247. package/src/api/files/services/FileService.spec.ts +1 -1
  248. package/src/api/jobs/index.ts +10 -3
  249. package/src/api/keys/controllers/AdminApiKeyController.ts +75 -0
  250. package/src/api/keys/controllers/ApiKeyController.ts +103 -0
  251. package/src/api/keys/entities/apiKeyEntity.ts +41 -0
  252. package/src/api/keys/index.ts +49 -0
  253. package/src/api/keys/schemas/adminApiKeyQuerySchema.ts +7 -0
  254. package/src/api/keys/schemas/adminApiKeyResourceSchema.ts +17 -0
  255. package/src/api/keys/schemas/createApiKeyBodySchema.ts +7 -0
  256. package/src/api/keys/schemas/createApiKeyResponseSchema.ts +11 -0
  257. package/src/api/keys/schemas/listApiKeyResponseSchema.ts +15 -0
  258. package/src/api/keys/schemas/revokeApiKeyParamsSchema.ts +5 -0
  259. package/src/api/keys/schemas/revokeApiKeyResponseSchema.ts +5 -0
  260. package/src/api/keys/services/ApiKeyService.spec.ts +553 -0
  261. package/src/api/keys/services/ApiKeyService.ts +306 -0
  262. package/src/api/logs/TODO.md +55 -0
  263. package/src/api/notifications/index.ts +10 -4
  264. package/src/api/parameters/index.ts +9 -30
  265. package/src/api/parameters/primitives/$config.ts +12 -4
  266. package/src/api/parameters/services/ConfigStore.ts +9 -3
  267. package/src/api/users/__tests__/ApiKeys-integration.spec.ts +1035 -0
  268. package/src/api/users/__tests__/ApiKeys.spec.ts +401 -0
  269. package/src/api/users/index.ts +14 -3
  270. package/src/api/users/primitives/$realm.ts +33 -5
  271. package/src/api/users/providers/RealmProvider.ts +1 -12
  272. package/src/api/users/services/SessionService.ts +1 -1
  273. package/src/api/verifications/controllers/VerificationController.ts +2 -0
  274. package/src/api/verifications/index.ts +10 -4
  275. package/src/batch/index.ts +9 -36
  276. package/src/batch/primitives/$batch.ts +0 -8
  277. package/src/batch/providers/BatchProvider.ts +29 -2
  278. package/src/bucket/__tests__/shared.ts +1 -1
  279. package/src/bucket/index.ts +13 -6
  280. package/src/bucket/primitives/$bucket.ts +1 -1
  281. package/src/bucket/providers/LocalFileStorageProvider.ts +1 -1
  282. package/src/bucket/providers/MemoryFileStorageProvider.ts +1 -1
  283. package/src/cache/core/__tests__/shared.ts +30 -0
  284. package/src/cache/core/index.ts +11 -6
  285. package/src/cache/core/primitives/$cache.spec.ts +5 -0
  286. package/src/cache/core/providers/CacheProvider.ts +17 -0
  287. package/src/cache/core/providers/MemoryCacheProvider.ts +300 -1
  288. package/src/cache/redis/__tests__/cache-redis.spec.ts +5 -0
  289. package/src/cache/redis/providers/RedisCacheProvider.ts +9 -0
  290. package/src/cli/apps/AlephaCli.ts +1 -14
  291. package/src/cli/apps/AlephaPackageBuilderCli.ts +10 -1
  292. package/src/cli/atoms/buildOptions.ts +99 -9
  293. package/src/cli/commands/build.ts +150 -37
  294. package/src/cli/commands/db.ts +22 -18
  295. package/src/cli/commands/deploy.ts +1 -1
  296. package/src/cli/commands/dev.ts +1 -20
  297. package/src/cli/commands/gen/env.ts +5 -2
  298. package/src/cli/commands/gen/openapi.ts +5 -2
  299. package/src/cli/commands/init.spec.ts +588 -0
  300. package/src/cli/commands/init.ts +115 -58
  301. package/src/cli/commands/lint.ts +7 -1
  302. package/src/cli/commands/typecheck.ts +11 -0
  303. package/src/cli/providers/AppEntryProvider.ts +1 -1
  304. package/src/cli/providers/ViteBuildProvider.ts +8 -50
  305. package/src/cli/providers/ViteDevServerProvider.ts +35 -16
  306. package/src/cli/services/AlephaCliUtils.ts +52 -121
  307. package/src/cli/services/PackageManagerUtils.ts +129 -11
  308. package/src/cli/services/ProjectScaffolder.spec.ts +97 -0
  309. package/src/cli/services/ProjectScaffolder.ts +148 -81
  310. package/src/cli/services/ViteUtils.ts +82 -0
  311. package/src/cli/{assets/claudeMd.ts → templates/agentMd.ts} +37 -24
  312. package/src/cli/templates/apiAppSecurityTs.ts +11 -0
  313. package/src/cli/templates/apiIndexTs.ts +30 -0
  314. package/src/cli/templates/gitignore.ts +39 -0
  315. package/src/cli/{assets → templates}/mainCss.ts +11 -2
  316. package/src/cli/templates/mainServerTs.ts +33 -0
  317. package/src/cli/templates/webAppRouterTs.ts +74 -0
  318. package/src/cli/templates/webHelloComponentTsx.ts +30 -0
  319. package/src/command/helpers/Runner.spec.ts +139 -0
  320. package/src/command/helpers/Runner.ts +7 -22
  321. package/src/command/index.ts +12 -4
  322. package/src/command/providers/CliProvider.spec.ts +1392 -0
  323. package/src/command/providers/CliProvider.ts +320 -47
  324. package/src/core/Alepha.ts +34 -27
  325. package/src/core/__tests__/Alepha-start.spec.ts +4 -4
  326. package/src/core/helpers/jsonSchemaToTypeBox.spec.ts +771 -0
  327. package/src/core/helpers/jsonSchemaToTypeBox.ts +62 -10
  328. package/src/core/index.shared.ts +1 -0
  329. package/src/core/index.ts +20 -0
  330. package/src/core/providers/EventManager.spec.ts +0 -71
  331. package/src/core/providers/EventManager.ts +3 -15
  332. package/src/core/providers/Json.ts +2 -14
  333. package/src/datetime/index.ts +15 -0
  334. package/src/email/index.ts +10 -5
  335. package/src/email/providers/LocalEmailProvider.spec.ts +1 -1
  336. package/src/email/providers/LocalEmailProvider.ts +1 -1
  337. package/src/fake/__tests__/keyName.example.ts +1 -1
  338. package/src/fake/__tests__/keyName.spec.ts +5 -5
  339. package/src/fake/index.ts +9 -6
  340. package/src/fake/providers/FakeProvider.spec.ts +258 -40
  341. package/src/fake/providers/FakeProvider.ts +133 -19
  342. package/src/lock/core/index.ts +11 -4
  343. package/src/logger/index.ts +17 -66
  344. package/src/mcp/index.ts +10 -27
  345. package/src/mcp/transports/SseMcpTransport.ts +0 -11
  346. package/src/orm/__tests__/PostgresProvider.spec.ts +2 -2
  347. package/src/orm/index.browser.ts +2 -2
  348. package/src/orm/index.bun.ts +5 -3
  349. package/src/orm/index.ts +23 -53
  350. package/src/orm/providers/drivers/BunSqliteProvider.ts +5 -1
  351. package/src/orm/providers/drivers/CloudflareD1Provider.ts +57 -30
  352. package/src/orm/providers/drivers/DatabaseProvider.ts +9 -1
  353. package/src/orm/providers/drivers/NodeSqliteProvider.ts +4 -1
  354. package/src/orm/services/Repository.ts +7 -3
  355. package/src/queue/core/index.ts +14 -6
  356. package/src/react/auth/__tests__/$auth.spec.ts +202 -0
  357. package/src/react/auth/hooks/useAuth.ts +32 -0
  358. package/src/react/auth/index.browser.ts +13 -0
  359. package/src/react/auth/index.shared.ts +2 -0
  360. package/src/react/auth/index.ts +48 -0
  361. package/src/react/auth/providers/ReactAuthProvider.ts +16 -0
  362. package/src/react/auth/services/ReactAuth.ts +135 -0
  363. package/src/react/core/__tests__/Router.spec.tsx +169 -0
  364. package/src/react/core/components/ClientOnly.tsx +49 -0
  365. package/src/react/core/components/ErrorBoundary.tsx +73 -0
  366. package/src/react/core/contexts/AlephaContext.ts +7 -0
  367. package/src/react/core/contexts/AlephaProvider.tsx +42 -0
  368. package/src/react/core/hooks/useAction.browser.spec.tsx +569 -0
  369. package/src/react/core/hooks/useAction.ts +480 -0
  370. package/src/react/core/hooks/useAlepha.ts +26 -0
  371. package/src/react/core/hooks/useClient.ts +17 -0
  372. package/src/react/core/hooks/useEvents.ts +51 -0
  373. package/src/react/core/hooks/useInject.ts +12 -0
  374. package/src/react/core/hooks/useStore.ts +52 -0
  375. package/src/react/core/index.ts +90 -0
  376. package/src/react/form/components/FormState.tsx +17 -0
  377. package/src/react/form/errors/FormValidationError.ts +18 -0
  378. package/src/react/form/hooks/useForm.browser.spec.tsx +366 -0
  379. package/src/react/form/hooks/useForm.ts +47 -0
  380. package/src/react/form/hooks/useFormState.ts +130 -0
  381. package/src/react/form/index.ts +44 -0
  382. package/src/react/form/services/FormModel.ts +614 -0
  383. package/src/react/head/helpers/SeoExpander.spec.ts +203 -0
  384. package/src/react/head/helpers/SeoExpander.ts +142 -0
  385. package/src/react/head/hooks/useHead.spec.tsx +288 -0
  386. package/src/react/head/hooks/useHead.ts +62 -0
  387. package/src/react/head/index.browser.ts +26 -0
  388. package/src/react/head/index.ts +44 -0
  389. package/src/react/head/interfaces/Head.ts +105 -0
  390. package/src/react/head/primitives/$head.ts +25 -0
  391. package/src/react/head/providers/BrowserHeadProvider.browser.spec.ts +196 -0
  392. package/src/react/head/providers/BrowserHeadProvider.ts +212 -0
  393. package/src/react/head/providers/HeadProvider.ts +168 -0
  394. package/src/react/head/providers/ServerHeadProvider.ts +31 -0
  395. package/src/react/i18n/__tests__/integration.spec.tsx +239 -0
  396. package/src/react/i18n/components/Localize.spec.tsx +357 -0
  397. package/src/react/i18n/components/Localize.tsx +35 -0
  398. package/src/react/i18n/hooks/useI18n.browser.spec.tsx +438 -0
  399. package/src/react/i18n/hooks/useI18n.ts +18 -0
  400. package/src/react/i18n/index.ts +41 -0
  401. package/src/react/i18n/primitives/$dictionary.ts +69 -0
  402. package/src/react/i18n/providers/I18nProvider.spec.ts +389 -0
  403. package/src/react/i18n/providers/I18nProvider.ts +278 -0
  404. package/src/react/router/__tests__/page-head-browser.browser.spec.ts +95 -0
  405. package/src/react/router/__tests__/page-head.spec.ts +48 -0
  406. package/src/react/router/__tests__/seo-head.spec.ts +125 -0
  407. package/src/react/router/atoms/ssrManifestAtom.ts +58 -0
  408. package/src/react/router/components/ErrorViewer.tsx +872 -0
  409. package/src/react/router/components/Link.tsx +23 -0
  410. package/src/react/router/components/NestedView.tsx +223 -0
  411. package/src/react/router/components/NotFound.tsx +30 -0
  412. package/src/react/router/constants/PAGE_PRELOAD_KEY.ts +6 -0
  413. package/src/react/router/contexts/RouterLayerContext.ts +12 -0
  414. package/src/react/router/errors/Redirection.ts +28 -0
  415. package/src/react/router/hooks/useActive.ts +52 -0
  416. package/src/react/router/hooks/useQueryParams.ts +63 -0
  417. package/src/react/router/hooks/useRouter.ts +20 -0
  418. package/src/react/router/hooks/useRouterState.ts +11 -0
  419. package/src/react/router/index.browser.ts +45 -0
  420. package/src/react/router/index.shared.ts +19 -0
  421. package/src/react/router/index.ts +146 -0
  422. package/src/react/router/primitives/$page.browser.spec.tsx +851 -0
  423. package/src/react/router/primitives/$page.spec.tsx +676 -0
  424. package/src/react/router/primitives/$page.ts +489 -0
  425. package/src/react/router/providers/ReactBrowserProvider.ts +312 -0
  426. package/src/react/router/providers/ReactBrowserRendererProvider.ts +25 -0
  427. package/src/react/router/providers/ReactBrowserRouterProvider.ts +168 -0
  428. package/src/react/router/providers/ReactPageProvider.ts +726 -0
  429. package/src/react/router/providers/ReactPreloadProvider.spec.ts +142 -0
  430. package/src/react/router/providers/ReactPreloadProvider.ts +85 -0
  431. package/src/react/router/providers/ReactServerProvider.spec.tsx +316 -0
  432. package/src/react/router/providers/ReactServerProvider.ts +487 -0
  433. package/src/react/router/providers/ReactServerTemplateProvider.spec.ts +210 -0
  434. package/src/react/router/providers/ReactServerTemplateProvider.ts +542 -0
  435. package/src/react/router/providers/SSRManifestProvider.ts +334 -0
  436. package/src/react/router/services/ReactPageServerService.ts +48 -0
  437. package/src/react/router/services/ReactPageService.ts +27 -0
  438. package/src/react/router/services/ReactRouter.ts +262 -0
  439. package/src/react/websocket/hooks/useRoom.tsx +242 -0
  440. package/src/react/websocket/index.ts +7 -0
  441. package/src/redis/__tests__/redis.spec.ts +13 -0
  442. package/src/redis/index.ts +9 -25
  443. package/src/redis/providers/BunRedisProvider.ts +9 -0
  444. package/src/redis/providers/NodeRedisProvider.ts +8 -0
  445. package/src/redis/providers/RedisProvider.ts +16 -0
  446. package/src/retry/index.ts +11 -2
  447. package/src/router/index.ts +15 -0
  448. package/src/scheduler/index.ts +11 -2
  449. package/src/security/__tests__/BasicAuth.spec.ts +2 -0
  450. package/src/security/__tests__/ServerSecurityProvider.spec.ts +90 -5
  451. package/src/security/index.ts +15 -10
  452. package/src/security/interfaces/IssuerResolver.ts +27 -0
  453. package/src/security/primitives/$issuer.ts +55 -0
  454. package/src/security/providers/SecurityProvider.ts +179 -0
  455. package/src/security/providers/ServerBasicAuthProvider.ts +6 -2
  456. package/src/security/providers/ServerSecurityProvider.ts +63 -41
  457. package/src/server/auth/index.ts +12 -7
  458. package/src/server/cache/index.ts +7 -22
  459. package/src/server/compress/index.ts +10 -2
  460. package/src/server/cookies/index.ts +7 -5
  461. package/src/server/cookies/primitives/$cookie.ts +33 -11
  462. package/src/server/core/index.ts +16 -6
  463. package/src/server/core/interfaces/ServerRequest.ts +83 -1
  464. package/src/server/core/primitives/$action.spec.ts +1 -1
  465. package/src/server/core/primitives/$action.ts +8 -3
  466. package/src/server/core/providers/NodeHttpServerProvider.spec.ts +9 -3
  467. package/src/server/core/providers/NodeHttpServerProvider.ts +9 -3
  468. package/src/server/core/services/ServerRequestParser.spec.ts +520 -0
  469. package/src/server/core/services/ServerRequestParser.ts +306 -13
  470. package/src/server/cors/index.ts +7 -21
  471. package/src/server/cors/primitives/$cors.ts +6 -2
  472. package/src/server/health/index.ts +8 -2
  473. package/src/server/helmet/index.ts +11 -3
  474. package/src/server/links/index.ts +11 -6
  475. package/src/server/metrics/index.ts +10 -3
  476. package/src/server/multipart/index.ts +9 -3
  477. package/src/server/proxy/index.ts +8 -2
  478. package/src/server/rate-limit/index.ts +21 -25
  479. package/src/server/rate-limit/primitives/$rateLimit.ts +6 -2
  480. package/src/server/rate-limit/providers/ServerRateLimitProvider.spec.ts +38 -14
  481. package/src/server/rate-limit/providers/ServerRateLimitProvider.ts +22 -56
  482. package/src/server/static/index.ts +8 -2
  483. package/src/server/static/providers/ServerStaticProvider.ts +1 -1
  484. package/src/server/swagger/index.ts +9 -4
  485. package/src/server/swagger/providers/ServerSwaggerProvider.ts +1 -1
  486. package/src/sms/index.ts +9 -5
  487. package/src/sms/providers/LocalSmsProvider.spec.ts +1 -1
  488. package/src/sms/providers/LocalSmsProvider.ts +1 -1
  489. package/src/system/index.browser.ts +36 -0
  490. package/src/system/index.ts +62 -0
  491. package/src/system/index.workerd.ts +1 -0
  492. package/src/{file → system}/providers/FileSystemProvider.ts +24 -0
  493. package/src/{file → system}/providers/MemoryFileSystemProvider.ts +116 -3
  494. package/src/system/providers/MemoryShellProvider.ts +164 -0
  495. package/src/{file → system}/providers/NodeFileSystemProvider.spec.ts +2 -2
  496. package/src/{file → system}/providers/NodeFileSystemProvider.ts +47 -2
  497. package/src/system/providers/NodeShellProvider.ts +184 -0
  498. package/src/system/providers/ShellProvider.ts +74 -0
  499. package/src/{file → system}/services/FileDetector.spec.ts +2 -2
  500. package/src/thread/index.ts +11 -2
  501. package/src/topic/core/index.ts +12 -5
  502. package/src/vite/tasks/buildClient.ts +2 -7
  503. package/src/vite/tasks/buildServer.ts +19 -13
  504. package/src/vite/tasks/generateCloudflare.ts +10 -7
  505. package/src/vite/tasks/generateDocker.ts +4 -0
  506. package/src/websocket/index.ts +12 -8
  507. package/dist/file/index.d.ts.map +0 -1
  508. package/dist/file/index.js.map +0 -1
  509. package/src/cli/assets/apiIndexTs.ts +0 -16
  510. package/src/cli/assets/mainServerTs.ts +0 -24
  511. package/src/cli/assets/webAppRouterTs.ts +0 -16
  512. package/src/cli/assets/webHelloComponentTsx.ts +0 -20
  513. package/src/cli/providers/ViteTemplateProvider.ts +0 -27
  514. package/src/file/index.ts +0 -43
  515. /package/src/cli/{assets → templates}/apiHelloControllerTs.ts +0 -0
  516. /package/src/cli/{assets → templates}/biomeJson.ts +0 -0
  517. /package/src/cli/{assets → templates}/dummySpecTs.ts +0 -0
  518. /package/src/cli/{assets → templates}/editorconfig.ts +0 -0
  519. /package/src/cli/{assets → templates}/mainBrowserTs.ts +0 -0
  520. /package/src/cli/{assets → templates}/tsconfigJson.ts +0 -0
  521. /package/src/cli/{assets → templates}/webIndexTs.ts +0 -0
  522. /package/src/{file → system}/errors/FileError.ts +0 -0
  523. /package/src/{file → system}/services/FileDetector.ts +0 -0
@@ -0,0 +1,90 @@
1
+ import { $module } from "alepha";
2
+
3
+ // ---------------------------------------------------------------------------------------------------------------------
4
+
5
+ export type * from "./components/ClientOnly.tsx";
6
+ export { default as ClientOnly } from "./components/ClientOnly.tsx";
7
+ export type * from "./components/ErrorBoundary.tsx";
8
+ export { default as ErrorBoundary } from "./components/ErrorBoundary.tsx";
9
+ export * from "./contexts/AlephaContext.ts";
10
+ export * from "./contexts/AlephaProvider.tsx";
11
+ export * from "./hooks/useAction.ts";
12
+ export * from "./hooks/useAlepha.ts";
13
+ export * from "./hooks/useClient.ts";
14
+ export * from "./hooks/useEvents.ts";
15
+ export * from "./hooks/useInject.ts";
16
+ export * from "./hooks/useStore.ts";
17
+
18
+ // ---------------------------------------------------------------------------------------------------------------------
19
+
20
+ declare module "alepha" {
21
+ interface Hooks {
22
+ /**
23
+ * Fires when a user action is starting.
24
+ * Action can be a form submission, a route transition, or a custom action.
25
+ */
26
+ "react:action:begin": {
27
+ type: string;
28
+ id?: string;
29
+ };
30
+ /**
31
+ * Fires when a user action has succeeded.
32
+ * Action can be a form submission, a route transition, or a custom action.
33
+ */
34
+ "react:action:success": {
35
+ type: string;
36
+ id?: string;
37
+ };
38
+ /**
39
+ * Fires when a user action has failed.
40
+ * Action can be a form submission, a route transition, or a custom action.
41
+ */
42
+ "react:action:error": {
43
+ type: string;
44
+ id?: string;
45
+ error: Error;
46
+ };
47
+ /**
48
+ * Fires when a user action has completed, regardless of success or failure.
49
+ * Action can be a form submission, a route transition, or a custom action.
50
+ */
51
+ "react:action:end": {
52
+ type: string;
53
+ id?: string;
54
+ };
55
+ }
56
+ }
57
+
58
+ // ---------------------------------------------------------------------------------------------------------------------
59
+
60
+ /**
61
+ * | type | quality | stability |
62
+ * |------|---------|-----------|
63
+ * | frontend | epic | stable |
64
+ *
65
+ * Full-stack React framework with server-side rendering.
66
+ *
67
+ * **Features:**
68
+ * - React page routes with type-safe params
69
+ * - Async action handler with loading/error/cancel states
70
+ * - Type-safe HTTP client access
71
+ * - Dependency injection in components
72
+ * - Global state management
73
+ * - Router navigation methods
74
+ * - Current route state access
75
+ * - Check if path is active
76
+ * - URL query parameters
77
+ * - Access route schema
78
+ * - Subscribe to Alepha events
79
+ * - Type-safe form handling with validation
80
+ * - Error handling wrapper component
81
+ * - Client-side only rendering component
82
+ * - Server-side rendering with hydration
83
+ * - Automatic code splitting
84
+ * - Event system for action tracking
85
+ *
86
+ * @module alepha.react
87
+ */
88
+ export const AlephaReact = $module({
89
+ name: "alepha.react.core",
90
+ });
@@ -0,0 +1,17 @@
1
+ import type { TObject } from "alepha";
2
+ import type { ReactNode } from "react";
3
+ import { useFormState } from "../hooks/useFormState.ts";
4
+ import type { FormModel } from "../services/FormModel.ts";
5
+
6
+ const FormState = <T extends TObject>(props: {
7
+ form: FormModel<T>;
8
+ children: (state: { loading: boolean; dirty: boolean }) => ReactNode;
9
+ }) => {
10
+ const formState = useFormState(props.form);
11
+ return props.children({
12
+ loading: formState.loading,
13
+ dirty: formState.dirty,
14
+ });
15
+ };
16
+
17
+ export default FormState;
@@ -0,0 +1,18 @@
1
+ import { TypeBoxError } from "alepha";
2
+
3
+ export class FormValidationError extends TypeBoxError {
4
+ readonly name = "ValidationError";
5
+
6
+ constructor(options: {
7
+ message: string;
8
+ path: string;
9
+ }) {
10
+ super({
11
+ message: options.message,
12
+ instancePath: options.path,
13
+ schemaPath: "",
14
+ keyword: "not",
15
+ params: {},
16
+ });
17
+ }
18
+ }
@@ -0,0 +1,366 @@
1
+ import { fireEvent, render } from "@testing-library/react";
2
+ import { Alepha, t } from "alepha";
3
+ import { AlephaLogger } from "alepha/logger";
4
+ import { AlephaContext } from "alepha/react";
5
+ import type { ReactNode } from "react";
6
+ import { describe, it } from "vitest";
7
+ import { useForm } from "../index.ts";
8
+
9
+ describe("useForm", () => {
10
+ const renderWithAlepha = (alepha: Alepha, element: ReactNode) => {
11
+ return render(
12
+ <AlephaContext.Provider value={alepha}>{element}</AlephaContext.Provider>,
13
+ );
14
+ };
15
+
16
+ it("should run handler on submit", async ({ expect }) => {
17
+ const alepha = Alepha.create().with(AlephaLogger);
18
+ const calls: Array<any> = [];
19
+ const Form = () => {
20
+ const form = useForm({
21
+ id: "test",
22
+ schema: t.object({
23
+ str: t.text(),
24
+ int: t.integer(),
25
+ nested: t.object({
26
+ str: t.text(),
27
+ another: t.object({
28
+ level: t.text(),
29
+ }),
30
+ }),
31
+ }),
32
+ handler: (values, args) => {
33
+ calls.push(values);
34
+ },
35
+ });
36
+
37
+ return (
38
+ <form {...form.props} data-testid="test-form">
39
+ <input {...form.input.str.props} />
40
+ <input {...form.input.int.props} />
41
+ <input {...form.input.nested.items.str.props} />
42
+ <input {...form.input.nested.items.another.items.level.props} />
43
+ <button type="submit">Submit</button>
44
+ </form>
45
+ );
46
+ };
47
+
48
+ await alepha.start();
49
+
50
+ const ui = renderWithAlepha(alepha, <Form />);
51
+
52
+ fireEvent.change(ui.getByTestId("test-str"), {
53
+ target: { value: "testuser" },
54
+ });
55
+
56
+ fireEvent.change(ui.getByTestId("test-int"), {
57
+ target: { value: "123" },
58
+ });
59
+
60
+ fireEvent.change(ui.getByTestId("test-nested.str"), {
61
+ target: { value: "nestedvalue" },
62
+ });
63
+
64
+ fireEvent.change(ui.getByTestId("test-nested.another.level"), {
65
+ target: { value: "anothervalue" },
66
+ });
67
+
68
+ fireEvent.submit(ui.getByText("Submit"));
69
+
70
+ await new Promise((resolve) => setTimeout(resolve, 100));
71
+
72
+ expect(calls[0]).toEqual({
73
+ str: "testuser",
74
+ int: 123,
75
+ nested: {
76
+ str: "nestedvalue",
77
+ another: {
78
+ level: "anothervalue",
79
+ },
80
+ },
81
+ });
82
+ });
83
+
84
+ it("should provide correct InputField types for nested objects", async ({
85
+ expect,
86
+ }) => {
87
+ const alepha = Alepha.create().with(AlephaLogger);
88
+
89
+ const Form = () => {
90
+ const form = useForm({
91
+ id: "types-test",
92
+ schema: t.object({
93
+ name: t.text(),
94
+ address: t.object({
95
+ street: t.text(),
96
+ city: t.text(),
97
+ country: t.object({
98
+ code: t.text(),
99
+ name: t.text(),
100
+ }),
101
+ }),
102
+ }),
103
+ handler: () => {},
104
+ });
105
+
106
+ // Verify nested object InputFields have items property
107
+ const addressInput = form.input.address;
108
+ const streetInput = addressInput.items.street;
109
+ const countryInput = addressInput.items.country;
110
+ const countryCodeInput = countryInput.items.code;
111
+
112
+ return (
113
+ <div data-testid="type-check">
114
+ <span data-testid="has-items">
115
+ {addressInput.items ? "true" : "false"}
116
+ </span>
117
+ <span data-testid="street-path">{streetInput.path}</span>
118
+ <span data-testid="country-code-path">{countryCodeInput.path}</span>
119
+ </div>
120
+ );
121
+ };
122
+
123
+ await alepha.start();
124
+ const ui = renderWithAlepha(alepha, <Form />);
125
+
126
+ expect(ui.getByTestId("has-items").textContent).toBe("true");
127
+ expect(ui.getByTestId("street-path").textContent).toBe("/address/street");
128
+ expect(ui.getByTestId("country-code-path").textContent).toBe(
129
+ "/address/country/code",
130
+ );
131
+ });
132
+
133
+ it("should provide ArrayInputField with items array for arrays", async ({
134
+ expect,
135
+ }) => {
136
+ const alepha = Alepha.create().with(AlephaLogger);
137
+
138
+ const Form = () => {
139
+ const form = useForm({
140
+ id: "array-test",
141
+ schema: t.object({
142
+ tags: t.array(t.text()),
143
+ contacts: t.array(
144
+ t.object({
145
+ name: t.text(),
146
+ email: t.text(),
147
+ }),
148
+ ),
149
+ }),
150
+ handler: () => {},
151
+ });
152
+
153
+ // Verify array InputFields have items property (initially empty)
154
+ const tagsInput = form.input.tags;
155
+ const contactsInput = form.input.contacts;
156
+
157
+ return (
158
+ <div data-testid="array-check">
159
+ <span data-testid="tags-has-items">
160
+ {Array.isArray(tagsInput.items) ? "true" : "false"}
161
+ </span>
162
+ <span data-testid="tags-items-length">{tagsInput.items.length}</span>
163
+ <span data-testid="contacts-has-items">
164
+ {Array.isArray(contactsInput.items) ? "true" : "false"}
165
+ </span>
166
+ <span data-testid="contacts-items-length">
167
+ {contactsInput.items.length}
168
+ </span>
169
+ <span data-testid="tags-path">{tagsInput.path}</span>
170
+ <span data-testid="contacts-path">{contactsInput.path}</span>
171
+ </div>
172
+ );
173
+ };
174
+
175
+ await alepha.start();
176
+ const ui = renderWithAlepha(alepha, <Form />);
177
+
178
+ // Arrays have items property as array (initially empty)
179
+ expect(ui.getByTestId("tags-has-items").textContent).toBe("true");
180
+ expect(ui.getByTestId("tags-items-length").textContent).toBe("0");
181
+ expect(ui.getByTestId("contacts-has-items").textContent).toBe("true");
182
+ expect(ui.getByTestId("contacts-items-length").textContent).toBe("0");
183
+ expect(ui.getByTestId("tags-path").textContent).toBe("/tags");
184
+ expect(ui.getByTestId("contacts-path").textContent).toBe("/contacts");
185
+ });
186
+
187
+ it("should update array values via set method", async ({ expect }) => {
188
+ const alepha = Alepha.create().with(AlephaLogger);
189
+ const calls: Array<any> = [];
190
+
191
+ const Form = () => {
192
+ const form = useForm({
193
+ id: "array-set-test",
194
+ schema: t.object({
195
+ tags: t.array(t.text()),
196
+ }),
197
+ handler: (values) => {
198
+ calls.push(values);
199
+ },
200
+ });
201
+
202
+ return (
203
+ <form {...form.props} data-testid="array-form">
204
+ <button
205
+ type="button"
206
+ data-testid="set-tags"
207
+ onClick={() => form.input.tags.set(["tag1", "tag2", "tag3"])}
208
+ >
209
+ Set Tags
210
+ </button>
211
+ <button type="submit">Submit</button>
212
+ </form>
213
+ );
214
+ };
215
+
216
+ await alepha.start();
217
+ const ui = renderWithAlepha(alepha, <Form />);
218
+
219
+ fireEvent.click(ui.getByTestId("set-tags"));
220
+ fireEvent.submit(ui.getByText("Submit"));
221
+
222
+ await new Promise((resolve) => setTimeout(resolve, 100));
223
+
224
+ expect(calls[0]).toEqual({
225
+ tags: ["tag1", "tag2", "tag3"],
226
+ });
227
+ });
228
+
229
+ it("should update array of objects via set method", async ({ expect }) => {
230
+ const alepha = Alepha.create().with(AlephaLogger);
231
+ const calls: Array<any> = [];
232
+
233
+ const Form = () => {
234
+ const form = useForm({
235
+ id: "array-objects-test",
236
+ schema: t.object({
237
+ contacts: t.array(
238
+ t.object({
239
+ name: t.text(),
240
+ email: t.text(),
241
+ }),
242
+ ),
243
+ }),
244
+ handler: (values) => {
245
+ calls.push(values);
246
+ },
247
+ });
248
+
249
+ return (
250
+ <form {...form.props} data-testid="array-objects-form">
251
+ <button
252
+ type="button"
253
+ data-testid="set-contacts"
254
+ onClick={() =>
255
+ form.input.contacts.set([
256
+ { name: "Alice", email: "alice@example.com" },
257
+ { name: "Bob", email: "bob@example.com" },
258
+ ])
259
+ }
260
+ >
261
+ Set Contacts
262
+ </button>
263
+ <button type="submit">Submit</button>
264
+ </form>
265
+ );
266
+ };
267
+
268
+ await alepha.start();
269
+ const ui = renderWithAlepha(alepha, <Form />);
270
+
271
+ fireEvent.click(ui.getByTestId("set-contacts"));
272
+ fireEvent.submit(ui.getByText("Submit"));
273
+
274
+ await new Promise((resolve) => setTimeout(resolve, 100));
275
+
276
+ expect(calls[0]).toEqual({
277
+ contacts: [
278
+ { name: "Alice", email: "alice@example.com" },
279
+ { name: "Bob", email: "bob@example.com" },
280
+ ],
281
+ });
282
+ });
283
+
284
+ it("should handle complex nested structures with objects and arrays", async ({
285
+ expect,
286
+ }) => {
287
+ const alepha = Alepha.create().with(AlephaLogger);
288
+ const calls: Array<any> = [];
289
+
290
+ const Form = () => {
291
+ const form = useForm({
292
+ id: "complex-test",
293
+ schema: t.object({
294
+ company: t.object({
295
+ name: t.text(),
296
+ address: t.object({
297
+ street: t.text(),
298
+ city: t.text(),
299
+ }),
300
+ }),
301
+ employees: t.array(
302
+ t.object({
303
+ name: t.text(),
304
+ role: t.text(),
305
+ }),
306
+ ),
307
+ }),
308
+ handler: (values) => {
309
+ calls.push(values);
310
+ },
311
+ });
312
+
313
+ return (
314
+ <form {...form.props} data-testid="complex-form">
315
+ <input {...form.input.company.items.name.props} />
316
+ <input {...form.input.company.items.address.items.street.props} />
317
+ <input {...form.input.company.items.address.items.city.props} />
318
+ <button
319
+ type="button"
320
+ data-testid="set-employees"
321
+ onClick={() =>
322
+ form.input.employees.set([
323
+ { name: "Alice", role: "Engineer" },
324
+ { name: "Bob", role: "Designer" },
325
+ ])
326
+ }
327
+ >
328
+ Set Employees
329
+ </button>
330
+ <button type="submit">Submit</button>
331
+ </form>
332
+ );
333
+ };
334
+
335
+ await alepha.start();
336
+ const ui = renderWithAlepha(alepha, <Form />);
337
+
338
+ fireEvent.change(ui.getByTestId("complex-test-company.name"), {
339
+ target: { value: "Acme Corp" },
340
+ });
341
+ fireEvent.change(ui.getByTestId("complex-test-company.address.street"), {
342
+ target: { value: "123 Main St" },
343
+ });
344
+ fireEvent.change(ui.getByTestId("complex-test-company.address.city"), {
345
+ target: { value: "New York" },
346
+ });
347
+ fireEvent.click(ui.getByTestId("set-employees"));
348
+ fireEvent.submit(ui.getByText("Submit"));
349
+
350
+ await new Promise((resolve) => setTimeout(resolve, 100));
351
+
352
+ expect(calls[0]).toEqual({
353
+ company: {
354
+ name: "Acme Corp",
355
+ address: {
356
+ street: "123 Main St",
357
+ city: "New York",
358
+ },
359
+ },
360
+ employees: [
361
+ { name: "Alice", role: "Engineer" },
362
+ { name: "Bob", role: "Designer" },
363
+ ],
364
+ });
365
+ });
366
+ });
@@ -0,0 +1,47 @@
1
+ import type { TObject } from "alepha";
2
+ import { useAlepha } from "alepha/react";
3
+ import { useId, useMemo } from "react";
4
+ import { type FormCtrlOptions, FormModel } from "../services/FormModel.ts";
5
+
6
+ /**
7
+ * Custom hook to create a form with validation and field management.
8
+ * This hook uses TypeBox schemas to define the structure and validation rules for the form.
9
+ * It provides a way to handle form submission, field creation, and value management.
10
+ *
11
+ * @example
12
+ * ```tsx
13
+ * import { t } from "alepha";
14
+ *
15
+ * const form = useForm({
16
+ * schema: t.object({
17
+ * username: t.text(),
18
+ * password: t.text(),
19
+ * }),
20
+ * handler: (values) => {
21
+ * console.log("Form submitted with values:", values);
22
+ * },
23
+ * });
24
+ *
25
+ * return (
26
+ * <form {...form.props}>
27
+ * <input {...form.input.username.props} />
28
+ * <input {...form.input.password.props} />
29
+ * <button type="submit">Submit</button>
30
+ * </form>
31
+ * );
32
+ * ```
33
+ */
34
+ export const useForm = <T extends TObject>(
35
+ options: FormCtrlOptions<T>,
36
+ deps: any[] = [],
37
+ ): FormModel<T> => {
38
+ const alepha = useAlepha();
39
+ const formId = useId();
40
+
41
+ return useMemo(() => {
42
+ return alepha.inject(FormModel<T>, {
43
+ lifetime: "transient",
44
+ args: [options.id || formId, options],
45
+ });
46
+ }, deps);
47
+ };
@@ -0,0 +1,130 @@
1
+ import { type TObject, TypeBoxError } from "alepha";
2
+ import { useAlepha } from "alepha/react";
3
+ import { useEffect, useState } from "react";
4
+ import type { FormModel } from "../services/FormModel.ts";
5
+
6
+ export interface UseFormStateReturn {
7
+ loading: boolean;
8
+ dirty: boolean;
9
+ values?: Record<string, any>;
10
+ error?: Error;
11
+ }
12
+
13
+ export const useFormState = <
14
+ T extends TObject,
15
+ Keys extends keyof UseFormStateReturn,
16
+ >(
17
+ target: FormModel<T> | { form: FormModel<T>; path: string },
18
+ _events: Keys[] = ["loading", "dirty", "error"] as Keys[],
19
+ ): Pick<UseFormStateReturn, Keys> => {
20
+ const alepha = useAlepha();
21
+ const events = _events as string[];
22
+
23
+ const [dirty, setDirty] = useState(false);
24
+ const [loading, setLoading] = useState(false);
25
+ const [error, setError] = useState<Error | undefined>(undefined);
26
+ const [values, setValues] = useState<Record<string, any> | undefined>(
27
+ undefined,
28
+ );
29
+
30
+ const form = "form" in target ? target.form : target;
31
+ const path = "path" in target ? target.path : undefined;
32
+
33
+ const hasValues = events.includes("values");
34
+ const hasErrors = events.includes("error");
35
+ const hasDirty = events.includes("dirty");
36
+ const hasLoading = events.includes("loading");
37
+
38
+ useEffect(() => {
39
+ const listeners: Function[] = [];
40
+
41
+ if (hasErrors || hasValues || hasDirty) {
42
+ listeners.push(
43
+ alepha.events.on("form:change", (event) => {
44
+ if (event.id === form.id) {
45
+ if (!path || event.path === path) {
46
+ if (hasDirty) {
47
+ setDirty(true);
48
+ }
49
+ if (hasErrors) {
50
+ setError(undefined);
51
+ }
52
+ }
53
+ if (hasValues) {
54
+ setValues(form.currentValues);
55
+ }
56
+ }
57
+ }),
58
+ );
59
+ }
60
+
61
+ if (hasValues) {
62
+ listeners.push(
63
+ alepha.events.on("form:reset", (event) => {
64
+ if (event.id === form.id) {
65
+ setValues(event.values);
66
+ }
67
+ }),
68
+ );
69
+ }
70
+
71
+ if (hasLoading) {
72
+ listeners.push(
73
+ alepha.events.on("form:submit:begin", (event) => {
74
+ if (event.id === form.id) {
75
+ setLoading(true);
76
+ }
77
+ }),
78
+ alepha.events.on("form:submit:end", (event) => {
79
+ if (event.id === form.id) {
80
+ setLoading(false);
81
+ }
82
+ }),
83
+ );
84
+ }
85
+
86
+ if (hasValues || hasDirty) {
87
+ listeners.push(
88
+ alepha.events.on("form:submit:success", (event) => {
89
+ if (event.id === form.id) {
90
+ if (hasValues) {
91
+ setValues(event.values);
92
+ }
93
+ if (hasDirty) {
94
+ setDirty(false);
95
+ }
96
+ }
97
+ }),
98
+ );
99
+ }
100
+
101
+ if (hasErrors) {
102
+ listeners.push(
103
+ alepha.events.on("form:submit:error", (event) => {
104
+ if (event.id === form.id) {
105
+ if (
106
+ !path ||
107
+ (event.error instanceof TypeBoxError &&
108
+ event.error.value.path === path)
109
+ ) {
110
+ setError(event.error);
111
+ }
112
+ }
113
+ }),
114
+ );
115
+ }
116
+
117
+ return () => {
118
+ for (const unsub of listeners) {
119
+ unsub();
120
+ }
121
+ };
122
+ }, []);
123
+
124
+ return {
125
+ dirty,
126
+ loading,
127
+ error,
128
+ values,
129
+ } as Pick<UseFormStateReturn, Keys>;
130
+ };