@wener/common 2.0.2 → 2.0.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (351) hide show
  1. package/lib/ai/qwen3vl/index.js +2 -0
  2. package/lib/ai/qwen3vl/index.js.map +1 -0
  3. package/lib/ai/qwen3vl/utils.js +31 -0
  4. package/lib/ai/qwen3vl/utils.js.map +1 -0
  5. package/lib/ai/vision/DocLayoutElementTypeSchema.js +28 -0
  6. package/lib/ai/vision/DocLayoutElementTypeSchema.js.map +1 -0
  7. package/lib/ai/vision/ImageAnnotationSchema.js +50 -0
  8. package/lib/ai/vision/ImageAnnotationSchema.js.map +1 -0
  9. package/lib/ai/vision/index.js +3 -0
  10. package/lib/ai/vision/index.js.map +1 -0
  11. package/lib/ai/vision/resolveImageAnnotation.js +105 -0
  12. package/lib/ai/vision/resolveImageAnnotation.js.map +1 -0
  13. package/lib/cn/ChineseResidentIdNo.js +21 -14
  14. package/lib/cn/ChineseResidentIdNo.js.map +1 -0
  15. package/lib/cn/ChineseResidentIdNo.test.js +1 -1
  16. package/lib/cn/DivisionCode.js +30 -25
  17. package/lib/cn/DivisionCode.js.map +1 -0
  18. package/lib/cn/DivisionCode.test.js +1 -1
  19. package/lib/cn/Mod11.js +38 -81
  20. package/lib/cn/Mod11.js.map +1 -0
  21. package/lib/cn/Mod31.js +41 -90
  22. package/lib/cn/Mod31.js.map +1 -0
  23. package/lib/cn/UnifiedSocialCreditCode.js +43 -34
  24. package/lib/cn/UnifiedSocialCreditCode.js.map +1 -0
  25. package/lib/cn/UnifiedSocialCreditCode.test.js +1 -1
  26. package/lib/cn/formatChineseAmount.js +77 -0
  27. package/lib/cn/formatChineseAmount.js.map +1 -0
  28. package/lib/cn/index.js +7 -1
  29. package/lib/cn/index.js.map +1 -0
  30. package/lib/cn/parseChineseNumber.js +94 -0
  31. package/lib/cn/parseChineseNumber.js.map +1 -0
  32. package/lib/cn/parseChineseNumber.test.js +278 -0
  33. package/lib/cn/pinyin/cartesianProduct.js +22 -0
  34. package/lib/cn/pinyin/cartesianProduct.js.map +1 -0
  35. package/lib/cn/pinyin/cartesianProduct.test.js +179 -0
  36. package/lib/cn/pinyin/data.json +23573 -0
  37. package/lib/cn/pinyin/loader.js +14 -0
  38. package/lib/cn/pinyin/loader.js.map +1 -0
  39. package/lib/cn/pinyin/preload.js +3 -0
  40. package/lib/cn/pinyin/preload.js.map +1 -0
  41. package/lib/cn/pinyin/toPinyin.test.js +167 -0
  42. package/lib/cn/pinyin/toPinyinPure.js +33 -0
  43. package/lib/cn/pinyin/toPinyinPure.js.map +1 -0
  44. package/lib/cn/pinyin/transform.js +14 -0
  45. package/lib/cn/pinyin/transform.js.map +1 -0
  46. package/lib/cn/types.d.js +2 -0
  47. package/lib/cn/types.d.js.map +1 -0
  48. package/lib/consola/createStandardConsolaReporter.js +6 -6
  49. package/lib/consola/createStandardConsolaReporter.js.map +1 -0
  50. package/lib/consola/formatLogObject.js +65 -145
  51. package/lib/consola/formatLogObject.js.map +1 -0
  52. package/lib/consola/formatLogObject.test.js +184 -0
  53. package/lib/consola/index.js +1 -0
  54. package/lib/consola/index.js.map +1 -0
  55. package/lib/data/formatSort.js +6 -5
  56. package/lib/data/formatSort.js.map +1 -0
  57. package/lib/data/index.js +1 -0
  58. package/lib/data/index.js.map +1 -0
  59. package/lib/data/maybeNumber.js +5 -7
  60. package/lib/data/maybeNumber.js.map +1 -0
  61. package/lib/data/parseSort.js +22 -28
  62. package/lib/data/parseSort.js.map +1 -0
  63. package/lib/data/resolvePagination.js +13 -17
  64. package/lib/data/resolvePagination.js.map +1 -0
  65. package/lib/data/types.d.js +2 -0
  66. package/lib/data/types.d.js.map +1 -0
  67. package/lib/dayjs/dayjs.js +21 -19
  68. package/lib/dayjs/dayjs.js.map +1 -0
  69. package/lib/dayjs/formatDuration.js +15 -14
  70. package/lib/dayjs/formatDuration.js.map +1 -0
  71. package/lib/dayjs/index.js +2 -0
  72. package/lib/dayjs/index.js.map +1 -0
  73. package/lib/dayjs/parseDuration.js +5 -8
  74. package/lib/dayjs/parseDuration.js.map +1 -0
  75. package/lib/dayjs/parseRelativeTime.js +90 -0
  76. package/lib/dayjs/parseRelativeTime.js.map +1 -0
  77. package/lib/dayjs/parseRelativeTime.test.js +247 -0
  78. package/lib/dayjs/resolveRelativeTime.js +158 -0
  79. package/lib/dayjs/resolveRelativeTime.js.map +1 -0
  80. package/lib/dayjs/resolveRelativeTime.test.js +310 -0
  81. package/lib/decimal/index.js +1 -0
  82. package/lib/decimal/index.js.map +1 -0
  83. package/lib/decimal/parseDecimal.js +3 -1
  84. package/lib/decimal/parseDecimal.js.map +1 -0
  85. package/lib/emittery/emitter.js +10 -0
  86. package/lib/emittery/emitter.js.map +1 -0
  87. package/lib/emittery/index.js +2 -0
  88. package/lib/emittery/index.js.map +1 -0
  89. package/lib/foundation/schema/SexType.js +5 -3
  90. package/lib/foundation/schema/SexType.js.map +1 -0
  91. package/lib/foundation/schema/index.js +1 -0
  92. package/lib/foundation/schema/index.js.map +1 -0
  93. package/lib/foundation/schema/parseSexType.js +1 -0
  94. package/lib/foundation/schema/parseSexType.js.map +1 -0
  95. package/lib/foundation/schema/types.js +4 -2
  96. package/lib/foundation/schema/types.js.map +1 -0
  97. package/lib/fs/FileSystemError.js +23 -0
  98. package/lib/fs/FileSystemError.js.map +1 -0
  99. package/lib/fs/IFileSystem.d.js +3 -0
  100. package/lib/fs/IFileSystem.d.js.map +1 -0
  101. package/lib/fs/MemoryFileSystem.test.js +188 -0
  102. package/lib/fs/createBrowserFileSystem.js +248 -0
  103. package/lib/fs/createBrowserFileSystem.js.map +1 -0
  104. package/lib/fs/createMemoryFileSystem.js +516 -0
  105. package/lib/fs/createMemoryFileSystem.js.map +1 -0
  106. package/lib/fs/createSandboxFileSystem.js +108 -0
  107. package/lib/fs/createSandboxFileSystem.js.map +1 -0
  108. package/lib/fs/createWebDavFileSystem.js +137 -0
  109. package/lib/fs/createWebDavFileSystem.js.map +1 -0
  110. package/lib/fs/findMimeType.js +17 -0
  111. package/lib/fs/findMimeType.js.map +1 -0
  112. package/lib/fs/index.js +8 -0
  113. package/lib/fs/index.js.map +1 -0
  114. package/lib/fs/orpc/FileSystemContract.js +93 -0
  115. package/lib/fs/orpc/FileSystemContract.js.map +1 -0
  116. package/lib/fs/orpc/createContractClientFileSystem.js +93 -0
  117. package/lib/fs/orpc/createContractClientFileSystem.js.map +1 -0
  118. package/lib/fs/orpc/index.js +3 -0
  119. package/lib/fs/orpc/index.js.map +1 -0
  120. package/lib/fs/orpc/server/createFileSystemContractImpl.js +63 -0
  121. package/lib/fs/orpc/server/createFileSystemContractImpl.js.map +1 -0
  122. package/lib/fs/orpc/server/index.js +2 -0
  123. package/lib/fs/orpc/server/index.js.map +1 -0
  124. package/lib/fs/s3/createS3MiniFileSystem.js +705 -0
  125. package/lib/fs/s3/createS3MiniFileSystem.js.map +1 -0
  126. package/lib/fs/s3/index.js +2 -0
  127. package/lib/fs/s3/index.js.map +1 -0
  128. package/lib/fs/s3/s3mini.test.js +584 -0
  129. package/lib/fs/scandir.js +59 -0
  130. package/lib/fs/scandir.js.map +1 -0
  131. package/lib/fs/server/createDatabaseFileSystem.js +750 -0
  132. package/lib/fs/server/createDatabaseFileSystem.js.map +1 -0
  133. package/lib/fs/server/createNodeFileSystem.js +401 -0
  134. package/lib/fs/server/createNodeFileSystem.js.map +1 -0
  135. package/lib/fs/server/dbfs.test.js +221 -0
  136. package/lib/fs/server/index.js +2 -0
  137. package/lib/fs/server/index.js.map +1 -0
  138. package/lib/fs/server/loadTestDatabase.js +127 -0
  139. package/lib/fs/server/loadTestDatabase.js.map +1 -0
  140. package/lib/fs/tests/runFileSystemTest.js +318 -0
  141. package/lib/fs/tests/runFileSystemTest.js.map +1 -0
  142. package/lib/fs/types.js +27 -0
  143. package/lib/fs/types.js.map +1 -0
  144. package/lib/fs/utils/getFileUrl.js +35 -0
  145. package/lib/fs/utils/getFileUrl.js.map +1 -0
  146. package/lib/fs/utils.js +22 -0
  147. package/lib/fs/utils.js.map +1 -0
  148. package/lib/index.js +1 -0
  149. package/lib/index.js.map +1 -0
  150. package/lib/jsonschema/JsonSchema.js +146 -172
  151. package/lib/jsonschema/JsonSchema.js.map +1 -0
  152. package/lib/jsonschema/forEachJsonSchema.js +44 -0
  153. package/lib/jsonschema/forEachJsonSchema.js.map +1 -0
  154. package/lib/jsonschema/index.js +2 -0
  155. package/lib/jsonschema/index.js.map +1 -0
  156. package/lib/jsonschema/types.d.js +2 -0
  157. package/lib/jsonschema/types.d.js.map +1 -0
  158. package/lib/meta/defineFileType.js +20 -103
  159. package/lib/meta/defineFileType.js.map +1 -0
  160. package/lib/meta/defineInit.js +31 -250
  161. package/lib/meta/defineInit.js.map +1 -0
  162. package/lib/meta/defineMetadata.js +24 -140
  163. package/lib/meta/defineMetadata.js.map +1 -0
  164. package/lib/meta/index.js +1 -0
  165. package/lib/meta/index.js.map +1 -0
  166. package/lib/orpc/createOpenApiContractClient.js +27 -0
  167. package/lib/orpc/createOpenApiContractClient.js.map +1 -0
  168. package/lib/orpc/createRpcContractClient.js +34 -0
  169. package/lib/orpc/createRpcContractClient.js.map +1 -0
  170. package/lib/orpc/index.js +3 -0
  171. package/lib/orpc/index.js.map +1 -0
  172. package/lib/orpc/resolveLinkPlugins.js +28 -0
  173. package/lib/orpc/resolveLinkPlugins.js.map +1 -0
  174. package/lib/password/PHC.js +63 -87
  175. package/lib/password/PHC.js.map +1 -0
  176. package/lib/password/PHC.test.js +11 -3
  177. package/lib/password/Password.js +29 -294
  178. package/lib/password/Password.js.map +1 -0
  179. package/lib/password/Password.test.js +35 -22
  180. package/lib/password/createArgon2PasswordAlgorithm.js +35 -191
  181. package/lib/password/createArgon2PasswordAlgorithm.js.map +1 -0
  182. package/lib/password/createBase64PasswordAlgorithm.js +8 -141
  183. package/lib/password/createBase64PasswordAlgorithm.js.map +1 -0
  184. package/lib/password/createBcryptPasswordAlgorithm.js +13 -168
  185. package/lib/password/createBcryptPasswordAlgorithm.js.map +1 -0
  186. package/lib/password/createPBKDF2PasswordAlgorithm.js +46 -228
  187. package/lib/password/createPBKDF2PasswordAlgorithm.js.map +1 -0
  188. package/lib/password/createScryptPasswordAlgorithm.js +55 -211
  189. package/lib/password/createScryptPasswordAlgorithm.js.map +1 -0
  190. package/lib/password/index.js +1 -0
  191. package/lib/password/index.js.map +1 -0
  192. package/lib/password/server/index.js +1 -0
  193. package/lib/password/server/index.js.map +1 -0
  194. package/lib/resource/Identifiable.js +2 -0
  195. package/lib/resource/Identifiable.js.map +1 -0
  196. package/lib/resource/ListQuery.js +21 -93
  197. package/lib/resource/ListQuery.js.map +1 -0
  198. package/lib/resource/getTitleOfResource.js +3 -5
  199. package/lib/resource/getTitleOfResource.js.map +1 -0
  200. package/lib/resource/index.js +1 -0
  201. package/lib/resource/index.js.map +1 -0
  202. package/lib/resource/schema/AnyResourceSchema.js +2 -1
  203. package/lib/resource/schema/AnyResourceSchema.js.map +1 -0
  204. package/lib/resource/schema/BaseResourceSchema.js +2 -1
  205. package/lib/resource/schema/BaseResourceSchema.js.map +1 -0
  206. package/lib/resource/schema/ResourceActionType.js +6 -4
  207. package/lib/resource/schema/ResourceActionType.js.map +1 -0
  208. package/lib/resource/schema/ResourceStatus.js +5 -3
  209. package/lib/resource/schema/ResourceStatus.js.map +1 -0
  210. package/lib/resource/schema/ResourceType.js +5 -3
  211. package/lib/resource/schema/ResourceType.js.map +1 -0
  212. package/lib/resource/schema/index.js +1 -0
  213. package/lib/resource/schema/index.js.map +1 -0
  214. package/lib/resource/schema/types.js +16 -20
  215. package/lib/resource/schema/types.js.map +1 -0
  216. package/lib/s3/formatS3Url.js +65 -0
  217. package/lib/s3/formatS3Url.js.map +1 -0
  218. package/lib/s3/formatS3Url.test.js +262 -0
  219. package/lib/s3/index.js +3 -0
  220. package/lib/s3/index.js.map +1 -0
  221. package/lib/s3/parseS3Url.js +65 -0
  222. package/lib/s3/parseS3Url.js.map +1 -0
  223. package/lib/s3/parseS3Url.test.js +270 -0
  224. package/lib/schema/SchemaRegistry.js +38 -38
  225. package/lib/schema/SchemaRegistry.js.map +1 -0
  226. package/lib/schema/TypeSchema.d.js +2 -0
  227. package/lib/schema/TypeSchema.d.js.map +1 -0
  228. package/lib/schema/createSchemaData.js +26 -125
  229. package/lib/schema/createSchemaData.js.map +1 -0
  230. package/lib/schema/findJsonSchemaByPath.js +13 -36
  231. package/lib/schema/findJsonSchemaByPath.js.map +1 -0
  232. package/lib/schema/formatZodError.js +140 -0
  233. package/lib/schema/formatZodError.js.map +1 -0
  234. package/lib/schema/formatZodError.test.js +196 -0
  235. package/lib/schema/getSchemaCache.js +5 -5
  236. package/lib/schema/getSchemaCache.js.map +1 -0
  237. package/lib/schema/getSchemaOptions.js +8 -11
  238. package/lib/schema/getSchemaOptions.js.map +1 -0
  239. package/lib/schema/index.js +2 -1
  240. package/lib/schema/index.js.map +1 -0
  241. package/lib/schema/toJsonSchema.js +47 -290
  242. package/lib/schema/toJsonSchema.js.map +1 -0
  243. package/lib/schema/validate.js +33 -45
  244. package/lib/schema/validate.js.map +1 -0
  245. package/lib/tools/generateSchema.js +39 -197
  246. package/lib/tools/generateSchema.js.map +1 -0
  247. package/lib/tools/renderJsonSchemaToMarkdownDoc.js +55 -143
  248. package/lib/tools/renderJsonSchemaToMarkdownDoc.js.map +1 -0
  249. package/lib/utils/buildBaseUrl.js +13 -0
  250. package/lib/utils/buildBaseUrl.js.map +1 -0
  251. package/lib/utils/buildRedactorFormSchema.js +59 -0
  252. package/lib/utils/buildRedactorFormSchema.js.map +1 -0
  253. package/lib/utils/getEstimateProcessTime.js +12 -11
  254. package/lib/utils/getEstimateProcessTime.js.map +1 -0
  255. package/lib/utils/index.js +3 -0
  256. package/lib/utils/index.js.map +1 -0
  257. package/lib/utils/resolveFeatureOptions.js +12 -0
  258. package/lib/utils/resolveFeatureOptions.js.map +1 -0
  259. package/package.json +61 -13
  260. package/src/ai/qwen3vl/index.ts +1 -0
  261. package/src/ai/qwen3vl/utils.ts +36 -0
  262. package/src/ai/vision/DocLayoutElementTypeSchema.ts +30 -0
  263. package/src/ai/vision/ImageAnnotationSchema.ts +60 -0
  264. package/src/ai/vision/index.ts +2 -0
  265. package/src/ai/vision/resolveImageAnnotation.ts +135 -0
  266. package/src/cn/ChineseResidentIdNo.test.ts +1 -1
  267. package/src/cn/ChineseResidentIdNo.ts +8 -0
  268. package/src/cn/DivisionCode.test.ts +1 -1
  269. package/src/cn/DivisionCode.ts +8 -0
  270. package/src/cn/UnifiedSocialCreditCode.test.ts +1 -1
  271. package/src/cn/UnifiedSocialCreditCode.ts +15 -0
  272. package/src/cn/__snapshots__/UnifiedSocialCreditCode.test.ts.snap +23 -0
  273. package/src/cn/formatChineseAmount.ts +61 -0
  274. package/src/cn/index.ts +7 -1
  275. package/src/cn/parseChineseNumber.test.ts +159 -0
  276. package/src/cn/parseChineseNumber.ts +97 -0
  277. package/src/cn/pinyin/cartesianProduct.test.ts +64 -0
  278. package/src/cn/pinyin/cartesianProduct.ts +24 -0
  279. package/src/cn/pinyin/data.json +23573 -0
  280. package/src/cn/pinyin/loader.ts +12 -0
  281. package/src/cn/pinyin/preload.ts +3 -0
  282. package/src/cn/pinyin/toPinyin.test.ts +12 -0
  283. package/src/cn/pinyin/toPinyinPure.ts +43 -0
  284. package/src/cn/pinyin/transform.ts +12 -0
  285. package/src/consola/formatLogObject.test.ts +27 -0
  286. package/src/consola/formatLogObject.ts +34 -6
  287. package/src/dayjs/dayjs.ts +18 -18
  288. package/src/dayjs/index.ts +3 -1
  289. package/src/dayjs/parseRelativeTime.test.ts +185 -0
  290. package/src/dayjs/parseRelativeTime.ts +115 -0
  291. package/src/dayjs/resolveRelativeTime.test.ts +357 -0
  292. package/src/dayjs/resolveRelativeTime.ts +167 -0
  293. package/src/emittery/emitter.ts +9 -0
  294. package/src/emittery/index.ts +1 -0
  295. package/src/fs/FileSystemError.ts +26 -0
  296. package/src/fs/IFileSystem.d.ts +102 -0
  297. package/src/fs/MemoryFileSystem.test.ts +37 -0
  298. package/src/fs/createBrowserFileSystem.ts +291 -0
  299. package/src/fs/createMemoryFileSystem.ts +604 -0
  300. package/src/fs/createSandboxFileSystem.ts +136 -0
  301. package/src/fs/createWebDavFileSystem.ts +172 -0
  302. package/src/fs/findMimeType.ts +23 -0
  303. package/src/fs/index.ts +8 -0
  304. package/src/fs/orpc/FileSystemContract.ts +92 -0
  305. package/src/fs/orpc/createContractClientFileSystem.ts +115 -0
  306. package/src/fs/orpc/index.ts +2 -0
  307. package/src/fs/orpc/server/createFileSystemContractImpl.ts +64 -0
  308. package/src/fs/orpc/server/index.ts +1 -0
  309. package/src/fs/s3/createS3MiniFileSystem.ts +830 -0
  310. package/src/fs/s3/index.ts +1 -0
  311. package/src/fs/s3/s3mini.test.ts +264 -0
  312. package/src/fs/scandir.ts +75 -0
  313. package/src/fs/server/createDatabaseFileSystem.ts +668 -0
  314. package/src/fs/server/createNodeFileSystem.ts +499 -0
  315. package/src/fs/server/dbfs.test.ts +47 -0
  316. package/src/fs/server/index.ts +1 -0
  317. package/src/fs/server/loadTestDatabase.ts +131 -0
  318. package/src/fs/tests/runFileSystemTest.ts +288 -0
  319. package/src/fs/types.ts +29 -0
  320. package/src/fs/utils/getFileUrl.ts +44 -0
  321. package/src/fs/utils.ts +23 -0
  322. package/src/jsonschema/JsonSchema.ts +118 -110
  323. package/src/jsonschema/forEachJsonSchema.ts +50 -0
  324. package/src/jsonschema/index.ts +1 -0
  325. package/src/orpc/createOpenApiContractClient.ts +52 -0
  326. package/src/orpc/createRpcContractClient.ts +50 -0
  327. package/src/orpc/index.ts +2 -0
  328. package/src/orpc/resolveLinkPlugins.ts +29 -0
  329. package/src/password/PHC.ts +3 -3
  330. package/src/password/Password.test.ts +1 -1
  331. package/src/password/createPBKDF2PasswordAlgorithm.ts +2 -2
  332. package/src/resource/schema/AnyResourceSchema.ts +16 -2
  333. package/src/s3/formatS3Url.test.ts +254 -0
  334. package/src/s3/formatS3Url.ts +84 -0
  335. package/src/s3/index.ts +2 -0
  336. package/src/s3/parseS3Url.test.ts +258 -0
  337. package/src/s3/parseS3Url.ts +88 -0
  338. package/src/schema/SchemaRegistry.ts +35 -33
  339. package/src/schema/formatZodError.test.ts +196 -0
  340. package/src/schema/formatZodError.ts +151 -0
  341. package/src/schema/getSchemaOptions.ts +2 -2
  342. package/src/schema/index.ts +1 -1
  343. package/src/utils/buildBaseUrl.ts +12 -0
  344. package/src/utils/buildRedactorFormSchema.ts +85 -0
  345. package/src/utils/index.ts +4 -0
  346. package/src/utils/resolveFeatureOptions.ts +14 -0
  347. package/src/cn/ChineseResidentIdNo.mod.ts +0 -7
  348. package/src/cn/DivisionCode.mod.ts +0 -7
  349. package/src/cn/UnifiedSocialCreditCode.mod.ts +0 -7
  350. package/src/cn/mod.ts +0 -3
  351. package/src/schema/SchemaRegistry.mod.ts +0 -1
