@wener/utils 1.1.9 → 1.1.11

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 (121) hide show
  1. package/dist/LICENSE.txt +135 -1
  2. package/dist/cjs/index-da9513d6.js +13 -0
  3. package/dist/cjs/index-da9513d6.js.map +1 -0
  4. package/dist/cjs/index.cjs +9 -1
  5. package/dist/cjs/index.cjs.map +1 -1
  6. package/dist/cjs/server.cjs +1 -1
  7. package/dist/cjs/server.cjs.map +1 -1
  8. package/dist/esm/index-c696799a.js +13 -0
  9. package/dist/esm/index-c696799a.js.map +1 -0
  10. package/dist/esm/index.js +9 -1
  11. package/dist/esm/index.js.map +1 -1
  12. package/dist/esm/server.js +1 -1
  13. package/dist/esm/server.js.map +1 -1
  14. package/dist/system/index-2dfef0f3.js +13 -0
  15. package/dist/system/index-2dfef0f3.js.map +1 -0
  16. package/dist/system/index.js +9 -1
  17. package/dist/system/index.js.map +1 -1
  18. package/dist/system/server.js +1 -1
  19. package/dist/system/server.js.map +1 -1
  20. package/lib/arrays/MaybeArray.js.map +1 -1
  21. package/lib/arrays/arrayFromAsync.js +9 -0
  22. package/lib/arrays/arrayFromAsync.js.map +1 -0
  23. package/lib/browsers/getFileFromDataTransfer.js.map +1 -1
  24. package/lib/crypto/getNodeCrypto.js +17 -0
  25. package/lib/crypto/getNodeCrypto.js.map +1 -0
  26. package/lib/crypto/getRandomValues.js +4 -2
  27. package/lib/crypto/getRandomValues.js.map +1 -1
  28. package/lib/crypto/pem/pem.js +37 -0
  29. package/lib/crypto/pem/pem.js.map +1 -0
  30. package/lib/crypto/randomUUID.js +1 -1
  31. package/lib/crypto/randomUUID.js.map +1 -1
  32. package/lib/crypto/ulid.js.map +1 -1
  33. package/lib/i18n/createTranslate.js.map +1 -1
  34. package/lib/index.js +9 -5
  35. package/lib/index.js.map +1 -1
  36. package/lib/io/ArrayBuffers.js +25 -13
  37. package/lib/io/ArrayBuffers.js.map +1 -1
  38. package/lib/io/Buffer.js +3 -0
  39. package/lib/io/Buffer.js.map +1 -1
  40. package/lib/io/base64.js +53 -0
  41. package/lib/io/base64.js.map +1 -0
  42. package/lib/isomorphics/structuredClone.js +1 -1
  43. package/lib/isomorphics/structuredClone.js.map +1 -1
  44. package/lib/langs/MaybeFunction.js +9 -0
  45. package/lib/langs/MaybeFunction.js.map +1 -0
  46. package/lib/langs/deepEqual.js.map +1 -1
  47. package/lib/{validations → langs}/isClass.js +0 -0
  48. package/lib/langs/isClass.js.map +1 -0
  49. package/lib/{validations → langs}/isDefined.js +0 -0
  50. package/lib/langs/isDefined.js.map +1 -0
  51. package/lib/{validations → langs}/isEmptyObject.js +0 -0
  52. package/lib/langs/isEmptyObject.js.map +1 -0
  53. package/lib/{validations → langs}/isPlainObject.js +1 -1
  54. package/lib/langs/isPlainObject.js.map +1 -0
  55. package/lib/langs/memoize.js +24 -0
  56. package/lib/langs/memoize.js.map +1 -0
  57. package/lib/{validations → langs}/parseBoolean.js +0 -0
  58. package/lib/langs/parseBoolean.js.map +1 -0
  59. package/lib/langs/shallowEqual.js.map +1 -1
  60. package/lib/logging/createChildLogger.js.map +1 -1
  61. package/lib/logging/createLogger.js.map +1 -1
  62. package/lib/modules/isModule.js.map +1 -1
  63. package/lib/objects/get.js.map +1 -1
  64. package/lib/objects/set.js +4 -1
  65. package/lib/objects/set.js.map +1 -1
  66. package/lib/server.js +5 -4
  67. package/lib/server.js.map +1 -1
  68. package/lib/servers/createProxyFetch.js +32 -0
  69. package/lib/servers/createProxyFetch.js.map +1 -0
  70. package/lib/{server → servers}/polyfillBrowser.js +0 -0
  71. package/lib/servers/polyfillBrowser.js.map +1 -0
  72. package/lib/{server → servers}/polyfillCrypto.js +0 -0
  73. package/lib/servers/polyfillCrypto.js.map +1 -0
  74. package/lib/{server → servers}/polyfillFetch.js +0 -0
  75. package/lib/servers/polyfillFetch.js.map +1 -0
  76. package/lib/{server → servers}/polyfillJsDom.js +6 -1
  77. package/lib/servers/polyfillJsDom.js.map +1 -0
  78. package/lib/strings/camelCase.js.map +1 -1
  79. package/lib/strings/formatBytes.js.map +1 -1
  80. package/lib/strings/renderTemplate.js.map +1 -1
  81. package/package.json +5 -4
  82. package/src/arrays/arrayFromAsync.ts +5 -0
  83. package/src/crypto/getNodeCrypto.ts +17 -0
  84. package/src/crypto/getRandomValues.ts +6 -16
  85. package/src/crypto/hashing.test.ts +2 -2
  86. package/src/crypto/pem/pem.test.ts +63 -0
  87. package/src/crypto/pem/pem.test.ts.md +24 -0
  88. package/src/crypto/pem/pem.test.ts.snap +0 -0
  89. package/src/crypto/pem/pem.ts +74 -0
  90. package/src/crypto/randomUUID.ts +1 -1
  91. package/src/fetch/index.ts +1 -0
  92. package/src/index.ts +10 -7
  93. package/src/io/ArrayBuffers.base64.test.ts +3 -2
  94. package/src/io/ArrayBuffers.ts +30 -14
  95. package/src/io/base64.ts +64 -0
  96. package/src/langs/MaybeFunction.ts +9 -0
  97. package/src/langs/README.md +4 -0
  98. package/src/{validations → langs}/isClass.ts +0 -0
  99. package/src/{validations → langs}/isDefined.ts +0 -0
  100. package/src/{validations → langs}/isEmptyObject.ts +0 -0
  101. package/src/{validations → langs}/isFunction.ts +0 -0
  102. package/src/{validations → langs}/isPlainObject.ts +1 -1
  103. package/src/langs/memoize.ts +32 -0
  104. package/src/{validations → langs}/parseBoolean.ts +0 -0
  105. package/src/server.ts +5 -4
  106. package/src/servers/createProxyFetch.ts +36 -0
  107. package/src/{server → servers}/polyfillBrowser.test.ts +2 -1
  108. package/src/{server → servers}/polyfillBrowser.ts +0 -0
  109. package/src/{server → servers}/polyfillCrypto.ts +0 -0
  110. package/src/{server → servers}/polyfillFetch.ts +0 -0
  111. package/src/{server → servers}/polyfillJsDom.ts +0 -0
  112. package/src/typedoc.ts +2 -2
  113. package/lib/server/polyfillBrowser.js.map +0 -1
  114. package/lib/server/polyfillCrypto.js.map +0 -1
  115. package/lib/server/polyfillFetch.js.map +0 -1
  116. package/lib/server/polyfillJsDom.js.map +0 -1
  117. package/lib/validations/isClass.js.map +0 -1
  118. package/lib/validations/isDefined.js.map +0 -1
  119. package/lib/validations/isEmptyObject.js.map +0 -1
  120. package/lib/validations/isPlainObject.js.map +0 -1
  121. package/lib/validations/parseBoolean.js.map +0 -1
