alepha 0.21.1 → 0.22.0

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 (481) hide show
  1. package/README.md +0 -1
  2. package/dist/api/audits/index.browser.js.map +1 -1
  3. package/dist/api/audits/index.d.ts +393 -403
  4. package/dist/api/audits/index.d.ts.map +1 -1
  5. package/dist/api/audits/index.js +25 -56
  6. package/dist/api/audits/index.js.map +1 -1
  7. package/dist/api/files/index.browser.js +31 -1
  8. package/dist/api/files/index.browser.js.map +1 -1
  9. package/dist/api/files/index.d.ts +353 -207
  10. package/dist/api/files/index.d.ts.map +1 -1
  11. package/dist/api/files/index.js +211 -45
  12. package/dist/api/files/index.js.map +1 -1
  13. package/dist/api/jobs/index.browser.js +2 -2
  14. package/dist/api/jobs/index.browser.js.map +1 -1
  15. package/dist/api/jobs/index.d.ts +289 -292
  16. package/dist/api/jobs/index.d.ts.map +1 -1
  17. package/dist/api/jobs/index.js +39 -33
  18. package/dist/api/jobs/index.js.map +1 -1
  19. package/dist/api/keys/index.d.ts +211 -216
  20. package/dist/api/keys/index.d.ts.map +1 -1
  21. package/dist/api/keys/index.js.map +1 -1
  22. package/dist/api/notifications/index.browser.js.map +1 -1
  23. package/dist/api/notifications/index.d.ts +188 -195
  24. package/dist/api/notifications/index.d.ts.map +1 -1
  25. package/dist/api/notifications/index.js.map +1 -1
  26. package/dist/api/oauth/index.d.ts +71 -76
  27. package/dist/api/oauth/index.d.ts.map +1 -1
  28. package/dist/api/oauth/index.js.map +1 -1
  29. package/dist/api/organizations/index.browser.js.map +1 -1
  30. package/dist/api/organizations/index.d.ts +104 -109
  31. package/dist/api/organizations/index.d.ts.map +1 -1
  32. package/dist/api/organizations/index.js.map +1 -1
  33. package/dist/api/parameters/index.browser.js +43 -16
  34. package/dist/api/parameters/index.browser.js.map +1 -1
  35. package/dist/api/parameters/index.d.ts +488 -344
  36. package/dist/api/parameters/index.d.ts.map +1 -1
  37. package/dist/api/parameters/index.js +175 -35
  38. package/dist/api/parameters/index.js.map +1 -1
  39. package/dist/api/payments/index.d.ts +396 -402
  40. package/dist/api/payments/index.d.ts.map +1 -1
  41. package/dist/api/payments/index.js.map +1 -1
  42. package/dist/api/subscriptions/index.d.ts +644 -652
  43. package/dist/api/subscriptions/index.d.ts.map +1 -1
  44. package/dist/api/subscriptions/index.js +1 -1
  45. package/dist/api/subscriptions/index.js.map +1 -1
  46. package/dist/api/users/index.browser.js +7 -0
  47. package/dist/api/users/index.browser.js.map +1 -1
  48. package/dist/api/users/index.d.ts +1072 -1005
  49. package/dist/api/users/index.d.ts.map +1 -1
  50. package/dist/api/users/index.js +311 -87
  51. package/dist/api/users/index.js.map +1 -1
  52. package/dist/api/verifications/index.browser.js +1 -0
  53. package/dist/api/verifications/index.browser.js.map +1 -1
  54. package/dist/api/verifications/index.d.ts +144 -133
  55. package/dist/api/verifications/index.d.ts.map +1 -1
  56. package/dist/api/verifications/index.js +5 -1
  57. package/dist/api/verifications/index.js.map +1 -1
  58. package/dist/background/index.d.ts +95 -0
  59. package/dist/background/index.d.ts.map +1 -0
  60. package/dist/background/index.js +121 -0
  61. package/dist/background/index.js.map +1 -0
  62. package/dist/background/index.workerd.js +110 -0
  63. package/dist/background/index.workerd.js.map +1 -0
  64. package/dist/batch/index.d.ts +5 -7
  65. package/dist/batch/index.d.ts.map +1 -1
  66. package/dist/batch/index.js.map +1 -1
  67. package/dist/bin/index.js.map +1 -1
  68. package/dist/bucket/index.d.ts +76 -54
  69. package/dist/bucket/index.d.ts.map +1 -1
  70. package/dist/bucket/index.js +58 -11
  71. package/dist/bucket/index.js.map +1 -1
  72. package/dist/bucket/index.workerd.js +200 -5
  73. package/dist/bucket/index.workerd.js.map +1 -1
  74. package/dist/cache/core/index.d.ts +7 -10
  75. package/dist/cache/core/index.d.ts.map +1 -1
  76. package/dist/cache/core/index.js.map +1 -1
  77. package/dist/cache/core/index.workerd.js.map +1 -1
  78. package/dist/cache/database/index.d.ts +22 -26
  79. package/dist/cache/database/index.d.ts.map +1 -1
  80. package/dist/cache/database/index.js.map +1 -1
  81. package/dist/cache/redis/index.d.ts +4 -7
  82. package/dist/cache/redis/index.d.ts.map +1 -1
  83. package/dist/cache/redis/index.js.map +1 -1
  84. package/dist/captcha/index.d.ts +3 -6
  85. package/dist/captcha/index.d.ts.map +1 -1
  86. package/dist/captcha/index.js.map +1 -1
  87. package/dist/cli/config/index.d.ts.map +1 -1
  88. package/dist/cli/config/index.js.map +1 -1
  89. package/dist/cli/core/index.d.ts +460 -188
  90. package/dist/cli/core/index.d.ts.map +1 -1
  91. package/dist/cli/core/index.js +411 -551
  92. package/dist/cli/core/index.js.map +1 -1
  93. package/dist/cli/devtools/index.d.ts +3 -5
  94. package/dist/cli/devtools/index.d.ts.map +1 -1
  95. package/dist/cli/devtools/index.js.map +1 -1
  96. package/dist/cli/i18n/index.d.ts +8 -12
  97. package/dist/cli/i18n/index.d.ts.map +1 -1
  98. package/dist/cli/i18n/index.js.map +1 -1
  99. package/dist/cli/platform/index.d.ts +127 -1382
  100. package/dist/cli/platform/index.d.ts.map +1 -1
  101. package/dist/cli/platform/index.js +188 -2493
  102. package/dist/cli/platform/index.js.map +1 -1
  103. package/dist/cli/platform-lib/index.d.ts +1446 -0
  104. package/dist/cli/platform-lib/index.d.ts.map +1 -0
  105. package/dist/cli/platform-lib/index.js +2597 -0
  106. package/dist/cli/platform-lib/index.js.map +1 -0
  107. package/dist/cli/vendor/index.d.ts +17 -21
  108. package/dist/cli/vendor/index.d.ts.map +1 -1
  109. package/dist/cli/vendor/index.js.map +1 -1
  110. package/dist/command/index.d.ts +21 -20
  111. package/dist/command/index.d.ts.map +1 -1
  112. package/dist/command/index.js +39 -10
  113. package/dist/command/index.js.map +1 -1
  114. package/dist/container/core/index.d.ts +236 -0
  115. package/dist/container/core/index.d.ts.map +1 -0
  116. package/dist/container/core/index.js +231 -0
  117. package/dist/container/core/index.js.map +1 -0
  118. package/dist/container/core/index.workerd.js +192 -0
  119. package/dist/container/core/index.workerd.js.map +1 -0
  120. package/dist/core/index.browser.js +27 -1
  121. package/dist/core/index.browser.js.map +1 -1
  122. package/dist/core/index.d.ts +48 -24
  123. package/dist/core/index.d.ts.map +1 -1
  124. package/dist/core/index.js +27 -1
  125. package/dist/core/index.js.map +1 -1
  126. package/dist/core/index.native.js +27 -1
  127. package/dist/core/index.native.js.map +1 -1
  128. package/dist/core/index.workerd.js +27 -1
  129. package/dist/core/index.workerd.js.map +1 -1
  130. package/dist/crypto/index.browser.js.map +1 -1
  131. package/dist/crypto/index.d.ts +5 -8
  132. package/dist/crypto/index.d.ts.map +1 -1
  133. package/dist/crypto/index.js.map +1 -1
  134. package/dist/datetime/index.d.ts +3 -4
  135. package/dist/datetime/index.d.ts.map +1 -1
  136. package/dist/datetime/index.js.map +1 -1
  137. package/dist/email/brevo/index.d.ts +2 -4
  138. package/dist/email/brevo/index.d.ts.map +1 -1
  139. package/dist/email/brevo/index.js.map +1 -1
  140. package/dist/email/cloudflare/index.d.ts +103 -0
  141. package/dist/email/cloudflare/index.d.ts.map +1 -0
  142. package/dist/email/cloudflare/index.js +153 -0
  143. package/dist/email/cloudflare/index.js.map +1 -0
  144. package/dist/email/core/index.d.ts +6 -9
  145. package/dist/email/core/index.d.ts.map +1 -1
  146. package/dist/email/core/index.js.map +1 -1
  147. package/dist/email/core/index.workerd.js.map +1 -1
  148. package/dist/email/smtp/index.d.ts +10 -13
  149. package/dist/email/smtp/index.d.ts.map +1 -1
  150. package/dist/email/smtp/index.js +107 -32
  151. package/dist/email/smtp/index.js.map +1 -1
  152. package/dist/fake/index.d.ts +1 -2
  153. package/dist/fake/index.d.ts.map +1 -1
  154. package/dist/fake/index.js.map +1 -1
  155. package/dist/lock/core/index.d.ts +9 -14
  156. package/dist/lock/core/index.d.ts.map +1 -1
  157. package/dist/lock/core/index.js.map +1 -1
  158. package/dist/lock/redis/index.d.ts +2 -4
  159. package/dist/lock/redis/index.d.ts.map +1 -1
  160. package/dist/lock/redis/index.js.map +1 -1
  161. package/dist/logger/index.d.ts +105 -76
  162. package/dist/logger/index.d.ts.map +1 -1
  163. package/dist/logger/index.js +196 -174
  164. package/dist/logger/index.js.map +1 -1
  165. package/dist/mcp/index.d.ts +16 -20
  166. package/dist/mcp/index.d.ts.map +1 -1
  167. package/dist/mcp/index.js.map +1 -1
  168. package/dist/orm/core/index.browser.js.map +1 -1
  169. package/dist/orm/core/index.bun.js +21 -3
  170. package/dist/orm/core/index.bun.js.map +1 -1
  171. package/dist/orm/core/index.d.ts +76 -62
  172. package/dist/orm/core/index.d.ts.map +1 -1
  173. package/dist/orm/core/index.js +22 -4
  174. package/dist/orm/core/index.js.map +1 -1
  175. package/dist/orm/postgres/index.bun.js.map +1 -1
  176. package/dist/orm/postgres/index.d.ts +28 -20
  177. package/dist/orm/postgres/index.d.ts.map +1 -1
  178. package/dist/orm/postgres/index.js.map +1 -1
  179. package/dist/queue/core/index.d.ts +12 -15
  180. package/dist/queue/core/index.d.ts.map +1 -1
  181. package/dist/queue/core/index.js.map +1 -1
  182. package/dist/queue/core/index.workerd.js.map +1 -1
  183. package/dist/queue/redis/index.d.ts +3 -5
  184. package/dist/queue/redis/index.d.ts.map +1 -1
  185. package/dist/queue/redis/index.js.map +1 -1
  186. package/dist/react/auth/index.browser.js +9 -2
  187. package/dist/react/auth/index.browser.js.map +1 -1
  188. package/dist/react/auth/index.d.ts +14 -9
  189. package/dist/react/auth/index.d.ts.map +1 -1
  190. package/dist/react/auth/index.js +9 -2
  191. package/dist/react/auth/index.js.map +1 -1
  192. package/dist/react/core/index.d.ts +7 -8
  193. package/dist/react/core/index.d.ts.map +1 -1
  194. package/dist/react/core/index.js +6 -3
  195. package/dist/react/core/index.js.map +1 -1
  196. package/dist/react/form/index.d.ts +38 -5
  197. package/dist/react/form/index.d.ts.map +1 -1
  198. package/dist/react/form/index.js +91 -3
  199. package/dist/react/form/index.js.map +1 -1
  200. package/dist/react/head/index.browser.js.map +1 -1
  201. package/dist/react/head/index.d.ts +2 -4
  202. package/dist/react/head/index.d.ts.map +1 -1
  203. package/dist/react/head/index.js.map +1 -1
  204. package/dist/react/i18n/index.d.ts +47 -11
  205. package/dist/react/i18n/index.d.ts.map +1 -1
  206. package/dist/react/i18n/index.js +33 -1
  207. package/dist/react/i18n/index.js.map +1 -1
  208. package/dist/react/intro/index.d.ts +1 -2
  209. package/dist/react/intro/index.d.ts.map +1 -1
  210. package/dist/react/intro/index.js +2 -2
  211. package/dist/react/intro/index.js.map +1 -1
  212. package/dist/react/router/index.browser.js +66 -20
  213. package/dist/react/router/index.browser.js.map +1 -1
  214. package/dist/react/router/index.d.ts +327 -222
  215. package/dist/react/router/index.d.ts.map +1 -1
  216. package/dist/react/router/index.js +66 -30
  217. package/dist/react/router/index.js.map +1 -1
  218. package/dist/react/testing/index.d.ts +1 -2
  219. package/dist/react/testing/index.d.ts.map +1 -1
  220. package/dist/react/testing/index.js +16 -17
  221. package/dist/react/testing/index.js.map +1 -1
  222. package/dist/react/ui/index.d.ts +20 -25
  223. package/dist/react/ui/index.d.ts.map +1 -1
  224. package/dist/react/ui/index.js.map +1 -1
  225. package/dist/redis/index.bun.js.map +1 -1
  226. package/dist/redis/index.d.ts +17 -19
  227. package/dist/redis/index.d.ts.map +1 -1
  228. package/dist/redis/index.js.map +1 -1
  229. package/dist/retry/index.d.ts +2 -4
  230. package/dist/retry/index.d.ts.map +1 -1
  231. package/dist/retry/index.js.map +1 -1
  232. package/dist/router/index.d.ts.map +1 -1
  233. package/dist/router/index.js.map +1 -1
  234. package/dist/scheduler/index.d.ts +10 -13
  235. package/dist/scheduler/index.d.ts.map +1 -1
  236. package/dist/scheduler/index.js.map +1 -1
  237. package/dist/scheduler/index.workerd.js.map +1 -1
  238. package/dist/security/index.browser.js.map +1 -1
  239. package/dist/security/index.d.ts +45 -48
  240. package/dist/security/index.d.ts.map +1 -1
  241. package/dist/security/index.js.map +1 -1
  242. package/dist/server/auth/index.browser.js.map +1 -1
  243. package/dist/server/auth/index.d.ts +167 -172
  244. package/dist/server/auth/index.d.ts.map +1 -1
  245. package/dist/server/auth/index.js +4 -8
  246. package/dist/server/auth/index.js.map +1 -1
  247. package/dist/server/cookies/index.browser.js.map +1 -1
  248. package/dist/server/cookies/index.d.ts +5 -7
  249. package/dist/server/cookies/index.d.ts.map +1 -1
  250. package/dist/server/cookies/index.js.map +1 -1
  251. package/dist/server/core/index.browser.js.map +1 -1
  252. package/dist/server/core/index.d.ts +88 -73
  253. package/dist/server/core/index.d.ts.map +1 -1
  254. package/dist/server/core/index.js +19 -0
  255. package/dist/server/core/index.js.map +1 -1
  256. package/dist/server/cors/index.d.ts +11 -14
  257. package/dist/server/cors/index.d.ts.map +1 -1
  258. package/dist/server/cors/index.js.map +1 -1
  259. package/dist/server/etag/index.d.ts +6 -9
  260. package/dist/server/etag/index.d.ts.map +1 -1
  261. package/dist/server/etag/index.js.map +1 -1
  262. package/dist/server/health/index.d.ts +18 -21
  263. package/dist/server/health/index.d.ts.map +1 -1
  264. package/dist/server/health/index.js.map +1 -1
  265. package/dist/server/links/index.browser.js +2 -0
  266. package/dist/server/links/index.browser.js.map +1 -1
  267. package/dist/server/links/index.d.ts +63 -67
  268. package/dist/server/links/index.d.ts.map +1 -1
  269. package/dist/server/links/index.js +2 -0
  270. package/dist/server/links/index.js.map +1 -1
  271. package/dist/server/metrics/index.d.ts +5 -7
  272. package/dist/server/metrics/index.d.ts.map +1 -1
  273. package/dist/server/metrics/index.js.map +1 -1
  274. package/dist/server/proxy/index.d.ts +3 -5
  275. package/dist/server/proxy/index.d.ts.map +1 -1
  276. package/dist/server/proxy/index.js.map +1 -1
  277. package/dist/server/rate-limit/index.d.ts +10 -13
  278. package/dist/server/rate-limit/index.d.ts.map +1 -1
  279. package/dist/server/rate-limit/index.js.map +1 -1
  280. package/dist/server/static/index.d.ts +3 -5
  281. package/dist/server/static/index.d.ts.map +1 -1
  282. package/dist/server/static/index.js.map +1 -1
  283. package/dist/server/swagger/index.d.ts +5 -8
  284. package/dist/server/swagger/index.d.ts.map +1 -1
  285. package/dist/server/swagger/index.js.map +1 -1
  286. package/dist/sms/index.d.ts +3 -5
  287. package/dist/sms/index.d.ts.map +1 -1
  288. package/dist/sms/index.js.map +1 -1
  289. package/dist/system/index.browser.js.map +1 -1
  290. package/dist/system/index.d.ts +2 -4
  291. package/dist/system/index.d.ts.map +1 -1
  292. package/dist/system/index.js.map +1 -1
  293. package/dist/system/index.workerd.js.map +1 -1
  294. package/dist/topic/core/index.d.ts +4 -6
  295. package/dist/topic/core/index.d.ts.map +1 -1
  296. package/dist/topic/core/index.js.map +1 -1
  297. package/dist/topic/redis/index.d.ts +5 -8
  298. package/dist/topic/redis/index.d.ts.map +1 -1
  299. package/dist/topic/redis/index.js.map +1 -1
  300. package/package.json +57 -12
  301. package/src/api/audits/__tests__/AuditService.spec.ts +18 -110
  302. package/src/api/audits/controllers/AdminAuditController.ts +14 -0
  303. package/src/api/audits/services/AuditService.ts +21 -88
  304. package/src/api/files/__tests__/FileService.spec.ts +207 -2
  305. package/src/api/files/controllers/FileController.ts +41 -1
  306. package/src/api/files/index.ts +3 -0
  307. package/src/api/files/providers/FileAccessProvider.ts +23 -1
  308. package/src/api/files/schemas/fileCreatorSummarySchema.ts +22 -0
  309. package/src/api/files/schemas/fileResourceSchema.ts +10 -1
  310. package/src/api/files/services/FileService.ts +175 -75
  311. package/src/api/jobs/__tests__/$job.spec.ts +24 -1
  312. package/src/api/jobs/index.ts +4 -3
  313. package/src/api/jobs/primitives/$job.ts +7 -3
  314. package/src/api/jobs/providers/DirectJobDispatcher.ts +17 -36
  315. package/src/api/jobs/providers/JobProvider.ts +53 -24
  316. package/src/api/jobs/schemas/jobConfigAtom.ts +1 -1
  317. package/src/api/jobs/schemas/jobExecutionResourceSchema.ts +4 -1
  318. package/src/api/keys/schemas/adminApiKeyResourceSchema.ts +3 -1
  319. package/src/api/parameters/__tests__/$parameter.spec.ts +19 -2
  320. package/src/api/parameters/audits/ParameterAudits.ts +17 -0
  321. package/src/api/parameters/controllers/AdminParameterController.ts +95 -19
  322. package/src/api/parameters/index.ts +3 -0
  323. package/src/api/parameters/schemas/activateParameterBodySchema.ts +3 -3
  324. package/src/api/parameters/schemas/createParameterVersionBodySchema.ts +3 -2
  325. package/src/api/parameters/schemas/parameterCreatorSummarySchema.ts +25 -0
  326. package/src/api/parameters/schemas/parameterResponseSchema.ts +5 -0
  327. package/src/api/parameters/schemas/rollbackParameterBodySchema.ts +4 -2
  328. package/src/api/parameters/services/ParameterProvider.ts +69 -6
  329. package/src/api/subscriptions/jobs/SubscriptionJobs.ts +1 -1
  330. package/src/api/users/__tests__/AdminSessionController.spec.ts +37 -0
  331. package/src/api/users/audits/SessionAudits.ts +33 -0
  332. package/src/api/users/audits/UserAudits.ts +19 -43
  333. package/src/api/users/controllers/AdminUserController.ts +66 -1
  334. package/src/api/users/entities/sessions.ts +6 -0
  335. package/src/api/users/entities/users.ts +2 -0
  336. package/src/api/users/index.ts +9 -1
  337. package/src/api/users/primitives/$realm.ts +3 -0
  338. package/src/api/users/schemas/sessionResourceSchema.ts +16 -0
  339. package/src/api/users/schemas/updateUserSchema.ts +1 -8
  340. package/src/api/users/schemas/userQuerySchema.ts +7 -0
  341. package/src/api/users/services/CredentialService.ts +52 -32
  342. package/src/api/users/services/IdentityService.ts +2 -1
  343. package/src/api/users/services/RegistrationService.ts +2 -1
  344. package/src/api/users/services/SessionCrudService.ts +19 -2
  345. package/src/api/users/services/SessionService.ts +39 -19
  346. package/src/api/users/services/UserService.ts +106 -8
  347. package/src/api/verifications/entities/verifications.ts +8 -0
  348. package/src/api/verifications/services/VerificationService.ts +14 -0
  349. package/src/background/__tests__/BackgroundTaskProvider.spec.ts +96 -0
  350. package/src/background/index.ts +37 -0
  351. package/src/background/index.workerd.ts +28 -0
  352. package/src/background/providers/BackgroundTaskProvider.ts +70 -0
  353. package/src/background/providers/WorkerdBackgroundTaskProvider.ts +43 -0
  354. package/src/bucket/__tests__/$bucket.spec.ts +18 -0
  355. package/src/bucket/__tests__/LocalFileStorageProvider.spec.ts +5 -0
  356. package/src/bucket/__tests__/MemoryFileStorageProvider.spec.ts +5 -0
  357. package/src/bucket/__tests__/NodeS3BucketProvider.spec.ts +23 -4
  358. package/src/bucket/__tests__/shared.ts +30 -0
  359. package/src/bucket/index.ts +5 -5
  360. package/src/bucket/index.workerd.ts +11 -4
  361. package/src/bucket/primitives/$bucket.ts +27 -0
  362. package/src/bucket/providers/FileStorageProvider.ts +13 -0
  363. package/src/bucket/providers/LocalFileStorageProvider.ts +17 -1
  364. package/src/bucket/providers/MemoryFileStorageProvider.ts +7 -0
  365. package/src/bucket/providers/{CloudflareR2Provider.ts → R2FileStorageProvider.ts} +10 -1
  366. package/src/bucket/providers/{NodeS3BucketProvider.ts → S3FileStorageProvider.ts} +27 -5
  367. package/src/cli/core/__tests__/BuildDockerTask.spec.ts +49 -1
  368. package/src/cli/core/__tests__/init.spec.ts +0 -219
  369. package/src/cli/core/atoms/buildOptions.ts +15 -0
  370. package/src/cli/core/commands/__tests__/BuildCommand.spec.ts +43 -0
  371. package/src/cli/core/commands/build.ts +122 -32
  372. package/src/cli/core/commands/init.ts +0 -12
  373. package/src/cli/core/commands/pack.ts +133 -0
  374. package/src/cli/core/index.ts +3 -0
  375. package/src/cli/core/providers/ViteDevServerProvider.ts +40 -16
  376. package/src/cli/core/services/PackageManagerUtils.ts +0 -16
  377. package/src/cli/core/services/ProjectScaffolder.ts +29 -291
  378. package/src/cli/core/tasks/BuildAssetsTask.ts +3 -0
  379. package/src/cli/core/tasks/BuildClientTask.ts +3 -0
  380. package/src/cli/core/tasks/BuildCloudflareTask.ts +453 -13
  381. package/src/cli/core/tasks/BuildCompressTask.ts +3 -0
  382. package/src/cli/core/tasks/BuildDockerTask.ts +43 -2
  383. package/src/cli/core/tasks/BuildPrerenderTask.ts +3 -0
  384. package/src/cli/core/tasks/BuildPwaTask.ts +3 -0
  385. package/src/cli/core/tasks/BuildServerTask.ts +3 -0
  386. package/src/cli/core/tasks/BuildTask.ts +42 -0
  387. package/src/cli/core/templates/apiIndexTs.ts +1 -22
  388. package/src/cli/core/templates/mainCss.ts +0 -1
  389. package/src/cli/core/templates/webAppRouterTs.ts +0 -99
  390. package/src/cli/core/templates/webIndexTs.ts +1 -22
  391. package/src/cli/platform/__tests__/SecretsCommand.spec.ts +5 -3
  392. package/src/cli/platform/commands/SecretsCommand.ts +8 -6
  393. package/src/cli/platform/commands/platform.ts +265 -122
  394. package/src/cli/platform/index.ts +25 -53
  395. package/src/cli/{platform → platform-lib}/__tests__/CloudflareAdapter.spec.ts +426 -169
  396. package/src/cli/{platform → platform-lib}/__tests__/NamingService.spec.ts +81 -36
  397. package/src/cli/{platform → platform-lib}/__tests__/PlatformInspector.spec.ts +4 -48
  398. package/src/cli/{platform → platform-lib}/__tests__/VercelAdapter.spec.ts +56 -85
  399. package/src/cli/{platform → platform-lib}/adapters/CloudflareAdapter.ts +412 -199
  400. package/src/cli/{platform → platform-lib}/adapters/PlatformAdapter.ts +68 -33
  401. package/src/cli/{platform → platform-lib}/adapters/VercelAdapter.ts +11 -25
  402. package/src/cli/{platform → platform-lib}/atoms/platformOptions.ts +34 -6
  403. package/src/cli/platform-lib/index.ts +67 -0
  404. package/src/cli/platform-lib/services/NamingService.ts +136 -0
  405. package/src/cli/platform-lib/services/PlatformInspector.ts +153 -0
  406. package/src/cli/{platform → platform-lib}/services/PlatformOrchestrator.ts +80 -127
  407. package/src/cli/{platform → platform-lib}/services/WranglerApi.ts +4 -2
  408. package/src/command/__tests__/Runner.spec.ts +20 -0
  409. package/src/command/helpers/EnvUtils.ts +19 -3
  410. package/src/command/helpers/Runner.ts +12 -2
  411. package/src/command/providers/CliProvider.ts +34 -1
  412. package/src/container/core/__tests__/$container.spec.ts +83 -0
  413. package/src/container/core/index.ts +50 -0
  414. package/src/container/core/index.workerd.ts +53 -0
  415. package/src/container/core/interfaces/ContainerOptions.ts +69 -0
  416. package/src/container/core/primitives/$container.ts +100 -0
  417. package/src/container/core/providers/CloudflareContainerProvider.ts +70 -0
  418. package/src/container/core/providers/ContainerProvider.ts +92 -0
  419. package/src/container/core/providers/MockContainerProvider.ts +62 -0
  420. package/src/container/core/providers/NodeContainerProvider.ts +53 -0
  421. package/src/core/Alepha.ts +49 -1
  422. package/src/core/__tests__/$env.spec.ts +42 -0
  423. package/src/core/__tests__/dump.spec.ts +47 -0
  424. package/src/email/cloudflare/__tests__/CloudflareEmailProvider.spec.ts +182 -0
  425. package/src/email/cloudflare/index.ts +35 -0
  426. package/src/email/cloudflare/providers/CloudflareEmailProvider.ts +205 -0
  427. package/src/logger/__tests__/Logger.spec.ts +55 -0
  428. package/src/logger/index.ts +13 -0
  429. package/src/logger/services/Logger.ts +31 -1
  430. package/src/orm/__tests__/orm-showcase-tests.ts +27 -0
  431. package/src/orm/__tests__/orm-showcase.spec.ts +12 -0
  432. package/src/orm/core/interfaces/PgQuery.ts +4 -1
  433. package/src/orm/core/services/Repository.ts +38 -13
  434. package/src/react/auth/hooks/useAuth.ts +10 -5
  435. package/src/react/core/__tests__/useQuery.browser.spec.tsx +25 -0
  436. package/src/react/core/hooks/useAction.ts +14 -3
  437. package/src/react/core/hooks/useQuery.ts +24 -4
  438. package/src/react/form/hooks/useFormQuerySync.ts +0 -0
  439. package/src/react/form/index.ts +1 -0
  440. package/src/react/form/services/FormModel.ts +18 -2
  441. package/src/react/i18n/components/Translate.tsx +47 -0
  442. package/src/react/i18n/index.ts +2 -0
  443. package/src/react/intro/components/GettingStartedAdminSlide.tsx +2 -2
  444. package/src/react/router/__tests__/$page.spec.tsx +3 -2
  445. package/src/react/router/__tests__/page-can.spec.ts +18 -13
  446. package/src/react/router/hooks/useQueryParams.ts +114 -14
  447. package/src/react/router/primitives/$page.ts +85 -4
  448. package/src/react/router/providers/ReactBrowserProvider.ts +12 -3
  449. package/src/react/router/providers/ReactBrowserRouterProvider.ts +3 -7
  450. package/src/react/router/providers/ReactServerProvider.ts +4 -13
  451. package/src/react/ui/services/SchemaControl.ts +3 -4
  452. package/src/server/core/providers/ServerMultipartProvider.ts +19 -0
  453. package/src/server/links/providers/LinkProvider.ts +10 -0
  454. package/src/cli/core/templates/componentsJsonTs.ts +0 -39
  455. package/src/cli/core/templates/saasAdminLayoutTsx.ts +0 -77
  456. package/src/cli/core/templates/saasAdminPagesTsx.ts +0 -26
  457. package/src/cli/core/templates/saasAuthLayoutTsx.ts +0 -22
  458. package/src/cli/core/templates/saasAuthPagesTsx.ts +0 -62
  459. package/src/cli/core/templates/saasRealmProviderTs.ts +0 -52
  460. package/src/cli/platform/hooks/PlatformHook.ts +0 -51
  461. package/src/cli/platform/services/NamingService.ts +0 -55
  462. package/src/cli/platform/services/PlatformInspector.ts +0 -140
  463. package/src/orm/REFACTORING.md +0 -330
  464. /package/dist/orm/core/{chunk-o8xxKEmq.js → chunk-B4FMCO8f.js} +0 -0
  465. /package/dist/react/testing/{chunk-6Ep1yQYe.js → chunk-BpyX8vjI.js} +0 -0
  466. /package/src/cli/{platform → platform-lib}/__tests__/GitHubSecretStore.spec.ts +0 -0
  467. /package/src/cli/{platform → platform-lib}/__tests__/PlatformCacheProvider.spec.ts +0 -0
  468. /package/src/cli/{platform → platform-lib}/__tests__/PlatformOrchestrator.spec.ts +0 -0
  469. /package/src/cli/{platform → platform-lib}/__tests__/SecretFilterService.spec.ts +0 -0
  470. /package/src/cli/{platform → platform-lib}/__tests__/detectResources.spec.ts +0 -0
  471. /package/src/cli/{platform → platform-lib}/providers/GitHubSecretStore.ts +0 -0
  472. /package/src/cli/{platform → platform-lib}/providers/MemorySecretStore.ts +0 -0
  473. /package/src/cli/{platform → platform-lib}/providers/PlatformCacheProvider.ts +0 -0
  474. /package/src/cli/{platform → platform-lib}/providers/SecretStoreProvider.ts +0 -0
  475. /package/src/cli/{platform → platform-lib}/schemas/cloudflare.ts +0 -0
  476. /package/src/cli/{platform → platform-lib}/schemas/platform.ts +0 -0
  477. /package/src/cli/{platform → platform-lib}/schemas/vercel.ts +0 -0
  478. /package/src/cli/{platform → platform-lib}/services/CloudflareApi.ts +0 -0
  479. /package/src/cli/{platform → platform-lib}/services/SecretFilterService.ts +0 -0
  480. /package/src/cli/{platform → platform-lib}/services/VercelApi.ts +0 -0
  481. /package/src/cli/{platform → platform-lib}/services/VercelCli.ts +0 -0