@@ -0,0 +1,50 @@
1
+ import type { JsonSchemaDef } from './types';
2
+
3
+ export type VisitJsonSchemaContext = { path: string[]; parent?: JsonSchemaDef };
4
+
5
+ export function forEachJsonSchema(js: JsonSchemaDef, cb: (js: JsonSchemaDef, ctx: VisitJsonSchemaContext) => void) {
6
+ const ctx: VisitJsonSchemaContext = { path: [] };
7
+ const _visit = (
8
+ js: JsonSchemaDef,
9
+ _f: (js: JsonSchemaDef, ctx: VisitJsonSchemaContext) => void,
10
+ parent: JsonSchemaDef | undefined,
11
+ path: string[],
12
+ k?: string,
13
+ ) => {
14
+ if (!js) {
15
+ return;
16
+ }
17
+ ctx.path = path;
18
+ ctx.parent = parent;
19
+ _f(js, ctx);
20
+ if (js.properties) {
21
+ for (const [k, v] of Object.entries(js.properties)) {
22
+ if (v) {
23
+ _visit(v, _f, js, path.concat(k), 'properties');
24
+ }
25
+ }
26
+ } else if (js.items) {
27
+ // array should be `key[].subkey` or `key.*.subkey`?
28
+ if (Array.isArray(js.items)) {
29
+ for (const v of js.items) {
30
+ _visit(v, _f, js, path, 'items');
31
+ }
32
+ } else {
33
+ _visit(js.items, _f, js, path, 'items');
34
+ }
35
+ } else if (js.anyOf) {
36
+ for (const v of js.anyOf) {
37
+ _visit(v, _f, js, path, 'anyOf');
38
+ }
39
+ } else if (js.oneOf) {
40
+ for (const v of js.oneOf) {
41
+ _visit(v, _f, js, path, 'oneOf');
42
+ }
43
+ } else if (js.allOf) {
44
+ for (const v of js.allOf) {
45
+ _visit(v, _f, js, path, 'allOf');
46
+ }
47
+ }
48
+ };
49
+ _visit(js, cb, undefined, []);
50
+ }
@@ -1,2 +1,3 @@
1
1
  export { JsonSchema } from './JsonSchema';
