@anjianshi/utils 2.9.1 → 3.0.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 (164) hide show
  1. package/README.md +10 -0
  2. package/eslint.config.cjs +33 -0
  3. package/package.json +19 -10
  4. package/publish-prepare.cjs +16 -0
  5. package/src/env-browser/device.ts +62 -0
  6. package/src/env-browser/global.ts +21 -0
  7. package/src/env-browser/load-script.ts +13 -0
  8. package/src/env-browser/logging.ts +58 -0
  9. package/src/env-browser/manage-vconsole.ts +54 -0
  10. package/src/env-node/crypto-random.ts +30 -0
  11. package/src/env-node/fs.ts +50 -0
  12. package/src/env-node/index.ts +5 -0
  13. package/src/env-node/logging/handlers.ts +190 -0
  14. package/src/env-node/logging/index.ts +16 -0
  15. package/{env-react/emotion-register-globals.d.ts → src/env-react/emotion-register-globals.ts} +5 -2
  16. package/src/env-react/emotion.tsx +42 -0
  17. package/src/env-react/hooks.ts +59 -0
  18. package/src/env-react/index.ts +1 -0
  19. package/src/env-react/react-register-globals.ts +53 -0
  20. package/src/env-service/controllers.ts +93 -0
  21. package/src/env-service/env-reader.ts +141 -0
  22. package/src/env-service/index.ts +6 -0
  23. package/src/env-service/prisma/adapt-logging.ts +39 -0
  24. package/src/env-service/prisma/extensions/exist.ts +21 -0
  25. package/src/env-service/prisma/extensions/find-and-count.ts +24 -0
  26. package/src/env-service/prisma/extensions/soft-delete.ts +162 -0
  27. package/src/env-service/prisma/extensions/with-transaction.ts +65 -0
  28. package/src/env-service/prisma/index.ts +6 -0
  29. package/src/env-service/prisma/transaction-contexted.ts +80 -0
  30. package/src/env-service/redis-cache.ts +142 -0
  31. package/src/env-service/tasks.ts +45 -0
  32. package/src/index.ts +4 -0
  33. package/src/init-dayjs.ts +8 -0
  34. package/src/lang/async.ts +47 -0
  35. package/src/lang/color.ts +119 -0
  36. package/src/lang/index.ts +8 -0
  37. package/src/lang/object.ts +39 -0
  38. package/src/lang/random.ts +25 -0
  39. package/src/lang/result.ts +78 -0
  40. package/src/lang/string.ts +95 -0
  41. package/src/lang/time.ts +19 -0
  42. package/{lang/types.d.ts → src/lang/types.ts} +43 -23
  43. package/src/logging/adapt.ts +49 -0
  44. package/src/logging/formatters.ts +23 -0
  45. package/src/logging/index.ts +106 -0
  46. package/src/md5.ts +318 -0
  47. package/src/safe-request.ts +193 -0
  48. package/src/url.ts +185 -0
  49. package/src/validators/array.ts +97 -0
  50. package/src/validators/base.ts +145 -0
  51. package/src/validators/boolean.ts +21 -0
  52. package/src/validators/datetime.ts +39 -0
  53. package/src/validators/factory.ts +244 -0
  54. package/src/validators/index.ts +9 -0
  55. package/src/validators/number.ts +54 -0
  56. package/src/validators/object.ts +101 -0
  57. package/src/validators/one-of.ts +33 -0
  58. package/src/validators/string.ts +72 -0
  59. package/env-browser/device.d.ts +0 -24
  60. package/env-browser/device.js +0 -50
  61. package/env-browser/global.d.ts +0 -10
  62. package/env-browser/global.js +0 -15
  63. package/env-browser/load-script.d.ts +0 -5
  64. package/env-browser/load-script.js +0 -13
  65. package/env-browser/logging.d.ts +0 -18
  66. package/env-browser/logging.js +0 -49
  67. package/env-browser/manage-vconsole.d.ts +0 -16
  68. package/env-browser/manage-vconsole.js +0 -38
  69. package/env-node/crypto-random.d.ts +0 -13
  70. package/env-node/crypto-random.js +0 -28
  71. package/env-node/fs.d.ts +0 -19
  72. package/env-node/fs.js +0 -48
  73. package/env-node/index.d.ts +0 -6
  74. package/env-node/index.js +0 -6
  75. package/env-node/logging/handlers.d.ts +0 -58
  76. package/env-node/logging/handlers.js +0 -154
  77. package/env-node/logging/index.d.ts +0 -11
  78. package/env-node/logging/index.js +0 -14
  79. package/env-node/safe-request.d.ts +0 -26
  80. package/env-node/safe-request.js +0 -40
  81. package/env-react/emotion-register-globals.js +0 -5
  82. package/env-react/emotion.d.ts +0 -20
  83. package/env-react/emotion.jsx +0 -34
  84. package/env-react/hooks.d.ts +0 -4
  85. package/env-react/hooks.js +0 -16
  86. package/env-react/index.d.ts +0 -1
  87. package/env-react/index.js +0 -1
  88. package/env-react/react-register-globals.d.ts +0 -21
  89. package/env-react/react-register-globals.js +0 -19
  90. package/env-service/controllers.d.ts +0 -30
  91. package/env-service/controllers.js +0 -41
  92. package/env-service/env-reader.d.ts +0 -55
  93. package/env-service/env-reader.js +0 -79
  94. package/env-service/index.d.ts +0 -6
  95. package/env-service/index.js +0 -6
  96. package/env-service/prisma/adapt-logging.d.ts +0 -21
  97. package/env-service/prisma/adapt-logging.js +0 -30
  98. package/env-service/prisma/extensions/exist.d.ts +0 -10
  99. package/env-service/prisma/extensions/exist.js +0 -16
  100. package/env-service/prisma/extensions/find-and-count.d.ts +0 -7
  101. package/env-service/prisma/extensions/find-and-count.js +0 -19
  102. package/env-service/prisma/extensions/soft-delete.d.ts +0 -52
  103. package/env-service/prisma/extensions/soft-delete.js +0 -123
  104. package/env-service/prisma/extensions/with-transaction.d.ts +0 -9
  105. package/env-service/prisma/extensions/with-transaction.js +0 -54
  106. package/env-service/prisma/index.d.ts +0 -6
  107. package/env-service/prisma/index.js +0 -6
  108. package/env-service/prisma/transaction-contexted.d.ts +0 -11
  109. package/env-service/prisma/transaction-contexted.js +0 -52
  110. package/env-service/redis-cache.d.ts +0 -39
  111. package/env-service/redis-cache.js +0 -116
  112. package/env-service/tasks.d.ts +0 -12
  113. package/env-service/tasks.js +0 -37
  114. package/index.d.ts +0 -3
  115. package/index.js +0 -3
  116. package/init-dayjs.d.ts +0 -2
  117. package/init-dayjs.js +0 -7
  118. package/lang/async.d.ts +0 -19
  119. package/lang/async.js +0 -34
  120. package/lang/color.d.ts +0 -37
  121. package/lang/color.js +0 -111
  122. package/lang/index.d.ts +0 -8
  123. package/lang/index.js +0 -8
  124. package/lang/may-success.d.ts +0 -40
  125. package/lang/may-success.js +0 -27
  126. package/lang/object.d.ts +0 -12
  127. package/lang/object.js +0 -41
  128. package/lang/random.d.ts +0 -13
  129. package/lang/random.js +0 -24
  130. package/lang/string.d.ts +0 -29
  131. package/lang/string.js +0 -92
  132. package/lang/time.d.ts +0 -10
  133. package/lang/time.js +0 -18
  134. package/lang/types.js +0 -28
  135. package/logging/adapt.d.ts +0 -10
  136. package/logging/adapt.js +0 -43
  137. package/logging/formatters.d.ts +0 -10
  138. package/logging/formatters.js +0 -22
  139. package/logging/index.d.ts +0 -45
  140. package/logging/index.js +0 -90
  141. package/md5.d.ts +0 -30
  142. package/md5.js +0 -308
  143. package/url.d.ts +0 -77
  144. package/url.js +0 -149
  145. package/validators/array.d.ts +0 -30
  146. package/validators/array.js +0 -47
  147. package/validators/base.d.ts +0 -82
  148. package/validators/base.js +0 -42
  149. package/validators/boolean.d.ts +0 -3
  150. package/validators/boolean.js +0 -22
  151. package/validators/datetime.d.ts +0 -12
  152. package/validators/datetime.js +0 -30
  153. package/validators/factory.d.ts +0 -70
  154. package/validators/factory.js +0 -121
  155. package/validators/index.d.ts +0 -9
  156. package/validators/index.js +0 -9
  157. package/validators/number.d.ts +0 -19
  158. package/validators/number.js +0 -26
  159. package/validators/object.d.ts +0 -28
  160. package/validators/object.js +0 -49
  161. package/validators/one-of.d.ts +0 -10
  162. package/validators/one-of.js +0 -15
  163. package/validators/string.d.ts +0 -22
  164. package/validators/string.js +0 -35