@@ -1,6 +1,4 @@
1
- import * as _$alepha from "alepha";
2
1
  import { Static } from "alepha";
3
- import * as _$typebox from "typebox";
4
2
 
5
3
  //#region ../../src/cli/devtools/atoms/devtoolsOptions.d.ts
6
4
  /**
@@ -8,13 +6,13 @@ import * as _$typebox from "typebox";
8
6
  *
9
7
  * Filled from the `devtools` section of `alepha.config.ts`.
10
8
  */
11
- declare const devtoolsOptions: _$alepha.Atom<_$typebox.TOptional<_$typebox.TObject<{
9
+ declare const devtoolsOptions: import("alepha").Atom<import("typebox").TOptional<import("typebox").TObject<{
12
10
  /**
13
11
  * Hide the floating devtools button in the browser.
14
12
  *
15
13
  * The devtools UI is still accessible at `/__devtools/`.
16
14
  */
17
- hideButton: _$typebox.TOptional<_$typebox.TBoolean>;
15
+ hideButton: import("typebox").TOptional<import("typebox").TBoolean>;
18
16
  }>>, "alepha.cli.devtools.options">;
19
17
  /**
20
18
  * Type for devtools options.
@@ -40,7 +38,7 @@ type DevtoolsOptions = Static<typeof devtoolsOptions.schema>;
40
38
  *
41
39
  * @module alepha.devtools.plugin
42
40
  */
43
- declare const AlephaCliDevtoolsPlugin: _$alepha.Service<_$alepha.Module>;
41
+ declare const AlephaCliDevtoolsPlugin: import("alepha").Service<import("alepha").Module>;
44
42
  declare const devtools: (options?: DevtoolsOptions) => () => void;
45
43
  //#endregion
46
44
  export { AlephaCliDevtoolsPlugin, DevtoolsOptions, devtools, devtoolsOptions };
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","names":[],"sources":["../../../src/cli/devtools/atoms/devtoolsOptions.ts","../../../src/cli/devtools/index.ts"],"mappings":";;;;;;;;;AAOA;cAAa,eAAA,EAAe,QAAA,CAAA,IAAA,CAAA,SAAA,CAAA,SAAA,WAAA,OAAA;;;;;;kCAa1B,SAAA,CAAA,QAAA;AAAA;;;;KAKU,eAAA,GAAkB,MAAA,QAAc,eAAA,CAAgB,MAAA;;;;;;;AAlB5D;;;;;;;;;;;;;;cCyGa,uBAAA,EAAuB,QAAA,CAAA,OAAA,CAgFlC,QAAA,CAhFkC,MAAA;AAAA,cAkFvB,QAAA,GAAY,OAAA,GAAS,eAAA"}
1
+ {"version":3,"file":"index.d.ts","names":[],"sources":["../../../src/cli/devtools/atoms/devtoolsOptions.ts","../../../src/cli/devtools/index.ts"],"mappings":";;;;;AAOA;;;cAAa,eAAA,mBAAe,IAAA,mBAAA,SAAA,mBAAA,OAAA;EAAA;;;;;;;;AAkB5B;;KAAY,eAAA,GAAkB,MAAM,QAAQ,eAAA,CAAgB,MAAA;;;;;AAlB5D;;;;;;;;;;;;AAkBA;;;;cCuFa,uBAAA,mBAAuB,OAAA,kBAAA,MAAA;AAAA,cAkFvB,QAAA,GAAY,OAA6B,GAApB,eAAoB"}
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":[],"sources":["../../../src/cli/devtools/atoms/devtoolsOptions.ts","../../../src/cli/devtools/index.ts"],"sourcesContent":["import { $atom, type Static, t } from \"alepha\";\n\n/**\n * Devtools configuration atom.\n *\n * Filled from the `devtools` section of `alepha.config.ts`.\n */\nexport const devtoolsOptions = $atom({\n name: \"alepha.cli.devtools.options\",\n description: \"Devtools plugin configuration\",\n schema: t.optional(\n t.object({\n /**\n * Hide the floating devtools button in the browser.\n *\n * The devtools UI is still accessible at `/__devtools/`.\n */\n hideButton: t.optional(t.boolean({ default: false })),\n }),\n ),\n});\n\n/**\n * Type for devtools options.\n */\nexport type DevtoolsOptions = Static<typeof devtoolsOptions.schema>;\n","import { readFile } from \"node:fs/promises\";\nimport { createRequire } from \"node:module\";\nimport { dirname, join } from \"node:path\";\nimport { $context, $module, AlephaError } from \"alepha\";\nimport { ViteDevServerProvider } from \"alepha/cli\";\nimport {\n type DevtoolsOptions,\n devtoolsOptions,\n} from \"./atoms/devtoolsOptions.ts\";\n\n// ---------------------------------------------------------------------------------------------------------------------\n\nconst DEVTOOLS_OVERLAY_SCRIPT = `\n(function () {\n if (window.__alepha_devtools_injected) return;\n window.__alepha_devtools_injected = true;\n\n const STORAGE_KEY = \"alepha-devtools-open\";\n\n // Button\n const btn = document.createElement(\"button\");\n btn.id = \"alepha-devtools-btn\";\n btn.innerHTML = \\`<svg width=\"18\" height=\"18\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M12 15a3 3 0 1 0 0-6 3 3 0 0 0 0 6Z\"/><path d=\"M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 1 1-2.83 2.83l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 1 1-4 0v-.09a1.65 1.65 0 0 0-1-1.51 1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 1 1-2.83-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 1 1 0-4h.09a1.65 1.65 0 0 0 1.51-1 1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 1 1 2.83-2.83l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 1 1 4 0v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 1 1 2.83 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 1 1 0 4h-.09a1.65 1.65 0 0 0-1.51 1Z\"/></svg>\\`;\n Object.assign(btn.style, {\n position: \"fixed\", bottom: \"16px\", left: \"16px\", zIndex: \"99998\",\n width: \"36px\", height: \"36px\", borderRadius: \"50%\",\n background: \"rgba(255,255,255,0.85)\", color: \"#3f3f46\",\n border: \"none\", backdropFilter: \"blur(8px)\", WebkitBackdropFilter: \"blur(8px)\",\n cursor: \"pointer\", display: \"flex\", alignItems: \"center\", justifyContent: \"center\",\n boxShadow: \"0 1px 2px rgba(0,0,0,0.06), 0 4px 12px rgba(0,0,0,0.08)\",\n transition: \"transform 0.2s ease, box-shadow 0.2s ease\",\n padding: \"0\", fontSize: \"0\",\n });\n btn.addEventListener(\"mouseenter\", () => {\n btn.style.transform = \"translateY(-1px) rotate(45deg)\";\n btn.style.boxShadow = \"0 2px 4px rgba(0,0,0,0.08), 0 8px 20px rgba(0,0,0,0.12)\";\n });\n btn.addEventListener(\"mouseleave\", () => {\n btn.style.transform = \"translateY(0) rotate(0deg)\";\n btn.style.boxShadow = \"0 1px 2px rgba(0,0,0,0.06), 0 4px 12px rgba(0,0,0,0.08)\";\n });\n\n // Overlay\n const overlay = document.createElement(\"div\");\n overlay.id = \"alepha-devtools-overlay\";\n Object.assign(overlay.style, {\n position: \"fixed\", inset: \"0\", zIndex: \"99999\",\n background: \"rgba(0,0,0,0.6)\", backdropFilter: \"blur(2px)\",\n display: \"none\", alignItems: \"center\", justifyContent: \"center\",\n });\n\n // Panel\n const panel = document.createElement(\"div\");\n Object.assign(panel.style, {\n width: \"90vw\", height: \"85vh\", maxWidth: \"1400px\",\n borderRadius: \"12px\", overflow: \"hidden\",\n boxShadow: \"0 8px 32px rgba(0,0,0,0.5)\",\n border: \"1px solid #2a2a4a\",\n });\n\n const iframe = document.createElement(\"iframe\");\n iframe.style.cssText = \"width:100%;height:100%;border:none;\";\n\n panel.appendChild(iframe);\n overlay.appendChild(panel);\n document.body.appendChild(btn);\n document.body.appendChild(overlay);\n\n function open() {\n if (!iframe.src) iframe.src = \"/__devtools/\";\n overlay.style.display = \"flex\";\n btn.style.display = \"none\";\n sessionStorage.setItem(STORAGE_KEY, \"1\");\n }\n\n function close() {\n overlay.style.display = \"none\";\n btn.style.display = \"flex\";\n sessionStorage.removeItem(STORAGE_KEY);\n }\n\n btn.addEventListener(\"click\", open);\n overlay.addEventListener(\"click\", (e) => {\n if (e.target === overlay) close();\n });\n document.addEventListener(\"keydown\", (e) => {\n if (e.key === \"Escape\" && overlay.style.display === \"flex\") close();\n });\n\n // Restore state after HMR\n if (sessionStorage.getItem(STORAGE_KEY)) open();\n})();\n`;\n\n/**\n * CLI plugin that integrates @alepha/devtools into the Vite dev server.\n *\n * This module is intentionally lightweight — it does NOT statically import\n * `@alepha/devtools` (which pulls in `alepha/react` and `.tsx` files).\n * Instead, it lazy-loads devtools via Vite's SSR module loader at runtime.\n *\n * Usage in `alepha.config.ts`:\n * ```ts\n * import { devtools } from \"alepha/cli/devtools\";\n *\n * export default defineConfig({\n * plugins: [devtools()],\n * });\n * ```\n *\n * @module alepha.devtools.plugin\n */\nexport const AlephaCliDevtoolsPlugin = $module({\n name: \"alepha.cli.plugins.devtools\",\n atoms: [devtoolsOptions],\n register: (alepha) => {\n const vite = alepha.inject(ViteDevServerProvider) as ViteDevServerProvider;\n\n const require = createRequire(import.meta.url);\n const pkgPath = require.resolve(\"@alepha/devtools/package.json\");\n const assetsPath = join(dirname(pkgPath), \"assets/ui\");\n\n process.env.VITE_ALEPHA_DEVTOOLS = \"true\";\n\n vite.addVitePlugin({\n name: \"alepha-devtools\",\n configureServer: (server) => {\n // Reload endpoint\n server.middlewares.use((req, res, next) => {\n if (req.url !== \"/__devtools/api/reload\" || req.method !== \"POST\") {\n return next();\n }\n\n vite.reload();\n res.writeHead(200, { \"content-type\": \"application/json\" });\n res.end(JSON.stringify({ ok: true }));\n });\n\n // Serve devtools HTML\n server.middlewares.use(async (req, res, next) => {\n const url = req.url || \"/\";\n\n if (\n !url.startsWith(\"/__devtools\") ||\n !req.headers.accept?.includes(\"text/html\")\n ) {\n return next();\n }\n\n const indexPath = join(assetsPath, \"index.html\");\n\n try {\n let html = await readFile(indexPath, \"utf-8\");\n html = html.replace(\n \"<head>\",\n `<head><script type=\"module\" src=\"/@vite/client\"></script>`,\n );\n\n res.writeHead(200, { \"content-type\": \"text/html\" });\n res.end(html);\n } catch {\n next();\n }\n });\n },\n transformIndexHtml: () => {\n const options = alepha.store.get(devtoolsOptions);\n if (options?.hideButton) return [];\n\n return [\n {\n tag: \"script\",\n attrs: { type: \"module\" },\n children: DEVTOOLS_OVERLAY_SCRIPT,\n injectTo: \"head\",\n },\n ];\n },\n });\n\n vite.onAlephaLoaded(async (appAlepha, server) => {\n try {\n const mod = await server.ssrLoadModule(\"@alepha/devtools\");\n appAlepha.with(mod.AlephaDevtools);\n } catch (err) {\n throw new AlephaError(\n \"Failed to load @alepha/devtools. Make sure the package is installed\",\n { cause: err },\n );\n }\n });\n },\n});\n\nexport const devtools = (options: DevtoolsOptions = {}) => {\n return () => {\n const { alepha } = $context();\n alepha.with(AlephaCliDevtoolsPlugin).set(devtoolsOptions, options);\n };\n};\n\n// ---------------------------------------------------------------------------------------------------------------------\n\nexport * from \"./atoms/devtoolsOptions.ts\";\n"],"mappings":";;;;;;;;;;;AAOA,MAAa,kBAAkB,MAAM;CACnC,MAAM;CACN,aAAa;CACb,QAAQ,EAAE,SACR,EAAE,OAAO;;;;;;AAMP,YAAY,EAAE,SAAS,EAAE,QAAQ,EAAE,SAAS,OAAO,CAAC,CAAC,EACtD,CAAC,CACH;CACF,CAAC;;;ACRF,MAAM,0BAA0B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAoGhC,MAAa,0BAA0B,QAAQ;CAC7C,MAAM;CACN,OAAO,CAAC,gBAAgB;CACxB,WAAW,WAAW;EACpB,MAAM,OAAO,OAAO,OAAO,sBAAsB;EAIjD,MAAM,aAAa,KAAK,QAFR,cAAc,OAAO,KAAK,IACnB,CAAC,QAAQ,gCACO,CAAC,EAAE,YAAY;EAEtD,QAAQ,IAAI,uBAAuB;EAEnC,KAAK,cAAc;GACjB,MAAM;GACN,kBAAkB,WAAW;IAE3B,OAAO,YAAY,KAAK,KAAK,KAAK,SAAS;KACzC,IAAI,IAAI,QAAQ,4BAA4B,IAAI,WAAW,QACzD,OAAO,MAAM;KAGf,KAAK,QAAQ;KACb,IAAI,UAAU,KAAK,EAAE,gBAAgB,oBAAoB,CAAC;KAC1D,IAAI,IAAI,KAAK,UAAU,EAAE,IAAI,MAAM,CAAC,CAAC;MACrC;IAGF,OAAO,YAAY,IAAI,OAAO,KAAK,KAAK,SAAS;KAG/C,IACE,EAHU,IAAI,OAAO,KAGhB,WAAW,cAAc,IAC9B,CAAC,IAAI,QAAQ,QAAQ,SAAS,YAAY,EAE1C,OAAO,MAAM;KAGf,MAAM,YAAY,KAAK,YAAY,aAAa;KAEhD,IAAI;MACF,IAAI,OAAO,MAAM,SAAS,WAAW,QAAQ;MAC7C,OAAO,KAAK,QACV,UACA,6DACD;MAED,IAAI,UAAU,KAAK,EAAE,gBAAgB,aAAa,CAAC;MACnD,IAAI,IAAI,KAAK;aACP;MACN,MAAM;;MAER;;GAEJ,0BAA0B;IAExB,IADgB,OAAO,MAAM,IAAI,gBACtB,EAAE,YAAY,OAAO,EAAE;IAElC,OAAO,CACL;KACE,KAAK;KACL,OAAO,EAAE,MAAM,UAAU;KACzB,UAAU;KACV,UAAU;KACX,CACF;;GAEJ,CAAC;EAEF,KAAK,eAAe,OAAO,WAAW,WAAW;GAC/C,IAAI;IACF,MAAM,MAAM,MAAM,OAAO,cAAc,mBAAmB;IAC1D,UAAU,KAAK,IAAI,eAAe;YAC3B,KAAK;IACZ,MAAM,IAAI,YACR,uEACA,EAAE,OAAO,KAAK,CACf;;IAEH;;CAEL,CAAC;AAEF,MAAa,YAAY,UAA2B,EAAE,KAAK;CACzD,aAAa;EACX,MAAM,EAAE,WAAW,UAAU;EAC7B,OAAO,KAAK,wBAAwB,CAAC,IAAI,iBAAiB,QAAQ"}
1
+ {"version":3,"file":"index.js","names":[],"sources":["../../../src/cli/devtools/atoms/devtoolsOptions.ts","../../../src/cli/devtools/index.ts"],"sourcesContent":["import { $atom, type Static, t } from \"alepha\";\n\n/**\n * Devtools configuration atom.\n *\n * Filled from the `devtools` section of `alepha.config.ts`.\n */\nexport const devtoolsOptions = $atom({\n name: \"alepha.cli.devtools.options\",\n description: \"Devtools plugin configuration\",\n schema: t.optional(\n t.object({\n /**\n * Hide the floating devtools button in the browser.\n *\n * The devtools UI is still accessible at `/__devtools/`.\n */\n hideButton: t.optional(t.boolean({ default: false })),\n }),\n ),\n});\n\n/**\n * Type for devtools options.\n */\nexport type DevtoolsOptions = Static<typeof devtoolsOptions.schema>;\n","import { readFile } from \"node:fs/promises\";\nimport { createRequire } from \"node:module\";\nimport { dirname, join } from \"node:path\";\nimport { $context, $module, AlephaError } from \"alepha\";\nimport { ViteDevServerProvider } from \"alepha/cli\";\nimport {\n type DevtoolsOptions,\n devtoolsOptions,\n} from \"./atoms/devtoolsOptions.ts\";\n\n// ---------------------------------------------------------------------------------------------------------------------\n\nconst DEVTOOLS_OVERLAY_SCRIPT = `\n(function () {\n if (window.__alepha_devtools_injected) return;\n window.__alepha_devtools_injected = true;\n\n const STORAGE_KEY = \"alepha-devtools-open\";\n\n // Button\n const btn = document.createElement(\"button\");\n btn.id = \"alepha-devtools-btn\";\n btn.innerHTML = \\`<svg width=\"18\" height=\"18\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M12 15a3 3 0 1 0 0-6 3 3 0 0 0 0 6Z\"/><path d=\"M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 1 1-2.83 2.83l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 1 1-4 0v-.09a1.65 1.65 0 0 0-1-1.51 1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 1 1-2.83-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 1 1 0-4h.09a1.65 1.65 0 0 0 1.51-1 1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 1 1 2.83-2.83l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 1 1 4 0v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 1 1 2.83 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 1 1 0 4h-.09a1.65 1.65 0 0 0-1.51 1Z\"/></svg>\\`;\n Object.assign(btn.style, {\n position: \"fixed\", bottom: \"16px\", left: \"16px\", zIndex: \"99998\",\n width: \"36px\", height: \"36px\", borderRadius: \"50%\",\n background: \"rgba(255,255,255,0.85)\", color: \"#3f3f46\",\n border: \"none\", backdropFilter: \"blur(8px)\", WebkitBackdropFilter: \"blur(8px)\",\n cursor: \"pointer\", display: \"flex\", alignItems: \"center\", justifyContent: \"center\",\n boxShadow: \"0 1px 2px rgba(0,0,0,0.06), 0 4px 12px rgba(0,0,0,0.08)\",\n transition: \"transform 0.2s ease, box-shadow 0.2s ease\",\n padding: \"0\", fontSize: \"0\",\n });\n btn.addEventListener(\"mouseenter\", () => {\n btn.style.transform = \"translateY(-1px) rotate(45deg)\";\n btn.style.boxShadow = \"0 2px 4px rgba(0,0,0,0.08), 0 8px 20px rgba(0,0,0,0.12)\";\n });\n btn.addEventListener(\"mouseleave\", () => {\n btn.style.transform = \"translateY(0) rotate(0deg)\";\n btn.style.boxShadow = \"0 1px 2px rgba(0,0,0,0.06), 0 4px 12px rgba(0,0,0,0.08)\";\n });\n\n // Overlay\n const overlay = document.createElement(\"div\");\n overlay.id = \"alepha-devtools-overlay\";\n Object.assign(overlay.style, {\n position: \"fixed\", inset: \"0\", zIndex: \"99999\",\n background: \"rgba(0,0,0,0.6)\", backdropFilter: \"blur(2px)\",\n display: \"none\", alignItems: \"center\", justifyContent: \"center\",\n });\n\n // Panel\n const panel = document.createElement(\"div\");\n Object.assign(panel.style, {\n width: \"90vw\", height: \"85vh\", maxWidth: \"1400px\",\n borderRadius: \"12px\", overflow: \"hidden\",\n boxShadow: \"0 8px 32px rgba(0,0,0,0.5)\",\n border: \"1px solid #2a2a4a\",\n });\n\n const iframe = document.createElement(\"iframe\");\n iframe.style.cssText = \"width:100%;height:100%;border:none;\";\n\n panel.appendChild(iframe);\n overlay.appendChild(panel);\n document.body.appendChild(btn);\n document.body.appendChild(overlay);\n\n function open() {\n if (!iframe.src) iframe.src = \"/__devtools/\";\n overlay.style.display = \"flex\";\n btn.style.display = \"none\";\n sessionStorage.setItem(STORAGE_KEY, \"1\");\n }\n\n function close() {\n overlay.style.display = \"none\";\n btn.style.display = \"flex\";\n sessionStorage.removeItem(STORAGE_KEY);\n }\n\n btn.addEventListener(\"click\", open);\n overlay.addEventListener(\"click\", (e) => {\n if (e.target === overlay) close();\n });\n document.addEventListener(\"keydown\", (e) => {\n if (e.key === \"Escape\" && overlay.style.display === \"flex\") close();\n });\n\n // Restore state after HMR\n if (sessionStorage.getItem(STORAGE_KEY)) open();\n})();\n`;\n\n/**\n * CLI plugin that integrates @alepha/devtools into the Vite dev server.\n *\n * This module is intentionally lightweight — it does NOT statically import\n * `@alepha/devtools` (which pulls in `alepha/react` and `.tsx` files).\n * Instead, it lazy-loads devtools via Vite's SSR module loader at runtime.\n *\n * Usage in `alepha.config.ts`:\n * ```ts\n * import { devtools } from \"alepha/cli/devtools\";\n *\n * export default defineConfig({\n * plugins: [devtools()],\n * });\n * ```\n *\n * @module alepha.devtools.plugin\n */\nexport const AlephaCliDevtoolsPlugin = $module({\n name: \"alepha.cli.plugins.devtools\",\n atoms: [devtoolsOptions],\n register: (alepha) => {\n const vite = alepha.inject(ViteDevServerProvider) as ViteDevServerProvider;\n\n const require = createRequire(import.meta.url);\n const pkgPath = require.resolve(\"@alepha/devtools/package.json\");\n const assetsPath = join(dirname(pkgPath), \"assets/ui\");\n\n process.env.VITE_ALEPHA_DEVTOOLS = \"true\";\n\n vite.addVitePlugin({\n name: \"alepha-devtools\",\n configureServer: (server) => {\n // Reload endpoint\n server.middlewares.use((req, res, next) => {\n if (req.url !== \"/__devtools/api/reload\" || req.method !== \"POST\") {\n return next();\n }\n\n vite.reload();\n res.writeHead(200, { \"content-type\": \"application/json\" });\n res.end(JSON.stringify({ ok: true }));\n });\n\n // Serve devtools HTML\n server.middlewares.use(async (req, res, next) => {\n const url = req.url || \"/\";\n\n if (\n !url.startsWith(\"/__devtools\") ||\n !req.headers.accept?.includes(\"text/html\")\n ) {\n return next();\n }\n\n const indexPath = join(assetsPath, \"index.html\");\n\n try {\n let html = await readFile(indexPath, \"utf-8\");\n html = html.replace(\n \"<head>\",\n `<head><script type=\"module\" src=\"/@vite/client\"></script>`,\n );\n\n res.writeHead(200, { \"content-type\": \"text/html\" });\n res.end(html);\n } catch {\n next();\n }\n });\n },\n transformIndexHtml: () => {\n const options = alepha.store.get(devtoolsOptions);\n if (options?.hideButton) return [];\n\n return [\n {\n tag: \"script\",\n attrs: { type: \"module\" },\n children: DEVTOOLS_OVERLAY_SCRIPT,\n injectTo: \"head\",\n },\n ];\n },\n });\n\n vite.onAlephaLoaded(async (appAlepha, server) => {\n try {\n const mod = await server.ssrLoadModule(\"@alepha/devtools\");\n appAlepha.with(mod.AlephaDevtools);\n } catch (err) {\n throw new AlephaError(\n \"Failed to load @alepha/devtools. Make sure the package is installed\",\n { cause: err },\n );\n }\n });\n },\n});\n\nexport const devtools = (options: DevtoolsOptions = {}) => {\n return () => {\n const { alepha } = $context();\n alepha.with(AlephaCliDevtoolsPlugin).set(devtoolsOptions, options);\n };\n};\n\n// ---------------------------------------------------------------------------------------------------------------------\n\nexport * from \"./atoms/devtoolsOptions.ts\";\n"],"mappings":";;;;;;;;;;;AAOA,MAAa,kBAAkB,MAAM;CACnC,MAAM;CACN,aAAa;CACb,QAAQ,EAAE,SACR,EAAE,OAAO;;;;;;AAMP,YAAY,EAAE,SAAS,EAAE,QAAQ,EAAE,SAAS,MAAM,CAAC,CAAC,EACtD,CAAC,CACH;AACF,CAAC;;;ACRD,MAAM,0BAA0B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAoGhC,MAAa,0BAA0B,QAAQ;CAC7C,MAAM;CACN,OAAO,CAAC,eAAe;CACvB,WAAW,WAAW;EACpB,MAAM,OAAO,OAAO,OAAO,qBAAqB;EAIhD,MAAM,aAAa,KAAK,QAFR,cAAc,OAAO,KAAK,GACpB,EAAE,QAAQ,+BACM,CAAC,GAAG,WAAW;EAErD,QAAQ,IAAI,uBAAuB;EAEnC,KAAK,cAAc;GACjB,MAAM;GACN,kBAAkB,WAAW;IAE3B,OAAO,YAAY,KAAK,KAAK,KAAK,SAAS;KACzC,IAAI,IAAI,QAAQ,4BAA4B,IAAI,WAAW,QACzD,OAAO,KAAK;KAGd,KAAK,OAAO;KACZ,IAAI,UAAU,KAAK,EAAE,gBAAgB,mBAAmB,CAAC;KACzD,IAAI,IAAI,KAAK,UAAU,EAAE,IAAI,KAAK,CAAC,CAAC;IACtC,CAAC;IAGD,OAAO,YAAY,IAAI,OAAO,KAAK,KAAK,SAAS;KAG/C,IACE,EAHU,IAAI,OAAO,KAGhB,WAAW,aAAa,KAC7B,CAAC,IAAI,QAAQ,QAAQ,SAAS,WAAW,GAEzC,OAAO,KAAK;KAGd,MAAM,YAAY,KAAK,YAAY,YAAY;KAE/C,IAAI;MACF,IAAI,OAAO,MAAM,SAAS,WAAW,OAAO;MAC5C,OAAO,KAAK,QACV,UACA,4DACF;MAEA,IAAI,UAAU,KAAK,EAAE,gBAAgB,YAAY,CAAC;MAClD,IAAI,IAAI,IAAI;KACd,QAAQ;MACN,KAAK;KACP;IACF,CAAC;GACH;GACA,0BAA0B;IAExB,IADgB,OAAO,MAAM,IAAI,eACvB,GAAG,YAAY,OAAO,CAAC;IAEjC,OAAO,CACL;KACE,KAAK;KACL,OAAO,EAAE,MAAM,SAAS;KACxB,UAAU;KACV,UAAU;IACZ,CACF;GACF;EACF,CAAC;EAED,KAAK,eAAe,OAAO,WAAW,WAAW;GAC/C,IAAI;IACF,MAAM,MAAM,MAAM,OAAO,cAAc,kBAAkB;IACzD,UAAU,KAAK,IAAI,cAAc;GACnC,SAAS,KAAK;IACZ,MAAM,IAAI,YACR,uEACA,EAAE,OAAO,IAAI,CACf;GACF;EACF,CAAC;CACH;AACF,CAAC;AAED,MAAa,YAAY,UAA2B,CAAC,MAAM;CACzD,aAAa;EACX,MAAM,EAAE,WAAW,SAAS;EAC5B,OAAO,KAAK,uBAAuB,EAAE,IAAI,iBAAiB,OAAO;CACnE;AACF"}
@@ -1,10 +1,6 @@
1
- import * as _$alepha from "alepha";
2
1
  import { Static } from "alepha";