2
2
  export type { JsonSchemaDef } from './types';
3
+ export { forEachJsonSchema } from './forEachJsonSchema';
@@ -0,0 +1,52 @@
1
+ import { createORPCClient, type ClientContext } from '@orpc/client';
2
+ import type { LinkFetchClientOptions } from '@orpc/client/fetch';
3
+ import type { BatchLinkPluginOptions, DedupeRequestsPluginOptions } from '@orpc/client/plugins';
4
+ import type { StandardLinkPlugin } from '@orpc/client/standard';
5
+ import type { AnyContractRouter, ContractRouterClient } from '@orpc/contract';
6
+ import type { JsonifiedClient } from '@orpc/openapi-client';
7
+ import { OpenAPILink } from '@orpc/openapi-client/fetch';
8
+ import type { PartialRequired } from '@wener/utils';
9
+ import { buildBaseUrl } from '../utils/buildBaseUrl';
10
+ import { resolveLinkPlugins } from './resolveLinkPlugins';
11
+
12
+ export function createOpenApiContractClient<TContract extends AnyContractRouter>(
13
+ contract: TContract,
14
+ options: PartialRequired<CreateContractClientOptions, 'url'>,
15
+ ): JsonifiedClient<ContractRouterClient<TContract, ClientContext>> {
16
+ let { url, baseUrl, apiKey, getApiKey, getHeaders } = options;
17
+ const baseHeaders = new Headers(options.headers);
18
+ if (apiKey) {
19
+ baseHeaders.set('Authorization', `Bearer ${apiKey}`);
20
+ }
21
+ url = buildBaseUrl(url, baseUrl);
22
+ return createORPCClient(
23
+ new OpenAPILink(contract, {
24
+ url,
25
+ headers: () => {
26
+ let headers = new Headers(baseHeaders);
27
+
28
+ let accessToken = getApiKey?.();
29
+ if (accessToken) {
30
+ headers.set('Authorization', `Bearer ${accessToken}`);
31
+ }
32
+ headers = getHeaders?.(headers) ?? headers;
33
+ return headers;
34
+ },
35
+ plugins: resolveLinkPlugins(options),
36
+ fetch: options.fetch,
37
+ }),
38
+ ) as JsonifiedClient<ContractRouterClient<TContract, ClientContext>>;
39
+ }
40
+
41
+ export type CreateContractClientOptions = {
42
+ url?: string;
43
+ baseUrl?: string;
44
+ apiKey?: string;
45
+ headers?: Record<string, string> | Headers;
46
+ getApiKey?: () => string | void;
47
+ getHeaders?: (headers: Headers) => Headers | void;
48
+ plugins?: StandardLinkPlugin<any>[];
49
+ fetch?: LinkFetchClientOptions<any>['fetch'];
50
+ batch?: BatchLinkPluginOptions<any> | boolean;
51
+ dedup?: DedupeRequestsPluginOptions<any> | boolean;
52
+ };
@@ -0,0 +1,50 @@
1
+ import { createORPCClient } from '@orpc/client';
2
+ import { RPCLink } from '@orpc/client/fetch';
3
+ import type { AnyContractRouter, ContractRouterClient } from '@orpc/contract';
4
+ import { buildBaseUrl } from '../utils/buildBaseUrl';
5
+ import type { CreateContractClientOptions } from './createOpenApiContractClient';
6
+ import { resolveLinkPlugins } from './resolveLinkPlugins';
7
+
8
+ export type CreateRpcContractClientOptions = CreateContractClientOptions;
9
+
10
+ export function createRpcContractClient<C extends AnyContractRouter>(
11
+ options: CreateRpcContractClientOptions = {},
12
+ ): ContractRouterClient<C> {
13
+ let {
14
+ url = '/api/rpc',
15
+ baseUrl = globalThis.location?.origin || 'http://localhost',
16
+ apiKey,
17
+ getApiKey,
18
+ getHeaders,
19
+ } = options;
20
+ url = buildBaseUrl(url, baseUrl);
21
+
22
+ const baseHeaders = new Headers(options.headers);
23
+ if (apiKey) {
24
+ baseHeaders.set('Authorization', `Bearer ${apiKey}`);
25
+ }
26
+
27
+ const plugins = resolveLinkPlugins({
28
+ dedup: true,
29
+ batch: true,
30
+ ...options,
31
+ });
32
+ const link = new RPCLink({
33
+ url,
34
+ headers: () => {
35
+ let headers = new Headers(baseHeaders);
36
+
37
+ let accessToken = getApiKey?.();
38
+ if (accessToken) {
39
+ headers.set('Authorization', `Bearer ${accessToken}`);
40
+ }
41
+ headers = getHeaders?.(headers) ?? headers;
42
+ return headers;
43
+ },
44
+ fetch: options.fetch,
45
+ plugins,
46
+ });
47
+
48
+ const client: ContractRouterClient<C> = createORPCClient(link);
49
+ return client;
50
+ }
@@ -0,0 +1,2 @@
1
+ export { createRpcContractClient } from './createRpcContractClient';
2
+ export { createOpenApiContractClient } from './createOpenApiContractClient';
@@ -0,0 +1,29 @@
1
+ import { BatchLinkPlugin, DedupeRequestsPlugin } from '@orpc/client/plugins';
2
+ import { resolveFeatureOptions } from '../utils/resolveFeatureOptions';
3
+ import type { CreateRpcContractClientOptions } from './createRpcContractClient';
4
+
5
+ export function resolveLinkPlugins({ plugins = [], dedup, batch }: CreateRpcContractClientOptions) {
6
+ batch = resolveFeatureOptions(batch, {
7
+ groups: [
8
+ {
9
+ condition: () => true,
10
+ context: {},
11
+ },
12
+ ],
13
+ });
14
+ if (batch) {
15
+ plugins.push(new BatchLinkPlugin(batch));
16
+ }
17
+ dedup = resolveFeatureOptions(dedup, {
18
+ groups: [
19
+ {
20
+ condition: () => true,
21
+ context: {},
22
+ },
23
+ ],
24
+ });
25
+ if (dedup) {
26
+ plugins.push(new DedupeRequestsPlugin(dedup));
27
+ }
28
+ return plugins;
29
+ }
@@ -103,7 +103,7 @@ export namespace PHC {
103
103
  if (typeof value === 'number') {
104
104
  opts.params![k] = value.toString();
105
105
  } else if (value instanceof Uint8Array) {
106
- opts.params![k] = toBase64(value).split('=')[0];
106
+ opts.params![k] = toBase64(value as BufferSource).split('=')[0];
107
107
  }
108
108
  });
