@digitaldefiance/i18n-lib 3.7.5 → 3.8.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (490) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +261 -0
  3. package/package.json +12 -4
  4. package/src/active-context.ts +4 -0
  5. package/src/builders/i18n-builder.ts +82 -0
  6. package/src/builders/{index.d.ts → index.ts} +1 -1
  7. package/src/component-definition.ts +11 -0
  8. package/src/component-registration.ts +29 -0
  9. package/src/component-registry.ts +432 -0
  10. package/src/context-error-type.ts +7 -0
  11. package/src/core/component-store.ts +241 -0
  12. package/src/core/context-manager.ts +113 -0
  13. package/src/core/enum-registry.ts +106 -0
  14. package/src/core/i18n-engine.ts +710 -0
  15. package/src/core/index.ts +16 -0
  16. package/src/core/language-registry.ts +345 -0
  17. package/src/core-component-id.ts +5 -0
  18. package/src/core-i18n.ts +270 -0
  19. package/src/core-plugin-factory.ts +111 -0
  20. package/src/core-string-key.ts +59 -0
  21. package/src/create-translation-adapter.ts +93 -0
  22. package/src/enum-registry.ts +152 -0
  23. package/src/errors/base.ts +8 -0
  24. package/src/errors/context-error.ts +122 -0
  25. package/src/errors/enhanced-error-base.ts +260 -0
  26. package/src/errors/handleable.ts +152 -0
  27. package/src/errors/i18n-error.ts +494 -0
  28. package/src/errors/index.ts +15 -0
  29. package/src/errors/simple-typed-error.ts +81 -0
  30. package/src/errors/{index.d.ts → translatable-exports.ts} +3 -5
  31. package/src/errors/translatable-generic.ts +245 -0
  32. package/src/errors/translatable-handleable-generic.ts +222 -0
  33. package/src/errors/translatable.ts +138 -0
  34. package/src/errors/typed-handleable.ts +138 -0
  35. package/src/errors/typed.ts +617 -0
  36. package/src/gender/{gender-categories.d.ts → gender-categories.ts} +6 -2
  37. package/src/gender/gender-resolver.ts +40 -0
  38. package/src/gender/{index.d.ts → index.ts} +0 -1
  39. package/src/global-active-context.ts +266 -0
  40. package/src/icu/ast.ts +56 -0
  41. package/src/icu/compiler.ts +96 -0
  42. package/src/icu/formatter-registry.ts +36 -0
  43. package/src/icu/formatters/base-formatter.ts +8 -0
  44. package/src/icu/formatters/date-formatter.ts +30 -0
  45. package/src/icu/formatters/number-formatter.ts +32 -0
  46. package/src/icu/formatters/plural-formatter.ts +12 -0
  47. package/src/icu/formatters/select-formatter.ts +7 -0
  48. package/src/icu/formatters/selectordinal-formatter.ts +17 -0
  49. package/src/icu/formatters/time-formatter.ts +30 -0
  50. package/src/icu/helpers.ts +34 -0
  51. package/src/icu/parser.ts +242 -0
  52. package/src/icu/runtime.ts +37 -0
  53. package/src/icu/tokenizer.ts +212 -0
  54. package/src/icu/validator.ts +163 -0
  55. package/src/{index.d.ts → index.ts} +46 -14
  56. package/src/interfaces/active-context.interface.ts +41 -0
  57. package/src/interfaces/component-config.interface.ts +17 -0
  58. package/src/interfaces/engine-config.interface.ts +22 -0
  59. package/src/interfaces/global-active-context.ts +39 -0
  60. package/src/interfaces/handleable-error-options.ts +13 -0
  61. package/src/interfaces/handleable.ts +20 -0
  62. package/src/interfaces/i18n-engine.interface.ts +57 -0
  63. package/src/interfaces/index.ts +13 -0
  64. package/src/interfaces/language-definition.interface.ts +17 -0
  65. package/src/interfaces/translation-options.interface.ts +15 -0
  66. package/src/interfaces/validation-result.interface.ts +24 -0
  67. package/src/language-codes.ts +40 -0
  68. package/src/language-definition.ts +13 -0
  69. package/src/plugin-i18n-engine.ts +707 -0
  70. package/src/pluralization/{index.d.ts → index.ts} +1 -1
  71. package/src/pluralization/language-plural-map.ts +186 -0
  72. package/src/pluralization/{plural-categories.d.ts → plural-categories.ts} +5 -3
  73. package/src/pluralization/plural-rules.ts +228 -0
  74. package/src/registry-config.ts +16 -0
  75. package/src/registry-error-type.ts +19 -0
  76. package/src/registry-error.ts +100 -0
  77. package/src/strict-types.ts +35 -0
  78. package/src/strings/de.ts +75 -0
  79. package/src/strings/en-GB.ts +74 -0
  80. package/src/strings/en-US.ts +74 -0
  81. package/src/strings/es.ts +74 -0
  82. package/src/strings/fr.ts +75 -0
  83. package/src/strings/ja.ts +73 -0
  84. package/src/strings/uk.ts +73 -0
  85. package/src/strings/zh-CN.ts +72 -0
  86. package/src/template.ts +72 -0
  87. package/src/translation-engine.ts +18 -0
  88. package/src/translation-request.ts +12 -0
  89. package/src/translation-response.ts +8 -0
  90. package/src/types/engine.ts +55 -0
  91. package/src/types/{index.d.ts → index.ts} +1 -1
  92. package/src/types/{plural-types.d.ts → plural-types.ts} +29 -3
  93. package/src/types.ts +142 -0
  94. package/src/utils/currency.ts +141 -0
  95. package/src/utils/html-escape.ts +55 -0
  96. package/src/utils/{index.d.ts → index.ts} +0 -1
  97. package/src/utils/lru-cache.ts +76 -0
  98. package/src/utils/{plural-helpers.d.ts → plural-helpers.ts} +14 -4
  99. package/src/utils/{safe-object.js → safe-object.ts} +37 -34
  100. package/src/utils/string-utils.ts +77 -0
  101. package/src/utils/timezone.ts +76 -0
  102. package/src/utils/validation.ts +66 -0
  103. package/src/utils.ts +215 -0
  104. package/src/validation/{index.d.ts → index.ts} +0 -1
  105. package/src/validation/plural-validator.ts +168 -0
  106. package/src/validation-config.ts +11 -0
  107. package/src/validation-result.ts +12 -0
  108. package/src/active-context.d.ts +0 -36
  109. package/src/active-context.d.ts.map +0 -1
  110. package/src/active-context.js +0 -3
  111. package/src/active-context.js.map +0 -1
  112. package/src/builders/i18n-builder.d.ts +0 -26
  113. package/src/builders/i18n-builder.d.ts.map +0 -1
  114. package/src/builders/i18n-builder.js +0 -70
  115. package/src/builders/i18n-builder.js.map +0 -1
  116. package/src/builders/index.d.ts.map +0 -1
  117. package/src/builders/index.js +0 -8
  118. package/src/builders/index.js.map +0 -1
  119. package/src/component-definition.d.ts +0 -12
  120. package/src/component-definition.d.ts.map +0 -1
  121. package/src/component-definition.js +0 -3
  122. package/src/component-definition.js.map +0 -1
  123. package/src/component-registration.d.ts +0 -13
  124. package/src/component-registration.d.ts.map +0 -1
  125. package/src/component-registration.js +0 -3
  126. package/src/component-registration.js.map +0 -1
  127. package/src/component-registry.d.ts +0 -102
  128. package/src/component-registry.d.ts.map +0 -1
  129. package/src/component-registry.js +0 -282
  130. package/src/component-registry.js.map +0 -1
  131. package/src/context-error-type.d.ts +0 -8
  132. package/src/context-error-type.d.ts.map +0 -1
  133. package/src/context-error-type.js +0 -12
  134. package/src/context-error-type.js.map +0 -1
  135. package/src/core/component-store.d.ts +0 -93
  136. package/src/core/component-store.d.ts.map +0 -1
  137. package/src/core/component-store.js +0 -198
  138. package/src/core/component-store.js.map +0 -1
  139. package/src/core/context-manager.d.ts +0 -72
  140. package/src/core/context-manager.d.ts.map +0 -1
  141. package/src/core/context-manager.js +0 -98
  142. package/src/core/context-manager.js.map +0 -1
  143. package/src/core/enum-registry.d.ts +0 -48
  144. package/src/core/enum-registry.d.ts.map +0 -1
  145. package/src/core/enum-registry.js +0 -85
  146. package/src/core/enum-registry.js.map +0 -1
  147. package/src/core/i18n-engine.d.ts +0 -241
  148. package/src/core/i18n-engine.d.ts.map +0 -1
  149. package/src/core/i18n-engine.js +0 -568
  150. package/src/core/i18n-engine.js.map +0 -1
  151. package/src/core/index.d.ts +0 -9
  152. package/src/core/index.d.ts.map +0 -1
  153. package/src/core/index.js +0 -12
  154. package/src/core/index.js.map +0 -1
  155. package/src/core/language-registry.d.ts +0 -180
  156. package/src/core/language-registry.d.ts.map +0 -1
  157. package/src/core/language-registry.js +0 -295
  158. package/src/core/language-registry.js.map +0 -1
  159. package/src/core-i18n.d.ts +0 -91
  160. package/src/core-i18n.d.ts.map +0 -1
  161. package/src/core-i18n.js +0 -287
  162. package/src/core-i18n.js.map +0 -1
  163. package/src/core-string-key.d.ts +0 -52
  164. package/src/core-string-key.d.ts.map +0 -1
  165. package/src/core-string-key.js +0 -61
  166. package/src/core-string-key.js.map +0 -1
  167. package/src/create-translation-adapter.d.ts +0 -33
  168. package/src/create-translation-adapter.d.ts.map +0 -1
  169. package/src/create-translation-adapter.js +0 -72
  170. package/src/create-translation-adapter.js.map +0 -1
  171. package/src/enum-registry.d.ts +0 -65
  172. package/src/enum-registry.d.ts.map +0 -1
  173. package/src/enum-registry.js +0 -123
  174. package/src/enum-registry.js.map +0 -1
  175. package/src/errors/context-error.d.ts +0 -50
  176. package/src/errors/context-error.d.ts.map +0 -1
  177. package/src/errors/context-error.js +0 -75
  178. package/src/errors/context-error.js.map +0 -1
  179. package/src/errors/enhanced-error-base.d.ts +0 -125
  180. package/src/errors/enhanced-error-base.d.ts.map +0 -1
  181. package/src/errors/enhanced-error-base.js +0 -165
  182. package/src/errors/enhanced-error-base.js.map +0 -1
  183. package/src/errors/handleable.d.ts +0 -83
  184. package/src/errors/handleable.d.ts.map +0 -1
  185. package/src/errors/handleable.js +0 -136
  186. package/src/errors/handleable.js.map +0 -1
  187. package/src/errors/i18n-error.d.ts +0 -211
  188. package/src/errors/i18n-error.d.ts.map +0 -1
  189. package/src/errors/i18n-error.js +0 -358
  190. package/src/errors/i18n-error.js.map +0 -1
  191. package/src/errors/index.d.ts.map +0 -1
  192. package/src/errors/index.js +0 -17
  193. package/src/errors/index.js.map +0 -1
  194. package/src/errors/simple-typed-error.d.ts +0 -53
  195. package/src/errors/simple-typed-error.d.ts.map +0 -1
  196. package/src/errors/simple-typed-error.js +0 -51
  197. package/src/errors/simple-typed-error.js.map +0 -1
  198. package/src/errors/translatable-generic.d.ts +0 -84
  199. package/src/errors/translatable-generic.d.ts.map +0 -1
  200. package/src/errors/translatable-generic.js +0 -134
  201. package/src/errors/translatable-generic.js.map +0 -1
  202. package/src/errors/translatable-handleable-generic.d.ts +0 -116
  203. package/src/errors/translatable-handleable-generic.d.ts.map +0 -1
  204. package/src/errors/translatable-handleable-generic.js +0 -121
  205. package/src/errors/translatable-handleable-generic.js.map +0 -1
  206. package/src/errors/translatable.d.ts +0 -59
  207. package/src/errors/translatable.d.ts.map +0 -1
  208. package/src/errors/translatable.js +0 -80
  209. package/src/errors/translatable.js.map +0 -1
  210. package/src/errors/typed-handleable.d.ts +0 -62
  211. package/src/errors/typed-handleable.d.ts.map +0 -1
  212. package/src/errors/typed-handleable.js +0 -106
  213. package/src/errors/typed-handleable.js.map +0 -1
  214. package/src/errors/typed.d.ts +0 -206
  215. package/src/errors/typed.d.ts.map +0 -1
  216. package/src/errors/typed.js +0 -452
  217. package/src/errors/typed.js.map +0 -1
  218. package/src/gender/gender-categories.d.ts.map +0 -1
  219. package/src/gender/gender-categories.js +0 -15
  220. package/src/gender/gender-categories.js.map +0 -1
  221. package/src/gender/gender-resolver.d.ts +0 -14
  222. package/src/gender/gender-resolver.d.ts.map +0 -1
  223. package/src/gender/gender-resolver.js +0 -35
  224. package/src/gender/gender-resolver.js.map +0 -1
  225. package/src/gender/index.d.ts.map +0 -1
  226. package/src/gender/index.js +0 -6
  227. package/src/gender/index.js.map +0 -1
  228. package/src/global-active-context.d.ts +0 -50
  229. package/src/global-active-context.d.ts.map +0 -1
  230. package/src/global-active-context.js +0 -185
  231. package/src/global-active-context.js.map +0 -1
  232. package/src/icu/ast.d.ts +0 -48
  233. package/src/icu/ast.d.ts.map +0 -1
  234. package/src/icu/ast.js +0 -16
  235. package/src/icu/ast.js.map +0 -1
  236. package/src/icu/compiler.d.ts +0 -16
  237. package/src/icu/compiler.d.ts.map +0 -1
  238. package/src/icu/compiler.js +0 -87
  239. package/src/icu/compiler.js.map +0 -1
  240. package/src/icu/formatter-registry.d.ts +0 -10
  241. package/src/icu/formatter-registry.d.ts.map +0 -1
  242. package/src/icu/formatter-registry.js +0 -34
  243. package/src/icu/formatter-registry.js.map +0 -1
  244. package/src/icu/formatters/base-formatter.d.ts +0 -8
  245. package/src/icu/formatters/base-formatter.d.ts.map +0 -1
  246. package/src/icu/formatters/base-formatter.js +0 -3
  247. package/src/icu/formatters/base-formatter.js.map +0 -1
  248. package/src/icu/formatters/date-formatter.d.ts +0 -5
  249. package/src/icu/formatters/date-formatter.d.ts.map +0 -1
  250. package/src/icu/formatters/date-formatter.js +0 -31
  251. package/src/icu/formatters/date-formatter.js.map +0 -1
  252. package/src/icu/formatters/number-formatter.d.ts +0 -5
  253. package/src/icu/formatters/number-formatter.d.ts.map +0 -1
  254. package/src/icu/formatters/number-formatter.js +0 -33
  255. package/src/icu/formatters/number-formatter.js.map +0 -1
  256. package/src/icu/formatters/plural-formatter.d.ts +0 -5
  257. package/src/icu/formatters/plural-formatter.d.ts.map +0 -1
  258. package/src/icu/formatters/plural-formatter.js +0 -15
  259. package/src/icu/formatters/plural-formatter.js.map +0 -1
  260. package/src/icu/formatters/select-formatter.d.ts +0 -5
  261. package/src/icu/formatters/select-formatter.d.ts.map +0 -1
  262. package/src/icu/formatters/select-formatter.js +0 -10
  263. package/src/icu/formatters/select-formatter.js.map +0 -1
  264. package/src/icu/formatters/selectordinal-formatter.d.ts +0 -5
  265. package/src/icu/formatters/selectordinal-formatter.d.ts.map +0 -1
  266. package/src/icu/formatters/selectordinal-formatter.js +0 -22
  267. package/src/icu/formatters/selectordinal-formatter.js.map +0 -1
  268. package/src/icu/formatters/time-formatter.d.ts +0 -5
  269. package/src/icu/formatters/time-formatter.d.ts.map +0 -1
  270. package/src/icu/formatters/time-formatter.js +0 -31
  271. package/src/icu/formatters/time-formatter.js.map +0 -1
  272. package/src/icu/helpers.d.ts +0 -9
  273. package/src/icu/helpers.d.ts.map +0 -1
  274. package/src/icu/helpers.js +0 -31
  275. package/src/icu/helpers.js.map +0 -1
  276. package/src/icu/parser.d.ts +0 -31
  277. package/src/icu/parser.d.ts.map +0 -1
  278. package/src/icu/parser.js +0 -203
  279. package/src/icu/parser.js.map +0 -1
  280. package/src/icu/runtime.d.ts +0 -10
  281. package/src/icu/runtime.d.ts.map +0 -1
  282. package/src/icu/runtime.js +0 -33
  283. package/src/icu/runtime.js.map +0 -1
  284. package/src/icu/tokenizer.d.ts +0 -37
  285. package/src/icu/tokenizer.d.ts.map +0 -1
  286. package/src/icu/tokenizer.js +0 -187
  287. package/src/icu/tokenizer.js.map +0 -1
  288. package/src/icu/validator.d.ts +0 -11
  289. package/src/icu/validator.d.ts.map +0 -1
  290. package/src/icu/validator.js +0 -140
  291. package/src/icu/validator.js.map +0 -1
  292. package/src/index.d.ts.map +0 -1
  293. package/src/index.js +0 -76
  294. package/src/index.js.map +0 -1
  295. package/src/interfaces/component-config.interface.d.ts +0 -16
  296. package/src/interfaces/component-config.interface.d.ts.map +0 -1
  297. package/src/interfaces/component-config.interface.js +0 -6
  298. package/src/interfaces/component-config.interface.js.map +0 -1
  299. package/src/interfaces/engine-config.interface.d.ts +0 -22
  300. package/src/interfaces/engine-config.interface.d.ts.map +0 -1
  301. package/src/interfaces/engine-config.interface.js +0 -6
  302. package/src/interfaces/engine-config.interface.js.map +0 -1
  303. package/src/interfaces/global-active-context.d.ts +0 -23
  304. package/src/interfaces/global-active-context.d.ts.map +0 -1
  305. package/src/interfaces/global-active-context.js +0 -3
  306. package/src/interfaces/global-active-context.js.map +0 -1
  307. package/src/interfaces/handleable-error-options.d.ts +0 -14
  308. package/src/interfaces/handleable-error-options.d.ts.map +0 -1
  309. package/src/interfaces/handleable-error-options.js +0 -3
  310. package/src/interfaces/handleable-error-options.js.map +0 -1
  311. package/src/interfaces/handleable.d.ts +0 -21
  312. package/src/interfaces/handleable.d.ts.map +0 -1
  313. package/src/interfaces/handleable.js +0 -3
  314. package/src/interfaces/handleable.js.map +0 -1
  315. package/src/interfaces/i18n-engine.interface.d.ts +0 -46
  316. package/src/interfaces/i18n-engine.interface.d.ts.map +0 -1
  317. package/src/interfaces/i18n-engine.interface.js +0 -6
  318. package/src/interfaces/i18n-engine.interface.js.map +0 -1
  319. package/src/interfaces/index.d.ts +0 -11
  320. package/src/interfaces/index.d.ts.map +0 -1
  321. package/src/interfaces/index.js +0 -14
  322. package/src/interfaces/index.js.map +0 -1
  323. package/src/interfaces/language-definition.interface.d.ts +0 -17
  324. package/src/interfaces/language-definition.interface.d.ts.map +0 -1
  325. package/src/interfaces/language-definition.interface.js +0 -6
  326. package/src/interfaces/language-definition.interface.js.map +0 -1
  327. package/src/interfaces/translation-options.interface.d.ts +0 -15
  328. package/src/interfaces/translation-options.interface.d.ts.map +0 -1
  329. package/src/interfaces/translation-options.interface.js +0 -6
  330. package/src/interfaces/translation-options.interface.js.map +0 -1
  331. package/src/interfaces/validation-result.interface.d.ts +0 -24
  332. package/src/interfaces/validation-result.interface.d.ts.map +0 -1
  333. package/src/interfaces/validation-result.interface.js +0 -6
  334. package/src/interfaces/validation-result.interface.js.map +0 -1
  335. package/src/language-codes.d.ts +0 -28
  336. package/src/language-codes.d.ts.map +0 -1
  337. package/src/language-codes.js +0 -32
  338. package/src/language-codes.js.map +0 -1
  339. package/src/language-definition.d.ts +0 -14
  340. package/src/language-definition.d.ts.map +0 -1
  341. package/src/language-definition.js +0 -3
  342. package/src/language-definition.js.map +0 -1
  343. package/src/plugin-i18n-engine.d.ts +0 -164
  344. package/src/plugin-i18n-engine.d.ts.map +0 -1
  345. package/src/plugin-i18n-engine.js +0 -489
  346. package/src/plugin-i18n-engine.js.map +0 -1
  347. package/src/pluralization/index.d.ts.map +0 -1
  348. package/src/pluralization/index.js +0 -10
  349. package/src/pluralization/index.js.map +0 -1
  350. package/src/pluralization/language-plural-map.d.ts +0 -29
  351. package/src/pluralization/language-plural-map.d.ts.map +0 -1
  352. package/src/pluralization/language-plural-map.js +0 -155
  353. package/src/pluralization/language-plural-map.js.map +0 -1
  354. package/src/pluralization/plural-categories.d.ts.map +0 -1
  355. package/src/pluralization/plural-categories.js +0 -8
  356. package/src/pluralization/plural-categories.js.map +0 -1
  357. package/src/pluralization/plural-rules.d.ts +0 -102
  358. package/src/pluralization/plural-rules.d.ts.map +0 -1
  359. package/src/pluralization/plural-rules.js +0 -263
  360. package/src/pluralization/plural-rules.js.map +0 -1
  361. package/src/registry-config.d.ts +0 -16
  362. package/src/registry-config.d.ts.map +0 -1
  363. package/src/registry-config.js +0 -3
  364. package/src/registry-config.js.map +0 -1
  365. package/src/registry-error-type.d.ts +0 -20
  366. package/src/registry-error-type.d.ts.map +0 -1
  367. package/src/registry-error-type.js +0 -24
  368. package/src/registry-error-type.js.map +0 -1
  369. package/src/registry-error.d.ts +0 -19
  370. package/src/registry-error.d.ts.map +0 -1
  371. package/src/registry-error.js +0 -49
  372. package/src/registry-error.js.map +0 -1
  373. package/src/strict-types.d.ts +0 -19
  374. package/src/strict-types.d.ts.map +0 -1
  375. package/src/strict-types.js +0 -18
  376. package/src/strict-types.js.map +0 -1
  377. package/src/strings/de.d.ts +0 -3
  378. package/src/strings/de.d.ts.map +0 -1
  379. package/src/strings/de.js +0 -57
  380. package/src/strings/de.js.map +0 -1
  381. package/src/strings/en-GB.d.ts +0 -3
  382. package/src/strings/en-GB.d.ts.map +0 -1
  383. package/src/strings/en-GB.js +0 -57
  384. package/src/strings/en-GB.js.map +0 -1
  385. package/src/strings/en-US.d.ts +0 -3
  386. package/src/strings/en-US.d.ts.map +0 -1
  387. package/src/strings/en-US.js +0 -57
  388. package/src/strings/en-US.js.map +0 -1
  389. package/src/strings/es.d.ts +0 -3
  390. package/src/strings/es.d.ts.map +0 -1
  391. package/src/strings/es.js +0 -57
  392. package/src/strings/es.js.map +0 -1
  393. package/src/strings/fr.d.ts +0 -3
  394. package/src/strings/fr.d.ts.map +0 -1
  395. package/src/strings/fr.js +0 -57
  396. package/src/strings/fr.js.map +0 -1
  397. package/src/strings/ja.d.ts +0 -3
  398. package/src/strings/ja.d.ts.map +0 -1
  399. package/src/strings/ja.js +0 -57
  400. package/src/strings/ja.js.map +0 -1
  401. package/src/strings/uk.d.ts +0 -3
  402. package/src/strings/uk.d.ts.map +0 -1
  403. package/src/strings/uk.js +0 -57
  404. package/src/strings/uk.js.map +0 -1
  405. package/src/strings/zh-CN.d.ts +0 -3
  406. package/src/strings/zh-CN.d.ts.map +0 -1
  407. package/src/strings/zh-CN.js +0 -57
  408. package/src/strings/zh-CN.js.map +0 -1
  409. package/src/template.d.ts +0 -13
  410. package/src/template.d.ts.map +0 -1
  411. package/src/template.js +0 -40
  412. package/src/template.js.map +0 -1
  413. package/src/translation-engine.d.ts +0 -9
  414. package/src/translation-engine.d.ts.map +0 -1
  415. package/src/translation-engine.js +0 -3
  416. package/src/translation-engine.js.map +0 -1
  417. package/src/translation-request.d.ts +0 -10
  418. package/src/translation-request.d.ts.map +0 -1
  419. package/src/translation-request.js +0 -3
  420. package/src/translation-request.js.map +0 -1
  421. package/src/translation-response.d.ts +0 -9
  422. package/src/translation-response.d.ts.map +0 -1
  423. package/src/translation-response.js +0 -3
  424. package/src/translation-response.js.map +0 -1
  425. package/src/types/engine.d.ts +0 -47
  426. package/src/types/engine.d.ts.map +0 -1
  427. package/src/types/engine.js +0 -8
  428. package/src/types/engine.js.map +0 -1
  429. package/src/types/index.d.ts.map +0 -1
  430. package/src/types/index.js +0 -9
  431. package/src/types/index.js.map +0 -1
  432. package/src/types/plural-types.d.ts.map +0 -1
  433. package/src/types/plural-types.js +0 -39
  434. package/src/types/plural-types.js.map +0 -1
  435. package/src/types.d.ts +0 -96
  436. package/src/types.d.ts.map +0 -1
  437. package/src/types.js +0 -23
  438. package/src/types.js.map +0 -1
  439. package/src/utils/currency.d.ts +0 -81
  440. package/src/utils/currency.d.ts.map +0 -1
  441. package/src/utils/currency.js +0 -101
  442. package/src/utils/currency.js.map +0 -1
  443. package/src/utils/html-escape.d.ts +0 -22
  444. package/src/utils/html-escape.d.ts.map +0 -1
  445. package/src/utils/html-escape.js +0 -53
  446. package/src/utils/html-escape.js.map +0 -1
  447. package/src/utils/index.d.ts.map +0 -1
  448. package/src/utils/index.js +0 -12
  449. package/src/utils/index.js.map +0 -1
  450. package/src/utils/lru-cache.d.ts +0 -42
  451. package/src/utils/lru-cache.d.ts.map +0 -1
  452. package/src/utils/lru-cache.js +0 -73
  453. package/src/utils/lru-cache.js.map +0 -1
  454. package/src/utils/plural-helpers.d.ts.map +0 -1
  455. package/src/utils/plural-helpers.js +0 -35
  456. package/src/utils/plural-helpers.js.map +0 -1
  457. package/src/utils/safe-object.d.ts +0 -39
  458. package/src/utils/safe-object.d.ts.map +0 -1
  459. package/src/utils/safe-object.js.map +0 -1
  460. package/src/utils/string-utils.d.ts +0 -28
  461. package/src/utils/string-utils.d.ts.map +0 -1
  462. package/src/utils/string-utils.js +0 -63
  463. package/src/utils/string-utils.js.map +0 -1
  464. package/src/utils/timezone.d.ts +0 -50
  465. package/src/utils/timezone.d.ts.map +0 -1
  466. package/src/utils/timezone.js +0 -74
  467. package/src/utils/timezone.js.map +0 -1
  468. package/src/utils/validation.d.ts +0 -40
  469. package/src/utils/validation.d.ts.map +0 -1
  470. package/src/utils/validation.js +0 -69
  471. package/src/utils/validation.js.map +0 -1
  472. package/src/utils.d.ts +0 -65
  473. package/src/utils.d.ts.map +0 -1
  474. package/src/utils.js +0 -129
  475. package/src/utils.js.map +0 -1
  476. package/src/validation/index.d.ts.map +0 -1
  477. package/src/validation/index.js +0 -5
  478. package/src/validation/index.js.map +0 -1
  479. package/src/validation/plural-validator.d.ts +0 -46
  480. package/src/validation/plural-validator.d.ts.map +0 -1
  481. package/src/validation/plural-validator.js +0 -123
  482. package/src/validation/plural-validator.js.map +0 -1
  483. package/src/validation-config.d.ts +0 -12
  484. package/src/validation-config.d.ts.map +0 -1
  485. package/src/validation-config.js +0 -3
  486. package/src/validation-config.js.map +0 -1
  487. package/src/validation-result.d.ts +0 -13
  488. package/src/validation-result.d.ts.map +0 -1
  489. package/src/validation-result.js +0 -3
  490. package/src/validation-result.js.map +0 -1
