alepha 0.15.1 → 0.15.2

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 (507) hide show
  1. package/dist/api/audits/index.d.ts +342 -365
  2. package/dist/api/audits/index.d.ts.map +1 -1
  3. package/dist/api/audits/index.js +10 -33
  4. package/dist/api/audits/index.js.map +1 -1
  5. package/dist/api/files/index.d.ts +180 -173
  6. package/dist/api/files/index.d.ts.map +1 -1
  7. package/dist/api/files/index.js +10 -3
  8. package/dist/api/files/index.js.map +1 -1
  9. package/dist/api/jobs/index.d.ts +162 -155
  10. package/dist/api/jobs/index.d.ts.map +1 -1
  11. package/dist/api/jobs/index.js +10 -3
  12. package/dist/api/jobs/index.js.map +1 -1
  13. package/dist/api/keys/index.d.ts +413 -0
  14. package/dist/api/keys/index.d.ts.map +1 -0
  15. package/dist/api/keys/index.js +476 -0
  16. package/dist/api/keys/index.js.map +1 -0
  17. package/dist/api/notifications/index.d.ts +10 -4
  18. package/dist/api/notifications/index.d.ts.map +1 -1
  19. package/dist/api/notifications/index.js +10 -4
  20. package/dist/api/notifications/index.js.map +1 -1
  21. package/dist/api/parameters/index.d.ts +294 -301
  22. package/dist/api/parameters/index.d.ts.map +1 -1
  23. package/dist/api/parameters/index.js +30 -37
  24. package/dist/api/parameters/index.js.map +1 -1
  25. package/dist/api/users/index.d.ts +1079 -769
  26. package/dist/api/users/index.d.ts.map +1 -1
  27. package/dist/api/users/index.js +2534 -218
  28. package/dist/api/users/index.js.map +1 -1
  29. package/dist/api/verifications/index.d.ts +10 -4
  30. package/dist/api/verifications/index.d.ts.map +1 -1
  31. package/dist/api/verifications/index.js +12 -4
  32. package/dist/api/verifications/index.js.map +1 -1
  33. package/dist/batch/index.d.ts +20 -40
  34. package/dist/batch/index.d.ts.map +1 -1
  35. package/dist/batch/index.js +31 -44
  36. package/dist/batch/index.js.map +1 -1
  37. package/dist/bucket/index.d.ts +432 -8
  38. package/dist/bucket/index.d.ts.map +1 -1
  39. package/dist/bucket/index.js +1856 -12
  40. package/dist/bucket/index.js.map +1 -1
  41. package/dist/cache/core/index.d.ts +179 -7
  42. package/dist/cache/core/index.d.ts.map +1 -1
  43. package/dist/cache/core/index.js +213 -7
  44. package/dist/cache/core/index.js.map +1 -1
  45. package/dist/cache/redis/index.d.ts +1 -0
  46. package/dist/cache/redis/index.d.ts.map +1 -1
  47. package/dist/cache/redis/index.js +4 -0
  48. package/dist/cache/redis/index.js.map +1 -1
  49. package/dist/cli/index.d.ts +488 -5612
  50. package/dist/cli/index.d.ts.map +1 -1
  51. package/dist/cli/index.js +2326 -311
  52. package/dist/cli/index.js.map +1 -1
  53. package/dist/command/index.d.ts +194 -46
  54. package/dist/command/index.d.ts.map +1 -1
  55. package/dist/command/index.js +1995 -60
  56. package/dist/command/index.js.map +1 -1
  57. package/dist/core/index.browser.js +42 -19
  58. package/dist/core/index.browser.js.map +1 -1
  59. package/dist/core/index.d.ts +34 -13
  60. package/dist/core/index.d.ts.map +1 -1
  61. package/dist/core/index.js +62 -19
  62. package/dist/core/index.js.map +1 -1
  63. package/dist/core/index.native.js +42 -19
  64. package/dist/core/index.native.js.map +1 -1
  65. package/dist/datetime/index.d.ts +15 -0
  66. package/dist/datetime/index.d.ts.map +1 -1
  67. package/dist/datetime/index.js +15 -0
  68. package/dist/datetime/index.js.map +1 -1
  69. package/dist/email/index.d.ts +315 -20
  70. package/dist/email/index.d.ts.map +1 -1
  71. package/dist/email/index.js +1852 -7
  72. package/dist/email/index.js.map +1 -1
  73. package/dist/fake/index.d.ts +90 -8
  74. package/dist/fake/index.d.ts.map +1 -1
  75. package/dist/fake/index.js +91 -20
  76. package/dist/fake/index.js.map +1 -1
  77. package/dist/lock/core/index.d.ts +11 -4
  78. package/dist/lock/core/index.d.ts.map +1 -1
  79. package/dist/lock/core/index.js +11 -4
  80. package/dist/lock/core/index.js.map +1 -1
  81. package/dist/logger/index.d.ts +17 -66
  82. package/dist/logger/index.d.ts.map +1 -1
  83. package/dist/logger/index.js +14 -63
  84. package/dist/logger/index.js.map +1 -1
  85. package/dist/mcp/index.d.ts +15 -35
  86. package/dist/mcp/index.d.ts.map +1 -1
  87. package/dist/mcp/index.js +12 -35
  88. package/dist/mcp/index.js.map +1 -1
  89. package/dist/orm/index.browser.js +3 -3
  90. package/dist/orm/index.browser.js.map +1 -1
  91. package/dist/orm/index.bun.js +7 -4
  92. package/dist/orm/index.bun.js.map +1 -1
  93. package/dist/orm/index.d.ts +514 -540
  94. package/dist/orm/index.d.ts.map +1 -1
  95. package/dist/orm/index.js +24 -49
  96. package/dist/orm/index.js.map +1 -1
  97. package/dist/queue/core/index.d.ts +18 -10
  98. package/dist/queue/core/index.d.ts.map +1 -1
  99. package/dist/queue/core/index.js +14 -6
  100. package/dist/queue/core/index.js.map +1 -1
  101. package/dist/react/auth/index.browser.js +108 -0
  102. package/dist/react/auth/index.browser.js.map +1 -0
  103. package/dist/react/auth/index.d.ts +100 -0
  104. package/dist/react/auth/index.d.ts.map +1 -0
  105. package/dist/react/auth/index.js +145 -0
  106. package/dist/react/auth/index.js.map +1 -0
  107. package/dist/react/core/index.d.ts +469 -0
  108. package/dist/react/core/index.d.ts.map +1 -0
  109. package/dist/react/core/index.js +464 -0
  110. package/dist/react/core/index.js.map +1 -0
  111. package/dist/react/form/index.d.ts +232 -0
  112. package/dist/react/form/index.d.ts.map +1 -0
  113. package/dist/react/form/index.js +432 -0
  114. package/dist/react/form/index.js.map +1 -0
  115. package/dist/react/head/index.browser.js +423 -0
  116. package/dist/react/head/index.browser.js.map +1 -0
  117. package/dist/react/head/index.d.ts +288 -0
  118. package/dist/react/head/index.d.ts.map +1 -0
  119. package/dist/react/head/index.js +465 -0
  120. package/dist/react/head/index.js.map +1 -0
  121. package/dist/react/i18n/index.d.ts +175 -0
  122. package/dist/react/i18n/index.d.ts.map +1 -0
  123. package/dist/react/i18n/index.js +224 -0
  124. package/dist/react/i18n/index.js.map +1 -0
  125. package/dist/react/router/index.browser.js +1980 -0
  126. package/dist/react/router/index.browser.js.map +1 -0
  127. package/dist/react/router/index.d.ts +2068 -0
  128. package/dist/react/router/index.d.ts.map +1 -0
  129. package/dist/react/router/index.js +4932 -0
  130. package/dist/react/router/index.js.map +1 -0
  131. package/dist/react/websocket/index.d.ts +117 -0
  132. package/dist/react/websocket/index.d.ts.map +1 -0
  133. package/dist/react/websocket/index.js +107 -0
  134. package/dist/react/websocket/index.js.map +1 -0
  135. package/dist/redis/index.bun.js +4 -0
  136. package/dist/redis/index.bun.js.map +1 -1
  137. package/dist/redis/index.d.ts +22 -25
  138. package/dist/redis/index.d.ts.map +1 -1
  139. package/dist/redis/index.js +16 -25
  140. package/dist/redis/index.js.map +1 -1
  141. package/dist/retry/index.d.ts +11 -2
  142. package/dist/retry/index.d.ts.map +1 -1
  143. package/dist/retry/index.js +11 -2
  144. package/dist/retry/index.js.map +1 -1
  145. package/dist/scheduler/index.d.ts +11 -2
  146. package/dist/scheduler/index.d.ts.map +1 -1
  147. package/dist/scheduler/index.js +11 -2
  148. package/dist/scheduler/index.js.map +1 -1
  149. package/dist/security/index.d.ts +110 -19
  150. package/dist/security/index.d.ts.map +1 -1
  151. package/dist/security/index.js +157 -26
  152. package/dist/security/index.js.map +1 -1
  153. package/dist/server/auth/index.d.ts +179 -174
  154. package/dist/server/auth/index.d.ts.map +1 -1
  155. package/dist/server/auth/index.js +12 -7
  156. package/dist/server/auth/index.js.map +1 -1
  157. package/dist/server/cache/index.d.ts +7 -22
  158. package/dist/server/cache/index.d.ts.map +1 -1
  159. package/dist/server/cache/index.js +7 -22
  160. package/dist/server/cache/index.js.map +1 -1
  161. package/dist/server/compress/index.d.ts +10 -2
  162. package/dist/server/compress/index.d.ts.map +1 -1
  163. package/dist/server/compress/index.js +10 -2
  164. package/dist/server/compress/index.js.map +1 -1
  165. package/dist/server/cookies/index.d.ts +40 -16
  166. package/dist/server/cookies/index.d.ts.map +1 -1
  167. package/dist/server/cookies/index.js +7 -5
  168. package/dist/server/cookies/index.js.map +1 -1
  169. package/dist/server/core/index.d.ts +115 -14
  170. package/dist/server/core/index.d.ts.map +1 -1
  171. package/dist/server/core/index.js +231 -14
  172. package/dist/server/core/index.js.map +1 -1
  173. package/dist/server/cors/index.d.ts +13 -23
  174. package/dist/server/cors/index.d.ts.map +1 -1
  175. package/dist/server/cors/index.js +7 -21
  176. package/dist/server/cors/index.js.map +1 -1
  177. package/dist/server/health/index.d.ts +25 -19
  178. package/dist/server/health/index.d.ts.map +1 -1
  179. package/dist/server/health/index.js +8 -2
  180. package/dist/server/health/index.js.map +1 -1
  181. package/dist/server/helmet/index.d.ts +11 -3
  182. package/dist/server/helmet/index.d.ts.map +1 -1
  183. package/dist/server/helmet/index.js +11 -3
  184. package/dist/server/helmet/index.js.map +1 -1
  185. package/dist/server/links/index.d.ts +50 -45
  186. package/dist/server/links/index.d.ts.map +1 -1
  187. package/dist/server/links/index.js +11 -6
  188. package/dist/server/links/index.js.map +1 -1
  189. package/dist/server/metrics/index.d.ts +10 -3
  190. package/dist/server/metrics/index.d.ts.map +1 -1
  191. package/dist/server/metrics/index.js +10 -3
  192. package/dist/server/metrics/index.js.map +1 -1
  193. package/dist/server/multipart/index.d.ts +9 -3
  194. package/dist/server/multipart/index.d.ts.map +1 -1
  195. package/dist/server/multipart/index.js +9 -3
  196. package/dist/server/multipart/index.js.map +1 -1
  197. package/dist/server/proxy/index.d.ts +8 -2
  198. package/dist/server/proxy/index.d.ts.map +1 -1
  199. package/dist/server/proxy/index.js +8 -2
  200. package/dist/server/proxy/index.js.map +1 -1
  201. package/dist/server/rate-limit/index.d.ts +30 -35
  202. package/dist/server/rate-limit/index.d.ts.map +1 -1
  203. package/dist/server/rate-limit/index.js +18 -55
  204. package/dist/server/rate-limit/index.js.map +1 -1
  205. package/dist/server/static/index.d.ts +137 -4
  206. package/dist/server/static/index.d.ts.map +1 -1
  207. package/dist/server/static/index.js +1848 -5
  208. package/dist/server/static/index.js.map +1 -1
  209. package/dist/server/swagger/index.d.ts +301 -6
  210. package/dist/server/swagger/index.d.ts.map +1 -1
  211. package/dist/server/swagger/index.js +1849 -6
  212. package/dist/server/swagger/index.js.map +1 -1
  213. package/dist/sms/index.d.ts +301 -7
  214. package/dist/sms/index.d.ts.map +1 -1
  215. package/dist/sms/index.js +1851 -7
  216. package/dist/sms/index.js.map +1 -1
  217. package/dist/system/index.browser.js +496 -0
  218. package/dist/system/index.browser.js.map +1 -0
  219. package/dist/{file → system}/index.d.ts +335 -16
  220. package/dist/system/index.d.ts.map +1 -0
  221. package/dist/{file → system}/index.js +412 -20
  222. package/dist/system/index.js.map +1 -0
  223. package/dist/thread/index.d.ts +11 -2
  224. package/dist/thread/index.d.ts.map +1 -1
  225. package/dist/thread/index.js +11 -2
  226. package/dist/thread/index.js.map +1 -1
  227. package/dist/topic/core/index.d.ts +12 -5
  228. package/dist/topic/core/index.d.ts.map +1 -1
  229. package/dist/topic/core/index.js +12 -5
  230. package/dist/topic/core/index.js.map +1 -1
  231. package/dist/vite/index.d.ts +4 -6271
  232. package/dist/vite/index.d.ts.map +1 -1
  233. package/dist/vite/index.js +8 -3
  234. package/dist/vite/index.js.map +1 -1
  235. package/dist/websocket/index.d.ts +12 -8
  236. package/dist/websocket/index.d.ts.map +1 -1
  237. package/dist/websocket/index.js +12 -8
  238. package/dist/websocket/index.js.map +1 -1
  239. package/package.json +80 -11
  240. package/src/api/audits/index.ts +10 -33
  241. package/src/api/files/__tests__/$bucket.spec.ts +1 -1
  242. package/src/api/files/controllers/AdminFileStatsController.spec.ts +1 -1
  243. package/src/api/files/controllers/FileController.spec.ts +1 -1
  244. package/src/api/files/index.ts +10 -3
  245. package/src/api/files/jobs/FileJobs.spec.ts +1 -1
  246. package/src/api/files/services/FileService.spec.ts +1 -1
  247. package/src/api/jobs/index.ts +10 -3
  248. package/src/api/keys/controllers/AdminApiKeyController.ts +75 -0
  249. package/src/api/keys/controllers/ApiKeyController.ts +103 -0
  250. package/src/api/keys/entities/apiKeyEntity.ts +41 -0
  251. package/src/api/keys/index.ts +49 -0
  252. package/src/api/keys/schemas/adminApiKeyQuerySchema.ts +7 -0
  253. package/src/api/keys/schemas/adminApiKeyResourceSchema.ts +17 -0
  254. package/src/api/keys/schemas/createApiKeyBodySchema.ts +7 -0
  255. package/src/api/keys/schemas/createApiKeyResponseSchema.ts +11 -0
  256. package/src/api/keys/schemas/listApiKeyResponseSchema.ts +15 -0
  257. package/src/api/keys/schemas/revokeApiKeyParamsSchema.ts +5 -0
  258. package/src/api/keys/schemas/revokeApiKeyResponseSchema.ts +5 -0
  259. package/src/api/keys/services/ApiKeyService.spec.ts +553 -0
  260. package/src/api/keys/services/ApiKeyService.ts +306 -0
  261. package/src/api/logs/TODO.md +52 -0
  262. package/src/api/notifications/index.ts +10 -4
  263. package/src/api/parameters/index.ts +9 -30
  264. package/src/api/parameters/primitives/$config.ts +12 -4
  265. package/src/api/parameters/services/ConfigStore.ts +9 -3
  266. package/src/api/users/__tests__/ApiKeys-integration.spec.ts +1035 -0
  267. package/src/api/users/__tests__/ApiKeys.spec.ts +401 -0
  268. package/src/api/users/index.ts +14 -3
  269. package/src/api/users/primitives/$realm.ts +33 -5
  270. package/src/api/users/providers/RealmProvider.ts +1 -12
  271. package/src/api/users/services/SessionService.ts +1 -1
  272. package/src/api/verifications/controllers/VerificationController.ts +2 -0
  273. package/src/api/verifications/index.ts +10 -4
  274. package/src/batch/index.ts +9 -36
  275. package/src/batch/primitives/$batch.ts +0 -8
  276. package/src/batch/providers/BatchProvider.ts +29 -2
  277. package/src/bucket/__tests__/shared.ts +1 -1
  278. package/src/bucket/index.ts +13 -6
  279. package/src/bucket/primitives/$bucket.ts +1 -1
  280. package/src/bucket/providers/LocalFileStorageProvider.ts +1 -1
  281. package/src/bucket/providers/MemoryFileStorageProvider.ts +1 -1
  282. package/src/cache/core/__tests__/shared.ts +30 -0
  283. package/src/cache/core/index.ts +11 -6
  284. package/src/cache/core/primitives/$cache.spec.ts +5 -0
  285. package/src/cache/core/providers/CacheProvider.ts +17 -0
  286. package/src/cache/core/providers/MemoryCacheProvider.ts +300 -1
  287. package/src/cache/redis/__tests__/cache-redis.spec.ts +5 -0
  288. package/src/cache/redis/providers/RedisCacheProvider.ts +9 -0
  289. package/src/cli/apps/AlephaCli.ts +1 -14
  290. package/src/cli/apps/AlephaPackageBuilderCli.ts +1 -1
  291. package/src/cli/commands/build.ts +1 -5
  292. package/src/cli/commands/db.ts +17 -11
  293. package/src/cli/commands/deploy.ts +1 -1
  294. package/src/cli/commands/dev.ts +1 -20
  295. package/src/cli/commands/gen/env.ts +5 -2
  296. package/src/cli/commands/gen/openapi.ts +5 -2
  297. package/src/cli/commands/init.spec.ts +544 -0
  298. package/src/cli/commands/init.ts +89 -55
  299. package/src/cli/commands/lint.ts +7 -1
  300. package/src/cli/commands/typecheck.ts +11 -0
  301. package/src/cli/providers/AppEntryProvider.ts +1 -1
  302. package/src/cli/providers/ViteBuildProvider.ts +8 -50
  303. package/src/cli/providers/ViteDevServerProvider.ts +36 -8
  304. package/src/cli/services/AlephaCliUtils.ts +37 -122
  305. package/src/cli/services/PackageManagerUtils.ts +127 -11
  306. package/src/cli/services/ProjectScaffolder.ts +122 -77
  307. package/src/cli/services/ViteUtils.ts +82 -0
  308. package/src/cli/{assets/claudeMd.ts → templates/agentMd.ts} +32 -24
  309. package/src/cli/templates/gitignore.ts +39 -0
  310. package/src/cli/{assets → templates}/mainCss.ts +11 -2
  311. package/src/cli/templates/mainServerTs.ts +33 -0
  312. package/src/cli/templates/webAppRouterTs.ts +50 -0
  313. package/src/cli/{assets → templates}/webHelloComponentTsx.ts +2 -2
  314. package/src/command/helpers/Runner.spec.ts +4 -0
  315. package/src/command/helpers/Runner.ts +3 -21
  316. package/src/command/index.ts +12 -4
  317. package/src/command/providers/CliProvider.spec.ts +1067 -0
  318. package/src/command/providers/CliProvider.ts +203 -40
  319. package/src/core/Alepha.ts +2 -2
  320. package/src/core/__tests__/Alepha-start.spec.ts +4 -4
  321. package/src/core/helpers/jsonSchemaToTypeBox.spec.ts +771 -0
  322. package/src/core/helpers/jsonSchemaToTypeBox.ts +62 -10
  323. package/src/core/index.shared.ts +1 -0
  324. package/src/core/index.ts +20 -0
  325. package/src/core/providers/EventManager.spec.ts +0 -71
  326. package/src/core/providers/EventManager.ts +3 -15
  327. package/src/core/providers/Json.ts +2 -14
  328. package/src/datetime/index.ts +15 -0
  329. package/src/email/index.ts +10 -5
  330. package/src/email/providers/LocalEmailProvider.spec.ts +1 -1
  331. package/src/email/providers/LocalEmailProvider.ts +1 -1
  332. package/src/fake/__tests__/keyName.example.ts +1 -1
  333. package/src/fake/__tests__/keyName.spec.ts +5 -5
  334. package/src/fake/index.ts +9 -6
  335. package/src/fake/providers/FakeProvider.spec.ts +258 -40
  336. package/src/fake/providers/FakeProvider.ts +133 -19
  337. package/src/lock/core/index.ts +11 -4
  338. package/src/logger/index.ts +17 -66
  339. package/src/mcp/index.ts +10 -27
  340. package/src/mcp/transports/SseMcpTransport.ts +0 -11
  341. package/src/orm/__tests__/PostgresProvider.spec.ts +2 -2
  342. package/src/orm/index.browser.ts +2 -2
  343. package/src/orm/index.bun.ts +4 -2
  344. package/src/orm/index.ts +21 -47
  345. package/src/orm/providers/drivers/BunSqliteProvider.ts +1 -0
  346. package/src/orm/services/Repository.ts +7 -3
  347. package/src/queue/core/index.ts +14 -6
  348. package/src/react/auth/__tests__/$auth.spec.ts +202 -0
  349. package/src/react/auth/hooks/useAuth.ts +32 -0
  350. package/src/react/auth/index.browser.ts +13 -0
  351. package/src/react/auth/index.shared.ts +2 -0
  352. package/src/react/auth/index.ts +48 -0
  353. package/src/react/auth/providers/ReactAuthProvider.ts +16 -0
  354. package/src/react/auth/services/ReactAuth.ts +135 -0
  355. package/src/react/core/__tests__/Router.spec.tsx +169 -0
  356. package/src/react/core/components/ClientOnly.tsx +49 -0
  357. package/src/react/core/components/ErrorBoundary.tsx +73 -0
  358. package/src/react/core/contexts/AlephaContext.ts +7 -0
  359. package/src/react/core/contexts/AlephaProvider.tsx +42 -0
  360. package/src/react/core/hooks/useAction.browser.spec.tsx +569 -0
  361. package/src/react/core/hooks/useAction.ts +480 -0
  362. package/src/react/core/hooks/useAlepha.ts +26 -0
  363. package/src/react/core/hooks/useClient.ts +17 -0
  364. package/src/react/core/hooks/useEvents.ts +51 -0
  365. package/src/react/core/hooks/useInject.ts +12 -0
  366. package/src/react/core/hooks/useStore.ts +52 -0
  367. package/src/react/core/index.ts +90 -0
  368. package/src/react/form/components/FormState.tsx +17 -0
  369. package/src/react/form/errors/FormValidationError.ts +18 -0
  370. package/src/react/form/hooks/useForm.browser.spec.tsx +366 -0
  371. package/src/react/form/hooks/useForm.ts +47 -0
  372. package/src/react/form/hooks/useFormState.ts +130 -0
  373. package/src/react/form/index.ts +44 -0
  374. package/src/react/form/services/FormModel.ts +614 -0
  375. package/src/react/head/helpers/SeoExpander.spec.ts +203 -0
  376. package/src/react/head/helpers/SeoExpander.ts +142 -0
  377. package/src/react/head/hooks/useHead.spec.tsx +288 -0
  378. package/src/react/head/hooks/useHead.ts +62 -0
  379. package/src/react/head/index.browser.ts +26 -0
  380. package/src/react/head/index.ts +44 -0
  381. package/src/react/head/interfaces/Head.ts +105 -0
  382. package/src/react/head/primitives/$head.ts +25 -0
  383. package/src/react/head/providers/BrowserHeadProvider.browser.spec.ts +196 -0
  384. package/src/react/head/providers/BrowserHeadProvider.ts +212 -0
  385. package/src/react/head/providers/HeadProvider.ts +168 -0
  386. package/src/react/head/providers/ServerHeadProvider.ts +31 -0
  387. package/src/react/i18n/__tests__/integration.spec.tsx +239 -0
  388. package/src/react/i18n/components/Localize.spec.tsx +357 -0
  389. package/src/react/i18n/components/Localize.tsx +35 -0
  390. package/src/react/i18n/hooks/useI18n.browser.spec.tsx +438 -0
  391. package/src/react/i18n/hooks/useI18n.ts +18 -0
  392. package/src/react/i18n/index.ts +41 -0
  393. package/src/react/i18n/primitives/$dictionary.ts +69 -0
  394. package/src/react/i18n/providers/I18nProvider.spec.ts +389 -0
  395. package/src/react/i18n/providers/I18nProvider.ts +278 -0
  396. package/src/react/router/__tests__/page-head-browser.browser.spec.ts +95 -0
  397. package/src/react/router/__tests__/page-head.spec.ts +48 -0
  398. package/src/react/router/__tests__/seo-head.spec.ts +125 -0
  399. package/src/react/router/atoms/ssrManifestAtom.ts +58 -0
  400. package/src/react/router/components/ErrorViewer.tsx +872 -0
  401. package/src/react/router/components/Link.tsx +23 -0
  402. package/src/react/router/components/NestedView.tsx +223 -0
  403. package/src/react/router/components/NotFound.tsx +30 -0
  404. package/src/react/router/constants/PAGE_PRELOAD_KEY.ts +6 -0
  405. package/src/react/router/contexts/RouterLayerContext.ts +12 -0
  406. package/src/react/router/errors/Redirection.ts +28 -0
  407. package/src/react/router/hooks/useActive.ts +52 -0
  408. package/src/react/router/hooks/useQueryParams.ts +63 -0
  409. package/src/react/router/hooks/useRouter.ts +20 -0
  410. package/src/react/router/hooks/useRouterState.ts +11 -0
  411. package/src/react/router/index.browser.ts +45 -0
  412. package/src/react/router/index.shared.ts +19 -0
  413. package/src/react/router/index.ts +142 -0
  414. package/src/react/router/primitives/$page.browser.spec.tsx +851 -0
  415. package/src/react/router/primitives/$page.spec.tsx +708 -0
  416. package/src/react/router/primitives/$page.ts +497 -0
  417. package/src/react/router/providers/ReactBrowserProvider.ts +309 -0
  418. package/src/react/router/providers/ReactBrowserRendererProvider.ts +25 -0
  419. package/src/react/router/providers/ReactBrowserRouterProvider.ts +168 -0
  420. package/src/react/router/providers/ReactPageProvider.ts +726 -0
  421. package/src/react/router/providers/ReactServerProvider.spec.tsx +316 -0
  422. package/src/react/router/providers/ReactServerProvider.ts +558 -0
  423. package/src/react/router/providers/ReactServerTemplateProvider.ts +979 -0
  424. package/src/react/router/providers/SSRManifestProvider.ts +334 -0
  425. package/src/react/router/services/ReactPageServerService.ts +48 -0
  426. package/src/react/router/services/ReactPageService.ts +27 -0
  427. package/src/react/router/services/ReactRouter.ts +262 -0
  428. package/src/react/websocket/hooks/useRoom.tsx +242 -0
  429. package/src/react/websocket/index.ts +7 -0
  430. package/src/redis/__tests__/redis.spec.ts +13 -0
  431. package/src/redis/index.ts +9 -25
  432. package/src/redis/providers/BunRedisProvider.ts +9 -0
  433. package/src/redis/providers/NodeRedisProvider.ts +8 -0
  434. package/src/redis/providers/RedisProvider.ts +16 -0
  435. package/src/retry/index.ts +11 -2
  436. package/src/router/index.ts +15 -0
  437. package/src/scheduler/index.ts +11 -2
  438. package/src/security/__tests__/BasicAuth.spec.ts +2 -0
  439. package/src/security/__tests__/ServerSecurityProvider.spec.ts +13 -5
  440. package/src/security/index.ts +15 -10
  441. package/src/security/interfaces/IssuerResolver.ts +27 -0
  442. package/src/security/primitives/$issuer.ts +55 -0
  443. package/src/security/providers/SecurityProvider.ts +179 -0
  444. package/src/security/providers/ServerBasicAuthProvider.ts +6 -2
  445. package/src/security/providers/ServerSecurityProvider.ts +36 -22
  446. package/src/server/auth/index.ts +12 -7
  447. package/src/server/cache/index.ts +7 -22
  448. package/src/server/compress/index.ts +10 -2
  449. package/src/server/cookies/index.ts +7 -5
  450. package/src/server/cookies/primitives/$cookie.ts +33 -11
  451. package/src/server/core/index.ts +16 -6
  452. package/src/server/core/interfaces/ServerRequest.ts +83 -1
  453. package/src/server/core/primitives/$action.spec.ts +1 -1
  454. package/src/server/core/primitives/$action.ts +8 -3
  455. package/src/server/core/providers/NodeHttpServerProvider.ts +9 -3
  456. package/src/server/core/services/ServerRequestParser.spec.ts +520 -0
  457. package/src/server/core/services/ServerRequestParser.ts +306 -13
  458. package/src/server/cors/index.ts +7 -21
  459. package/src/server/cors/primitives/$cors.ts +6 -2
  460. package/src/server/health/index.ts +8 -2
  461. package/src/server/helmet/index.ts +11 -3
  462. package/src/server/links/index.ts +11 -6
  463. package/src/server/metrics/index.ts +10 -3
  464. package/src/server/multipart/index.ts +9 -3
  465. package/src/server/proxy/index.ts +8 -2
  466. package/src/server/rate-limit/index.ts +21 -25
  467. package/src/server/rate-limit/primitives/$rateLimit.ts +6 -2
  468. package/src/server/rate-limit/providers/ServerRateLimitProvider.spec.ts +38 -14
  469. package/src/server/rate-limit/providers/ServerRateLimitProvider.ts +22 -56
  470. package/src/server/static/index.ts +8 -2
  471. package/src/server/static/providers/ServerStaticProvider.ts +1 -1
  472. package/src/server/swagger/index.ts +9 -4
  473. package/src/server/swagger/providers/ServerSwaggerProvider.ts +1 -1
  474. package/src/sms/index.ts +9 -5
  475. package/src/sms/providers/LocalSmsProvider.spec.ts +1 -1
  476. package/src/sms/providers/LocalSmsProvider.ts +1 -1
  477. package/src/system/index.browser.ts +11 -0
  478. package/src/system/index.ts +62 -0
  479. package/src/{file → system}/providers/FileSystemProvider.ts +16 -0
  480. package/src/{file → system}/providers/MemoryFileSystemProvider.ts +116 -3
  481. package/src/system/providers/MemoryShellProvider.ts +164 -0
  482. package/src/{file → system}/providers/NodeFileSystemProvider.spec.ts +2 -2
  483. package/src/{file → system}/providers/NodeFileSystemProvider.ts +36 -0
  484. package/src/system/providers/NodeShellProvider.ts +184 -0
  485. package/src/system/providers/ShellProvider.ts +74 -0
  486. package/src/{file → system}/services/FileDetector.spec.ts +2 -2
  487. package/src/thread/index.ts +11 -2
  488. package/src/topic/core/index.ts +12 -5
  489. package/src/vite/tasks/buildClient.ts +2 -7
  490. package/src/vite/tasks/buildServer.ts +17 -1
  491. package/src/websocket/index.ts +12 -8
  492. package/dist/file/index.d.ts.map +0 -1
  493. package/dist/file/index.js.map +0 -1
  494. package/src/cli/assets/mainServerTs.ts +0 -24
  495. package/src/cli/assets/webAppRouterTs.ts +0 -16
  496. package/src/cli/providers/ViteTemplateProvider.ts +0 -27
  497. package/src/file/index.ts +0 -43
  498. /package/src/cli/{assets → templates}/apiHelloControllerTs.ts +0 -0
  499. /package/src/cli/{assets → templates}/apiIndexTs.ts +0 -0
  500. /package/src/cli/{assets → templates}/biomeJson.ts +0 -0
  501. /package/src/cli/{assets → templates}/dummySpecTs.ts +0 -0
  502. /package/src/cli/{assets → templates}/editorconfig.ts +0 -0
  503. /package/src/cli/{assets → templates}/mainBrowserTs.ts +0 -0
  504. /package/src/cli/{assets → templates}/tsconfigJson.ts +0 -0
  505. /package/src/cli/{assets → templates}/webIndexTs.ts +0 -0
  506. /package/src/{file → system}/errors/FileError.ts +0 -0
  507. /package/src/{file → system}/services/FileDetector.ts +0 -0