@@ -0,0 +1,63 @@
1
+ import test from 'ava';
2
+ import { ArrayBuffers } from '../../io/ArrayBuffers';
3
+ import { PEM } from './pem';
4
+
5
+ const { decode, encode } = PEM;
6
+
7
+ test('pem', (t) => {
8
+ const pem = `-----BEGIN RSA PRIVATE KEY-----
9
+ Proc-Type: 4,ENCRYPTED
10
+ DEK-Info: DES-EDE3-CBC,ABC
11
+
12
+ MDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAw
13
+ MDAwMDAwMDAwMDAwMDAwMA==
14
+ -----END RSA PRIVATE KEY-----
15
+ `;
16
+ const input = `Nice\n${pem}Hello
17
+ `;
18
+ const out = {
19
+ block: {
20
+ type: 'RSA PRIVATE KEY',
21
+ header: {
22
+ 'Proc-Type': '4,ENCRYPTED',
23
+ 'DEK-Info': 'DES-EDE3-CBC,ABC',
24
+ },
25
+ bytes: ArrayBuffers.from('0'.repeat(64)),
26
+ },
27
+ head: 'Nice\n',
28
+ tail: 'Hello\n',
29
+ };
30
+ t.deepEqual(decode(input), out);
31
+ t.is(encode(out.block), pem);
32
+ });
33
+ test('cases', (t) => {
34
+ for (const b of [
35
+ {
36
+ type: 'RSA PRIVATE KEY',
37
+ bytes: 'Hello',
38
+ },
39
+ {
40
+ type: 'RSA PRIVATE KEY',
41
+ header: {
42
+ Name: 'N-:1/=',
43
+ Age: '1234',
44
+ },
45
+ bytes: ArrayBuffers.from('Nice'),
46
+ },
47
+ ]) {
48
+ const s = encode(b);
49
+ const act = decode(s);
50
+ t.deepEqual(
51
+ {
52
+ ...act.block,
53
+ bytes: ArrayBuffers.toString(act.block.bytes),
54
+ },
55
+ {
56
+ header: {},
57
+ ...b,
58
+ bytes: ArrayBuffers.toString(b.bytes),
59
+ },
60
+ );
61
+ t.snapshot(s);
62
+ }
63
+ });
@@ -0,0 +1,24 @@
1
+ # Snapshot report for `packages/utils/src/crypto/pem.test.ts`
2
+
3
+ The actual snapshot is saved in `pem.test.ts.snap`.
4
+
5
+ Generated by [AVA](https://avajs.dev).
6
+
7
+ ## cases
8
+
9
+ > Snapshot 1
10
+
11
+ `-----BEGIN RSA PRIVATE KEY-----␊
12
+ SGVsbG8=␊
13
+ -----END RSA PRIVATE KEY-----␊
14
+ `
15
+
16
+ > Snapshot 2
17
+
18
+ `-----BEGIN RSA PRIVATE KEY-----␊
19
+ Name: N-:1/=␊
20
+ Age: 1234␊
21
+
22
+ TmljZQ==␊
23
+ -----END RSA PRIVATE KEY-----␊
24
+ `
Binary file
@@ -0,0 +1,74 @@
1
+ import { ArrayBuffers } from '../../io/ArrayBuffers';
2
+
3
+ export interface Block {
4
+ /**
5
+ * @see https://github.com/openssl/openssl/blob/master/include/openssl/pem.h#L35-L60 openssl/pem.h
6
+ */
7
+ type:
8
+ | string
9
+ | `${'RSA' | 'DSA'} ${'PRIVATE' | 'PUBLIC'} KEY`
10
+ | `${'X509' | 'TRUSTED'} CERTIFICATE`
11
+ | 'CERTIFICATE'
12
+ | 'X509 CRL'
13
+ | 'CERTIFICATE REQUEST'
14
+ | 'NEW CERTIFICATE REQUEST'
15
+ | 'ANY PRIVATE KEY'
16
+ | 'PUBLIC KEY'
17
+ | 'PKCS7'
18
+ | 'PKCS #7 SIGNED DATA'
19
+ | 'ENCRYPTED PRIVATE KEY'
20
+ | 'PRIVATE KEY'
21
+ | 'DH PARAMETERS'
22
+ | 'X9.42 DH PARAMETERS'
23
+ | 'SSL SESSION PARAMETERS'
24
+ | 'DSA PARAMETERS'
25
+ | 'ECDSA PUBLIC KEY'
26
+ | 'EC PARAMETERS'
27
+ | 'EC PRIVATE KEY'
28
+ | 'PARAMETERS'
29
+ | 'CMS'
30
+ | 'SM2 PARAMETERS';
31
+
32
+ header: Record<string, string>;
33
+ bytes: BufferSource;
34
+ }
35
+
36
+ export class PEM {
37
+ static decode(data: string): { block: Block; tail: string; head: string } {
38
+ const match = data.match(
39
+ /^-----BEGIN (?<type>[^\r\n-]+)-----$\r?\n(?<headers>(^[^:\r\n]+:[^\n\r]+\r?\n)+\r?\n)?(?<data>[a-zA-Z0-9/_=\n\r+]+?)^-----END \1-----$\r?\n?/ms,
40
+ );
41
+ if (!match?.groups) throw new Error('Invalid PEM data');
42
+ const { type, headers = '', data: b64 } = match.groups;
43
+ const header = headers
44
+ .split('\n')
45
+ .filter((v) => v.trim())
46
+ .map((h) => {
47
+ const [k, ...v] = h.split(':');
48
+ return [k.trim(), v.join(':').trim()];
49
+ })
50
+ .reduce((a, [k, v]) => ({ ...a, [k]: v }), {});
51
+ return {
52
+ block: {
53
+ type,
54
+ header,
55
+ // avoid replaceAll
56
+ bytes: ArrayBuffers.from(b64.replace(/[\r\n]/g, ''), 'base64'),
57
+ },
58
+ head: data.slice(0, match.index || 0),
59
+ tail: data.slice((match.index || 0) + match[0].length),
60
+ };
61
+ }
62
+
63
+ static encode(block: { type: string; bytes: string | BufferSource; header?: Record<string, string> }): string {
64
+ const { type, header, bytes } = block;
65
+ const headers = Object.entries(header || {})
66
+ .map(([k, v]) => `${k}: ${v}`)
67
+ .join('\n');
68
+ return `-----BEGIN ${type}-----\n${headers}${headers.length ? '\n\n' : ''}${
69
+ ArrayBuffers.toString(bytes, 'base64')
70
+ .match(/.{1,64}/g)
71
+ ?.join('\n') || ''
72
+ }\n-----END ${type}-----\n`;
73
+ }
74
+ }
@@ -1,7 +1,7 @@
1
1
  /**
2
2
  * generate random UUIDv4
3
3
  */
4
- export const randomUUID: () => string = globalThis.crypto?.randomUUID.bind(globalThis.crypto) || _randomUUID;
4
+ export const randomUUID: () => string = globalThis.crypto?.randomUUID?.bind(globalThis.crypto) || _randomUUID;
5
5
 
6
6
  /**
7
7
  * @internal
@@ -0,0 +1 @@
1
+ export type FetchLike = (url: string | Request, init?: RequestInit) => Promise<Response>;
package/src/index.ts CHANGED
@@ -6,6 +6,7 @@ export {
6
6
  objectOfMaybeArray,
7
7
  type MaybeArray,
8
8
  } from './arrays/MaybeArray';
9
+ export { arrayFromAsync } from './arrays/arrayFromAsync';
9
10
 
10
11
  // object
11
12
  export { get } from './objects/get';
@@ -27,15 +28,16 @@ export { shallowEqual } from './langs/shallowEqual';
27
28
  export { deepEqual } from './langs/deepEqual';
28
29
  export { classOf } from './langs/classOf';
29
30
  export { shallowClone } from './langs/shallowClone';
31
+ export { isClass } from './langs/isClass';
32
+ export { isDefined } from './langs/isDefined';
33
+ export { isEmptyObject } from './langs/isEmptyObject';
34
+ export { isPlainObject } from './langs/isPlainObject';
35
+ export { parseBoolean } from './langs/parseBoolean';
36
+ export { maybeFunction, type MaybeFunction } from './langs/MaybeFunction';
37
+ export { memoize } from './langs/memoize';
30
38
 
31
- // assertions
32
- export { isClass } from './validations/isClass';
33
- export { isDefined } from './validations/isDefined';
34
- export { isEmptyObject } from './validations/isEmptyObject';
35
39
  export { isUUID } from './validations/isUUID';
36
- export { isPlainObject } from './validations/isPlainObject';
37
40
  export { parseTimestamp } from './validations/parseTimestamp';
38
- export { parseBoolean } from './validations/parseBoolean';
39
41
 
40
42
  // modules
41
43
  export { parseModuleId, type ParsedModuleId } from './modules/parseModuleId';
@@ -78,8 +80,9 @@ export { getRandomValues } from './crypto/getRandomValues';
78
80
  export { sha1, sha256, sha384, sha512 } from './crypto/hashing';
79
81
  export { hex } from './crypto/base';
80
82
  export { isULID, createULID, ulid, parseULID } from './crypto/ulid';
83
+ export { PEM } from './crypto/pem/pem';
81
84
 
82
85
  // misc
83
86
  export { createRandom } from './maths/random';
84
87
 
85
- export type FetchLike = (url: string, init?: RequestInit) => Promise<Response>;
88
+ export { type FetchLike } from './fetch';
@@ -1,8 +1,9 @@
1
1
  import test from 'ava';
2
2
  import { ArrayBuffers } from './ArrayBuffers';
3
3
 
4
- test.before(() => {
5
- ArrayBuffers.setAllowedNativeBuffer(false);
4
+ test.before((t) => {
5
+ t.true(ArrayBuffers.isNativeBufferAvailable());
6
+ ArrayBuffers.setNativeBufferAllowed(false);
6
7
  });
7
8
 
8
9
  test('base64: ignore whitespace', function (t) {
@@ -1,4 +1,5 @@
1
1
  import { classOf } from '../langs/classOf';
2
+ import { decodeBase64ToArrayBuffer, encodeArrayBufferToBase64 } from './base64';
2
3
  import { isBuffer } from './isBuffer';
3
4
 
4
5
  /**
@@ -67,14 +68,20 @@ type ToStringEncoding =
67
68
  | 'hex';
68
69
 
69
70
  export class ArrayBuffers {
70
- static #_allowedNativeBuffer: boolean = true;
71
+ static #nativeBufferAllowed: boolean = true;
72
+ static #isBufferAvailable: undefined | boolean;
71
73
 
72
- static #isNativeBufferValid() {
73
- return this.#_allowedNativeBuffer && !(globalThis.Buffer as any)?.isPollyfill?.();
74
+ static isNativeBufferAvailable() {
75
+ // eslint-disable-next-line no-return-assign
76
+ return (this.#isBufferAvailable ??= !(globalThis.Buffer as any)?.isPollyfill?.());
74
77
  }
75
78
 
76
- static setAllowedNativeBuffer(v: boolean) {
77
- this.#_allowedNativeBuffer = v;
79
+ static isNativeBufferAllowed() {
80
+ return this.#nativeBufferAllowed && this.#isBufferAvailable;
81
+ }
82
+
83
+ static setNativeBufferAllowed(v: boolean) {
84
+ this.#nativeBufferAllowed = v;
78
85
  }
79
86
 
80
87
  static isArrayBuffer = (v: any): v is ArrayBuffer => {
@@ -100,7 +107,7 @@ export class ArrayBuffers {
100
107
  return v as I;
101
108
  }
102
109
  if (ArrayBuffer.isView(v) || isBuffer(v)) {
103
- if (ArrayBuffers.#isNativeBufferValid() && (TypedArray as any) === Buffer) {
110
+ if (ArrayBuffers.isNativeBufferAllowed() && (TypedArray as any) === Buffer) {
104
111
  // new Buffer() is deprecated
105
112
  return Buffer.from(v.buffer, byteOffset, byteLength) as I;
106
113
  }
@@ -112,9 +119,17 @@ export class ArrayBuffers {
112
119
  static toString = (buf: BufferSource | string, encoding: ToStringEncoding = 'utf8') => {
113
120
  // 'ascii' 'utf16le' | 'ucs2' | 'ucs-2' | 'base64' | 'base64url' | 'latin1' | 'binary' | 'hex'
114
121
  if (typeof buf === 'string') {
115
- return buf;
122
+ switch (encoding) {
123
+ case 'base64':
124
+ return btoa(buf);
125
+ case 'utf-8':
126
+ case 'utf8':
127
+ return buf;
128
+ default:
129
+ throw new Error(`[ArrayBuffers.toString] Unsupported encoding for string: ${encoding}`);
130
+ }
116
131
  }
117
- if (ArrayBuffers.#isNativeBufferValid()) {
132
+ if (ArrayBuffers.isNativeBufferAllowed()) {
118
133
  return Buffer.from(ArrayBuffers.asView(Uint8Array, buf)).toString(encoding);
119
134
  }
120
135
  // reference
@@ -125,8 +140,7 @@ export class ArrayBuffers {
125
140
  return [...view].map((b) => hexLookupTable[b]).join('');
126
141
  }
127
142
  case 'base64': {
128
- const view: Uint8Array = ArrayBuffers.asView(Uint8Array, buf);
129
- return btoa(String.fromCharCode(...view));
143
+ return encodeArrayBufferToBase64(ArrayBuffers.asView(Uint8Array, buf));
130
144
  }
131
145
  case 'utf8':
132
146
  // falls through
@@ -185,7 +199,7 @@ export class ArrayBuffers {
185
199
  return new ArrayBuffer(0);
186
200
  }
187
201
  if (typeof v === 'string') {
188
- if (ArrayBuffers.#isNativeBufferValid()) {
202
+ if (ArrayBuffers.isNativeBufferAllowed()) {
189
203
  return Buffer.from(v, encoding);
190
204
  }
191
205
 
@@ -195,8 +209,10 @@ export class ArrayBuffers {
195
209
  case 'utf8':
196
210
  return new TextEncoder().encode(v).buffer;
197
211
  case 'base64':
198
- // replaceAll
199
- return Uint8Array.from(atob(v.replace(/[^0-9a-zA-Z=+/_ \r\n]/g, '')), (c) => c.charCodeAt(0));
212
+ // replaceAll need higher version of nodejs
213
+ return decodeBase64ToArrayBuffer(v.replace(/[^0-9a-zA-Z=+/_]/g, ''));
214
+ // error in nodejs 18
215
+ // return Uint8Array.from(atob(v.replace(/[^0-9a-zA-Z=+/_ \r\n]/g, '')), (c) => c.charCodeAt(0));
200
216
  default:
201
217
  throw new Error(`[ArrayBuffers.from] Unknown encoding: ${encoding}`);
202
218
  }
@@ -244,7 +260,7 @@ export class ArrayBuffers {
244
260
  const length = buffers.reduce((a, b) => a + b.byteLength, 0);
245
261
  const r = result ? new Uint8Array(result) : new Uint8Array(length);
246
262
  for (const buffer of buffers) {
247
- if (!buffer || !buffer.byteLength) continue;
263
+ if (!buffer?.byteLength) continue;
248
264
  let n: Uint8Array;
249
265
  if (buffer instanceof ArrayBuffer) {
250
266
  n = new Uint8Array(buffer);
@@ -0,0 +1,64 @@
1
+ // https://github.com/niklasvh/base64-arraybuffer/blob/master/src/index.ts
2
+
3
+ const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';
4
+
5
+ // Use a lookup table to find the index.
6
+ const lookup = typeof Uint8Array === 'undefined' ? [] : new Uint8Array(256);
7
+ for (let i = 0; i < chars.length; i++) {
8
+ lookup[chars.charCodeAt(i)] = i;
9
+ }
10
+
11
+ export function encodeArrayBufferToBase64(arraybuffer: ArrayBuffer): string {
12
+ const bytes = new Uint8Array(arraybuffer);
13
+ const len = bytes.length;
14
+ let base64 = '';
15
+
16
+ for (let i = 0; i < len; i += 3) {
17
+ base64 += chars[bytes[i] >> 2];
18
+ base64 += chars[((bytes[i] & 3) << 4) | (bytes[i + 1] >> 4)];
19
+ base64 += chars[((bytes[i + 1] & 15) << 2) | (bytes[i + 2] >> 6)];
20
+ base64 += chars[bytes[i + 2] & 63];
21
+ }
22
+
23
+ if (len % 3 === 2) {
24
+ base64 = base64.substring(0, base64.length - 1) + '=';
25
+ } else if (len % 3 === 1) {
26
+ base64 = base64.substring(0, base64.length - 2) + '==';
27
+ }
28
+
29
+ return base64;
30
+ }
31
+
32
+ export function decodeBase64ToArrayBuffer(base64: string): ArrayBuffer {
33
+ const len = base64.length;
34
+ let bufferLength = base64.length * 0.75;
35
+ let i;
36
+ let p = 0;
37
+ let encoded1;
38
+ let encoded2;
39
+ let encoded3;
40
+ let encoded4;
41
+
42
+ if (base64[base64.length - 1] === '=') {
43
+ bufferLength--;
44
+ if (base64[base64.length - 2] === '=') {
45
+ bufferLength--;
46
+ }
47
+ }
48
+
49
+ const arraybuffer = new ArrayBuffer(bufferLength);
50
+ const bytes = new Uint8Array(arraybuffer);
51
+
52
+ for (i = 0; i < len; i += 4) {
53
+ encoded1 = lookup[base64.charCodeAt(i)];
54
+ encoded2 = lookup[base64.charCodeAt(i + 1)];
55
+ encoded3 = lookup[base64.charCodeAt(i + 2)];
56
+ encoded4 = lookup[base64.charCodeAt(i + 3)];
57
+
58
+ bytes[p++] = (encoded1 << 2) | (encoded2 >> 4);
59
+ bytes[p++] = ((encoded2 & 15) << 4) | (encoded3 >> 2);
60
+ bytes[p++] = ((encoded3 & 3) << 6) | (encoded4 & 63);
61
+ }
62
+
63
+ return arraybuffer;
64
+ }
@@ -0,0 +1,9 @@
1
+ export type MaybeFunction<T, P extends any[] = any[]> = T | ((...args: P) => T);
2
+
3
+ export function maybeFunction<T, P extends any[] = any[]>(v: MaybeFunction<T, P>, ...args: P): T {
4
+ // https://github.com/microsoft/TypeScript/issues/37663#issuecomment-759728342
5
+ if (v instanceof Function) {
6
+ return v(...args);
7
+ }
8
+ return v;
9
+ }
@@ -0,0 +1,4 @@
1
+ # Core
2
+
3
+ - https://github.com/angus-c/just
4
+ - lodash
File without changes
File without changes
File without changes
File without changes
@@ -1,4 +1,4 @@
1
- import { classOf } from '../langs/classOf';
1
+ import { classOf } from './classOf';
2
2
 
3
3
  // see: https://github.com/mesqueeb/is-what/blob/88d6e4ca92fb2baab6003c54e02eedf4e729e5ab/src/index.ts
4
4
 
@@ -0,0 +1,32 @@
1
+ // https://github.com/angus-c/just/blob/master/packages/function-memoize/index.mjs
2
+ type func = (...args: any) => any;
3
+
4
+ export function memoize<T extends func>(
5
+ callback: T,
6
+ {
7
+ resolver = (...args: Parameters<T>) => JSON.stringify(args),
8
+ }: { resolver?: (...args: Parameters<T>) => string } = {},
9
+ ): T {
10
+ if (typeof callback !== 'function') {
11
+ throw new Error('`callback` should be a function');
12
+ }
13
+
14
+ if (resolver !== undefined && typeof resolver !== 'function') {
15
+ throw new Error('`resolver` should be a function');
16
+ }
17
+
18
+ const cache: Record<string, any> = {};
19
+
20
+ const memoized = function (this: any) {
21
+ const args = Array.prototype.slice.call(arguments); // to simplify JSON.stringify
22
+ const key = resolver.apply(this, args as Parameters<T>);
23
+
24
+ if (!(key in cache)) {
25
+ cache[key] = callback.apply(this, args);
26
+ }
27
+
28
+ return cache[key];
29
+ };
30
+ memoized.cache = cache;
31
+ return memoized as func as T;
32
+ }
File without changes
package/src/server.ts CHANGED
@@ -1,4 +1,5 @@
1
- export { polyfillCrypto } from './server/polyfillCrypto';
2
- export { polyfillFetch } from './server/polyfillFetch';
3
- export { polyfillJsDom } from './server/polyfillJsDom';
4
- export { polyfillBrowser } from './server/polyfillBrowser';
1
+ export { polyfillCrypto } from './servers/polyfillCrypto';
2
+ export { polyfillFetch } from './servers/polyfillFetch';
3
+ export { polyfillJsDom } from './servers/polyfillJsDom';
4
+ export { polyfillBrowser } from './servers/polyfillBrowser';
5
+ export { createProxyFetch } from './servers/createProxyFetch';
@@ -0,0 +1,36 @@
1
+ import { FetchLike } from '../fetch';
2
+
3
+ export function createProxyFetch(proxy?: string, fetch?: FetchLike): FetchLike {
4
+ if (!proxy) {
5
+ return fetch || globalThis.fetch;
6
+ }
7
+
8
+ let agent: any;
9
+ const Request = globalThis.Request;
10
+ let NodeRequest: any;
11
+ let NodeFetch: any;
12
+ return async (url, init?: RequestInit) => {
13
+ if (!agent) {
14
+ const { default: createHttpsProxyAgent } = await import('https-proxy-agent');
15
+ agent = createHttpsProxyAgent(proxy);
16
+ }
17
+
18
+ // node-fetch 才可以,node v18 fetch 不支持
19
+ if (!NodeRequest) {
20
+ ({ Request: NodeRequest, default: NodeFetch } = await import('node-fetch'));
21
+ }
22
+
23
+ fetch ||= NodeFetch;
24
+
25
+ if (url instanceof Request) {
26
+ return (fetch as any)(new Request(url, { agent } as any));
27
+ }
28
+ if ((url as any) instanceof NodeRequest) {
29
+ return (fetch as any)(new NodeRequest(url, { agent } as any));
30
+ }
31
+ return (fetch as any)(url, {
32
+ ...init,
33
+ agent,
34
+ } as any);
35
+ };
36
+ }
@@ -4,7 +4,8 @@ import { polyfillBrowser } from './polyfillBrowser';
4
4
  import { polyfillFetch } from './polyfillFetch';
5
5
 
6
6
  test.before(async (t) => {
7
- t.true(polyfillFetch(nodeFetch));
7
+ // return false in nodejs18
8
+ polyfillFetch(nodeFetch);
8
9
  t.false(polyfillFetch());
9
10
  await polyfillBrowser();
10
11
  });
File without changes
File without changes
File without changes
File without changes
package/src/typedoc.ts CHANGED
@@ -1,2 +1,2 @@
1
- export * from './index'
2
- export * from './server'
1
+ export * from './index';
2
+ export * from './server';
@@ -1 +0,0 @@
1
- {"version":3,"file":"polyfillBrowser.js","sources":["../../src/server/polyfillBrowser.ts"],"sourcesContent":["import { polyfillCrypto } from './polyfillCrypto';\nimport { polyfillFetch } from './polyfillFetch';\nimport { polyfillJsDom } from './polyfillJsDom';\n\n/**\n * Polyfills the browser environment with the necessary APIs for the server.\n * Currently, this includes:\n * - `window`\n * - `document`\n * - `fetch`\n * - `crypto`\n */\nexport async function polyfillBrowser() {\n await polyfillCrypto();\n await polyfillFetch();\n await polyfillJsDom();\n}\n"],"names":[],"mappings":";;;;AAYA,eAAsB,eAAkB,GAAA;AACtC,EAAA,MAAM,cAAe,EAAA,CAAA;AACrB,EAAA,MAAM,aAAc,EAAA,CAAA;AACpB,EAAA,MAAM,aAAc,EAAA,CAAA;AACtB;;;;"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"polyfillCrypto.js","sources":["../../src/server/polyfillCrypto.ts"],"sourcesContent":["export async function polyfillCrypto() {\n if ('crypto' in globalThis) {\n return false;\n }\n globalThis.crypto = (await import('node:crypto')).webcrypto as Crypto;\n return true;\n}\n"],"names":[],"mappings":"AAAA,eAAsB,cAAiB,GAAA;AACrC,EAAA,IAAI,YAAY,UAAY,EAAA;AAC1B,IAAO,OAAA,KAAA,CAAA;AAAA,GACT;AACA,EAAW,UAAA,CAAA,MAAA,GAAA,CAAU,MAAM,OAAO,aAAgB,CAAA,EAAA,SAAA,CAAA;AAClD,EAAO,OAAA,IAAA,CAAA;AACT;;;;"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"polyfillFetch.js","sources":["../../src/server/polyfillFetch.ts"],"sourcesContent":["import type { MaybePromise } from '../asyncs/MaybePromise';\n\nexport function polyfillFetch(nodeFetch: typeof import('node-fetch')): boolean;\nexport function polyfillFetch(nodeFetch?: undefined): Promise<boolean>;\nexport function polyfillFetch(nodeFetch?: typeof import('node-fetch')): MaybePromise<boolean> {\n if ('fetch' in globalThis) {\n return false;\n }\n // sync mode\n if (nodeFetch) {\n const { default: fetch, Response, Headers, Request, AbortError, FetchError, FormData, Blob, File } = nodeFetch;\n Object.assign(globalThis, {\n fetch,\n Response,\n Headers,\n Request,\n AbortError,\n FetchError,\n FormData,\n Blob,\n File,\n });\n return true;\n }\n return import('node-fetch').then((v) => polyfillFetch(v));\n}\n"],"names":[],"mappings":"AAIO,SAAS,cAAc,SAAgE,EAAA;AAC5F,EAAA,IAAI,WAAW,UAAY,EAAA;AACzB,IAAO,OAAA,KAAA,CAAA;AAAA,GACT;AAEA,EAAA,IAAI,SAAW,EAAA;AACb,IAAM,MAAA,EAAE,OAAS,EAAA,KAAA,EAAO,QAAU,EAAA,OAAA,EAAS,OAAS,EAAA,UAAA,EAAY,UAAY,EAAA,QAAA,EAAU,IAAM,EAAA,IAAA,EAAS,GAAA,SAAA,CAAA;AACrG,IAAA,MAAA,CAAO,OAAO,UAAY,EAAA;AAAA,MACxB,KAAA;AAAA,MACA,QAAA;AAAA,MACA,OAAA;AAAA,MACA,OAAA;AAAA,MACA,UAAA;AAAA,MACA,UAAA;AAAA,MACA,QAAA;AAAA,MACA,IAAA;AAAA,MACA,IAAA;AAAA,KACD,CAAA,CAAA;AACD,IAAO,OAAA,IAAA,CAAA;AAAA,GACT;AACA,EAAA,OAAO,OAAO,YAAc,CAAA,CAAA,IAAA,CAAK,CAAC,CAAM,KAAA,aAAA,CAAc,CAAC,CAAC,CAAA,CAAA;AAC1D;;;;"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"polyfillJsDom.js","sources":["../../src/server/polyfillJsDom.ts"],"sourcesContent":["import type { ConstructorOptions, ResourceLoaderConstructorOptions } from 'jsdom';\n\nexport async function polyfillJsDom() {\n if (typeof window !== 'undefined') {\n return false;\n }\n\n const { ResourceLoader, JSDOM } = await import('jsdom');\n\n // https://github.com/lukechilds/window/blob/master/src/index.js\n // eslint-disable-next-line @typescript-eslint/no-extraneous-class\n class Window {\n constructor(opts: ResourceLoaderConstructorOptions & ConstructorOptions = {}) {\n const { proxy, strictSSL, userAgent, ...jsdomOpts } = opts;\n const resources = new ResourceLoader({\n proxy,\n strictSSL,\n userAgent,\n });\n return new JSDOM(\n '',\n Object.assign(jsdomOpts, {\n resources,\n }),\n ).window;\n }\n }\n\n // https://github.com/lukechilds/browser-env/blob/master/src/index.js\n // Default jsdom config.\n // These settings must override any custom settings to make sure we can iterate\n // over the window object.\n const defaultJsdomConfig = {\n // features: {\n // FetchExternalResources: false,\n // ProcessExternalResources: false,\n // },\n };\n // IIFE executed on import to return an array of global Node.js properties that\n // conflict with global browser properties.\n const protectedProperties = (() =>\n Object.getOwnPropertyNames(new Window(defaultJsdomConfig)).filter(\n (prop) => typeof globalThis[prop as keyof typeof globalThis] !== 'undefined',\n ))();\n\n function installEnv(...args: any[]) {\n // Sets up global browser environment\n // Extract options from args\n const properties = args.filter((arg: any) => Array.isArray(arg))[0];\n const userJsdomConfig = args.filter((arg: any) => !Array.isArray(arg))[0];\n\n // Create window object\n const window = new Window(Object.assign({}, userJsdomConfig, defaultJsdomConfig));\n\n // Get all global browser properties\n Object.getOwnPropertyNames(window)\n\n // Remove protected properties\n .filter((prop) => !protectedProperties.includes(prop))\n\n // If we're only applying specific required properties remove everything else\n .filter((prop) => !(properties && properties.indexOf(prop) === -1))\n .filter((prop) => {\n switch (prop) {\n case 'undefined':\n return false;\n }\n return true;\n })\n\n // Copy what's left to the Node.js global scope\n .forEach((prop) => {\n // console.debug(`define globalThis.${prop}`);\n Object.defineProperty(globalThis, prop, {\n configurable: true,\n get: () => window[prop as keyof Window] as any,\n });\n });\n\n return window;\n }\n\n installEnv({ url: 'http://localhost' });\n return true;\n}\n"],"names":["window"],"mappings":"AAEA,eAAsB,aAAgB,GAAA;AACpC,EAAI,IAAA,OAAO,WAAW,WAAa,EAAA;AACjC,IAAO,OAAA,KAAA,CAAA;AAAA,GACT;AAEA,EAAA,MAAM,EAAE,cAAA,EAAgB,KAAM,EAAA,GAAI,MAAM,OAAO,OAAA,CAAA,CAAA;AAI/C,EAAA,MAAM,MAAO,CAAA;AAAA,IACX,WAAA,CAAY,IAA8D,GAAA,EAAI,EAAA;AAC5E,MAAA,MAAM,EAAE,KAAA,EAAO,SAAW,EAAA,SAAA,EAAA,GAAc,WAAc,GAAA,IAAA,CAAA;AACtD,MAAM,MAAA,SAAA,GAAY,IAAI,cAAe,CAAA;AAAA,QACnC,KAAA;AAAA,QACA,SAAA;AAAA,QACA,SAAA;AAAA,OACD,CAAA,CAAA;AACD,MAAA,OAAO,IAAI,KAAA;AAAA,QACT,EAAA;AAAA,QACA,MAAA,CAAO,OAAO,SAAW,EAAA;AAAA,UACvB,SAAA;AAAA,SACD,CAAA;AAAA,OACD,CAAA,MAAA,CAAA;AAAA,KACJ;AAAA,GACF;AAMA,EAAA,MAAM,qBAAqB,EAK3B,CAAA;AAGA,EAAM,MAAA,mBAAA,GAAA,CAAuB,MAC3B,MAAO,CAAA,mBAAA,CAAoB,IAAI,MAAO,CAAA,kBAAkB,CAAC,CAAE,CAAA,MAAA;AAAA,IACzD,CAAC,IAAA,KAAS,OAAO,UAAA,CAAW,IAAqC,CAAA,KAAA,WAAA;AAAA,GAChE,GAAA,CAAA;AAEL,EAAA,SAAS,cAAc,IAAa,EAAA;AAGlC,IAAM,MAAA,UAAA,GAAa,KAAK,MAAO,CAAA,CAAC,QAAa,KAAM,CAAA,OAAA,CAAQ,GAAG,CAAC,CAAE,CAAA,CAAA,CAAA,CAAA;AACjE,IAAM,MAAA,eAAA,GAAkB,IAAK,CAAA,MAAA,CAAO,CAAC,GAAA,KAAa,CAAC,KAAM,CAAA,OAAA,CAAQ,GAAG,CAAC,CAAE,CAAA,CAAA,CAAA,CAAA;AAGvE,IAAMA,MAAAA,OAAAA,GAAS,IAAI,MAAO,CAAA,MAAA,CAAO,OAAO,EAAC,EAAG,eAAiB,EAAA,kBAAkB,CAAC,CAAA,CAAA;AAGhF,IAAO,MAAA,CAAA,mBAAA,CAAoBA,OAAM,CAAA,CAG9B,MAAO,CAAA,CAAC,SAAS,CAAC,mBAAA,CAAoB,QAAS,CAAA,IAAI,CAAC,CAAA,CAGpD,OAAO,CAAC,IAAA,KAAS,EAAE,UAAA,IAAc,UAAW,CAAA,OAAA,CAAQ,IAAI,CAAA,KAAM,CAAG,CAAA,CAAA,CAAA,CACjE,MAAO,CAAA,CAAC,IAAS,KAAA;AAChB,MAAA,QAAQ,IAAM;AAAA,QACZ,KAAK,WAAA;AACH,UAAO,OAAA,KAAA,CAAA;AAAA,OACX;AACA,MAAO,OAAA,IAAA,CAAA;AAAA,KACR,CAAA,CAGA,OAAQ,CAAA,CAAC,IAAS,KAAA;AAEjB,MAAO,MAAA,CAAA,cAAA,CAAe,YAAY,IAAM,EAAA;AAAA,QACtC,YAAc,EAAA,IAAA;AAAA,QACd,GAAA,EAAK,MAAMA,OAAO,CAAA,IAAA,CAAA;AAAA,OACnB,CAAA,CAAA;AAAA,KACF,CAAA,CAAA;AAEH,IAAOA,OAAAA,OAAAA,CAAAA;AAAA,GACT;AAEA,EAAW,UAAA,CAAA,EAAE,GAAK,EAAA,kBAAA,EAAoB,CAAA,CAAA;AACtC,EAAO,OAAA,IAAA,CAAA;AACT;;;;"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"isClass.js","sources":["../../src/validations/isClass.ts"],"sourcesContent":["export function isClass(func: any) {\n return typeof func === 'function' && /^class\\s/.test(Function.prototype.toString.call(func));\n}\n"],"names":[],"mappings":"AAAO,SAAS,QAAQ,IAAW,EAAA;AACjC,EAAO,OAAA,OAAO,IAAS,KAAA,UAAA,IAAc,UAAW,CAAA,IAAA,CAAK,SAAS,SAAU,CAAA,QAAA,CAAS,IAAK,CAAA,IAAI,CAAC,CAAA,CAAA;AAC7F;;;;"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"isDefined.js","sources":["../../src/validations/isDefined.ts"],"sourcesContent":["export function isDefined<T = any>(v: T): v is NonNullable<T> {\n return v !== null && v !== undefined;\n}\n"],"names":[],"mappings":"AAAO,SAAS,UAAmB,CAA2B,EAAA;AAC5D,EAAO,OAAA,CAAA,KAAM,QAAQ,CAAM,KAAA,KAAA,CAAA,CAAA;AAC7B;;;;"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"isEmptyObject.js","sources":["../../src/validations/isEmptyObject.ts"],"sourcesContent":["import { isPlainObject } from './isPlainObject';\n\nexport function isEmptyObject(o: any) {\n return isPlainObject(o) && Object.keys(o).length === 0;\n}\n"],"names":[],"mappings":";;AAEO,SAAS,cAAc,CAAQ,EAAA;AACpC,EAAA,OAAO,cAAc,CAAC,CAAA,IAAK,OAAO,IAAK,CAAA,CAAC,EAAE,MAAW,KAAA,CAAA,CAAA;AACvD;;;;"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"isPlainObject.js","sources":["../../src/validations/isPlainObject.ts"],"sourcesContent":["import { classOf } from '../langs/classOf';\n\n// see: https://github.com/mesqueeb/is-what/blob/88d6e4ca92fb2baab6003c54e02eedf4e729e5ab/src/index.ts\n\nexport function isPlainObject(value: any): value is Record<string, any> {\n if (classOf(value) !== 'Object') {\n return false;\n }\n return value.constructor === Object && Object.getPrototypeOf(value) === Object.prototype;\n}\n"],"names":[],"mappings":";;AAIO,SAAS,cAAc,KAA0C,EAAA;AACtE,EAAI,IAAA,OAAA,CAAQ,KAAK,CAAA,KAAM,QAAU,EAAA;AAC/B,IAAO,OAAA,KAAA,CAAA;AAAA,GACT;AACA,EAAA,OAAO,MAAM,WAAgB,KAAA,MAAA,IAAU,OAAO,cAAe,CAAA,KAAK,MAAM,MAAO,CAAA,SAAA,CAAA;AACjF;;;;"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"parseBoolean.js","sources":["../../src/validations/parseBoolean.ts"],"sourcesContent":["export function parseBoolean(s: string | boolean | number | null | undefined, strict: true): boolean | undefined;\nexport function parseBoolean(s: string | boolean | number | null | undefined): boolean;\nexport function parseBoolean(s?: string | boolean | number | null, strict = false): boolean | undefined {\n if (typeof s === 'boolean') {\n return s;\n }\n if (typeof s === 'string') {\n switch (s.toLowerCase()) {\n case 'f':\n case 'false':\n case '0':\n return false;\n case '1':\n case 't':\n case 'true':\n return true;\n }\n } else if (typeof s === 'number') {\n switch (s) {\n case 0:\n return false;\n case 1:\n return true;\n }\n }\n if (strict) {\n return undefined;\n }\n return Boolean(s);\n}\n"],"names":[],"mappings":"AAEgB,SAAA,YAAA,CAAa,CAAsC,EAAA,MAAA,GAAS,KAA4B,EAAA;AACtG,EAAI,IAAA,OAAO,MAAM,SAAW,EAAA;AAC1B,IAAO,OAAA,CAAA,CAAA;AAAA,GACT;AACA,EAAI,IAAA,OAAO,MAAM,QAAU,EAAA;AACzB,IAAQ,QAAA,CAAA,CAAE,aAAe;AAAA,MACvB,KAAK,GAAA,CAAA;AAAA,MACL,KAAK,OAAA,CAAA;AAAA,MACL,KAAK,GAAA;AACH,QAAO,OAAA,KAAA,CAAA;AAAA,MACT,KAAK,GAAA,CAAA;AAAA,MACL,KAAK,GAAA,CAAA;AAAA,MACL,KAAK,MAAA;AACH,QAAO,OAAA,IAAA,CAAA;AAAA,KACX;AAAA,GACF,MAAA,IAAW,OAAO,CAAA,KAAM,QAAU,EAAA;AAChC,IAAA,QAAQ,CAAG;AAAA,MACT,KAAK,CAAA;AACH,QAAO,OAAA,KAAA,CAAA;AAAA,MACT,KAAK,CAAA;AACH,QAAO,OAAA,IAAA,CAAA;AAAA,KACX;AAAA,GACF;AACA,EAAA,IAAI,MAAQ,EAAA;AACV,IAAO,OAAA,KAAA,CAAA,CAAA;AAAA,GACT;AACA,EAAA,OAAO,QAAQ,CAAC,CAAA,CAAA;AAClB;;;;"}