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,389 @@
1
+ import { Alepha } from "alepha";
2
+ import { describe, test } from "vitest";
3
+ import { AlephaReactI18n } from "../index.ts";
4
+ import { $dictionary } from "../primitives/$dictionary.ts";
5
+ import { I18nProvider } from "./I18nProvider.ts";
6
+
7
+ describe("I18nProvider", () => {
8
+ test("should register dictionaries on initialization", async ({ expect }) => {
9
+ class App {
10
+ en = $dictionary({
11
+ lazy: async () => ({ default: { hello: "Hello" } }),
12
+ });
13
+
14
+ fr = $dictionary({
15
+ lazy: async () => ({ default: { hello: "Bonjour" } }),
16
+ });
17
+ }
18
+
19
+ const alepha = Alepha.create().with(AlephaReactI18n);
20
+ const app = alepha.inject(App);
21
+ const i18n = alepha.inject(I18nProvider);
22
+
23
+ expect(i18n.registry).toHaveLength(2);
24
+ expect(i18n.registry[0].lang).toBe("en");
25
+ expect(i18n.registry[1].lang).toBe("fr");
26
+ });
27
+
28
+ test("should load all translations on server start", async ({ expect }) => {
29
+ class App {
30
+ en = $dictionary({
31
+ lazy: async () => ({ default: { hello: "Hello", world: "World" } }),
32
+ });
33
+
34
+ fr = $dictionary({
35
+ lazy: async () => ({ default: { hello: "Bonjour", world: "Monde" } }),
36
+ });
37
+ }
38
+
39
+ const alepha = Alepha.create().with(AlephaReactI18n);
40
+ const app = alepha.inject(App);
41
+ const i18n = alepha.inject(I18nProvider);
42
+
43
+ await alepha.start();
44
+
45
+ expect(i18n.registry[0].translations).toEqual({
46
+ hello: "Hello",
47
+ world: "World",
48
+ });
49
+ expect(i18n.registry[1].translations).toEqual({
50
+ hello: "Bonjour",
51
+ world: "Monde",
52
+ });
53
+ });
54
+
55
+ test("should default to fallback language", async ({ expect }) => {
56
+ class App {
57
+ en = $dictionary({
58
+ lazy: async () => ({ default: { hello: "Hello" } }),
59
+ });
60
+ }
61
+
62
+ const alepha = Alepha.create().with(AlephaReactI18n);
63
+ const app = alepha.inject(App);
64
+ const i18n = alepha.inject(I18nProvider);
65
+
66
+ expect(i18n.lang).toBe("en");
67
+ });
68
+
69
+ test("should translate keys in current language", async ({ expect }) => {
70
+ class App {
71
+ en = $dictionary({
72
+ lazy: async () => ({ default: { greeting: "Hello, World!" } }),
73
+ });
74
+
75
+ fr = $dictionary({
76
+ lazy: async () => ({ default: { greeting: "Bonjour, Monde!" } }),
77
+ });
78
+ }
79
+
80
+ const alepha = Alepha.create().with(AlephaReactI18n);
81
+ const app = alepha.inject(App);
82
+ const i18n = alepha.inject(I18nProvider);
83
+
84
+ await alepha.start();
85
+
86
+ expect(i18n.tr("greeting")).toBe("Hello, World!");
87
+
88
+ await i18n.setLang("fr");
89
+ expect(i18n.tr("greeting")).toBe("Bonjour, Monde!");
90
+ });
91
+
92
+ test("should fallback to fallback language when key missing", async ({
93
+ expect,
94
+ }) => {
95
+ class App {
96
+ en = $dictionary({
97
+ lazy: async () => ({
98
+ default: { hello: "Hello", world: "World" },
99
+ }),
100
+ });
101
+
102
+ fr = $dictionary({
103
+ lazy: async () => ({ default: { hello: "Bonjour" } }),
104
+ });
105
+ }
106
+
107
+ const alepha = Alepha.create().with(AlephaReactI18n);
108
+ const app = alepha.inject(App);
109
+ const i18n = alepha.inject(I18nProvider);
110
+
111
+ await alepha.start();
112
+ await i18n.setLang("fr");
113
+
114
+ expect(i18n.tr("hello")).toBe("Bonjour");
115
+ expect(i18n.tr("world")).toBe("World"); // Falls back to English
116
+ });
117
+
118
+ test("should return key when translation not found", async ({ expect }) => {
119
+ class App {
120
+ en = $dictionary({
121
+ lazy: async () => ({ default: { hello: "Hello" } }),
122
+ });
123
+ }
124
+
125
+ const alepha = Alepha.create().with(AlephaReactI18n);
126
+ const app = alepha.inject(App);
127
+ const i18n = alepha.inject(I18nProvider);
128
+
129
+ await alepha.start();
130
+
131
+ expect(i18n.tr("missing.key")).toBe("missing.key");
132
+ });
133
+
134
+ test("should support default option when key missing", async ({ expect }) => {
135
+ class App {
136
+ en = $dictionary({
137
+ lazy: async () => ({ default: { hello: "Hello" } }),
138
+ });
139
+ }
140
+
141
+ const alepha = Alepha.create().with(AlephaReactI18n);
142
+ const app = alepha.inject(App);
143
+ const i18n = alepha.inject(I18nProvider);
144
+
145
+ await alepha.start();
146
+
147
+ expect(i18n.tr("missing", { default: "Default Text" })).toBe(
148
+ "Default Text",
149
+ );
150
+ });
151
+
152
+ test("should support variable interpolation", async ({ expect }) => {
153
+ class App {
154
+ en = $dictionary({
155
+ lazy: async () => ({
156
+ default: { greeting: "Hello, $1! You have $2 messages." },
157
+ }),
158
+ });
159
+ }
160
+
161
+ const alepha = Alepha.create().with(AlephaReactI18n);
162
+ const app = alepha.inject(App);
163
+ const i18n = alepha.inject(I18nProvider);
164
+
165
+ await alepha.start();
166
+
167
+ expect(i18n.tr("greeting", { args: ["John", "5"] })).toBe(
168
+ "Hello, John! You have 5 messages.",
169
+ );
170
+ });
171
+
172
+ test("should list all available languages", async ({ expect }) => {
173
+ class App {
174
+ en = $dictionary({
175
+ lazy: async () => ({ default: { hello: "Hello" } }),
176
+ });
177
+
178
+ fr = $dictionary({
179
+ lazy: async () => ({ default: { hello: "Bonjour" } }),
180
+ });
181
+
182
+ de = $dictionary({
183
+ lazy: async () => ({ default: { hello: "Hallo" } }),
184
+ });
185
+ }
186
+
187
+ const alepha = Alepha.create().with(AlephaReactI18n);
188
+ const app = alepha.inject(App);
189
+ const i18n = alepha.inject(I18nProvider);
190
+
191
+ const languages = i18n.languages;
192
+ expect(languages).toContain("en");
193
+ expect(languages).toContain("fr");
194
+ expect(languages).toContain("de");
195
+ expect(languages).toHaveLength(3);
196
+ });
197
+
198
+ test("should support custom language codes", async ({ expect }) => {
199
+ class App {
200
+ enUS = $dictionary({
201
+ lang: "en-US",
202
+ lazy: async () => ({ default: { color: "color" } }),
203
+ });
204
+
205
+ enGB = $dictionary({
206
+ lang: "en-GB",
207
+ lazy: async () => ({ default: { color: "colour" } }),
208
+ });
209
+ }
210
+
211
+ const alepha = Alepha.create().with(AlephaReactI18n);
212
+ const app = alepha.inject(App);
213
+ const i18n = alepha.inject(I18nProvider);
214
+
215
+ await alepha.start();
216
+
217
+ expect(i18n.registry[0].lang).toBe("en-US");
218
+ expect(i18n.registry[1].lang).toBe("en-GB");
219
+ });
220
+
221
+ test("should support custom dictionary names", async ({ expect }) => {
222
+ class App {
223
+ english = $dictionary({
224
+ name: "custom-en",
225
+ lang: "en",
226
+ lazy: async () => ({ default: { hello: "Hello" } }),
227
+ });
228
+ }
229
+
230
+ const alepha = Alepha.create().with(AlephaReactI18n);
231
+ const app = alepha.inject(App);
232
+ const i18n = alepha.inject(I18nProvider);
233
+
234
+ expect(i18n.registry[0].name).toBe("custom-en");
235
+ });
236
+
237
+ test("should update state when language changes", async ({ expect }) => {
238
+ class App {
239
+ en = $dictionary({
240
+ lazy: async () => ({ default: { hello: "Hello" } }),
241
+ });
242
+
243
+ fr = $dictionary({
244
+ lazy: async () => ({ default: { hello: "Bonjour" } }),
245
+ });
246
+ }
247
+
248
+ const alepha = Alepha.create().with(AlephaReactI18n);
249
+ const app = alepha.inject(App);
250
+ const i18n = alepha.inject(I18nProvider);
251
+
252
+ await alepha.start();
253
+
254
+ // State is set lazily, so we should set a language first
255
+ await i18n.setLang("en");
256
+ expect(alepha.store.get("alepha.react.i18n.lang")).toBe("en");
257
+
258
+ await i18n.setLang("fr");
259
+ expect(alepha.store.get("alepha.react.i18n.lang")).toBe("fr");
260
+ });
261
+
262
+ test("should have number formatter for current language", async ({
263
+ expect,
264
+ }) => {
265
+ class App {
266
+ en = $dictionary({
267
+ lazy: async () => ({ default: {} }),
268
+ });
269
+ }
270
+
271
+ const alepha = Alepha.create().with(AlephaReactI18n);
272
+ const app = alepha.inject(App);
273
+ const i18n = alepha.inject(I18nProvider);
274
+
275
+ await alepha.start();
276
+
277
+ expect(i18n.numberFormat).toBeDefined();
278
+ expect(i18n.numberFormat.format).toBeTypeOf("function");
279
+ });
280
+
281
+ test("should handle multiple translations with same key across dictionaries", async ({
282
+ expect,
283
+ }) => {
284
+ class App {
285
+ en = $dictionary({
286
+ lazy: async () => ({
287
+ default: {
288
+ app: "Application",
289
+ settings: "Settings",
290
+ },
291
+ }),
292
+ });
293
+
294
+ fr = $dictionary({
295
+ lazy: async () => ({
296
+ default: {
297
+ app: "Programme",
298
+ settings: "Paramètres",
299
+ },
300
+ }),
301
+ });
302
+
303
+ es = $dictionary({
304
+ lazy: async () => ({
305
+ default: {
306
+ app: "Aplicación",
307
+ settings: "Configuración",
308
+ },
309
+ }),
310
+ });
311
+ }
312
+
313
+ const alepha = Alepha.create().with(AlephaReactI18n);
314
+ const app = alepha.inject(App);
315
+ const i18n = alepha.inject(I18nProvider);
316
+
317
+ await alepha.start();
318
+
319
+ expect(i18n.tr("app")).toBe("Application");
320
+ expect(i18n.tr("settings")).toBe("Settings");
321
+
322
+ await i18n.setLang("fr");
323
+ expect(i18n.tr("app")).toBe("Programme");
324
+ expect(i18n.tr("settings")).toBe("Paramètres");
325
+
326
+ await i18n.setLang("es");
327
+ expect(i18n.tr("app")).toBe("Aplicación");
328
+ expect(i18n.tr("settings")).toBe("Configuración");
329
+ });
330
+
331
+ test("should handle empty dictionaries", async ({ expect }) => {
332
+ class App {
333
+ en = $dictionary({
334
+ lazy: async () => ({ default: {} }),
335
+ });
336
+ }
337
+
338
+ const alepha = Alepha.create().with(AlephaReactI18n);
339
+ const app = alepha.inject(App);
340
+ const i18n = alepha.inject(I18nProvider);
341
+
342
+ await alepha.start();
343
+
344
+ expect(i18n.tr("anything")).toBe("anything");
345
+ });
346
+
347
+ test("should handle complex interpolation patterns", async ({ expect }) => {
348
+ class App {
349
+ en = $dictionary({
350
+ lazy: async () => ({
351
+ default: {
352
+ complex: "User $1 has $2 followers and follows $3 people",
353
+ },
354
+ }),
355
+ });
356
+ }
357
+
358
+ const alepha = Alepha.create().with(AlephaReactI18n);
359
+ const app = alepha.inject(App);
360
+ const i18n = alepha.inject(I18nProvider);
361
+
362
+ await alepha.start();
363
+
364
+ expect(i18n.tr("complex", { args: ["Alice", "1000", "500"] })).toBe(
365
+ "User Alice has 1000 followers and follows 500 people",
366
+ );
367
+ });
368
+
369
+ test("should handle partial interpolation args", async ({ expect }) => {
370
+ class App {
371
+ en = $dictionary({
372
+ lazy: async () => ({
373
+ default: { message: "Hello $1, you have $2 messages and $3 alerts" },
374
+ }),
375
+ });
376
+ }
377
+
378
+ const alepha = Alepha.create().with(AlephaReactI18n);
379
+ const app = alepha.inject(App);
380
+ const i18n = alepha.inject(I18nProvider);
381
+
382
+ await alepha.start();
383
+
384
+ // Provide fewer args than placeholders
385
+ expect(i18n.tr("message", { args: ["Bob"] })).toBe(
386
+ "Hello Bob, you have $2 messages and $3 alerts",
387
+ );
388
+ });
389
+ });
@@ -0,0 +1,278 @@
1
+ import { $hook, $inject, Alepha, TypeBoxError, TypeProvider, t } from "alepha";
2
+ import { type DateTime, DateTimeProvider } from "alepha/datetime";
3
+ import { $logger } from "alepha/logger";
4
+ import { $cookie } from "alepha/server/cookies";
5
+ import type { ServiceDictionary } from "../hooks/useI18n.ts";
6
+
7
+ export class I18nProvider<
8
+ S extends object,
9
+ K extends keyof ServiceDictionary<S>,
10
+ > {
11
+ protected log = $logger();
12
+ protected alepha = $inject(Alepha);
13
+ protected dateTimeProvider = $inject(DateTimeProvider);
14
+
15
+ protected cookie = $cookie({
16
+ name: "lang",
17
+ schema: t.text(),
18
+ ttl: [1, "year"],
19
+ });
20
+
21
+ public readonly registry: Array<{
22
+ target: string;
23
+ name: string;
24
+ lang: string;
25
+ loader: () => Promise<Record<string, string>>;
26
+ translations: Record<string, string>;
27
+ }> = [];
28
+
29
+ options = {
30
+ fallbackLang: "en",
31
+ };
32
+
33
+ public dateFormat: { format: (value: Date) => string } =
34
+ new Intl.DateTimeFormat(this.lang);
35
+
36
+ public numberFormat: { format: (value: number) => string } =
37
+ new Intl.NumberFormat(this.lang);
38
+
39
+ public get languages() {
40
+ const languages = new Set<string>();
41
+
42
+ for (const item of this.registry) {
43
+ languages.add(item.lang);
44
+ }
45
+ languages.add(this.options.fallbackLang);
46
+
47
+ return Array.from(languages);
48
+ }
49
+
50
+ constructor() {
51
+ this.refreshLocale();
52
+ }
53
+
54
+ protected readonly onRender = $hook({
55
+ on: "server:onRequest",
56
+ priority: "last",
57
+ handler: async ({ request }) => {
58
+ this.alepha.store.set("alepha.react.i18n.lang", this.cookie.get(request));
59
+ },
60
+ });
61
+
62
+ protected readonly onStart = $hook({
63
+ on: "start",
64
+ handler: async () => {
65
+ if (this.alepha.isBrowser()) {
66
+ // get cookie lang
67
+ const cookieLang = this.cookie.get();
68
+ if (cookieLang) {
69
+ this.alepha.store.set("alepha.react.i18n.lang", cookieLang);
70
+ }
71
+
72
+ for (const item of this.registry) {
73
+ if (
74
+ item.lang === this.lang ||
75
+ item.lang === this.options.fallbackLang
76
+ ) {
77
+ this.log.trace("Loading language", {
78
+ lang: item.lang,
79
+ name: item.name,
80
+ target: item.target,
81
+ });
82
+ item.translations = await item.loader();
83
+ }
84
+ }
85
+ return;
86
+ }
87
+
88
+ for (const item of this.registry) {
89
+ item.translations = await item.loader();
90
+ }
91
+ },
92
+ });
93
+
94
+ protected refreshLocale() {
95
+ this.numberFormat = new Intl.NumberFormat(this.lang);
96
+ this.dateFormat = new Intl.DateTimeFormat(this.lang);
97
+ this.dateTimeProvider.setLocale(this.lang);
98
+ TypeProvider.setLocale(this.lang);
99
+ }
100
+
101
+ public setLang = async (lang: string) => {
102
+ if (this.alepha.isBrowser()) {
103
+ for (const item of this.registry) {
104
+ if (lang === item.lang) {
105
+ if (Object.keys(item.translations).length > 0) {
106
+ continue; // already loaded
107
+ }
108
+ item.translations = await item.loader();
109
+ }
110
+ }
111
+ this.cookie.set(lang);
112
+ }
113
+
114
+ this.alepha.store.set("alepha.react.i18n.lang", lang);
115
+ this.refreshLocale();
116
+ };
117
+
118
+ protected readonly mutate = $hook({
119
+ on: "state:mutate",
120
+ handler: async ({ key, value }) => {
121
+ if (key === "alepha.react.i18n.lang" && this.alepha.isBrowser()) {
122
+ let hasChanged = false;
123
+ for (const item of this.registry) {
124
+ if (value === item.lang) {
125
+ if (Object.keys(item.translations).length > 0) {
126
+ continue; // already loaded
127
+ }
128
+ item.translations = await item.loader();
129
+ hasChanged = true;
130
+ }
131
+ }
132
+
133
+ this.refreshLocale();
134
+
135
+ if (hasChanged) {
136
+ this.alepha.store.set("alepha.react.i18n.lang", value);
137
+ }
138
+ }
139
+ },
140
+ });
141
+
142
+ public get lang(): string {
143
+ return (
144
+ this.alepha.store.get("alepha.react.i18n.lang") ||
145
+ this.options.fallbackLang
146
+ );
147
+ }
148
+
149
+ public translate = (key: string, args: string[] = []) => {
150
+ for (const item of this.registry) {
151
+ if (item.lang === this.lang) {
152
+ if (item.translations[key]) {
153
+ return this.render(item.translations[key], args); // append lang for fallback
154
+ }
155
+ }
156
+ }
157
+
158
+ for (const item of this.registry) {
159
+ if (item.lang === this.options.fallbackLang) {
160
+ if (item.translations[key]) {
161
+ return this.render(item.translations[key], args); // append lang for fallback
162
+ }
163
+ }
164
+ }
165
+
166
+ return key; // fallback to the key itself if not found
167
+ };
168
+
169
+ public readonly l = (
170
+ value: I18nLocalizeType,
171
+ options: I18nLocalizeOptions = {},
172
+ ) => {
173
+ // Handle numbers
174
+ if (typeof value === "number" && !options.date) {
175
+ return new Intl.NumberFormat(this.lang, options.number).format(value);
176
+ }
177
+
178
+ // Handle dates
179
+ if (
180
+ value instanceof Date ||
181
+ this.dateTimeProvider.isDateTime(value) ||
182
+ (typeof value === "string" && options.date) ||
183
+ (typeof value === "number" && options.date)
184
+ ) {
185
+ // convert to DateTime with locale applied
186
+ let dt = this.dateTimeProvider.of(value);
187
+
188
+ // apply timezone if specified
189
+ if (options.timezone) {
190
+ dt = dt.tz(options.timezone);
191
+ }
192
+
193
+ // format using dayjs format string
194
+ if (typeof options.date === "string") {
195
+ if (options.date === "fromNow") {
196
+ return dt.locale(this.lang).fromNow();
197
+ }
198
+ return dt.locale(this.lang).format(options.date);
199
+ }
200
+
201
+ // format using Intl.DateTimeFormatOptions
202
+ if (options.date) {
203
+ return new Intl.DateTimeFormat(
204
+ this.lang,
205
+ options.timezone
206
+ ? { ...options.date, timeZone: options.timezone }
207
+ : options.date,
208
+ ).format(dt.toDate());
209
+ }
210
+
211
+ // default formatting with timezone
212
+ if (options.timezone) {
213
+ return new Intl.DateTimeFormat(this.lang, {
214
+ timeZone: options.timezone,
215
+ }).format(dt.toDate());
216
+ }
217
+
218
+ // default formatting
219
+ return new Intl.DateTimeFormat(this.lang).format(dt.toDate());
220
+ }
221
+
222
+ // handle TypeBox errors
223
+ if (value instanceof TypeBoxError) {
224
+ return TypeProvider.translateError(value, this.lang);
225
+ }
226
+
227
+ // return string values as-is
228
+ return value;
229
+ };
230
+
231
+ public readonly tr = (
232
+ key: keyof ServiceDictionary<S>[K],
233
+ options: {
234
+ args?: string[];
235
+ default?: string;
236
+ } = {},
237
+ ) => {
238
+ const translation = this.translate(key as string, options.args || []);
239
+ if (translation === key && options.default) {
240
+ return options.default;
241
+ }
242
+ return translation;
243
+ };
244
+
245
+ protected render(item: string, args: string[]): string {
246
+ let result = item;
247
+ for (let i = 0; i < args.length; i++) {
248
+ result = result.replace(`$${i + 1}`, args[i]);
249
+ }
250
+ return result;
251
+ }
252
+ }
253
+
254
+ export type I18nLocalizeType = string | number | Date | DateTime | TypeBoxError;
255
+
256
+ export interface I18nLocalizeOptions {
257
+ /**
258
+ * Options for number formatting (when value is a number)
259
+ * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Intl/NumberFormat/NumberFormat
260
+ */
261
+ number?: Intl.NumberFormatOptions;
262
+ /**
263
+ * Options for date formatting (when value is a Date or DateTime)
264
+ * Can be:
265
+ * - A dayjs format string (e.g., "LLL", "YYYY-MM-DD", "dddd, MMMM D YYYY")
266
+ * - "fromNow" for relative time (e.g., "2 hours ago")
267
+ * - Intl.DateTimeFormatOptions for native formatting
268
+ * @see https://day.js.org/docs/en/display/format
269
+ * @see https://day.js.org/docs/en/display/from-now
270
+ */
271
+ date?: string | "fromNow" | Intl.DateTimeFormatOptions;
272
+ /**
273
+ * Timezone to display dates in (when value is a Date or DateTime)
274
+ * Uses IANA timezone names (e.g., "America/New_York", "Europe/Paris", "Asia/Tokyo")
275
+ * @see https://day.js.org/docs/en/timezone/timezone
276
+ */
277
+ timezone?: string;
278
+ }