@@ -0,0 +1,202 @@
1
+ import { randomUUID } from "node:crypto";
2
+ import { Alepha } from "alepha";
3
+ import { DateTimeProvider } from "alepha/datetime";
4
+ import { $issuer, AlephaSecurity } from "alepha/security";
5
+ import { AlephaServer, HttpClient, ServerProvider } from "alepha/server";
6
+ import {
7
+ $auth,
8
+ alephaServerAuthRoutes,
9
+ type TokenResponse,
10
+ tokenResponseSchema,
11
+ tokensSchema,
12
+ } from "alepha/server/auth";
13
+ import { $client } from "alepha/server/links";
14
+ import { describe, test } from "vitest";
15
+ import { ReactAuth, type ReactAuthProvider } from "../index.ts";
16
+
17
+ describe("$auth", () => {
18
+ const user = {
19
+ id: randomUUID(),
20
+ name: "John Doe",
21
+ username: "john",
22
+ password: "***",
23
+ roles: ["admin"],
24
+ };
25
+
26
+ class App {
27
+ issuer = $issuer({
28
+ secret: "my-secret-key",
29
+ roles: [
30
+ {
31
+ name: "admin",
32
+ permissions: [{ name: "*" }],
33
+ },
34
+ ],
35
+ });
36
+
37
+ auth = $auth({
38
+ issuer: this.issuer,
39
+ credentials: {
40
+ account: () => user,
41
+ },
42
+ });
43
+
44
+ api = $client<ReactAuthProvider>();
45
+ }
46
+
47
+ const userinfo = (alepha: Alepha, token?: string) =>
48
+ alepha
49
+ .inject(HttpClient)
50
+ .fetch(
51
+ `${alepha.inject(ServerProvider).hostname}${alephaServerAuthRoutes.userinfo}`,
52
+ {
53
+ method: "GET",
54
+ headers: {
55
+ authorization: `Bearer ${token}`,
56
+ },
57
+ },
58
+ )
59
+ .then((it) => it.data);
60
+
61
+ const login = (alepha: Alepha) =>
62
+ alepha
63
+ .inject(HttpClient)
64
+ .fetch(
65
+ `${alepha.inject(ServerProvider).hostname}${alephaServerAuthRoutes.token}?provider=auth`,
66
+ {
67
+ method: "POST",
68
+ body: JSON.stringify({
69
+ username: user.username,
70
+ password: user.password,
71
+ }),
72
+ schema: {
73
+ response: tokenResponseSchema,
74
+ },
75
+ },
76
+ );
77
+
78
+ const refresh = (alepha: Alepha, tokens: TokenResponse) =>
79
+ alepha
80
+ .inject(HttpClient)
81
+ .fetch(
82
+ `${alepha.inject(ServerProvider).hostname}${alephaServerAuthRoutes.refresh}?provider=auth`,
83
+ {
84
+ method: "POST",
85
+ body: JSON.stringify({
86
+ refresh_token: tokens.refresh_token!,
87
+ access_token: tokens.access_token,
88
+ }),
89
+ schema: {
90
+ response: tokensSchema,
91
+ },
92
+ },
93
+ );
94
+
95
+ test("should login with credentials", async ({ expect }) => {
96
+ const alepha = Alepha.create()
97
+ .with(AlephaServer)
98
+ .with(AlephaSecurity)
99
+ .with(App);
100
+ const auth = alepha.inject(ReactAuth);
101
+ await alepha.start();
102
+
103
+ expect(auth.user).toBeUndefined();
104
+ await auth.login("auth", {
105
+ username: user.username,
106
+ password: user.password,
107
+ hostname: alepha.inject(ServerProvider).hostname,
108
+ });
109
+ expect(auth.user).toEqual({
110
+ id: user.id,
111
+ name: user.name,
112
+ roles: user.roles,
113
+ username: user.username,
114
+ });
115
+ });
116
+
117
+ test("should get userinfo", async ({ expect }) => {
118
+ const alepha = Alepha.create()
119
+ .with(AlephaServer)
120
+ .with(AlephaSecurity)
121
+ .with(App);
122
+ await alepha.start();
123
+
124
+ const { data: tokens } = await login(alepha);
125
+
126
+ expect(await userinfo(alepha, tokens.access_token)).toEqual({
127
+ user: {
128
+ id: user.id,
129
+ name: user.name,
130
+ roles: user.roles,
131
+ username: user.username,
132
+ sessionId: expect.any(String),
133
+ },
134
+ api: {
135
+ prefix: "/api",
136
+ links: [],
137
+ },
138
+ });
139
+ });
140
+
141
+ test("should reject expired token", async ({ expect }) => {
142
+ const alepha = Alepha.create()
143
+ .with(AlephaServer)
144
+ .with(AlephaSecurity)
145
+ .with(App);
146
+ await alepha.start();
147
+
148
+ const { data: tokens } = await login(alepha);
149
+
150
+ await alepha.inject(DateTimeProvider).travel(1, "hour");
151
+
152
+ expect(await userinfo(alepha, tokens.access_token)).toEqual({
153
+ api: {
154
+ prefix: "/api",
155
+ links: [],
156
+ },
157
+ });
158
+ });
159
+
160
+ test("should refresh expired token", async ({ expect }) => {
161
+ const alepha = Alepha.create()
162
+ .with(AlephaServer)
163
+ .with(AlephaSecurity)
164
+ .with(App);
165
+ await alepha.start();
166
+
167
+ const { data: tokens } = await login(alepha);
168
+
169
+ await alepha.inject(DateTimeProvider).travel(1, "hour");
170
+
171
+ const { data: tokens2 } = await refresh(alepha, tokens);
172
+
173
+ expect(await userinfo(alepha, tokens2.access_token)).toEqual({
174
+ user: {
175
+ id: user.id,
176
+ name: user.name,
177
+ roles: user.roles,
178
+ username: user.username,
179
+ },
180
+ api: {
181
+ prefix: "/api",
182
+ links: [],
183
+ },
184
+ });
185
+ });
186
+
187
+ test("should reject expired refresh token", async ({ expect }) => {
188
+ const alepha = Alepha.create()
189
+ .with(AlephaServer)
190
+ .with(AlephaSecurity)
191
+ .with(App);
192
+ await alepha.start();
193
+
194
+ const { data: tokens } = await login(alepha);
195
+
196
+ await alepha.inject(DateTimeProvider).travel(40, "days");
197
+
198
+ await expect(refresh(alepha, tokens)).rejects.toThrowError(
199
+ "Failed to refresh access token using the refresh token (issuer)",
200
+ );
201
+ });
202
+ });
@@ -0,0 +1,32 @@
1
+ import { useAlepha, useStore } from "alepha/react";
2
+ import { type HttpVirtualClient, LinkProvider } from "alepha/server/links";
3
+ import { ReactAuth } from "../services/ReactAuth.ts";
4
+
5
+ export const useAuth = <T extends object = any>() => {
6
+ const alepha = useAlepha();
7
+ const [user] = useStore("alepha.server.request.user");
8
+
9
+ return {
10
+ user,
11
+ logout: () => {
12
+ alepha.inject(ReactAuth).logout();
13
+ },
14
+ login: async (
15
+ provider: keyof T,
16
+ options: {
17
+ username?: string;
18
+ password?: string;
19
+ redirect?: string;
20
+ realm?: string;
21
+ [extra: string]: any;
22
+ } = {},
23
+ ) => {
24
+ await alepha.inject(ReactAuth).login(provider as string, options);
25
+ },
26
+ can: <Api extends object = any>(
27
+ name: keyof HttpVirtualClient<Api>,
28
+ ): boolean => {
29
+ return alepha.inject(LinkProvider).can(name as string);
30
+ },
31
+ };
32
+ };
@@ -0,0 +1,13 @@
1
+ import { $module } from "alepha";
2
+ import { ReactAuth } from "./services/ReactAuth.ts";
3
+
4
+ // ---------------------------------------------------------------------------------------------------------------------
5
+
6
+ export * from "./index.shared.ts";
7
+
8
+ // ---------------------------------------------------------------------------------------------------------------------
9
+
10
+ export const AlephaReactAuth = $module({
11
+ name: "alepha.react.auth",
12
+ services: [ReactAuth],
13
+ });
@@ -0,0 +1,2 @@
1
+ export * from "./hooks/useAuth.ts";
2
+ export * from "./services/ReactAuth.ts";
@@ -0,0 +1,48 @@
1
+ import { $module } from "alepha";
2
+ import { AlephaReact } from "alepha/react";
3
+ import type { UserAccount } from "alepha/security";
4
+ import { $auth, AlephaServerAuth } from "alepha/server/auth";
5
+ import { AlephaServerLinks } from "alepha/server/links";
6
+ import { ReactAuthProvider } from "./providers/ReactAuthProvider.ts";
7
+ import { ReactAuth } from "./services/ReactAuth.ts";
8
+
9
+ // ---------------------------------------------------------------------------------------------------------------------
10
+
11
+ export * from "./index.shared.ts";
12
+ export * from "./providers/ReactAuthProvider.ts";
13
+
14
+ // ---------------------------------------------------------------------------------------------------------------------
15
+
16
+ declare module "alepha/react/router" {
17
+ interface ReactRouterState {
18
+ user?: UserAccount;
19
+ }
20
+ }
21
+
22
+ // ---------------------------------------------------------------------------------------------------------------------
23
+
24
+ /**
25
+ * | type | quality | stability |
26
+ * |------|---------|-----------|
27
+ * | frontend | rare | stable |
28
+ *
29
+ * Auth-related React components and hooks.
30
+ *
31
+ * **Features:**
32
+ * - Login/logout components
33
+ * - Protected route wrappers
34
+ * - Auth state hooks
35
+ *
36
+ * @module alepha.react.auth
37
+ */
38
+ export const AlephaReactAuth = $module({
39
+ name: "alepha.react.auth",
40
+ primitives: [$auth],
41
+ services: [
42
+ AlephaReact,
43
+ AlephaServerLinks,
44
+ AlephaServerAuth,
45
+ ReactAuthProvider,
46
+ ReactAuth,
47
+ ],
48
+ });
@@ -0,0 +1,16 @@
1
+ import { $hook, $inject, Alepha } from "alepha";
2
+
3
+ export class ReactAuthProvider {
4
+ protected readonly alepha = $inject(Alepha);
5
+
6
+ public readonly onRender = $hook({
7
+ on: "react:server:render:begin",
8
+ handler: async ({ request, state }) => {
9
+ if (request?.user) {
10
+ const { token, realm, ...user } = request.user; // do not send token and realm to the client
11
+ this.alepha.store.set("alepha.server.request.user", user); // for hydration, browser, etc...
12
+ state.user = user;
13
+ }
14
+ },
15
+ });
16
+ }
@@ -0,0 +1,135 @@
1
+ import { $hook, $inject, Alepha } from "alepha";
2
+ import { $logger } from "alepha/logger";
3
+ import { ReactBrowserProvider, Redirection } from "alepha/react/router";
4
+ import type { UserAccountToken } from "alepha/security";
5
+ import { HttpClient } from "alepha/server";
6
+ import {
7
+ alephaServerAuthRoutes,
8
+ type Tokens,
9
+ tokenResponseSchema,
10
+ userinfoResponseSchema,
11
+ } from "alepha/server/auth";
12
+ import { LinkProvider } from "alepha/server/links";
13
+
14
+ /**
15
+ * Browser, SSR friendly, service to handle authentication.
16
+ */
17
+ export class ReactAuth {
18
+ protected readonly log = $logger();
19
+ protected readonly alepha = $inject(Alepha);
20
+ protected readonly httpClient = $inject(HttpClient);
21
+ protected readonly linkProvider = $inject(LinkProvider);
22
+
23
+ protected readonly onBeginTransition = $hook({
24
+ on: "react:transition:begin",
25
+ handler: async (event) => {
26
+ if (this.alepha.isBrowser()) {
27
+ Object.defineProperty(event.state, "user", {
28
+ get: () => this.user,
29
+ });
30
+ }
31
+ },
32
+ });
33
+
34
+ protected readonly onFetchRequest = $hook({
35
+ on: "client:onRequest",
36
+ handler: async ({ request }) => {
37
+ if (this.alepha.isBrowser() && this.user) {
38
+ // ensure cookies are sent with requests and refresh-able
39
+ request.credentials ??= "include";
40
+ }
41
+ },
42
+ });
43
+
44
+ /**
45
+ * Get the current authenticated user.
46
+ *
47
+ * Alias for `alepha.state.get("user")`
48
+ */
49
+ public get user(): UserAccountToken | undefined {
50
+ return this.alepha.store.get("alepha.server.request.user");
51
+ }
52
+
53
+ public async ping() {
54
+ const { data } = await this.httpClient.fetch(
55
+ alephaServerAuthRoutes.userinfo,
56
+ {
57
+ schema: { response: userinfoResponseSchema },
58
+ },
59
+ );
60
+
61
+ this.alepha.store.set("alepha.server.request.apiLinks", data.api);
62
+ this.alepha.store.set("alepha.server.request.user", data.user);
63
+
64
+ return data.user;
65
+ }
66
+
67
+ public can(action: string): boolean {
68
+ if (!this.user) {
69
+ return false;
70
+ }
71
+
72
+ return this.linkProvider.can(action);
73
+ }
74
+
75
+ public async login(
76
+ provider: string,
77
+ options: {
78
+ hostname?: string;
79
+ username?: string;
80
+ password?: string;
81
+ redirect?: string;
82
+ realm?: string;
83
+ [extra: string]: any;
84
+ },
85
+ ): Promise<Tokens> {
86
+ const realmParam = options.realm
87
+ ? `&realm=${encodeURIComponent(options.realm)}`
88
+ : "";
89
+
90
+ if (options.username || options.password) {
91
+ const { data } = await this.httpClient.fetch(
92
+ `${options.hostname || ""}${alephaServerAuthRoutes.token}?provider=${provider}${realmParam}`,
93
+ {
94
+ method: "POST",
95
+ body: JSON.stringify({
96
+ username: options.username,
97
+ password: options.password,
98
+ }),
99
+ schema: { response: tokenResponseSchema },
100
+ },
101
+ );
102
+
103
+ this.alepha.store.set("alepha.server.request.apiLinks", data.api);
104
+ this.alepha.store.set("alepha.server.request.user", data.user);
105
+
106
+ return data;
107
+ }
108
+
109
+ if (this.alepha.isBrowser()) {
110
+ const browser = this.alepha.inject(ReactBrowserProvider);
111
+ const redirect =
112
+ options.redirect ||
113
+ (browser.transitioning
114
+ ? window.location.origin + browser.transitioning.to
115
+ : window.location.href);
116
+
117
+ const href = `${window.location.origin}${alephaServerAuthRoutes.login}?provider=${provider}${realmParam}&redirect_uri=${encodeURIComponent(redirect)}`;
118
+
119
+ if (browser.transitioning) {
120
+ throw new Redirection(href);
121
+ } else {
122
+ window.location.href = href;
123
+ return {} as Tokens;
124
+ }
125
+ }
126
+
127
+ throw new Redirection(
128
+ `${alephaServerAuthRoutes.login}?provider=${provider}${realmParam}&redirect_uri=${options.redirect || "/"}`,
129
+ );
130
+ }
131
+
132
+ public logout() {
133
+ window.location.href = `${alephaServerAuthRoutes.logout}?post_logout_redirect_uri=${encodeURIComponent(window.location.origin)}`;
134
+ }
135
+ }
@@ -0,0 +1,169 @@
1
+ import { Alepha, t } from "alepha";
2
+ import { renderToString } from "react-dom/server";
3
+ import { test } from "vitest";
4
+ import { NestedView } from "../../router/index.ts";
5
+ import { ReactBrowserRouterProvider } from "../../router/providers/ReactBrowserRouterProvider.ts";
6
+
7
+ const setup = () => {
8
+ const alepha = Alepha.create();
9
+ const router = alepha.inject(ReactBrowserRouterProvider);
10
+
11
+ const render = async (path: string): Promise<string> => {
12
+ await router.transition(new URL(`http://localhost${path}`));
13
+ const state = alepha.store.get("alepha.react.router.state");
14
+ if (!state) {
15
+ throw new Error(
16
+ "Router state not found. Ensure the router is properly configured.",
17
+ );
18
+ }
19
+ const element = router.root(state);
20
+ return renderToString(element).replaceAll("<!-- -->", "");
21
+ };
22
+
23
+ return {
24
+ router,
25
+ render,
26
+ alepha,
27
+ };
28
+ };
29
+
30
+ test("Router - Basic", async ({ expect }) => {
31
+ const { router, render, alepha } = setup();
32
+
33
+ router.add({
34
+ name: "Test",
35
+ path: "/",
36
+ component: () => "Hey",
37
+ });
38
+
39
+ router.add({
40
+ name: "NotFound",
41
+ path: "/*",
42
+ component: () => "Not Found",
43
+ });
44
+
45
+ await alepha.start();
46
+
47
+ expect(await render("/")).toEqual("Hey");
48
+ expect(await render("/zz")).toEqual("Not Found");
49
+ });
50
+
51
+ test("Router - NestedView", async ({ expect }) => {
52
+ const { router, render, alepha } = setup();
53
+
54
+ router.add({
55
+ name: "Test",
56
+ component: () => (
57
+ <>
58
+ ((
59
+ <NestedView />
60
+ ))
61
+ </>
62
+ ),
63
+ children: [
64
+ {
65
+ name: "Home",
66
+ path: "/",
67
+ component: () => "Home",
68
+ },
69
+ {
70
+ name: "Hello",
71
+ path: "/hello/:name",
72
+ schema: {
73
+ params: t.object({
74
+ name: t.text(),
75
+ }),
76
+ },
77
+ loader: ({ params }) => params,
78
+ component: (props) => `Hello, ${props.name}!`,
79
+ },
80
+ ],
81
+ });
82
+
83
+ await alepha.start();
84
+
85
+ expect(await render("/")).toEqual("((Home))");
86
+ expect(await render("/hello/jack")).toEqual("((Hello, jack!))");
87
+ });
88
+
89
+ test("Router - All routes", async ({ expect }) => {
90
+ const { router, render, alepha } = setup();
91
+
92
+ router.add({
93
+ children: [
94
+ {
95
+ path: "/*",
96
+ component: () => "404",
97
+ },
98
+ {
99
+ component: () => "home",
100
+ },
101
+ {
102
+ path: "about",
103
+ component: () => "about",
104
+ },
105
+ {
106
+ path: "sub",
107
+ children: [
108
+ {
109
+ component: () => "a",
110
+ },
111
+ {
112
+ path: "b",
113
+ component: () => "b",
114
+ },
115
+ ],
116
+ },
117
+ {
118
+ path: "users",
119
+ component: () => <NestedView>yo</NestedView>,
120
+ children: [
121
+ {
122
+ path: "new",
123
+ component: () => "users/new",
124
+ },
125
+ {
126
+ path: ":id",
127
+ schema: { params: t.object({ id: t.text() }) },
128
+ loader: ({ params }) => {
129
+ if (params.id === "boom") throw new Error("boom");
130
+ return params;
131
+ },
132
+ children: [
133
+ {
134
+ loader: ({ params }) => params,
135
+ component: ({ id }) => `hey ${id}`,
136
+ },
137
+ {
138
+ path: "profile",
139
+ loader: ({ params }) => params,
140
+ component: ({ id }) => `profile of ${id}`,
141
+ },
142
+ ],
143
+ },
144
+ ],
145
+ errorHandler: (error) => {
146
+ return `Error: ${error.message}`;
147
+ },
148
+ },
149
+ ],
150
+ });
151
+
152
+ await alepha.start();
153
+
154
+ expect(await render("/")).toEqual("home");
155
+ expect(await render("/about")).toEqual("about");
156
+ expect(await render("/noop")).toEqual("404");
157
+ expect(await render("/noop/noop")).toEqual("404");
158
+ expect(await render("/sub")).toEqual("a");
159
+ expect(await render("/sub/")).toEqual("a");
160
+ expect(await render("/sub/b")).toEqual("b");
161
+ expect(await render("/sub/noop")).toEqual("404");
162
+ expect(await render("/users")).toEqual("yo");
163
+ expect(await render("/users/")).toEqual("yo");
164
+ expect(await render("/users/a")).toEqual("hey a");
165
+ expect(await render("/users/boom")).toEqual("Error: boom");
166
+ expect(await render("/users/new")).toEqual("users/new");
167
+ expect(await render("/users/hey/ho")).toEqual("404");
168
+ expect(await render("/users/a/profile")).toEqual("profile of a");
169
+ });
@@ -0,0 +1,49 @@
1
+ import {
2
+ type PropsWithChildren,
3
+ type ReactNode,
4
+ useEffect,
5
+ useState,
6
+ } from "react";
7
+
8
+ export interface ClientOnlyProps {
9
+ fallback?: ReactNode;
10
+ disabled?: boolean;
11
+ }
12
+
13
+ /**
14
+ * A small utility component that renders its children only on the client side.
15
+ *
16
+ * Optionally, you can provide a fallback React node that will be rendered.
17
+ *
18
+ * You should use this component when
19
+ * - you have code that relies on browser-specific APIs
20
+ * - you want to avoid server-side rendering for a specific part of your application
21
+ * - you want to prevent pre-rendering of a component
22
+ *
23
+ * @example
24
+ * ```tsx
25
+ * import { ClientOnly } from "alepha/react";
26
+ *
27
+ * const MyComponent = () => {
28
+ * // Avoids SSR issues with Date API
29
+ * return (
30
+ * <ClientOnly>
31
+ * {new Date().toLocaleTimeString()}
32
+ * </ClientOnly>
33
+ * );
34
+ * }
35
+ * ```
36
+ */
37
+ const ClientOnly = (props: PropsWithChildren<ClientOnlyProps>) => {
38
+ const [mounted, setMounted] = useState(false);
39
+
40
+ useEffect(() => setMounted(true), []);
41
+
42
+ if (props.disabled) {
43
+ return props.children;
44
+ }
45
+
46
+ return mounted ? props.children : props.fallback;
47
+ };
48
+
49
+ export default ClientOnly;