3
- import * as _$alepha_command0 from "alepha/command";
4
- import * as _$alepha_logger0 from "alepha/logger";
5
2
  import { ConsoleColorProvider } from "alepha/logger";
6
3
  import { FileSystemProvider } from "alepha/system";
7
- import * as _$typebox from "typebox";
8
4
 
9
5
  //#region ../../src/cli/i18n/atoms/i18nOptions.d.ts
10
6
  /**
@@ -13,14 +9,14 @@ import * as _$typebox from "typebox";
13
9
  * Filled from the `i18n` plugin in `alepha.config.ts`.
14
10
  * Read by `I18nCommand` to drive `alepha i18n check`.
15
11
  */
16
- declare const i18nOptions: _$alepha.Atom<_$typebox.TOptional<_$typebox.TObject<{
12
+ declare const i18nOptions: import("alepha").Atom<import("typebox").TOptional<import("typebox").TObject<{
17
13
  /**
18
14
  * Directories (relative to the project root) to scan both for
19
15
  * `$dictionary(...)` declarations and for translation key usage.
20
16
  *
21
17
  * @default ["src"]
22
18
  */
23
- scan: _$typebox.TOptional<_$typebox.TArray<_$typebox.TString>>;
19
+ scan: import("typebox").TOptional<import("typebox").TArray<import("typebox").TString>>;
24
20
  /**
25
21
  * Key prefixes that are constructed at runtime (e.g. via template
26
22
  * literals like `` tr(`archive.type.${kind}`) ``). Every key
@@ -32,7 +28,7 @@ declare const i18nOptions: _$alepha.Atom<_$typebox.TOptional<_$typebox.TObject<{
32
28
  *
33
29
  * @default []
34
30
  */
35
- dynamicPrefixes: _$typebox.TOptional<_$typebox.TArray<_$typebox.TString>>;
31
+ dynamicPrefixes: import("typebox").TOptional<import("typebox").TArray<import("typebox").TString>>;
36
32
  /**
37
33
  * Additional path substrings (matched against the full file
38
34
  * path) that should be excluded from the scan, on top of the
@@ -41,7 +37,7 @@ declare const i18nOptions: _$alepha.Atom<_$typebox.TOptional<_$typebox.TObject<{
41
37
  *
42
38
  * @default []
43
39
  */
44
- exclude: _$typebox.TOptional<_$typebox.TArray<_$typebox.TString>>;
40
+ exclude: import("typebox").TOptional<import("typebox").TArray<import("typebox").TString>>;
45
41
  }>>, "alepha.cli.i18n.options">;
46
42
  /**
47
43
  * Type for i18n options.
@@ -84,7 +80,7 @@ declare class I18nCheckService {
84
80
  //#endregion
85
81
  //#region ../../src/cli/i18n/commands/I18nCommand.d.ts
86
82
  declare class I18nCommand {
87
- protected readonly log: _$alepha_logger0.Logger;
83
+ protected readonly log: import("alepha/logger").Logger;
88
84
  protected readonly options: Readonly<{
89
85
  scan?: string[] | undefined;
90
86
  dynamicPrefixes?: string[] | undefined;
@@ -97,8 +93,8 @@ declare class I18nCommand {
97
93
  dynamicPrefixes: string[];
98
94
  exclude: string[];
99
95
  };
100
- protected readonly check: _$alepha_command0.CommandPrimitive<_$typebox.TObject<_$typebox.TProperties>, _$typebox.TSchema, _$typebox.TObject<_$typebox.TProperties>>;
101
- readonly i18n: _$alepha_command0.CommandPrimitive<_$typebox.TObject<_$typebox.TProperties>, _$typebox.TSchema, _$typebox.TObject<_$typebox.TProperties>>;
96
+ protected readonly check: import("alepha/command").CommandPrimitive<import("typebox").TObject<import("typebox").TProperties>, import("typebox").TSchema, import("typebox").TObject<import("typebox").TProperties>>;
97
+ readonly i18n: import("alepha/command").CommandPrimitive<import("typebox").TObject<import("typebox").TProperties>, import("typebox").TSchema, import("typebox").TObject<import("typebox").TProperties>>;
102
98
  }
103
99
  //#endregion
104
100
  //#region ../../src/cli/i18n/index.d.ts
@@ -129,7 +125,7 @@ declare class I18nCommand {
129
125
  * });
130
126
  * ```
131
127
  */
132
- declare const AlephaCliI18nPlugin: _$alepha.Service<_$alepha.Module>;
128
+ declare const AlephaCliI18nPlugin: import("alepha").Service<import("alepha").Module>;
133
129
  declare const i18n: (options?: I18nOptions) => () => void;
134
130
  //#endregion
135
131
  export { AlephaCliI18nPlugin, I18nCheckOptions, I18nCheckResult, I18nCheckService, I18nCommand, I18nOptions, i18n, i18nOptions };
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","names":[],"sources":["../../../src/cli/i18n/atoms/i18nOptions.ts","../../../src/cli/i18n/services/I18nCheckService.ts","../../../src/cli/i18n/commands/I18nCommand.ts","../../../src/cli/i18n/index.ts"],"mappings":";;;;;;;;;;;;;;;cAQa,WAAA,EAAW,QAAA,CAAA,IAAA,CAAA,SAAA,CAAA,SAAA,WAAA,OAAA;;AAAxB;;;;;6CAqCE,SAAA,CAAA,OAAA;;;;;;;;;;;;;EArCsB;;;;;;;;;;;;;KA0CZ,WAAA,GAAc,MAAA,QAAc,WAAA,CAAY,MAAA;;;UCbnC,gBAAA;EACf,IAAA;EACA,IAAA;EACA,eAAA;EACA,OAAA;AAAA;AAAA,UAGe,eAAA;;EAEf,SAAA;EDDA;ECGA,UAAA;EDHA;ECKA,YAAA;;EAEA,eAAA;;EAEA,MAAA;AAAA;AAAA,cAGW,gBAAA;EAAA,mBACQ,EAAA,EAAE,kBAAA;;;;;;;;;;;EAYf,KAAA,CAAM,OAAA,EAAS,gBAAA,GAAmB,OAAA,CAAQ,eAAA;AAAA;;;cChErC,WAAA;EAAA,mBACQ,GAAA,EADG,gBAAA,CACA,MAAA;EAAA,mBACH,OAAA,EAAO,QAAA;;;;;qBACP,YAAA,EAAY,gBAAA;EAAA,mBACZ,KAAA,EAAK,oBAAA;EAAA,UAEd,cAAA,CAAA;;;;;qBAQS,KAAA,EAAK,iBAAA,CAAA,gBAAA,CAAA,SAAA,CAAA,OAAA,CAVA,SAAA,CAUA,WAAA,GAAA,SAAA,CAAA,OAAA,EAAA,SAAA,CAAA,OAAA,CAAA,SAAA,CAAA,WAAA;EAAA,SAoDR,IAAA,EAAI,iBAAA,CAAA,gBAAA,CAAA,SAAA,CAAA,OAAA,CApDI,SAAA,CAoDJ,WAAA,GAAA,SAAA,CAAA,OAAA,EAAA,SAAA,CAAA,OAAA,CAAA,SAAA,CAAA,WAAA;AAAA;;;;;;;;;;;AFhEtB;;;;;;;;;;;;;;;;;;;cG0Ba,mBAAA,EAAmB,QAAA,CAAA,OAAA,CAI9B,QAAA,CAJ8B,MAAA;AAAA,cAMnB,IAAA,GAAQ,OAAA,GAAS,WAAA"}
1
+ {"version":3,"file":"index.d.ts","names":[],"sources":["../../../src/cli/i18n/atoms/i18nOptions.ts","../../../src/cli/i18n/services/I18nCheckService.ts","../../../src/cli/i18n/commands/I18nCommand.ts","../../../src/cli/i18n/index.ts"],"mappings":";;;;;;;;;AAQA;;cAAa,WAAA,mBAAW,IAAA,mBAAA,SAAA,mBAAA,OAAA;EAqCtB;;;;;;;;;;;;;;;;;;;EAKqB;;;;;;;AAAmC;;;;ACb1D;;KDaY,WAAA,GAAc,MAAM,QAAQ,WAAA,CAAY,MAAA;;;UCbnC,gBAAA;EACf,IAAA;EACA,IAAA;EACA,eAAA;EACA,OAAA;AAAA;AAAA,UAGe,eAAA;EDCf;ECCA,SAAA;EDtCsB;ECwCtB,UAAA;;EAEA,YAAA;;EAEA,eAAA;;EAEA,MAAA;AAAA;AAAA,cAGW,gBAAA;EAAA,mBACQ,EAAA,EAAE,kBAAA;;;;;;ADRvB;;;;;ECoBQ,KAAA,CAAM,OAAA,EAAS,gBAAA,GAAmB,OAAA,CAAQ,eAAA;AAAA;;;cChErC,WAAA;EAAA,mBACQ,GAAA,0BAAG,MAAA;EAAA,mBACH,OAAA,EAAO,QAAA;;;;;qBACP,YAAA,EAAY,gBAAA;EAAA,mBACZ,KAAA,EAAK,oBAAA;EAAA,UAEd,cAAA;;;;;qBAQS,KAAA,2BAAK,gBAAA,mBAAA,OAAA,mBAAA,WAAA,qBAAA,OAAA,oBAAA,OAAA,mBAAA,WAAA;EAAA,SAoDR,IAAA,2BAAI,gBAAA,mBAAA,OAAA,mBAAA,WAAA,qBAAA,OAAA,oBAAA,OAAA,mBAAA,WAAA;AAAA;;;;;;;AFhEtB;;;;;;;;;;;;;;;;;;;;;AA0CA;;cGhBa,mBAAA,mBAAmB,OAAA,kBAAA,MAAA;AAAA,cAMnB,IAAA,GAAQ,OAAyB,GAAhB,WAAgB"}
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":[],"sources":["../../../src/cli/i18n/atoms/i18nOptions.ts","../../../src/cli/i18n/services/I18nCheckService.ts","../../../src/cli/i18n/commands/I18nCommand.ts","../../../src/cli/i18n/index.ts"],"sourcesContent":["import { $atom, type Static, t } from \"alepha\";\n\n/**\n * i18n CLI configuration atom.\n *\n * Filled from the `i18n` plugin in `alepha.config.ts`.\n * Read by `I18nCommand` to drive `alepha i18n check`.\n */\nexport const i18nOptions = $atom({\n name: \"alepha.cli.i18n.options\",\n description: \"i18n unused-key check configuration\",\n schema: t.optional(\n t.object({\n /**\n * Directories (relative to the project root) to scan both for\n * `$dictionary(...)` declarations and for translation key usage.\n *\n * @default [\"src\"]\n */\n scan: t.optional(t.array(t.text())),\n\n /**\n * Key prefixes that are constructed at runtime (e.g. via template\n * literals like `` tr(`archive.type.${kind}`) ``). Every key\n * starting with one of these prefixes is exempted from the\n * unused check.\n *\n * Keep this list short and audit it when a feature is removed —\n * a stale prefix here means dead keys can hide.\n *\n * @default []\n */\n dynamicPrefixes: t.optional(t.array(t.text())),\n\n /**\n * Additional path substrings (matched against the full file\n * path) that should be excluded from the scan, on top of the\n * defaults (`node_modules`, `dist`, `__tests__`, `.spec.*`,\n * `.test.*`, `.alepha`).\n *\n * @default []\n */\n exclude: t.optional(t.array(t.text())),\n }),\n ),\n});\n\n/**\n * Type for i18n options.\n */\nexport type I18nOptions = Static<typeof i18nOptions.schema>;\n","import { $inject } from \"alepha\";\nimport { FileSystemProvider } from \"alepha/system\";\n\n/**\n * File extensions considered when scanning for dictionaries / usage.\n */\nconst SCAN_EXTS = [\".ts\", \".tsx\", \".mts\", \".cts\"];\n\n/**\n * Built-in path substrings that are always excluded from scanning.\n */\nconst DEFAULT_EXCLUDES = [\n \"/node_modules/\",\n \"/dist/\",\n \"/__tests__/\",\n \"/.alepha/\",\n \".spec.ts\",\n \".spec.tsx\",\n \".test.ts\",\n \".test.tsx\",\n];\n\n/**\n * Matches a quoted dotted property key on the left-hand side of a\n * dictionary entry: `\"some.dotted.key\":`. The \"at least one dot\"\n * requirement rules out unrelated quoted strings (e.g. JSON literals\n * inside helper text).\n */\nconst KEY_DECLARATION_RE = /\"([\\w-]+(?:\\.[\\w-]+)+)\"\\s*:/g;\n\n/**\n * Heuristic for spotting files that declare a dictionary. Conservative\n * by design — we'd rather scan a few unrelated files than miss a\n * dictionary tucked away in an unusual location.\n */\nconst DICTIONARY_MARKER = \"$dictionary\";\n\nexport interface I18nCheckOptions {\n root: string;\n scan: string[];\n dynamicPrefixes: string[];\n exclude: string[];\n}\n\nexport interface I18nCheckResult {\n /** Total number of keys discovered across all dictionary files. */\n totalKeys: number;\n /** Number of keys exempted via `dynamicPrefixes`. */\n exemptKeys: number;\n /** Number of source files scanned for references. */\n scannedFiles: number;\n /** Dictionary files that contributed keys. */\n dictionaryFiles: string[];\n /** Keys that have no quoted-literal reference anywhere in the scan. */\n unused: string[];\n}\n\nexport class I18nCheckService {\n protected readonly fs = $inject(FileSystemProvider);\n\n /**\n * Find unused translation keys.\n *\n * Discovery is fully static: we walk `scan` dirs, identify files\n * that import `$dictionary` (matched via the literal substring),\n * extract every `\"a.b.c\": ...` property key declared in them, then\n * grep the remaining source files for a quoted-literal occurrence\n * of each key. Anything matching a `dynamicPrefixes` entry is\n * exempted.\n */\n async check(options: I18nCheckOptions): Promise<I18nCheckResult> {\n const { root, scan, dynamicPrefixes, exclude } = options;\n const excludes = [...DEFAULT_EXCLUDES, ...exclude];\n\n const allFiles: string[] = [];\n for (const dir of scan) {\n const absDir = this.fs.join(root, dir);\n let entries: string[];\n try {\n entries = await this.fs.ls(absDir, { recursive: true });\n } catch {\n // Missing scan dir is silently skipped — config carries\n // optional paths (e.g. .vendor/**) that may not exist locally.\n continue;\n }\n for (const rel of entries) {\n const abs = this.fs.join(absDir, rel);\n if (!SCAN_EXTS.some((ext) => abs.endsWith(ext))) continue;\n if (excludes.some((sub) => abs.includes(sub))) continue;\n allFiles.push(abs);\n }\n }\n\n const dictionaryFiles: string[] = [];\n const allKeys = new Set<string>();\n const fileContents = new Map<string, string>();\n\n for (const file of allFiles) {\n const text = (await this.fs.readFile(file)).toString(\"utf8\");\n fileContents.set(file, text);\n if (!text.includes(DICTIONARY_MARKER)) continue;\n const before = allKeys.size;\n for (const m of text.matchAll(KEY_DECLARATION_RE)) {\n allKeys.add(m[1]);\n }\n if (allKeys.size > before) dictionaryFiles.push(file);\n }\n\n // Concatenate every non-dictionary file into one corpus so each\n // key is tested with a single regex run rather than O(files × keys).\n const dictionarySet = new Set(dictionaryFiles);\n const corpusParts: string[] = [];\n let scannedFiles = 0;\n for (const [file, text] of fileContents) {\n if (dictionarySet.has(file)) continue;\n corpusParts.push(text);\n scannedFiles++;\n }\n const corpus = corpusParts.join(\"\\n\");\n\n let exemptKeys = 0;\n const unused: string[] = [];\n for (const key of allKeys) {\n if (dynamicPrefixes.some((p) => key.startsWith(p))) {\n exemptKeys++;\n continue;\n }\n const literal = key.replace(/[.\\\\]/g, (c) => `\\\\${c}`);\n // Key must appear as a quoted string literal — `\"...\"`, `'...'`,\n // or `` `...` ``. Quotes on both sides rule out accidental\n // substring hits in longer keys.\n const re = new RegExp(`[\"'\\`]${literal}[\"'\\`]`);\n if (!re.test(corpus)) unused.push(key);\n }\n\n return {\n totalKeys: allKeys.size,\n exemptKeys,\n scannedFiles,\n dictionaryFiles,\n unused: unused.sort(),\n };\n }\n}\n","import { $inject, $state } from \"alepha\";\nimport { $command } from \"alepha/command\";\nimport { $logger, ConsoleColorProvider } from \"alepha/logger\";\nimport { i18nOptions } from \"../atoms/i18nOptions.ts\";\nimport { I18nCheckService } from \"../services/I18nCheckService.ts\";\n\nexport class I18nCommand {\n protected readonly log = $logger();\n protected readonly options = $state(i18nOptions);\n protected readonly checkService = $inject(I18nCheckService);\n protected readonly color = $inject(ConsoleColorProvider);\n\n protected resolveOptions() {\n return {\n scan: this.options?.scan ?? [\"src\"],\n dynamicPrefixes: this.options?.dynamicPrefixes ?? [],\n exclude: this.options?.exclude ?? [],\n };\n }\n\n protected readonly check = $command({\n name: \"check\",\n description: \"Report translation keys with no quoted-literal reference\",\n handler: async ({ root }) => {\n const opts = this.resolveOptions();\n const c = this.color;\n\n const result = await this.checkService.check({ root, ...opts });\n\n if (result.totalKeys === 0) {\n process.stdout.write(\n `\\n${c.set(\"ORANGE\", \"warn\")} No translation keys found. ` +\n `Did the dictionary location change? ` +\n `Searched: ${opts.scan.join(\", \")}\\n\\n`,\n );\n process.exit(2);\n }\n\n process.stdout.write(\n `\\nChecked ${c.set(\"CYAN\", String(result.totalKeys))} keys across ` +\n `${c.set(\"CYAN\", String(result.scannedFiles))} files ` +\n `(${result.dictionaryFiles.length} dictionary ` +\n `${result.dictionaryFiles.length === 1 ? \"file\" : \"files\"}).\\n`,\n );\n if (result.exemptKeys > 0) {\n process.stdout.write(\n ` exempt (dynamic prefixes): ${result.exemptKeys}\\n`,\n );\n }\n\n if (result.unused.length === 0) {\n process.stdout.write(\n `\\n${c.set(\"GREEN\", \"✓\")} All translations are referenced.\\n\\n`,\n );\n return;\n }\n\n process.stdout.write(\n `\\n${c.set(\"RED\", \"✗\")} Unused translations (${result.unused.length}):\\n`,\n );\n for (const k of result.unused) {\n process.stdout.write(` ${c.set(\"DIM\", \"-\")} ${k}\\n`);\n }\n process.stdout.write(\n `\\nEither delete the key from its dictionary, or add its prefix to ` +\n `${c.set(\"CYAN\", \"dynamicPrefixes\")} in alepha.config.ts ` +\n `if it's constructed at runtime.\\n\\n`,\n );\n process.exit(1);\n },\n });\n\n public readonly i18n = $command({\n name: \"i18n\",\n description: \"Internationalization tooling\",\n children: [this.check],\n handler: async ({ help }) => {\n help();\n },\n });\n}\n","import { $context, $module } from \"alepha\";\nimport { type I18nOptions, i18nOptions } from \"./atoms/i18nOptions.ts\";\nimport { I18nCommand } from \"./commands/I18nCommand.ts\";\nimport { I18nCheckService } from \"./services/I18nCheckService.ts\";\n\n// ---------------------------------------------------------------------------\n\n/**\n * CLI plugin for finding unused translation keys.\n *\n * Statically scans the project for `$dictionary(...)` calls, extracts\n * every declared key, and reports the ones that have no quoted-literal\n * reference anywhere else in the source tree. Designed to be wired\n * into `yarn v` (or any verify pipeline) so dead i18n entries can't\n * pile up unnoticed when a feature is removed.\n *\n * Commands:\n * - `alepha i18n check` — report unused translation keys\n *\n * Configuration in `alepha.config.ts`:\n *\n * ```typescript\n * import { i18n } from \"alepha/cli/i18n\";\n *\n * export default defineConfig({\n * plugins: [\n * i18n({\n * scan: [\"src\", \".vendor/@alepha/ui\"],\n * dynamicPrefixes: [\"archive.type.\", \"petitions.filter.\"],\n * }),\n * ],\n * });\n * ```\n */\nexport const AlephaCliI18nPlugin = $module({\n name: \"alepha.cli.plugins.i18n\",\n atoms: [i18nOptions],\n services: [I18nCommand, I18nCheckService],\n});\n\nexport const i18n = (options: I18nOptions = {}) => {\n return () => {\n const { alepha } = $context();\n alepha.with(AlephaCliI18nPlugin).set(i18nOptions, options);\n };\n};\n\n// ---------------------------------------------------------------------------\n\nexport * from \"./atoms/i18nOptions.ts\";\nexport * from \"./commands/I18nCommand.ts\";\nexport * from \"./services/I18nCheckService.ts\";\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":[],"sources":["../../../src/cli/i18n/atoms/i18nOptions.ts","../../../src/cli/i18n/services/I18nCheckService.ts","../../../src/cli/i18n/commands/I18nCommand.ts","../../../src/cli/i18n/index.ts"],"sourcesContent":["import { $atom, type Static, t } from \"alepha\";\n\n/**\n * i18n CLI configuration atom.\n *\n * Filled from the `i18n` plugin in `alepha.config.ts`.\n * Read by `I18nCommand` to drive `alepha i18n check`.\n */\nexport const i18nOptions = $atom({\n name: \"alepha.cli.i18n.options\",\n description: \"i18n unused-key check configuration\",\n schema: t.optional(\n t.object({\n /**\n * Directories (relative to the project root) to scan both for\n * `$dictionary(...)` declarations and for translation key usage.\n *\n * @default [\"src\"]\n */\n scan: t.optional(t.array(t.text())),\n\n /**\n * Key prefixes that are constructed at runtime (e.g. via template\n * literals like `` tr(`archive.type.${kind}`) ``). Every key\n * starting with one of these prefixes is exempted from the\n * unused check.\n *\n * Keep this list short and audit it when a feature is removed —\n * a stale prefix here means dead keys can hide.\n *\n * @default []\n */\n dynamicPrefixes: t.optional(t.array(t.text())),\n\n /**\n * Additional path substrings (matched against the full file\n * path) that should be excluded from the scan, on top of the\n * defaults (`node_modules`, `dist`, `__tests__`, `.spec.*`,\n * `.test.*`, `.alepha`).\n *\n * @default []\n */\n exclude: t.optional(t.array(t.text())),\n }),\n ),\n});\n\n/**\n * Type for i18n options.\n */\nexport type I18nOptions = Static<typeof i18nOptions.schema>;\n","import { $inject } from \"alepha\";\nimport { FileSystemProvider } from \"alepha/system\";\n\n/**\n * File extensions considered when scanning for dictionaries / usage.\n */\nconst SCAN_EXTS = [\".ts\", \".tsx\", \".mts\", \".cts\"];\n\n/**\n * Built-in path substrings that are always excluded from scanning.\n */\nconst DEFAULT_EXCLUDES = [\n \"/node_modules/\",\n \"/dist/\",\n \"/__tests__/\",\n \"/.alepha/\",\n \".spec.ts\",\n \".spec.tsx\",\n \".test.ts\",\n \".test.tsx\",\n];\n\n/**\n * Matches a quoted dotted property key on the left-hand side of a\n * dictionary entry: `\"some.dotted.key\":`. The \"at least one dot\"\n * requirement rules out unrelated quoted strings (e.g. JSON literals\n * inside helper text).\n */\nconst KEY_DECLARATION_RE = /\"([\\w-]+(?:\\.[\\w-]+)+)\"\\s*:/g;\n\n/**\n * Heuristic for spotting files that declare a dictionary. Conservative\n * by design — we'd rather scan a few unrelated files than miss a\n * dictionary tucked away in an unusual location.\n */\nconst DICTIONARY_MARKER = \"$dictionary\";\n\nexport interface I18nCheckOptions {\n root: string;\n scan: string[];\n dynamicPrefixes: string[];\n exclude: string[];\n}\n\nexport interface I18nCheckResult {\n /** Total number of keys discovered across all dictionary files. */\n totalKeys: number;\n /** Number of keys exempted via `dynamicPrefixes`. */\n exemptKeys: number;\n /** Number of source files scanned for references. */\n scannedFiles: number;\n /** Dictionary files that contributed keys. */\n dictionaryFiles: string[];\n /** Keys that have no quoted-literal reference anywhere in the scan. */\n unused: string[];\n}\n\nexport class I18nCheckService {\n protected readonly fs = $inject(FileSystemProvider);\n\n /**\n * Find unused translation keys.\n *\n * Discovery is fully static: we walk `scan` dirs, identify files\n * that import `$dictionary` (matched via the literal substring),\n * extract every `\"a.b.c\": ...` property key declared in them, then\n * grep the remaining source files for a quoted-literal occurrence\n * of each key. Anything matching a `dynamicPrefixes` entry is\n * exempted.\n */\n async check(options: I18nCheckOptions): Promise<I18nCheckResult> {\n const { root, scan, dynamicPrefixes, exclude } = options;\n const excludes = [...DEFAULT_EXCLUDES, ...exclude];\n\n const allFiles: string[] = [];\n for (const dir of scan) {\n const absDir = this.fs.join(root, dir);\n let entries: string[];\n try {\n entries = await this.fs.ls(absDir, { recursive: true });\n } catch {\n // Missing scan dir is silently skipped — config carries\n // optional paths (e.g. .vendor/**) that may not exist locally.\n continue;\n }\n for (const rel of entries) {\n const abs = this.fs.join(absDir, rel);\n if (!SCAN_EXTS.some((ext) => abs.endsWith(ext))) continue;\n if (excludes.some((sub) => abs.includes(sub))) continue;\n allFiles.push(abs);\n }\n }\n\n const dictionaryFiles: string[] = [];\n const allKeys = new Set<string>();\n const fileContents = new Map<string, string>();\n\n for (const file of allFiles) {\n const text = (await this.fs.readFile(file)).toString(\"utf8\");\n fileContents.set(file, text);\n if (!text.includes(DICTIONARY_MARKER)) continue;\n const before = allKeys.size;\n for (const m of text.matchAll(KEY_DECLARATION_RE)) {\n allKeys.add(m[1]);\n }\n if (allKeys.size > before) dictionaryFiles.push(file);\n }\n\n // Concatenate every non-dictionary file into one corpus so each\n // key is tested with a single regex run rather than O(files × keys).\n const dictionarySet = new Set(dictionaryFiles);\n const corpusParts: string[] = [];\n let scannedFiles = 0;\n for (const [file, text] of fileContents) {\n if (dictionarySet.has(file)) continue;\n corpusParts.push(text);\n scannedFiles++;\n }\n const corpus = corpusParts.join(\"\\n\");\n\n let exemptKeys = 0;\n const unused: string[] = [];\n for (const key of allKeys) {\n if (dynamicPrefixes.some((p) => key.startsWith(p))) {\n exemptKeys++;\n continue;\n }\n const literal = key.replace(/[.\\\\]/g, (c) => `\\\\${c}`);\n // Key must appear as a quoted string literal — `\"...\"`, `'...'`,\n // or `` `...` ``. Quotes on both sides rule out accidental\n // substring hits in longer keys.\n const re = new RegExp(`[\"'\\`]${literal}[\"'\\`]`);\n if (!re.test(corpus)) unused.push(key);\n }\n\n return {\n totalKeys: allKeys.size,\n exemptKeys,\n scannedFiles,\n dictionaryFiles,\n unused: unused.sort(),\n };\n }\n}\n","import { $inject, $state } from \"alepha\";\nimport { $command } from \"alepha/command\";\nimport { $logger, ConsoleColorProvider } from \"alepha/logger\";\nimport { i18nOptions } from \"../atoms/i18nOptions.ts\";\nimport { I18nCheckService } from \"../services/I18nCheckService.ts\";\n\nexport class I18nCommand {\n protected readonly log = $logger();\n protected readonly options = $state(i18nOptions);\n protected readonly checkService = $inject(I18nCheckService);\n protected readonly color = $inject(ConsoleColorProvider);\n\n protected resolveOptions() {\n return {\n scan: this.options?.scan ?? [\"src\"],\n dynamicPrefixes: this.options?.dynamicPrefixes ?? [],\n exclude: this.options?.exclude ?? [],\n };\n }\n\n protected readonly check = $command({\n name: \"check\",\n description: \"Report translation keys with no quoted-literal reference\",\n handler: async ({ root }) => {\n const opts = this.resolveOptions();\n const c = this.color;\n\n const result = await this.checkService.check({ root, ...opts });\n\n if (result.totalKeys === 0) {\n process.stdout.write(\n `\\n${c.set(\"ORANGE\", \"warn\")} No translation keys found. ` +\n `Did the dictionary location change? ` +\n `Searched: ${opts.scan.join(\", \")}\\n\\n`,\n );\n process.exit(2);\n }\n\n process.stdout.write(\n `\\nChecked ${c.set(\"CYAN\", String(result.totalKeys))} keys across ` +\n `${c.set(\"CYAN\", String(result.scannedFiles))} files ` +\n `(${result.dictionaryFiles.length} dictionary ` +\n `${result.dictionaryFiles.length === 1 ? \"file\" : \"files\"}).\\n`,\n );\n if (result.exemptKeys > 0) {\n process.stdout.write(\n ` exempt (dynamic prefixes): ${result.exemptKeys}\\n`,\n );\n }\n\n if (result.unused.length === 0) {\n process.stdout.write(\n `\\n${c.set(\"GREEN\", \"✓\")} All translations are referenced.\\n\\n`,\n );\n return;\n }\n\n process.stdout.write(\n `\\n${c.set(\"RED\", \"✗\")} Unused translations (${result.unused.length}):\\n`,\n );\n for (const k of result.unused) {\n process.stdout.write(` ${c.set(\"DIM\", \"-\")} ${k}\\n`);\n }\n process.stdout.write(\n `\\nEither delete the key from its dictionary, or add its prefix to ` +\n `${c.set(\"CYAN\", \"dynamicPrefixes\")} in alepha.config.ts ` +\n `if it's constructed at runtime.\\n\\n`,\n );\n process.exit(1);\n },\n });\n\n public readonly i18n = $command({\n name: \"i18n\",\n description: \"Internationalization tooling\",\n children: [this.check],\n handler: async ({ help }) => {\n help();\n },\n });\n}\n","import { $context, $module } from \"alepha\";\nimport { type I18nOptions, i18nOptions } from \"./atoms/i18nOptions.ts\";\nimport { I18nCommand } from \"./commands/I18nCommand.ts\";\nimport { I18nCheckService } from \"./services/I18nCheckService.ts\";\n\n// ---------------------------------------------------------------------------\n\n/**\n * CLI plugin for finding unused translation keys.\n *\n * Statically scans the project for `$dictionary(...)` calls, extracts\n * every declared key, and reports the ones that have no quoted-literal\n * reference anywhere else in the source tree. Designed to be wired\n * into `yarn v` (or any verify pipeline) so dead i18n entries can't\n * pile up unnoticed when a feature is removed.\n *\n * Commands:\n * - `alepha i18n check` — report unused translation keys\n *\n * Configuration in `alepha.config.ts`:\n *\n * ```typescript\n * import { i18n } from \"alepha/cli/i18n\";\n *\n * export default defineConfig({\n * plugins: [\n * i18n({\n * scan: [\"src\", \".vendor/@alepha/ui\"],\n * dynamicPrefixes: [\"archive.type.\", \"petitions.filter.\"],\n * }),\n * ],\n * });\n * ```\n */\nexport const AlephaCliI18nPlugin = $module({\n name: \"alepha.cli.plugins.i18n\",\n atoms: [i18nOptions],\n services: [I18nCommand, I18nCheckService],\n});\n\nexport const i18n = (options: I18nOptions = {}) => {\n return () => {\n const { alepha } = $context();\n alepha.with(AlephaCliI18nPlugin).set(i18nOptions, options);\n };\n};\n\n// ---------------------------------------------------------------------------\n\nexport * from \"./atoms/i18nOptions.ts\";\nexport * from \"./commands/I18nCommand.ts\";\nexport * from \"./services/I18nCheckService.ts\";\n"],"mappings":"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"}