109
109
  const pv = objectValues(opts.params);
@@ -125,7 +125,7 @@ export namespace PHC {
125
125
  throw new TypeError('salt must be a Buffer');
126
126
  }
127
127
 
128
- fields.push(toBase64(opts.salt).split('=')[0]);
128
+ fields.push(toBase64(opts.salt as BufferSource).split('=')[0]);
129
129
 
130
130
  if (typeof opts.hash !== 'undefined') {
131
131
  // Hash Validation
@@ -133,7 +133,7 @@ export namespace PHC {
133
133
  throw new TypeError('hash must be a Buffer');
134
134
  }
135
135
 
136
- fields.push(toBase64(opts.hash).split('=')[0]);
136
+ fields.push(toBase64(opts.hash as BufferSource).split('=')[0]);
137
137
  }
138
138
  }
139
139
 
@@ -28,7 +28,7 @@ describe('Password', () => {
28
28
 
29
29
  await check({ algorithm: '5' });
30
30
  await check({ algorithm: '6' });
31
- await check({ algorithm: '7' });
31
+ // await check({ algorithm: '7' });
32
32
  });
33
33
 
34
34
  test('should verify manual created hash', async () => {
@@ -33,7 +33,7 @@ export function createPBKDF2PasswordAlgorithm({
33
33
  'deriveBits',
34
34
  ]);
35
35
  let hash = await crypto.subtle.deriveBits(
36
- { name: 'PBKDF2', iterations: rounds, salt, hash: digest },
36
+ { name: 'PBKDF2', iterations: rounds, salt: salt as BufferSource, hash: digest },
37
37
  key,
38
38
  keylen * 8,
39
39
  );
@@ -51,7 +51,7 @@ export function createPBKDF2PasswordAlgorithm({
51
51
  'deriveBits',
52
52
  ]);
53
53
  let hash = await crypto.subtle.deriveBits(
54
- { name: 'PBKDF2', iterations: rounds, salt, hash: digest },
54
+ { name: 'PBKDF2', iterations: rounds, salt: salt as BufferSource, hash: digest },
55
55
  key,
56
56
  storedHash.length * 8,
57
57
  );
@@ -2,9 +2,23 @@ import { z } from 'zod/v4';
2
2
  import { SexTypeSchema } from '../../foundation/schema/SexType';
3
3
  import { rz } from './types';
4
4
 
5
- export type AnyResource = z.infer<typeof AnyResourceSchema>;
5
+ export type AnyResource = z.infer<typeof AnyResourceSchema> & {
6
+ owner?: AnyResource;
7
+
8
+ entity?: AnyResource;
9
+
10
+ customer?: AnyResource;
11
+ account?: AnyResource;
12
+ contact?: AnyResource;
13
+
14
+ user?: AnyResource;
15
+ createdBy?: AnyResource;
16
+ updatedBy?: AnyResource;
17
+ deletedBy?: AnyResource;
18
+ };
19
+
6
20
  export const AnyResourceSchema = z
7
- .object({
21
+ .looseObject({
8
22
  id: rz.resourceId.readonly().describe('ID'),
9
23
  uid: z.guid().nullish().readonly().describe('唯一ID'),
10
24
  tid: rz.resourceId.nullish().readonly().describe('租户ID'),
@@ -0,0 +1,254 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { formatS3Url } from './formatS3Url';
3
+ import type { ParsedS3Options } from './parseS3Url';
4
+
5
+ describe('formatS3Url', () => {
6
+ it('should format basic S3 URL with default endpoint', () => {
7
+ const options: ParsedS3Options = {
8
+ endpoint: 's3.amazonaws.com',
9
+ };
10
+ const result = formatS3Url(options);
11
+ expect(result).toBe('https://s3.amazonaws.com/');
12
+ });
13
+
14
+ it('should format S3 URL with custom endpoint', () => {
15
+ const options: ParsedS3Options = {
16
+ endpoint: 'custom-s3.com',
17
+ };
18
+ const result = formatS3Url(options);
19
+ expect(result).toBe('https://custom-s3.com/');
20
+ });
21
+
22
+ it('should format S3 URL with HTTP protocol', () => {
23
+ const options: ParsedS3Options = {
24
+ endpoint: 's3.amazonaws.com',
25
+ useSsl: false,
26
+ };
27
+ const result = formatS3Url(options);
28
+ expect(result).toBe('http://s3.amazonaws.com/');
29
+ });
30
+
31
+ it('should format S3 URL with port', () => {
32
+ const options: ParsedS3Options = {
33
+ endpoint: 's3.amazonaws.com',
34
+ port: 9000,
35
+ };
36
+ const result = formatS3Url(options);
37
+ expect(result).toBe('https://s3.amazonaws.com:9000/');
38
+ });
39
+
40
+ it('should format S3 URL with bucket in path style', () => {
41
+ const options: ParsedS3Options = {
42
+ endpoint: 's3.amazonaws.com',
43
+ bucket: 'my-bucket',
44
+ pathStyle: true,
45
+ };
46
+ const result = formatS3Url(options);
47
+ expect(result).toBe('https://s3.amazonaws.com/my-bucket/');
48
+ });
49
+
50
+ it('should format S3 URL with bucket in virtual hosted style', () => {
51
+ const options: ParsedS3Options = {
52
+ endpoint: 's3.amazonaws.com',
53
+ bucket: 'my-bucket',
54
+ pathStyle: false,
55
+ };
56
+ const result = formatS3Url(options);
57
+ expect(result).toBe('https://my-bucket.s3.amazonaws.com/');
58
+ });
59
+
60
+ it('should format S3 URL with credentials', () => {
61
+ const options: ParsedS3Options = {
62
+ endpoint: 's3.amazonaws.com',
63
+ accessKeyId: 'user',
64
+ secretAccessKey: 'pass',
65
+ };
66
+ const result = formatS3Url(options, { credentials: true });
67
+ expect(result).toBe('https://user:pass@s3.amazonaws.com/');
68
+ });
69
+
70
+ it('should format S3 URL with credentials and special characters', () => {
71
+ const options: ParsedS3Options = {
72
+ endpoint: 's3.amazonaws.com',
73
+ accessKeyId: 'user@example.com',
74
+ secretAccessKey: 'pass/word',
75
+ };
76
+ const result = formatS3Url(options, { credentials: true });
77
+ expect(result).toBe('https://user%40example.com:pass%2Fword@s3.amazonaws.com/');
78
+ });
79
+
80
+ it('should format S3 URL with region in query parameters', () => {
81
+ const options: ParsedS3Options = {
82
+ endpoint: 's3.amazonaws.com',
83
+ region: 'us-east-1',
84
+ };
85
+ const result = formatS3Url(options, { useParams: true });
86
+ expect(result).toBe('https://s3.amazonaws.com/?region=us-east-1');
87
+ });
88
+
89
+ it('should format S3 URL with pathStyle in query parameters', () => {
90
+ const options: ParsedS3Options = {
91
+ endpoint: 's3.amazonaws.com',
92
+ pathStyle: true,
93
+ };
94
+ const result = formatS3Url(options, { useParams: true });
95
+ expect(result).toBe('https://s3.amazonaws.com/?pathStyle=true');
96
+ });
97
+
98
+ it('should format S3 URL with all parameters', () => {
99
+ const options: ParsedS3Options = {
100
+ endpoint: 's3.amazonaws.com',
101
+ accessKeyId: 'user',
102
+ secretAccessKey: 'pass',
103
+ region: 'us-east-1',
104
+ port: 9000,
105
+ bucket: 'my-bucket',
106
+ pathStyle: true,
107
+ useSsl: true,
108
+ };
109
+ const result = formatS3Url(options, { credentials: true, useParams: true });
110
+ expect(result).toBe('https://user:pass@s3.amazonaws.com:9000/my-bucket/?region=us-east-1&pathStyle=true');
111
+ });
112
+
113
+ it('should handle full URL as endpoint', () => {
114
+ const options: ParsedS3Options = {
115
+ endpoint: 'https://custom-s3.com:9000',
116
+ bucket: 'my-bucket',
117
+ pathStyle: true,
118
+ };
119
+ const result = formatS3Url(options);
120
+ expect(result).toBe('https://custom-s3.com:9000/my-bucket/');
121
+ });
122
+
123
+ it('should handle complex virtual hosted style with existing bucket in hostname', () => {
124
+ const options: ParsedS3Options = {
125
+ endpoint: 'my-bucket.s3.amazonaws.com',
126
+ bucket: 'my-bucket',
127
+ pathStyle: false,
128
+ };
129
+ const result = formatS3Url(options);
130
+ expect(result).toBe('https://my-bucket.s3.amazonaws.com/');
131
+ });
132
+
133
+ it('should throw error for missing endpoint', () => {
134
+ const options = {} as ParsedS3Options;
135
+ expect(() => formatS3Url(options)).toThrow('Endpoint is required');
136
+ });
137
+
138
+ it('should handle endpoint without protocol by adding https://', () => {
139
+ const options: ParsedS3Options = {
140
+ endpoint: 'custom-s3.com',
141
+ };
142
+ const result = formatS3Url(options);
143
+ expect(result).toBe('https://custom-s3.com/');
144
+ });
145
+
146
+ it('should throw error for missing credentials when credentials=true', () => {
147
+ const options: ParsedS3Options = {
148
+ endpoint: 's3.amazonaws.com',
149
+ accessKeyId: 'user',
150
+ };
151
+ expect(() => formatS3Url(options, { credentials: true })).toThrow(
152
+ 'Access Key ID and Secret Access Key are required for credentials',
153
+ );
154
+ });
155
+
156
+ it('should throw error for invalid port', () => {
157
+ const options: ParsedS3Options = {
158
+ endpoint: 's3.amazonaws.com',
159
+ port: 99999,
160
+ };
161
+ expect(() => formatS3Url(options)).toThrow('Port must be a valid number between 1 and 65535');
162
+ });
163
+
164
+ it('should throw error for invalid bucket name', () => {
165
+ const options: ParsedS3Options = {
166
+ endpoint: 's3.amazonaws.com',
167
+ bucket: 'Invalid Bucket Name',
168
+ };
169
+ expect(() => formatS3Url(options)).toThrow('Invalid bucket name: Invalid Bucket Name');
170
+ });
171
+
172
+ it('should throw error for bucket name with consecutive dots', () => {
173
+ const options: ParsedS3Options = {
174
+ endpoint: 's3.amazonaws.com',
175
+ bucket: 'my..bucket',
176
+ };
177
+ expect(() => formatS3Url(options)).toThrow('Invalid bucket name: my..bucket');
178
+ });
179
+
180
+ it('should throw error for bucket name with hyphen at start', () => {
181
+ const options: ParsedS3Options = {
182
+ endpoint: 's3.amazonaws.com',
183
+ bucket: '-mybucket',
184
+ };
185
+ expect(() => formatS3Url(options)).toThrow('Invalid bucket name: -mybucket');
186
+ });
187
+
188
+ it('should throw error for bucket name with hyphen at end', () => {
189
+ const options: ParsedS3Options = {
190
+ endpoint: 's3.amazonaws.com',
191
+ bucket: 'mybucket-',
192
+ };
193
+ expect(() => formatS3Url(options)).toThrow('Invalid bucket name: mybucket-');
194
+ });
195
+
196
+ it('should handle bucket name with dot-hyphen (currently allowed)', () => {
197
+ const options: ParsedS3Options = {
198
+ endpoint: 's3.amazonaws.com',
199
+ bucket: 'my.bucket-name',
200
+ pathStyle: true,
201
+ };
202
+ const result = formatS3Url(options);
203
+ expect(result).toBe('https://s3.amazonaws.com/my.bucket-name/');
204
+ });
205
+
206
+ it('should throw error for bucket name that is too short', () => {
207
+ const options: ParsedS3Options = {
208
+ endpoint: 's3.amazonaws.com',
209
+ bucket: 'ab',
210
+ };
211
+ expect(() => formatS3Url(options)).toThrow('Invalid bucket name: ab');
212
+ });
213
+
214
+ it('should throw error for bucket name that is too long', () => {
215
+ const options: ParsedS3Options = {
216
+ endpoint: 's3.amazonaws.com',
217
+ bucket: 'a'.repeat(64),
218
+ };
219
+ expect(() => formatS3Url(options)).toThrow('Invalid bucket name: ' + 'a'.repeat(64));
220
+ });
221
+
222
+ it('should handle valid bucket names with various formats', () => {
223
+ const validBuckets = ['my-bucket', 'my.bucket', 'mybucket123', '123bucket', 'test-bucket-123'];
224
+
225
+ validBuckets.forEach((bucket) => {
226
+ const options: ParsedS3Options = {
227
+ endpoint: 's3.amazonaws.com',
228
+ bucket,
229
+ pathStyle: true,
230
+ };
231
+ expect(() => formatS3Url(options)).not.toThrow();
232
+ });
233
+ });
234
+
235
+ it('should handle pathname concatenation correctly in path style', () => {
236
+ const options: ParsedS3Options = {
237
+ endpoint: 'https://s3.amazonaws.com/path/to/something',
238
+ bucket: 'my-bucket',
239
+ pathStyle: true,
240
+ };
241
+ const result = formatS3Url(options);
242
+ expect(result).toBe('https://s3.amazonaws.com/my-bucket/path/to/something');
243
+ });
244
+
245
+ it('should handle URL with existing pathname in virtual hosted style', () => {
246
+ const options: ParsedS3Options = {
247
+ endpoint: 'https://s3.amazonaws.com/path/to/something',
248
+ bucket: 'my-bucket',
249
+ pathStyle: false,
250
+ };
251
+ const result = formatS3Url(options);
252
+ expect(result).toBe('https://my-bucket.s3.amazonaws.com/path/to/something');
253
+ });
254
+ });
@@ -0,0 +1,84 @@
1
+ import type { ParsedS3Options } from './parseS3Url';
2
+
3
+ function isValidBucketName(bucket: string): boolean {
4
+ const bucketRegex = /^[a-z0-9][a-z0-9.-]{1,61}[a-z0-9]$/;
5
+ return bucketRegex.test(bucket) && !bucket.includes('..') && !bucket.includes('.-') && !bucket.includes('-.');
6
+ }
7
+
8
+ export function formatS3Url(
9
+ o: ParsedS3Options,
10
+ {
11
+ credentials,
12
+ useParams,
13
+ }: {
14
+ credentials?: boolean;
15
+ useParams?: boolean;
16
+ } = {},
17
+ ): string {
18
+ if (!o || typeof o !== 'object') {
19
+ throw new Error('S3 options must be an object');
20
+ }
21
+
22
+ if (!o.endpoint) {
23
+ throw new Error('Endpoint is required');
24
+ }
25
+
26
+ let url: URL;
27
+ let ep = o.endpoint || 's3.amazonaws.com';
28
+ try {
29
+ if (URL.canParse(ep)) {
30
+ url = new URL(ep);
31
+ } else {
32
+ url = new URL(`https://${ep}`);
33
+ }
34
+ } catch (error) {
35
+ throw new Error(`Invalid endpoint: ${ep}`);
36
+ }
37
+
38
+ const { useSsl = true, region, pathStyle, port, accessKeyId, secretAccessKey, bucket } = o;
39
+ url.protocol = useSsl ? 'https:' : 'http:';
40
+
41
+ if (credentials) {
42
+ if (accessKeyId && secretAccessKey) {
43
+ url.username = encodeURIComponent(accessKeyId);
44
+ url.password = encodeURIComponent(secretAccessKey);
45
+ } else {
46
+ throw new Error('Access Key ID and Secret Access Key are required for credentials');
47
+ }
48
+ }
49
+
50
+ if (useParams) {
51
+ if (region) {
52
+ url.searchParams.set('region', region);
53
+ }
54
+ if (pathStyle !== undefined) {
55
+ url.searchParams.set('pathStyle', String(pathStyle));
56
+ }
57
+ }
58
+
59
+ if (port) {
60
+ const portNum = Number(port);
61
+ if (isNaN(portNum) || portNum < 1 || portNum > 65535) {
62
+ throw new Error('Port must be a valid number between 1 and 65535');
63
+ }
64
+ url.port = String(portNum);
65
+ }
66
+
67
+ if (bucket) {
68
+ if (!isValidBucketName(bucket)) {
69
+ throw new Error(`Invalid bucket name: ${bucket}`);
70
+ }
71
+
72
+ if (pathStyle) {
73
+ url.pathname = `/${bucket}${url.pathname}`;
74
+ } else {
75
+ // Check if bucket is already in hostname (virtual-hosted style)
76
+ const bucketPattern = new RegExp(`^${bucket.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')}\\.`);
77
+ if (!bucketPattern.test(url.hostname)) {
78
+ url.hostname = `${bucket}.${url.hostname}`;
79
+ }
80
+ }
81
+ }
82
+
83
+ return url.toString();
84
+ }
@@ -0,0 +1,2 @@
1
+ export { parseS3Url, type ParseS3UrlOptions } from './parseS3Url';
2
+ export { formatS3Url } from './formatS3Url';