@@ -0,0 +1,242 @@
1
+ /**
2
+ * ICU MessageFormat Parser
3
+ */
4
+
5
+ import { Tokenizer, Token, TokenType } from './tokenizer';
6
+ import {
7
+ MessageNode,
8
+ LiteralNode,
9
+ ArgumentNode,
10
+ PluralNode,
11
+ SelectNode,
12
+ SelectOrdinalNode,
13
+ NodeType,
14
+ ASTNode,
15
+ } from './ast';
16
+
17
+ export class ParseError extends Error {
18
+ constructor(message: string, public position: number) {
19
+ super(`Parse error at position ${position}: ${message}`);
20
+ this.name = 'ParseError';
21
+ }
22
+ }
23
+
24
+ export class Parser {
25
+ private tokens: Token[];
26
+ private position: number = 0;
27
+ private depth: number = 0;
28
+ private readonly MAX_DEPTH = 10;
29
+
30
+ constructor(input: string) {
31
+ if (input.length > 10000) {
32
+ throw new ParseError('Input exceeds maximum length', 0);
33
+ }
34
+ this.tokens = new Tokenizer(input).tokenize();
35
+ }
36
+
37
+ parse(): MessageNode {
38
+ return this.parseMessage();
39
+ }
40
+
41
+ private parseMessage(): MessageNode {
42
+ if (++this.depth > this.MAX_DEPTH) {
43
+ throw new ParseError(`Maximum nesting depth of ${this.MAX_DEPTH} exceeded`, this.position);
44
+ }
45
+
46
+ const elements: ASTNode[] = [];
47
+
48
+ while (!this.isAtEnd() && !this.check(TokenType.CLOSE_BRACE)) {
49
+ if (this.check(TokenType.TEXT)) {
50
+ elements.push(this.parseLiteral());
51
+ } else if (this.check(TokenType.OPEN_BRACE)) {
52
+ elements.push(this.parseArgument());
53
+ } else if (this.check(TokenType.HASH)) {
54
+ // # is treated as literal in message context
55
+ this.advance();
56
+ elements.push({ type: NodeType.LITERAL, value: '#' });
57
+ } else {
58
+ this.advance(); // Skip unexpected tokens
59
+ }
60
+ }
61
+
62
+ this.depth--;
63
+ return { type: NodeType.MESSAGE, elements };
64
+ }
65
+
66
+ private parseLiteral(): LiteralNode {
67
+ const token = this.advance();
68
+ return { type: NodeType.LITERAL, value: token.value };
69
+ }
70
+
71
+ private parseArgument(): ArgumentNode | PluralNode | SelectNode | SelectOrdinalNode {
72
+ this.consume(TokenType.OPEN_BRACE, 'Expected {');
73
+
74
+ const name = this.consume(TokenType.IDENTIFIER, 'Expected argument name').value;
75
+
76
+ // Simple argument: {name}
77
+ if (this.check(TokenType.CLOSE_BRACE)) {
78
+ this.advance();
79
+ return { type: NodeType.ARGUMENT, name };
80
+ }
81
+
82
+ // Formatted argument: {name, type} or {name, type, style}
83
+ this.consume(TokenType.COMMA, 'Expected ,');
84
+
85
+ const format = this.consume(TokenType.IDENTIFIER, 'Expected format type').value;
86
+
87
+ // Check for plural/select/selectordinal
88
+ if (format === 'plural') {
89
+ return this.parsePlural(name);
90
+ }
91
+ if (format === 'select') {
92
+ return this.parseSelect(name);
93
+ }
94
+ if (format === 'selectordinal') {
95
+ return this.parseSelectOrdinal(name);
96
+ }
97
+
98
+ // Regular formatted argument: {name, type, style}
99
+ let style: string | undefined;
100
+ if (this.check(TokenType.COMMA)) {
101
+ this.advance();
102
+ style = this.consume(TokenType.IDENTIFIER, 'Expected style').value;
103
+ }
104
+
105
+ this.consume(TokenType.CLOSE_BRACE, 'Expected }');
106
+
107
+ return { type: NodeType.ARGUMENT, name, format, style };
108
+ }
109
+
110
+ private parsePlural(name: string): PluralNode {
111
+ this.consume(TokenType.COMMA, 'Expected ,');
112
+ this.skipWhitespace();
113
+
114
+ const cases: Record<string, MessageNode> = {};
115
+ let offset: number | undefined;
116
+
117
+ // Parse offset if present
118
+ if (this.check(TokenType.IDENTIFIER) && this.peek().value === 'offset') {
119
+ this.advance(); // offset
120
+ this.skipWhitespace();
121
+ this.consume(TokenType.IDENTIFIER, 'Expected offset value'); // number
122
+ offset = 0; // Simplified for now
123
+ this.skipWhitespace();
124
+ }
125
+
126
+ // Parse cases
127
+ while (!this.check(TokenType.CLOSE_BRACE) && !this.isAtEnd()) {
128
+ this.skipWhitespace();
129
+ if (this.check(TokenType.CLOSE_BRACE)) break;
130
+
131
+ const caseKey = this.consume(TokenType.IDENTIFIER, 'Expected case key').value;
132
+ this.skipWhitespace();
133
+ this.consume(TokenType.OPEN_BRACE, 'Expected {');
134
+ const caseMessage = this.parseMessage();
135
+ this.consume(TokenType.CLOSE_BRACE, 'Expected }');
136
+ cases[caseKey] = caseMessage;
137
+ }
138
+
139
+ if (Object.keys(cases).length === 0) {
140
+ throw new ParseError('Plural must have at least one case', this.peek().position);
141
+ }
142
+
143
+ this.consume(TokenType.CLOSE_BRACE, 'Expected }');
144
+
145
+ return { type: NodeType.PLURAL, name, offset, cases };
146
+ }
147
+
148
+ private parseSelect(name: string): SelectNode {
149
+ this.consume(TokenType.COMMA, 'Expected ,');
150
+ this.skipWhitespace();
151
+
152
+ const cases: Record<string, MessageNode> = {};
153
+
154
+ // Parse cases
155
+ while (!this.check(TokenType.CLOSE_BRACE) && !this.isAtEnd()) {
156
+ this.skipWhitespace();
157
+ if (this.check(TokenType.CLOSE_BRACE)) break;
158
+
159
+ const caseKey = this.consume(TokenType.IDENTIFIER, 'Expected case key').value;
160
+ this.skipWhitespace();
161
+ this.consume(TokenType.OPEN_BRACE, 'Expected {');
162
+ const caseMessage = this.parseMessage();
163
+ this.consume(TokenType.CLOSE_BRACE, 'Expected }');
164
+ cases[caseKey] = caseMessage;
165
+ }
166
+
167
+ this.consume(TokenType.CLOSE_BRACE, 'Expected }');
168
+
169
+ return { type: NodeType.SELECT, name, cases };
170
+ }
171
+
172
+ private parseSelectOrdinal(name: string): SelectOrdinalNode {
173
+ this.consume(TokenType.COMMA, 'Expected ,');
174
+ this.skipWhitespace();
175
+
176
+ const cases: Record<string, MessageNode> = {};
177
+ let offset: number | undefined;
178
+
179
+ // Parse offset if present
180
+ if (this.check(TokenType.IDENTIFIER) && this.peek().value === 'offset') {
181
+ this.advance(); // offset
182
+ this.skipWhitespace();
183
+ this.consume(TokenType.IDENTIFIER, 'Expected offset value'); // number
184
+ offset = 0; // Simplified for now
185
+ this.skipWhitespace();
186
+ }
187
+
188
+ // Parse cases
189
+ while (!this.check(TokenType.CLOSE_BRACE) && !this.isAtEnd()) {
190
+ this.skipWhitespace();
191
+ if (this.check(TokenType.CLOSE_BRACE)) break;
192
+
193
+ const caseKey = this.consume(TokenType.IDENTIFIER, 'Expected case key').value;
194
+ this.skipWhitespace();
195
+ this.consume(TokenType.OPEN_BRACE, 'Expected {');
196
+ const caseMessage = this.parseMessage();
197
+ this.consume(TokenType.CLOSE_BRACE, 'Expected }');
198
+ cases[caseKey] = caseMessage;
199
+ }
200
+
201
+ this.consume(TokenType.CLOSE_BRACE, 'Expected }');
202
+
203
+ return { type: NodeType.SELECTORDINAL, name, offset, cases };
204
+ }
205
+
206
+ private check(type: TokenType): boolean {
207
+ if (this.isAtEnd()) return false;
208
+ return this.peek().type === type;
209
+ }
210
+
211
+ private advance(): Token {
212
+ if (!this.isAtEnd()) this.position++;
213
+ return this.previous();
214
+ }
215
+
216
+ private isAtEnd(): boolean {
217
+ return this.peek().type === TokenType.EOF;
218
+ }
219
+
220
+ private peek(): Token {
221
+ return this.tokens[this.position];
222
+ }
223
+
224
+ private previous(): Token {
225
+ return this.tokens[this.position - 1];
226
+ }
227
+
228
+ private consume(type: TokenType, message: string): Token {
229
+ if (this.check(type)) return this.advance();
230
+ throw new ParseError(message, this.peek().position);
231
+ }
232
+
233
+ private skipWhitespace(): void {
234
+ while (this.check(TokenType.TEXT) && this.peek().value.trim() === '') {
235
+ this.advance();
236
+ }
237
+ }
238
+ }
239
+
240
+ export function parse(input: string): MessageNode {
241
+ return new Parser(input).parse();
242
+ }
@@ -0,0 +1,37 @@
1
+ import { parse } from './parser';
2
+ import { validate } from './validator';
3
+ import { Compiler, CompiledMessage } from './compiler';
4
+ import { FormatterRegistry } from './formatter-registry';
5
+ import { FormatterContext } from './formatters/base-formatter';
6
+ import { LRUCache } from '../utils/lru-cache';
7
+
8
+ export class Runtime {
9
+ private compiler: Compiler;
10
+ private cache = new LRUCache<string, CompiledMessage>(1000);
11
+
12
+ constructor(registry?: FormatterRegistry) {
13
+ this.compiler = new Compiler(registry);
14
+ }
15
+
16
+ format(message: string, values: Record<string, any>, context: FormatterContext): string {
17
+ if (message.length > 10000) {
18
+ throw new Error('Message exceeds maximum length');
19
+ }
20
+
21
+ const cacheKey = `${context.locale}:${message}`;
22
+ let compiled = this.cache.get(cacheKey);
23
+
24
+ if (!compiled) {
25
+ const ast = parse(message);
26
+ validate(ast);
27
+ compiled = this.compiler.compile(ast);
28
+ this.cache.set(cacheKey, compiled);
29
+ }
30
+
31
+ return compiled(values, context);
32
+ }
33
+
34
+ clearCache(): void {
35
+ this.cache.clear();
36
+ }
37
+ }
@@ -0,0 +1,212 @@
1
+ /**
2
+ * ICU MessageFormat Tokenizer
3
+ */
4
+
5
+ export enum TokenType {
6
+ TEXT = 'TEXT',
7
+ OPEN_BRACE = 'OPEN_BRACE',
8
+ CLOSE_BRACE = 'CLOSE_BRACE',
9
+ COMMA = 'COMMA',
10
+ HASH = 'HASH',
11
+ IDENTIFIER = 'IDENTIFIER',
12
+ EOF = 'EOF',
13
+ }
14
+
15
+ export interface Token {
16
+ type: TokenType;
17
+ value: string;
18
+ position: number;
19
+ }
20
+
21
+ export class Tokenizer {
22
+ private input: string;
23
+ private position: number = 0;
24
+ private length: number;
25
+ private braceDepth: number = 0;
26
+ private lastTokenType: TokenType | null = null;
27
+ private formatDepths: Set<number> = new Set(); // Tracks which depths are format specs
28
+ private lastIdentifierWasAtFormatDepth: boolean = false;
29
+
30
+ constructor(input: string) {
31
+ this.input = input;
32
+ this.length = input.length;
33
+ }
34
+
35
+ tokenize(): Token[] {
36
+ const tokens: Token[] = [];
37
+ while (this.position < this.length) {
38
+ const token = this.nextToken();
39
+ if (token.type !== TokenType.EOF) {
40
+ tokens.push(token);
41
+ }
42
+ }
43
+ tokens.push({ type: TokenType.EOF, value: '', position: this.position });
44
+ return tokens;
45
+ }
46
+
47
+ private nextToken(): Token {
48
+ if (this.position >= this.length) {
49
+ return { type: TokenType.EOF, value: '', position: this.position };
50
+ }
51
+
52
+ const char = this.input[this.position];
53
+
54
+ // Handle escape sequences - quoted braces
55
+ if (char === "'") {
56
+ const next = this.peek();
57
+ if (next === '{' || next === '}') {
58
+ const pos = this.position;
59
+ this.position++; // Skip quote
60
+ this.position++; // Skip brace
61
+ return { type: TokenType.TEXT, value: next, position: pos };
62
+ }
63
+ }
64
+
65
+ // Special characters
66
+ if (char === '{') {
67
+ const prevDepth = this.braceDepth;
68
+ this.braceDepth++;
69
+ // Add to format depths if:
70
+ // 1. Start of message (null)
71
+ // 2. After TEXT - new format in message (unless after case name)
72
+ // 3. After CLOSE_BRACE at depth 0 - consecutive formats
73
+ // 4. After OPEN_BRACE - nested format (unless after case name)
74
+ // NOT after IDENTIFIER at format depth - that's a case name, next brace is message content
75
+ if (this.lastTokenType === null ||
76
+ (this.lastTokenType === TokenType.TEXT && !this.lastIdentifierWasAtFormatDepth) ||
77
+ (this.lastTokenType === TokenType.CLOSE_BRACE && prevDepth === 0) ||
78
+ (this.lastTokenType === TokenType.OPEN_BRACE && !this.lastIdentifierWasAtFormatDepth)) {
79
+ this.formatDepths.add(this.braceDepth);
80
+ }
81
+ this.lastIdentifierWasAtFormatDepth = false;
82
+ const token = { type: TokenType.OPEN_BRACE, value: char, position: this.position++ };
83
+ this.lastTokenType = TokenType.OPEN_BRACE;
84
+ return token;
85
+ }
86
+ if (char === '}') {
87
+ this.formatDepths.delete(this.braceDepth);
88
+ this.braceDepth--;
89
+ this.lastIdentifierWasAtFormatDepth = false; // Reset after closing
90
+ const token = { type: TokenType.CLOSE_BRACE, value: char, position: this.position++ };
91
+ this.lastTokenType = TokenType.CLOSE_BRACE;
92
+ return token;
93
+ }
94
+ if (char === ',') {
95
+ // Comma at depth 1+ is format separator, at depth 0 is text
96
+ if (this.braceDepth >= 1) {
97
+ const token = { type: TokenType.COMMA, value: char, position: this.position++ };
98
+ this.lastTokenType = TokenType.COMMA;
99
+ return token;
100
+ }
101
+ // At depth 0, comma is just text
102
+ return this.readText();
103
+ }
104
+ if (char === '#') {
105
+ return { type: TokenType.HASH, value: char, position: this.position++ };
106
+ }
107
+
108
+ // Whitespace - skip at format level OR after identifier at depth 1
109
+ if (this.isWhitespace(char) && (this.isAtFormatLevel() || (this.braceDepth === 1 && this.lastTokenType === TokenType.IDENTIFIER))) {
110
+ this.position++;
111
+ return this.nextToken();
112
+ }
113
+
114
+ // Identifier (variable names, keywords) - at format level
115
+ if (this.isIdentifierStart(char)) {
116
+ if (this.isAtFormatLevel()) {
117
+ return this.readIdentifier();
118
+ }
119
+ }
120
+
121
+ // Text (everything else)
122
+ return this.readText();
123
+ }
124
+
125
+ private readIdentifier(): Token {
126
+ const start = this.position;
127
+ while (this.position < this.length && this.isIdentifierChar(this.input[this.position])) {
128
+ this.position++;
129
+ }
130
+ const token = {
131
+ type: TokenType.IDENTIFIER,
132
+ value: this.input.substring(start, this.position),
133
+ position: start,
134
+ };
135
+ this.lastTokenType = TokenType.IDENTIFIER;
136
+ this.lastIdentifierWasAtFormatDepth = this.formatDepths.has(this.braceDepth);
137
+ return token;
138
+ }
139
+
140
+ private readText(): Token {
141
+ const start = this.position;
142
+ let text = '';
143
+
144
+ while (this.position < this.length) {
145
+ const char = this.input[this.position];
146
+
147
+ // Handle escaped quotes - stop before them
148
+ if (char === "'" && (this.peek() === '{' || this.peek() === '}')) {
149
+ break;
150
+ }
151
+
152
+ // Stop at special characters (but not if they're escaped)
153
+ if (char === '{' || char === '#') {
154
+ break;
155
+ }
156
+
157
+ // For }, only stop if we're inside braces
158
+ if (char === '}' && this.braceDepth > 0) {
159
+ break;
160
+ }
161
+
162
+ // Stop at comma if we're at format level (depth >= 1)
163
+ if (char === ',' && this.braceDepth >= 1) {
164
+ break;
165
+ }
166
+
167
+ text += char;
168
+ this.position++;
169
+ }
170
+
171
+ const token = {
172
+ type: TokenType.TEXT,
173
+ value: text,
174
+ position: start,
175
+ };
176
+ this.lastTokenType = TokenType.TEXT;
177
+ // Don't reset the identifier flag for whitespace-only text
178
+ if (text.trim() !== '') {
179
+ this.lastIdentifierWasAtFormatDepth = false;
180
+ }
181
+ return token;
182
+ }
183
+
184
+ private peek(): string {
185
+ return this.position + 1 < this.length ? this.input[this.position + 1] : '';
186
+ }
187
+
188
+ private isIdentifierStart(char: string): boolean {
189
+ return /[a-zA-Z_]/.test(char);
190
+ }
191
+
192
+ private isIdentifierChar(char: string): boolean {
193
+ return /[a-zA-Z0-9_]/.test(char);
194
+ }
195
+
196
+ private isWhitespace(char: string): boolean {
197
+ return /\s/.test(char);
198
+ }
199
+
200
+ private isAtFormatLevel(): boolean {
201
+ // We're at format level if current depth is marked as format AND:
202
+ // 1. After OPEN_BRACE (expecting variable name)
203
+ // 2. After COMMA (expecting format type or case name)
204
+ // 3. After CLOSE_BRACE (just finished a case message, expecting another case name)
205
+ if (!this.formatDepths.has(this.braceDepth)) {
206
+ return false;
207
+ }
208
+ return this.lastTokenType === TokenType.OPEN_BRACE ||
209
+ this.lastTokenType === TokenType.COMMA ||
210
+ this.lastTokenType === TokenType.CLOSE_BRACE;
211
+ }
212
+ }
@@ -0,0 +1,163 @@
1
+ import { MessageNode, NodeType, ArgumentNode, PluralNode, SelectNode, SelectOrdinalNode } from './ast';
2
+
3
+ export class ValidationError extends Error {
4
+ constructor(message: string) {
5
+ super(message);
6
+ this.name = 'ValidationError';
7
+ }
8
+ }
9
+
10
+ export interface ValidationOptions {
11
+ requireOtherCase?: boolean;
12
+ allowCircularReferences?: boolean;
13
+ maxDepth?: number;
14
+ }
15
+
16
+ const DEFAULT_OPTIONS: ValidationOptions = {
17
+ requireOtherCase: true,
18
+ allowCircularReferences: false,
19
+ maxDepth: 10,
20
+ };
21
+
22
+ export function validate(ast: MessageNode, options: ValidationOptions = {}): void {
23
+ const opts = { ...DEFAULT_OPTIONS, ...options };
24
+ const context = new ValidationContext(opts);
25
+ validateMessage(ast, context);
26
+ }
27
+
28
+ class ValidationContext {
29
+ private argumentNames = new Set<string>();
30
+ private currentDepth = 0;
31
+
32
+ constructor(public options: ValidationOptions) {}
33
+
34
+ addArgument(name: string): void {
35
+ this.argumentNames.add(name);
36
+ }
37
+
38
+ hasArgument(name: string): boolean {
39
+ return this.argumentNames.has(name);
40
+ }
41
+
42
+ enterDepth(): void {
43
+ this.currentDepth++;
44
+ if (this.currentDepth > this.options.maxDepth!) {
45
+ throw new ValidationError(`Maximum nesting depth of ${this.options.maxDepth} exceeded`);
46
+ }
47
+ }
48
+
49
+ exitDepth(): void {
50
+ this.currentDepth--;
51
+ }
52
+ }
53
+
54
+ function validateMessage(node: MessageNode, context: ValidationContext): void {
55
+ for (const element of node.elements) {
56
+ validateNode(element, context);
57
+ }
58
+ }
59
+
60
+ function validateNode(node: any, context: ValidationContext): void {
61
+ switch (node.type) {
62
+ case NodeType.LITERAL:
63
+ break;
64
+ case NodeType.ARGUMENT:
65
+ validateArgument(node, context);
66
+ break;
67
+ case NodeType.PLURAL:
68
+ validatePlural(node, context);
69
+ break;
70
+ case NodeType.SELECT:
71
+ validateSelect(node, context);
72
+ break;
73
+ case NodeType.SELECTORDINAL:
74
+ validateSelectOrdinal(node, context);
75
+ break;
76
+ default:
77
+ throw new ValidationError(`Unknown node type: ${node.type}`);
78
+ }
79
+ }
80
+
81
+ function validateArgument(node: ArgumentNode, context: ValidationContext): void {
82
+ if (!node.name || node.name.trim() === '') {
83
+ throw new ValidationError('Argument name cannot be empty');
84
+ }
85
+ context.addArgument(node.name);
86
+ }
87
+
88
+ function validatePlural(node: PluralNode, context: ValidationContext): void {
89
+ if (!node.name || node.name.trim() === '') {
90
+ throw new ValidationError('Plural variable name cannot be empty');
91
+ }
92
+ context.addArgument(node.name);
93
+
94
+ if (!node.cases || Object.keys(node.cases).length === 0) {
95
+ throw new ValidationError(`Plural '${node.name}' must have at least one case`);
96
+ }
97
+
98
+ if (context.options.requireOtherCase && !node.cases.other) {
99
+ throw new ValidationError(`Plural '${node.name}' must have an 'other' case`);
100
+ }
101
+
102
+ const validCases = ['zero', 'one', 'two', 'few', 'many', 'other'];
103
+ for (const caseName of Object.keys(node.cases)) {
104
+ if (!validCases.includes(caseName)) {
105
+ throw new ValidationError(`Invalid plural case '${caseName}' in '${node.name}'`);
106
+ }
107
+ }
108
+
109
+ context.enterDepth();
110
+ for (const caseMessage of Object.values(node.cases)) {
111
+ validateMessage(caseMessage, context);
112
+ }
113
+ context.exitDepth();
114
+ }
115
+
116
+ function validateSelect(node: SelectNode, context: ValidationContext): void {
117
+ if (!node.name || node.name.trim() === '') {
118
+ throw new ValidationError('Select variable name cannot be empty');
119
+ }
120
+ context.addArgument(node.name);
121
+
122
+ if (!node.cases || Object.keys(node.cases).length === 0) {
123
+ throw new ValidationError(`Select '${node.name}' must have at least one case`);
124
+ }
125
+
126
+ if (context.options.requireOtherCase && !node.cases.other) {
127
+ throw new ValidationError(`Select '${node.name}' must have an 'other' case`);
128
+ }
129
+
130
+ context.enterDepth();
131
+ for (const caseMessage of Object.values(node.cases)) {
132
+ validateMessage(caseMessage, context);
133
+ }
134
+ context.exitDepth();
135
+ }
136
+
137
+ function validateSelectOrdinal(node: SelectOrdinalNode, context: ValidationContext): void {
138
+ if (!node.name || node.name.trim() === '') {
139
+ throw new ValidationError('SelectOrdinal variable name cannot be empty');
140
+ }
141
+ context.addArgument(node.name);
142
+
143
+ if (!node.cases || Object.keys(node.cases).length === 0) {
144
+ throw new ValidationError(`SelectOrdinal '${node.name}' must have at least one case`);
145
+ }
146
+
147
+ if (context.options.requireOtherCase && !node.cases.other) {
148
+ throw new ValidationError(`SelectOrdinal '${node.name}' must have an 'other' case`);
149
+ }
150
+
151
+ const validCases = ['zero', 'one', 'two', 'few', 'many', 'other'];
152
+ for (const caseName of Object.keys(node.cases)) {
153
+ if (!validCases.includes(caseName)) {
154
+ throw new ValidationError(`Invalid selectordinal case '${caseName}' in '${node.name}'`);
155
+ }
156
+ }
157
+
158
+ context.enterDepth();
159
+ for (const caseMessage of Object.values(node.cases)) {
160
+ validateMessage(caseMessage, context);
161
+ }
162
+ context.exitDepth();
163
+ }