package/src/md5.ts ADDED
@@ -0,0 +1,318 @@
1
+ /* eslint-disable no-multi-assign */
2
+ /**
3
+ * MD5 算法来自:https://github.com/emn178/js-md5
4
+ */
5
+
6
+ export function md5(content: string | ArrayBuffer) {
7
+ const md5 = new MD5()
8
+ md5.update(content)
9
+ return md5.hex()
10
+ }
11
+
12
+ // -------------------------------------------------------------------
13
+
14
+ /**
15
+ * 使用方法:
16
+ * const md5 = new MD5()
17
+ * md5.update(xxx) // 对于大文件,可以拆分开,多次调用 md5.update()
18
+ * const hash = md5.hex()
19
+ */
20
+ export class MD5 {
21
+ private readonly buffer8: Uint8Array
22
+ private readonly blocks: Uint32Array
23
+
24
+ private h0 = 0
25
+ private h1 = 0
26
+ private h2 = 0
27
+ private h3 = 0
28
+ private start = 0
29
+ private bytes = 0
30
+ private hBytes = 0
31
+ private lastByteIndex = 0
32
+
33
+ private finalized = false
34
+ private hashed = false
35
+ private first = true
36
+
37
+ constructor() {
38
+ const buffer = new ArrayBuffer(68)
39
+ this.buffer8 = new Uint8Array(buffer)
40
+ this.blocks = new Uint32Array(buffer)
41
+ }
42
+
43
+ update(message: string | number[] | Uint8Array | ArrayBuffer) {
44
+ if (this.finalized) return
45
+
46
+ if (message instanceof ArrayBuffer) {
47
+ message = new Uint8Array(message)
48
+ }
49
+ if (typeof message !== 'string' && !Array.isArray(message) && !ArrayBuffer.isView(message)) {
50
+ throw new Error('input is invalid type')
51
+ }
52
+
53
+ const length = message.length
54
+ const { blocks, buffer8 } = this
55
+
56
+ let code: number
57
+ let index = 0
58
+ let i: number
59
+
60
+ while (index < length) {
61
+ if (this.hashed) {
62
+ this.hashed = false
63
+ blocks[0] = blocks[16]!
64
+ // prettier-ignore
65
+ blocks[16] = blocks[1] = blocks[2] = blocks[3] = blocks[4] = blocks[5] =
66
+ blocks[6] = blocks[7] = blocks[8] = blocks[9] = blocks[10] =
67
+ blocks[11] = blocks[12] = blocks[13] = blocks[14] = blocks[15] = 0
68
+ }
69
+
70
+ if (typeof message !== 'string') {
71
+ for (i = this.start; index < length && i < 64; ++index) {
72
+ buffer8[i++] = message[index]!
73
+ }
74
+ } else {
75
+ for (i = this.start; index < length && i < 64; ++index) {
76
+ code = message.charCodeAt(index)
77
+ if (code < 0x80) {
78
+ buffer8[i++] = code
79
+ } else if (code < 0x800) {
80
+ buffer8[i++] = 0xc0 | (code >> 6)
81
+ buffer8[i++] = 0x80 | (code & 0x3f)
82
+ } else if (code < 0xd800 || code >= 0xe000) {
83
+ buffer8[i++] = 0xe0 | (code >> 12)
84
+ buffer8[i++] = 0x80 | ((code >> 6) & 0x3f)
85
+ buffer8[i++] = 0x80 | (code & 0x3f)
86
+ } else {
87
+ code = 0x10000 + (((code & 0x3ff) << 10) | (message.charCodeAt(++index) & 0x3ff))
88
+ buffer8[i++] = 0xf0 | (code >> 18)
89
+ buffer8[i++] = 0x80 | ((code >> 12) & 0x3f)
90
+ buffer8[i++] = 0x80 | ((code >> 6) & 0x3f)
91
+ buffer8[i++] = 0x80 | (code & 0x3f)
92
+ }
93
+ }
94
+ }
95
+ this.lastByteIndex = i
96
+ this.bytes += i - this.start
97
+ if (i >= 64) {
98
+ this.start = i - 64
99
+ this.hash()
100
+ this.hashed = true
101
+ } else {
102
+ this.start = i
103
+ }
104
+ }
105
+ if (this.bytes > 4294967295) {
106
+ this.hBytes += (this.bytes / 4294967296) << 0
107
+ this.bytes = this.bytes % 4294967296
108
+ }
109
+ }
110
+
111
+ private finalize() {
112
+ if (this.finalized) return
113
+ this.finalized = true
114
+ const { blocks } = this
115
+ const i = this.lastByteIndex
116
+ const EXTRA = [128, 32768, 8388608, -2147483648]
117
+ blocks[i >> 2]! |= EXTRA[i & 3]!
118
+ if (i >= 56) {
119
+ if (!this.hashed) this.hash()
120
+ blocks[0] = blocks[16]!
121
+ // prettier-ignore
122
+ blocks[16] = blocks[1] = blocks[2] = blocks[3] = blocks[4] = blocks[5] =
123
+ blocks[6] = blocks[7] = blocks[8] = blocks[9] = blocks[10] =
124
+ blocks[11] = blocks[12] = blocks[13] = blocks[14] = blocks[15] = 0
125
+ }
126
+ blocks[14] = this.bytes << 3
127
+ blocks[15] = (this.hBytes << 3) | (this.bytes >>> 29)
128
+ this.hash()
129
+ }
130
+
131
+ private hash() {
132
+ let a: number, b: number, c: number, d: number, bc: number, da: number
133
+ const { blocks } = this
134
+
135
+ if (this.first) {
136
+ a = blocks[0]! - 680876937
137
+ a = (((a << 7) | (a >>> 25)) - 271733879) << 0
138
+ d = (-1732584194 ^ (a & 2004318071)) + blocks[1]! - 117830708
139
+ d = (((d << 12) | (d >>> 20)) + a) << 0
140
+ c = (-271733879 ^ (d & (a ^ -271733879))) + blocks[2]! - 1126478375
141
+ c = (((c << 17) | (c >>> 15)) + d) << 0
142
+ b = (a ^ (c & (d ^ a))) + blocks[3]! - 1316259209
143
+ b = (((b << 22) | (b >>> 10)) + c) << 0
144
+ } else {
145
+ a = this.h0
146
+ b = this.h1
147
+ c = this.h2
148
+ d = this.h3
149
+ a += (d ^ (b & (c ^ d))) + blocks[0]! - 680876936
150
+ a = (((a << 7) | (a >>> 25)) + b) << 0
151
+ d += (c ^ (a & (b ^ c))) + blocks[1]! - 389564586
152
+ d = (((d << 12) | (d >>> 20)) + a) << 0
153
+ c += (b ^ (d & (a ^ b))) + blocks[2]! + 606105819
154
+ c = (((c << 17) | (c >>> 15)) + d) << 0
155
+ b += (a ^ (c & (d ^ a))) + blocks[3]! - 1044525330
156
+ b = (((b << 22) | (b >>> 10)) + c) << 0
157
+ }
158
+
159
+ a += (d ^ (b & (c ^ d))) + blocks[4]! - 176418897
160
+ a = (((a << 7) | (a >>> 25)) + b) << 0
161
+ d += (c ^ (a & (b ^ c))) + blocks[5]! + 1200080426
162
+ d = (((d << 12) | (d >>> 20)) + a) << 0
163
+ c += (b ^ (d & (a ^ b))) + blocks[6]! - 1473231341
164
+ c = (((c << 17) | (c >>> 15)) + d) << 0
165
+ b += (a ^ (c & (d ^ a))) + blocks[7]! - 45705983
166
+ b = (((b << 22) | (b >>> 10)) + c) << 0
167
+ a += (d ^ (b & (c ^ d))) + blocks[8]! + 1770035416
168
+ a = (((a << 7) | (a >>> 25)) + b) << 0
169
+ d += (c ^ (a & (b ^ c))) + blocks[9]! - 1958414417
170
+ d = (((d << 12) | (d >>> 20)) + a) << 0
171
+ c += (b ^ (d & (a ^ b))) + blocks[10]! - 42063
172
+ c = (((c << 17) | (c >>> 15)) + d) << 0
173
+ b += (a ^ (c & (d ^ a))) + blocks[11]! - 1990404162
174
+ b = (((b << 22) | (b >>> 10)) + c) << 0
175
+ a += (d ^ (b & (c ^ d))) + blocks[12]! + 1804603682
176
+ a = (((a << 7) | (a >>> 25)) + b) << 0
177
+ d += (c ^ (a & (b ^ c))) + blocks[13]! - 40341101
178
+ d = (((d << 12) | (d >>> 20)) + a) << 0
179
+ c += (b ^ (d & (a ^ b))) + blocks[14]! - 1502002290
180
+ c = (((c << 17) | (c >>> 15)) + d) << 0
181
+ b += (a ^ (c & (d ^ a))) + blocks[15]! + 1236535329
182
+ b = (((b << 22) | (b >>> 10)) + c) << 0
183
+ a += (c ^ (d & (b ^ c))) + blocks[1]! - 165796510
184
+ a = (((a << 5) | (a >>> 27)) + b) << 0
185
+ d += (b ^ (c & (a ^ b))) + blocks[6]! - 1069501632
186
+ d = (((d << 9) | (d >>> 23)) + a) << 0
187
+ c += (a ^ (b & (d ^ a))) + blocks[11]! + 643717713
188
+ c = (((c << 14) | (c >>> 18)) + d) << 0
189
+ b += (d ^ (a & (c ^ d))) + blocks[0]! - 373897302
190
+ b = (((b << 20) | (b >>> 12)) + c) << 0
191
+ a += (c ^ (d & (b ^ c))) + blocks[5]! - 701558691
192
+ a = (((a << 5) | (a >>> 27)) + b) << 0
193
+ d += (b ^ (c & (a ^ b))) + blocks[10]! + 38016083
194
+ d = (((d << 9) | (d >>> 23)) + a) << 0
195
+ c += (a ^ (b & (d ^ a))) + blocks[15]! - 660478335
196
+ c = (((c << 14) | (c >>> 18)) + d) << 0
197
+ b += (d ^ (a & (c ^ d))) + blocks[4]! - 405537848
198
+ b = (((b << 20) | (b >>> 12)) + c) << 0
199
+ a += (c ^ (d & (b ^ c))) + blocks[9]! + 568446438
200
+ a = (((a << 5) | (a >>> 27)) + b) << 0
201
+ d += (b ^ (c & (a ^ b))) + blocks[14]! - 1019803690
202
+ d = (((d << 9) | (d >>> 23)) + a) << 0
203
+ c += (a ^ (b & (d ^ a))) + blocks[3]! - 187363961
204
+ c = (((c << 14) | (c >>> 18)) + d) << 0
205
+ b += (d ^ (a & (c ^ d))) + blocks[8]! + 1163531501
206
+ b = (((b << 20) | (b >>> 12)) + c) << 0
207
+ a += (c ^ (d & (b ^ c))) + blocks[13]! - 1444681467
208
+ a = (((a << 5) | (a >>> 27)) + b) << 0
209
+ d += (b ^ (c & (a ^ b))) + blocks[2]! - 51403784
210
+ d = (((d << 9) | (d >>> 23)) + a) << 0
211
+ c += (a ^ (b & (d ^ a))) + blocks[7]! + 1735328473
212
+ c = (((c << 14) | (c >>> 18)) + d) << 0
213
+ b += (d ^ (a & (c ^ d))) + blocks[12]! - 1926607734
214
+ b = (((b << 20) | (b >>> 12)) + c) << 0
215
+ bc = b ^ c
216
+ a += (bc ^ d) + blocks[5]! - 378558
217
+ a = (((a << 4) | (a >>> 28)) + b) << 0
218
+ d += (bc ^ a) + blocks[8]! - 2022574463
219
+ d = (((d << 11) | (d >>> 21)) + a) << 0
220
+ da = d ^ a
221
+ c += (da ^ b) + blocks[11]! + 1839030562
222
+ c = (((c << 16) | (c >>> 16)) + d) << 0
223
+ b += (da ^ c) + blocks[14]! - 35309556
224
+ b = (((b << 23) | (b >>> 9)) + c) << 0
225
+ bc = b ^ c
226
+ a += (bc ^ d) + blocks[1]! - 1530992060
227
+ a = (((a << 4) | (a >>> 28)) + b) << 0
228
+ d += (bc ^ a) + blocks[4]! + 1272893353
229
+ d = (((d << 11) | (d >>> 21)) + a) << 0
230
+ da = d ^ a
231
+ c += (da ^ b) + blocks[7]! - 155497632
232
+ c = (((c << 16) | (c >>> 16)) + d) << 0
233
+ b += (da ^ c) + blocks[10]! - 1094730640
234
+ b = (((b << 23) | (b >>> 9)) + c) << 0
235
+ bc = b ^ c
236
+ a += (bc ^ d) + blocks[13]! + 681279174
237
+ a = (((a << 4) | (a >>> 28)) + b) << 0
238
+ d += (bc ^ a) + blocks[0]! - 358537222
239
+ d = (((d << 11) | (d >>> 21)) + a) << 0
240
+ da = d ^ a
241
+ c += (da ^ b) + blocks[3]! - 722521979
242
+ c = (((c << 16) | (c >>> 16)) + d) << 0
243
+ b += (da ^ c) + blocks[6]! + 76029189
244
+ b = (((b << 23) | (b >>> 9)) + c) << 0
245
+ bc = b ^ c
246
+ a += (bc ^ d) + blocks[9]! - 640364487
247
+ a = (((a << 4) | (a >>> 28)) + b) << 0
248
+ d += (bc ^ a) + blocks[12]! - 421815835
249
+ d = (((d << 11) | (d >>> 21)) + a) << 0
250
+ da = d ^ a
251
+ c += (da ^ b) + blocks[15]! + 530742520
252
+ c = (((c << 16) | (c >>> 16)) + d) << 0
253
+ b += (da ^ c) + blocks[2]! - 995338651
254
+ b = (((b << 23) | (b >>> 9)) + c) << 0
255
+ a += (c ^ (b | ~d)) + blocks[0]! - 198630844
256
+ a = (((a << 6) | (a >>> 26)) + b) << 0
257
+ d += (b ^ (a | ~c)) + blocks[7]! + 1126891415
258
+ d = (((d << 10) | (d >>> 22)) + a) << 0
259
+ c += (a ^ (d | ~b)) + blocks[14]! - 1416354905
260
+ c = (((c << 15) | (c >>> 17)) + d) << 0
261
+ b += (d ^ (c | ~a)) + blocks[5]! - 57434055
262
+ b = (((b << 21) | (b >>> 11)) + c) << 0
263
+ a += (c ^ (b | ~d)) + blocks[12]! + 1700485571
264
+ a = (((a << 6) | (a >>> 26)) + b) << 0
265
+ d += (b ^ (a | ~c)) + blocks[3]! - 1894986606
266
+ d = (((d << 10) | (d >>> 22)) + a) << 0
267
+ c += (a ^ (d | ~b)) + blocks[10]! - 1051523
268
+ c = (((c << 15) | (c >>> 17)) + d) << 0
269
+ b += (d ^ (c | ~a)) + blocks[1]! - 2054922799
270
+ b = (((b << 21) | (b >>> 11)) + c) << 0
271
+ a += (c ^ (b | ~d)) + blocks[8]! + 1873313359
272
+ a = (((a << 6) | (a >>> 26)) + b) << 0
273
+ d += (b ^ (a | ~c)) + blocks[15]! - 30611744
274
+ d = (((d << 10) | (d >>> 22)) + a) << 0
275
+ c += (a ^ (d | ~b)) + blocks[6]! - 1560198380
276
+ c = (((c << 15) | (c >>> 17)) + d) << 0
277
+ b += (d ^ (c | ~a)) + blocks[13]! + 1309151649
278
+ b = (((b << 21) | (b >>> 11)) + c) << 0
279
+ a += (c ^ (b | ~d)) + blocks[4]! - 145523070
280
+ a = (((a << 6) | (a >>> 26)) + b) << 0
281
+ d += (b ^ (a | ~c)) + blocks[11]! - 1120210379
282
+ d = (((d << 10) | (d >>> 22)) + a) << 0
283
+ c += (a ^ (d | ~b)) + blocks[2]! + 718787259
284
+ c = (((c << 15) | (c >>> 17)) + d) << 0
285
+ b += (d ^ (c | ~a)) + blocks[9]! - 343485551
286
+ b = (((b << 21) | (b >>> 11)) + c) << 0
287
+
288
+ if (this.first) {
289
+ this.h0 = (a + 1732584193) << 0
290
+ this.h1 = (b - 271733879) << 0
291
+ this.h2 = (c - 1732584194) << 0
292
+ this.h3 = (d + 271733878) << 0
293
+ this.first = false
294
+ } else {
295
+ this.h0 = (this.h0 + a) << 0
296
+ this.h1 = (this.h1 + b) << 0
297
+ this.h2 = (this.h2 + c) << 0
298
+ this.h3 = (this.h3 + d) << 0
299
+ }
300
+ }
301
+
302
+ hex() {
303
+ this.finalize()
304
+
305
+ const { h0, h1, h2, h3 } = this
306
+ const HEX_CHARS = '0123456789abcdef'.split('')
307
+ return (
308
+ // prettier-ignore
309
+ HEX_CHARS[(h0 >> 4) & 0x0f]! + HEX_CHARS[h0 & 0x0f]! + HEX_CHARS[(h0 >> 12) & 0x0f]! + HEX_CHARS[(h0 >> 8) & 0x0f]! + HEX_CHARS[(h0 >> 20) & 0x0f]! +
310
+ HEX_CHARS[(h0 >> 16) & 0x0f]! + HEX_CHARS[(h0 >> 28) & 0x0f]! + HEX_CHARS[(h0 >> 24) & 0x0f]! + HEX_CHARS[(h1 >> 4) & 0x0f]! + HEX_CHARS[h1 & 0x0f]! +
311
+ HEX_CHARS[(h1 >> 12) & 0x0f]! + HEX_CHARS[(h1 >> 8) & 0x0f]! + HEX_CHARS[(h1 >> 20) & 0x0f]! + HEX_CHARS[(h1 >> 16) & 0x0f]! + HEX_CHARS[(h1 >> 28) & 0x0f]! +
312
+ HEX_CHARS[(h1 >> 24) & 0x0f]! + HEX_CHARS[(h2 >> 4) & 0x0f]! + HEX_CHARS[h2 & 0x0f]! + HEX_CHARS[(h2 >> 12) & 0x0f]! + HEX_CHARS[(h2 >> 8) & 0x0f]! +
313
+ HEX_CHARS[(h2 >> 20) & 0x0f]! + HEX_CHARS[(h2 >> 16) & 0x0f]! + HEX_CHARS[(h2 >> 28) & 0x0f]! + HEX_CHARS[(h2 >> 24) & 0x0f]! + HEX_CHARS[(h3 >> 4) & 0x0f]! +
314
+ HEX_CHARS[h3 & 0x0f]! + HEX_CHARS[(h3 >> 12) & 0x0f]! + HEX_CHARS[(h3 >> 8) & 0x0f]! + HEX_CHARS[(h3 >> 20) & 0x0f]! + HEX_CHARS[(h3 >> 16) & 0x0f]! +
315
+ HEX_CHARS[(h3 >> 28) & 0x0f]! + HEX_CHARS[(h3 >> 24) & 0x0f]!
316
+ )
317
+ }
318
+ }
@@ -0,0 +1,193 @@
1
+ import { sleep } from './lang/async.js'
2
+ import { failed, handleException, type Result } from './lang/result.js'
3
+ import { getLogger, type Logger } from './logging/index.js'
4
+ import { combineUrl } from './url.js'
5
+
6
+ export type { Options as RequestOptions, FormattedOptions as RequestFormattedOptions }
7
+
8
+ interface Options {
9
+ urlPrefix?: string
10
+ url?: string
11
+ query?: Record<string, string | number | undefined>
12
+ method?: string
13
+ headers?: Record<string, string>
14
+ body?: string | FormData | null
15
+
16
+ /**
17
+ * 向后端传递的数据。对于 GET 请求,会合并到 query 中;对于 POST 请求,会作为 POST body,代替 body 参数
18
+ * 注意:为了支持传入 interface 类型的值,Record 只能定义成 Record<string, any>
19
+ */
20
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
21
+ data?: FormData | Record<string, any>
22
+
23
+ /** 超时时间,不指定或设为 0 代表不限 */
24
+ timeout?: number
25
+ }
26
+
27
+ type FormattedOptions = Required<Pick<Options, 'url' | 'method' | 'headers' | 'body' | 'timeout'>>
28
+
29
+ type PredefinedOptions = Pick<Options, 'urlPrefix' | 'method' | 'headers' | 'timeout'>
30
+
31
+ /**
32
+ * 建立一个请求发起器,并可预设部分选项。
33
+ * 可以继承此类来自定义默认的错误处理逻辑。
34
+ *
35
+ * 请求失败时的 Failed 对象,其 code 为 HTTP status,没有 status 时为 0
36
+ * data 为解析出的响应内容,没有或解析失败则为 undefined
37
+ */
38
+ export class SafeRequestClient {
39
+ readonly logger: Logger
40
+ readonly prefefinedOptions: PredefinedOptions
41
+
42
+ constructor(options: PredefinedOptions & { loggerName?: string } = {}) {
43
+ this.logger = getLogger(options.loggerName ?? 'request')
44
+ this.prefefinedOptions = options
45
+ }
46
+
47
+ /** 生成一个快捷方式函数,调用它相当于调用 client.request() */
48
+ asFunction() {
49
+ return async <T>(inputUrl: string, inputOptions?: Options) =>
50
+ this.request<T>(inputUrl, inputOptions)
51
+ }
52
+
53
+ async request<T>(inputUrl: string, inputOptions?: Options): Promise<Result<T>> {
54
+ const options = await this.formatOptions({
55
+ url: inputUrl,
56
+ ...(inputOptions ?? {}),
57
+ })
58
+ const { url, method, headers, body, timeout } = options
59
+
60
+ try {
61
+ // 发起请求
62
+ const request = fetch(url, { method, headers, body })
63
+ let response: Response | undefined
64
+ try {
65
+ response = await (typeof timeout === 'number'
66
+ ? Promise.race([request, sleep(timeout)])
67
+ : request)
68
+ } catch (error) {
69
+ // 处理“请求发起失败”
70
+ return this.onRequestError(error as Error, url)
71
+ }
72
+
73
+ // 处理超时
74
+ if (response === undefined) {
75
+ return this.onTimeout(url)
76
+ }
77
+
78
+ // 处理“服务端返回失败状态”
79
+ if (!response.status.toString().startsWith('2')) {
80
+ // 此时服务端仍可能输出一些内容,试着解析出来
81
+ const responseDataRes = await this.parseResponse<T>(options, response)
82
+ const responseData = responseDataRes.success ? responseDataRes.data : undefined
83
+ return await this.onResponseError(url, response, responseData)
84
+ }
85
+
86
+ // 解析响应内容
87
+ return await this.parseResponse<T>(options, response)
88
+ } catch (error) {
89
+ this.logger.error('Unexpected error', error)
90
+ return failed('Request handle failed.')
91
+ }
92
+ }
93
+
94
+ async formatOptions(input: Options): Promise<FormattedOptions> {
95
+ const predefined = this.prefefinedOptions
96
+ const {
97
+ urlPrefix = predefined.urlPrefix ?? '',
98
+ url: rawUrl,
99
+ query = {},
100
+ method = predefined.method ?? 'GET',
101
+ headers: rawHeaders = {},
102
+ body: rawBody = null,
103
+ data,
104
+ timeout = predefined.timeout ?? 0,
105
+ } = input
106
+
107
+ const headers = {
108
+ ...(predefined.headers ?? {}),
109
+ ...rawHeaders,
110
+ }
111
+
112
+ let body: string | FormData | null = rawBody
113
+ if (data !== undefined) {
114
+ if (method === 'GET') {
115
+ Object.assign(query, data)
116
+ } else {
117
+ body = data instanceof FormData ? data : JSON.stringify(data)
118
+ headers['Content-Type'] = 'application/json; charset=utf-8'
119
+ }
120
+ }
121
+
122
+ const url = combineUrl(urlPrefix + (rawUrl ?? ''), query)
123
+
124
+ const options = {
125
+ method,
126
+ url,
127
+ headers,
128
+ body,
129
+ timeout,
130
+ }
131
+ Object.assign(options.headers, await this.getHeaders(options, input))
132
+ return options
133
+ }
134
+
135
+ /** 请求发起前调用此方法补充 Headers 内容 */
136
+ protected getHeaders(
137
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
138
+ options: FormattedOptions,
139
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
140
+ inputOptions: Options,
141
+ ): Record<string, string> | undefined | Promise<Record<string, string> | undefined> {
142
+ return undefined
143
+ }
144
+
145
+ protected async parseResponse<T>(options: FormattedOptions, response: Response) {
146
+ let result: Result<T>
147
+ result = await handleException(response.json())
148
+ if (result.success) return result
149
+
150
+ const contentType = (response.headers.get('Content-Type') ?? '').toLowerCase().trim()
151
+ if (contentType.startsWith('text/') || contentType === '') {
152
+ result = (await handleException(response.text())) as Result<T>
153
+ if (result.success) return result
154
+ }
155
+
156
+ result = (await handleException(response.arrayBuffer())) as Result<T>
157
+ return result
158
+ }
159
+
160
+ /** 若请求未成功发起,会触发此回调来生成失败信息 */
161
+ protected onRequestError(error: Error, url: string) {
162
+ this.logger.error('Request Failed', { url, error })
163
+ return failed('Request Failed', 0, undefined)
164
+ }
165
+
166
+ /** 请求成功发起,但服务端返回失败状态(如 500),会触发此回调来生成失败信息 */
167
+ // eslint-disable-next-line @typescript-eslint/require-await
168
+ protected async onResponseError(url: string, response: Response, responseData: unknown) {
169
+ this.logger.error('Response Error Status', {
170
+ url,
171
+ status: response.status,
172
+ data: responseData,
173
+ })
174
+ return failed(`Response Error Status - ${response.status}`, response.status, responseData)
175
+ }
176
+
177
+ /** 服务端返回内容解析失败时,会触发此回调来生成失败信息 */
178
+ protected onParseFailed(error: Error, response: Response, url: string) {
179
+ this.logger.error('Response Parse Failed', { url, response, error })
180
+ return failed('Response Parse Failed', response.status, undefined)
181
+ }
182
+
183
+ /** 处理超时 */
184
+ protected onTimeout(url: string) {
185
+ this.logger.warn('Request Timeout', url)
186
+ return failed('Request Timeout', 0, undefined)
187
+ }
188
+ }
189
+
190
+ /**
191
+ * 模块自带一个可直接调用发起请求的函数,跳过初始化实例
192
+ */
193
+ export const safeRequest = new SafeRequestClient().asFunction()
package/src/url.ts ADDED
@@ -0,0 +1,185 @@
1
+ /**
2
+ * URL 工具函数
3
+ * 部分灵感来自:https://www.npmjs.com/package/qs
4
+ *
5
+ * [名词定义]
6
+ * query: 指 a=1&b=2 格式的“查询字符串”或此类字符串的解析结果。
7
+ * search: URL 中的 search 部分,与 location.search 一致,空字符串或以 '?' 开头。
8
+ * hash: URL 中的 hash 部分,与 location.hash 一直,空字符串或以 '#' 开头。
9
+ */
10
+ import isPlainObject from 'lodash/isPlainObject.js'
11
+
12
+ /**
13
+ * 从 URL 中解析出 query 对象
14
+ * 注意:不带 ? 号的纯 query 内容需手动加上 ? 再传入。
15
+ *
16
+ * [array]
17
+ * 是否把重复出现的 key 保存为数组(默认不开启)
18
+ * a=1&a=2 => { a: [1,2] }
19
+ *
20
+ * [loose]
21
+ * 是否开启“宽松模式”(默认不开启)
22
+ * 1. hash 里的内容也会被解析,以兼容拼接错误的 URL(把 query 拼到了 hash 后面)。
23
+ * 2. 出现多个 ? 符号时,会把 ? 也当做 & 分隔符(index.html?a=1&b=2?c=3)
24
+ *
25
+ * [decode]
26
+ * 是否对 query 值进行 decode(默认开启)
27
+ */
28
+ function parseQuery(url: string, options?: { array?: false, loose?: boolean, decode?: boolean }): Record<string, string> // prettier-ignore
29
+ function parseQuery(url: string, options: { array: true, loose?: boolean, decode?: boolean }): Record<string, string | string[]> // prettier-ignore
30
+ function parseQuery(
31
+ url: string,
32
+ options?: { array?: boolean; loose?: boolean; decode?: boolean },
33
+ ): Record<string, string | string[]> {
34
+ const { array = false, loose = false, decode = true } = options ?? {}
35
+
36
+ // 正常状态下,将仅剩 a=1&b=1(即不会再有 ? 和 #);loose 模式下,可能为 a=1&b=2#c=3?d=4
37
+ const queryString = (loose ? /(\?|#)(.+)/ : /(\?)(.+?)(#|$)/).exec(url)?.[2] ?? ''
38
+ if (!queryString) return {}
39
+
40
+ const query: Record<string, string | string[]> = {}
41
+ const reg = /([^#?&]*)=([^#?&]*)/g
42
+ let re = reg.exec(queryString)
43
+ while (re) {
44
+ const [name, rawValue] = [re[1]!, re[2]!] as [string, string]
45
+ const value = decode ? safeDecode(rawValue) : rawValue
46
+ if (array && query[name] !== undefined) {
47
+ const prev = query[name]
48
+ query[name] = Array.isArray(prev) ? [...prev, value] : [prev, value]
49
+ } else {
50
+ query[name] = value
51
+ }
52
+ re = reg.exec(queryString)
53
+ }
54
+ return query
55
+ }
56
+ export { parseQuery }
57
+
58
+ /**
59
+ * 把对象合并成 query string。
60
+ * - 支持字符串、数值、布尔值、数组。
61
+ * - 布尔值会替换成 0 和 1。
62
+ * - 数组会多次赋值:{ a: [1,2,3] } => 'a=1&a=2&a=3',不支持嵌套数组
63
+ * - encode 为 true 时会对 value 执行 encodeURIComponent(默认为 true)
64
+ */
65
+ export type StringifyVal = string | number | boolean
66
+ export type StringifyQuery = Record<string, StringifyVal | StringifyVal[] | undefined>
67
+ export function stringifyQuery(obj: StringifyQuery, encode = true) {
68
+ if (!isPlainObject(obj)) return ''
69
+ return (
70
+ Object.entries(obj)
71
+ // 过滤值为 undefined 的项目,使其完全不出现在最终的 query 中
72
+ .filter((entry): entry is [string, StringifyVal | StringifyVal[]] => entry[1] !== undefined)
73
+ .map(([name, value]) => stringifyQueryItem(name, value, encode))
74
+ .join('&')
75
+ )
76
+ }
77
+ function stringifyQueryItem(
78
+ name: string,
79
+ value: StringifyVal | StringifyVal[],
80
+ encode: boolean,
81
+ ): string {
82
+ if (Array.isArray(value))
83
+ return value.map(subValue => stringifyQueryItem(name, subValue, encode)).join('&')
84
+ if (typeof value === 'boolean') value = value ? '1' : '0'
85
+ if (typeof value === 'number') value = value.toString()
86
+ if (encode) value = encodeURIComponent(value)
87
+ return `${name}=${value}`
88
+ }
89
+
90
+ /**
91
+ * 拆分 URL 的各个部分
92
+ *
93
+ * bare 为 true,则 search 不带 '?',hash 不带 '#'
94
+ * 否则和 location.search / hash 一样
95
+ * (默认为 true)
96
+ */
97
+ export function splitUrl(url: string, bare = true): { base: string; search: string; hash: string } {
98
+ let hashIndex = url.indexOf('#')
99
+ if (hashIndex === -1) hashIndex = url.length
100
+ const bareHash = url.slice(hashIndex + 1)
101
+
102
+ let searchIndex = url.slice(0, hashIndex).indexOf('?')
103
+ if (searchIndex === -1) searchIndex = hashIndex
104
+ const bareSearch = url.slice(searchIndex + 1, hashIndex)
105
+
106
+ return {
107
+ base: url.slice(0, searchIndex),
108
+ search: bare ? bareSearch : bareSearch ? '?' + bareSearch : '',
109
+ hash: bare ? bareHash : bareHash ? '#' + bareHash : '',
110
+ }
111
+ }
112
+
113
+ /**
114
+ * 把 query 和 hash 内容合并到 url 上
115
+ *
116
+ * query object 与现有 search 合并,替换同名项(值为数组的,用新数组代替老的,不会合并数组)
117
+ * hash string 带不带开头的 '#' 皆可。会代替 url 已有的 hash。
118
+ */
119
+ export function combineUrl(origUrl: string, query: StringifyQuery = {}, hash = '') {
120
+ if (hash.startsWith('#')) hash = hash.slice(1)
121
+
122
+ // 拆分原 url 的 search、hash
123
+ const { base, search: origSearch, hash: origHash } = splitUrl(origUrl)
124
+
125
+ // 拼接新 URL
126
+ let newUrl = base
127
+ const newSearch = stringifyQuery({ ...parseQuery(origSearch), ...query })
128
+ const newHash = hash || origHash
129
+ if (newSearch) newUrl += `?${newSearch}`
130
+ if (newHash) newUrl += `#${newHash}`
131
+ return newUrl
132
+ }
133
+
134
+ /**
135
+ * 移除路径中所有非必须的 "/"
136
+ * 清理后的字符串只有这几种可能的格式:''、'abc'、'abc/def'
137
+ * 例如 /abc/def 和 abc/def/ 都会变成 abc/def
138
+ *
139
+ * 注意:此操作不会统一大小写,因此不保证处理后两个字符串在代码层面全等(a === b)
140
+ */
141
+ export function clearSlash(path: string) {
142
+ if (path.startsWith('/')) path = path.slice(1)
143
+ if (path.endsWith('/')) path = path.slice(0, -1)
144
+ path = path.replace(/\/+/g, '/')
145
+ return path
146
+ }
147
+
148
+ /**
149
+ * 合并几段路径,保证合并处只有一个斜杠
150
+ */
151
+ export function joinPath(...nodes: string[]) {
152
+ // - node 为 '' 时忽略 node
153
+ // - path 可能的格式:'' 'a' 'a/' ‘/a/'
154
+ // - path 为 '',则 node 开头 '/' 保持原样
155
+ // - 否则,根据 path 结尾有没有 '/',决定 node 开头带不带 '/'
156
+ // - node 开头、结尾若有多个 '/' 均替换成单个
157
+ return nodes.reduce((path, node) => {
158
+ if (!node) return path
159
+ type Matched = RegExpExecArray & { 1: string; 2: string; 3: string }
160
+ const [, origPrefix, content, origSuffix] = /^(\/*)(.*?)(\/*)$/.exec(node)! as Matched
161
+ const prefix = (path === '' ? !!origPrefix : !path.endsWith('/')) ? '/' : ''
162
+ const suffix = origSuffix ? '/' : ''
163
+ const result = `${path}${prefix}${content}${suffix}`
164
+ return result
165
+ }, '')
166
+ }
167
+
168
+ /**
169
+ * decodeURIComponent() 对于不规范编码的字符串可能会报错(例如字符串里出现了“%”)
170
+ * 用此函数代替可避免此问题
171
+ */
172
+ export function safeDecode(content: string) {
173
+ try {
174
+ return decodeURIComponent(content)
175
+ } catch (e) {
176
+ return content
177
+ }
178
+ }
179
+
180
+ /**
181
+ * 将 URL 中的 http:// 协议改成 https://
182
+ */
183
+ export function ensureHttps(url: string | undefined) {
184
+ return url?.replace(/http:\/\//g, 'https://')
185
+ }