@planet-matrix/mobius-model 0.6.0 → 0.9.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.
- package/CHANGELOG.md +37 -0
- package/dist/index.js +706 -36
- package/dist/index.js.map +855 -59
- package/package.json +28 -16
- package/src/ai/README.md +1 -0
- package/src/ai/ai.ts +107 -0
- package/src/ai/chat-completion-ai/aihubmix-chat-completion.ts +78 -0
- package/src/ai/chat-completion-ai/chat-completion-ai.ts +270 -0
- package/src/ai/chat-completion-ai/chat-completion.ts +189 -0
- package/src/ai/chat-completion-ai/index.ts +7 -0
- package/src/ai/chat-completion-ai/lingyiwanwu-chat-completion.ts +78 -0
- package/src/ai/chat-completion-ai/ohmygpt-chat-completion.ts +78 -0
- package/src/ai/chat-completion-ai/openai-next-chat-completion.ts +78 -0
- package/src/ai/embedding-ai/embedding-ai.ts +63 -0
- package/src/ai/embedding-ai/embedding.ts +50 -0
- package/src/ai/embedding-ai/index.ts +4 -0
- package/src/ai/embedding-ai/openai-next-embedding.ts +23 -0
- package/src/ai/index.ts +4 -0
- package/src/aio/README.md +100 -0
- package/src/aio/content.ts +141 -0
- package/src/aio/index.ts +3 -0
- package/src/aio/json.ts +127 -0
- package/src/aio/prompt.ts +246 -0
- package/src/basic/README.md +20 -15
- package/src/basic/error.ts +19 -5
- package/src/basic/function.ts +2 -2
- package/src/basic/index.ts +1 -0
- package/src/basic/schedule.ts +111 -0
- package/src/basic/stream.ts +135 -25
- package/src/credential/README.md +107 -0
- package/src/credential/api-key.ts +158 -0
- package/src/credential/bearer.ts +73 -0
- package/src/credential/index.ts +4 -0
- package/src/credential/json-web-token.ts +96 -0
- package/src/credential/password.ts +170 -0
- package/src/cron/README.md +86 -0
- package/src/cron/cron.ts +87 -0
- package/src/cron/index.ts +1 -0
- package/src/drizzle/README.md +1 -0
- package/src/drizzle/drizzle.ts +1 -0
- package/src/drizzle/helper.ts +47 -0
- package/src/drizzle/index.ts +5 -0
- package/src/drizzle/infer.ts +52 -0
- package/src/drizzle/kysely.ts +8 -0
- package/src/drizzle/pagination.ts +200 -0
- package/src/email/README.md +1 -0
- package/src/email/index.ts +1 -0
- package/src/email/resend.ts +25 -0
- package/src/event/class-event-proxy.ts +6 -5
- package/src/event/common.ts +13 -3
- package/src/event/event-manager.ts +3 -3
- package/src/event/instance-event-proxy.ts +6 -5
- package/src/event/internal.ts +4 -4
- package/src/form/README.md +25 -0
- package/src/form/index.ts +1 -0
- package/src/form/inputor-controller/base.ts +874 -0
- package/src/form/inputor-controller/boolean.ts +39 -0
- package/src/form/inputor-controller/file.ts +39 -0
- package/src/form/inputor-controller/form.ts +181 -0
- package/src/form/inputor-controller/helper.ts +117 -0
- package/src/form/inputor-controller/index.ts +17 -0
- package/src/form/inputor-controller/multi-select.ts +99 -0
- package/src/form/inputor-controller/number.ts +116 -0
- package/src/form/inputor-controller/select.ts +109 -0
- package/src/form/inputor-controller/text.ts +82 -0
- package/src/http/READMD.md +1 -0
- package/src/http/api/api-core.ts +84 -0
- package/src/http/api/api-handler.ts +79 -0
- package/src/http/api/api-host.ts +47 -0
- package/src/http/api/api-result.ts +56 -0
- package/src/http/api/api-schema.ts +154 -0
- package/src/http/api/api-server.ts +130 -0
- package/src/http/api/api-test.ts +142 -0
- package/src/http/api/api-type.ts +37 -0
- package/src/http/api/api.ts +81 -0
- package/src/http/api/index.ts +11 -0
- package/src/http/api-adapter/api-core-node-http.ts +260 -0
- package/src/http/api-adapter/api-host-node-http.ts +156 -0
- package/src/http/api-adapter/api-result-arktype.ts +297 -0
- package/src/http/api-adapter/api-result-zod.ts +286 -0
- package/src/http/api-adapter/index.ts +5 -0
- package/src/http/bin/gen-api-list/gen-api-list.ts +126 -0
- package/src/http/bin/gen-api-list/index.ts +1 -0
- package/src/http/bin/gen-api-test/gen-api-test.ts +136 -0
- package/src/http/bin/gen-api-test/index.ts +1 -0
- package/src/http/bin/gen-api-type/calc-code.ts +25 -0
- package/src/http/bin/gen-api-type/gen-api-type.ts +127 -0
- package/src/http/bin/gen-api-type/index.ts +2 -0
- package/src/http/bin/index.ts +2 -0
- package/src/http/index.ts +3 -0
- package/src/huawei/README.md +1 -0
- package/src/huawei/index.ts +2 -0
- package/src/huawei/moderation/index.ts +1 -0
- package/src/huawei/moderation/moderation.ts +355 -0
- package/src/huawei/obs/esdk-obs-nodejs.d.ts +87 -0
- package/src/huawei/obs/index.ts +1 -0
- package/src/huawei/obs/obs.ts +42 -0
- package/src/index.ts +19 -2
- package/src/json/README.md +92 -0
- package/src/json/index.ts +1 -0
- package/src/json/repair.ts +18 -0
- package/src/log/logger.ts +15 -4
- package/src/openai/README.md +1 -0
- package/src/openai/index.ts +1 -0
- package/src/openai/openai.ts +510 -0
- package/src/orchestration/README.md +9 -7
- package/src/orchestration/dispatching/dispatcher.ts +83 -0
- package/src/orchestration/dispatching/index.ts +2 -0
- package/src/orchestration/dispatching/selector/base-selector.ts +39 -0
- package/src/orchestration/dispatching/selector/down-count-selector.ts +119 -0
- package/src/orchestration/dispatching/selector/index.ts +2 -0
- package/src/orchestration/index.ts +2 -0
- package/src/orchestration/scheduling/index.ts +2 -0
- package/src/orchestration/scheduling/scheduler.ts +103 -0
- package/src/orchestration/scheduling/task.ts +32 -0
- package/src/random/README.md +8 -7
- package/src/random/base.ts +66 -0
- package/src/random/index.ts +5 -1
- package/src/random/random-boolean.ts +40 -0
- package/src/random/random-integer.ts +60 -0
- package/src/random/random-number.ts +72 -0
- package/src/random/random-string.ts +66 -0
- package/src/request/README.md +108 -0
- package/src/request/fetch/base.ts +108 -0
- package/src/request/fetch/browser.ts +285 -0
- package/src/request/fetch/general.ts +20 -0
- package/src/request/fetch/index.ts +4 -0
- package/src/request/fetch/nodejs.ts +285 -0
- package/src/request/index.ts +2 -0
- package/src/request/request/base.ts +250 -0
- package/src/request/request/general.ts +64 -0
- package/src/request/request/index.ts +3 -0
- package/src/request/request/resource.ts +68 -0
- package/src/result/README.md +4 -0
- package/src/result/controller.ts +54 -0
- package/src/result/either.ts +193 -0
- package/src/result/index.ts +2 -0
- package/src/route/README.md +105 -0
- package/src/route/adapter/browser.ts +122 -0
- package/src/route/adapter/driver.ts +56 -0
- package/src/route/adapter/index.ts +2 -0
- package/src/route/index.ts +3 -0
- package/src/route/router/index.ts +2 -0
- package/src/route/router/route.ts +630 -0
- package/src/route/router/router.ts +1642 -0
- package/src/route/uri/hash.ts +308 -0
- package/src/route/uri/index.ts +7 -0
- package/src/route/uri/pathname.ts +376 -0
- package/src/route/uri/search.ts +413 -0
- package/src/socket/README.md +105 -0
- package/src/socket/client/index.ts +2 -0
- package/src/socket/client/socket-unit.ts +660 -0
- package/src/socket/client/socket.ts +203 -0
- package/src/socket/common/index.ts +2 -0
- package/src/socket/common/socket-unit-common.ts +23 -0
- package/src/socket/common/socket-unit-heartbeat.ts +427 -0
- package/src/socket/index.ts +3 -0
- package/src/socket/server/index.ts +3 -0
- package/src/socket/server/server.ts +183 -0
- package/src/socket/server/socket-unit.ts +449 -0
- package/src/socket/server/socket.ts +264 -0
- package/src/storage/table.ts +3 -3
- package/src/timer/expiration/expiration-manager.ts +3 -3
- package/src/timer/expiration/remaining-manager.ts +3 -3
- package/src/tube/README.md +99 -0
- package/src/tube/helper.ts +138 -0
- package/src/tube/index.ts +2 -0
- package/src/tube/tube.ts +880 -0
- package/src/weixin/README.md +1 -0
- package/src/weixin/index.ts +2 -0
- package/src/weixin/official-account/authorization.ts +159 -0
- package/src/weixin/official-account/index.ts +2 -0
- package/src/weixin/official-account/js-api.ts +134 -0
- package/src/weixin/open/index.ts +1 -0
- package/src/weixin/open/oauth2.ts +133 -0
- package/tests/unit/ai/ai.spec.ts +85 -0
- package/tests/unit/aio/content.spec.ts +105 -0
- package/tests/unit/aio/json.spec.ts +147 -0
- package/tests/unit/aio/prompt.spec.ts +111 -0
- package/tests/unit/basic/error.spec.ts +16 -4
- package/tests/unit/basic/schedule.spec.ts +74 -0
- package/tests/unit/basic/stream.spec.ts +90 -37
- package/tests/unit/credential/api-key.spec.ts +37 -0
- package/tests/unit/credential/bearer.spec.ts +23 -0
- package/tests/unit/credential/json-web-token.spec.ts +23 -0
- package/tests/unit/credential/password.spec.ts +41 -0
- package/tests/unit/cron/cron.spec.ts +84 -0
- package/tests/unit/event/class-event-proxy.spec.ts +3 -3
- package/tests/unit/event/event-manager.spec.ts +3 -3
- package/tests/unit/event/instance-event-proxy.spec.ts +3 -3
- package/tests/unit/form/inputor-controller/base.spec.ts +458 -0
- package/tests/unit/form/inputor-controller/boolean.spec.ts +30 -0
- package/tests/unit/form/inputor-controller/file.spec.ts +27 -0
- package/tests/unit/form/inputor-controller/form.spec.ts +120 -0
- package/tests/unit/form/inputor-controller/helper.spec.ts +67 -0
- package/tests/unit/form/inputor-controller/multi-select.spec.ts +34 -0
- package/tests/unit/form/inputor-controller/number.spec.ts +36 -0
- package/tests/unit/form/inputor-controller/select.spec.ts +49 -0
- package/tests/unit/form/inputor-controller/text.spec.ts +34 -0
- package/tests/unit/http/api/api-core-host.spec.ts +207 -0
- package/tests/unit/http/api/api-schema.spec.ts +120 -0
- package/tests/unit/http/api/api-server.spec.ts +363 -0
- package/tests/unit/http/api/api-test.spec.ts +117 -0
- package/tests/unit/http/api/api.spec.ts +121 -0
- package/tests/unit/http/api-adapter/node-http.spec.ts +191 -0
- package/tests/unit/json/repair.spec.ts +11 -0
- package/tests/unit/log/logger.spec.ts +19 -4
- package/tests/unit/openai/openai.spec.ts +64 -0
- package/tests/unit/orchestration/dispatching/dispatcher.spec.ts +41 -0
- package/tests/unit/orchestration/dispatching/selector/down-count-selector.spec.ts +81 -0
- package/tests/unit/orchestration/scheduling/scheduler.spec.ts +103 -0
- package/tests/unit/random/base.spec.ts +58 -0
- package/tests/unit/random/random-boolean.spec.ts +25 -0
- package/tests/unit/random/random-integer.spec.ts +32 -0
- package/tests/unit/random/random-number.spec.ts +33 -0
- package/tests/unit/random/random-string.spec.ts +22 -0
- package/tests/unit/request/fetch/browser.spec.ts +222 -0
- package/tests/unit/request/fetch/general.spec.ts +43 -0
- package/tests/unit/request/fetch/nodejs.spec.ts +225 -0
- package/tests/unit/request/request/base.spec.ts +385 -0
- package/tests/unit/request/request/general.spec.ts +161 -0
- package/tests/unit/route/router/route.spec.ts +431 -0
- package/tests/unit/route/router/router.spec.ts +407 -0
- package/tests/unit/route/uri/hash.spec.ts +72 -0
- package/tests/unit/route/uri/pathname.spec.ts +147 -0
- package/tests/unit/route/uri/search.spec.ts +107 -0
- package/tests/unit/socket/client.spec.ts +208 -0
- package/tests/unit/socket/server.spec.ts +135 -0
- package/tests/unit/socket/socket-unit-heartbeat.spec.ts +214 -0
- package/tests/unit/tube/helper.spec.ts +139 -0
- package/tests/unit/tube/tube.spec.ts +501 -0
- package/src/random/string.ts +0 -35
- package/tests/unit/random/string.spec.ts +0 -11
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import { getRandomUnitValue } from "./base.ts"
|
|
2
|
+
|
|
3
|
+
// 默认整数下界为 0。
|
|
4
|
+
// 当 randomInteger 只传入一个端点时,会与 0 共同构成整数区间。
|
|
5
|
+
const internalDefaultRandomIntegerValue = 0
|
|
6
|
+
|
|
7
|
+
// Step 1: 校验整数范围端点。
|
|
8
|
+
//
|
|
9
|
+
// randomInteger 只接受安全整数,避免把小数或超出安全范围的值带入整数区间计算。
|
|
10
|
+
const internalAssertRandomIntegerBound = (value: number, name: string): void => {
|
|
11
|
+
if (Number.isSafeInteger(value) === false) {
|
|
12
|
+
throw new TypeError(`Expected ${name} to be a safe integer, got: ${value}`)
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
// Step 2: 规范化随机整数区间。
|
|
17
|
+
//
|
|
18
|
+
// 支持两种调用方式:
|
|
19
|
+
// - randomInteger(max) -> [0, max] 或 [max, 0]
|
|
20
|
+
// - randomInteger(a, b) -> [min(a, b), max(a, b)]
|
|
21
|
+
const internalNormalizeRandomIntegerRange = (a: number, b?: number | undefined): [number, number] => {
|
|
22
|
+
internalAssertRandomIntegerBound(a, b === undefined ? "max" : "a")
|
|
23
|
+
|
|
24
|
+
if (b === undefined) {
|
|
25
|
+
return a >= internalDefaultRandomIntegerValue
|
|
26
|
+
? [internalDefaultRandomIntegerValue, a]
|
|
27
|
+
: [a, internalDefaultRandomIntegerValue]
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
internalAssertRandomIntegerBound(b, "b")
|
|
31
|
+
return a <= b ? [a, b] : [b, a]
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* 生成指定区间内的随机整数。
|
|
36
|
+
*
|
|
37
|
+
* 当传入一个参数时,将其视为区间另一端点,并与 `0` 共同构成整数区间;当传入两个参数时,会按较小值到较大值规范化区间。
|
|
38
|
+
* 结果同时包含下界与上界;若上下界相同,则直接返回该值。
|
|
39
|
+
*
|
|
40
|
+
* @example
|
|
41
|
+
* ```
|
|
42
|
+
* const sample1 = randomInteger(5)
|
|
43
|
+
* // Expect: true
|
|
44
|
+
* const example1 = sample1 >= 0 && sample1 <= 5
|
|
45
|
+
* const sample2 = randomInteger(8, 3)
|
|
46
|
+
* // Expect: true
|
|
47
|
+
* const example2 = sample2 >= 3 && sample2 <= 8
|
|
48
|
+
* // Expect: 4
|
|
49
|
+
* const example3 = randomInteger(4, 4)
|
|
50
|
+
* ```
|
|
51
|
+
*/
|
|
52
|
+
export const randomInteger = (a: number, b?: number | undefined): number => {
|
|
53
|
+
const [min, max] = internalNormalizeRandomIntegerRange(a, b)
|
|
54
|
+
|
|
55
|
+
if (min === max) {
|
|
56
|
+
return min
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
return Math.floor(getRandomUnitValue() * (max - min + 1)) + min
|
|
60
|
+
}
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { getRandomUnitValue } from "./base.ts"
|
|
2
|
+
|
|
3
|
+
// 默认下界为 0。
|
|
4
|
+
// 当调用方未传入任何参数,randomNumber 会从 0 开始生成随机数。
|
|
5
|
+
const internalDefaultRandomNumberMin = 0
|
|
6
|
+
|
|
7
|
+
// 默认上界为 1。
|
|
8
|
+
// 这使 randomNumber() 的行为与常见的随机小数语义保持一致。
|
|
9
|
+
const internalDefaultRandomNumberMax = 1
|
|
10
|
+
|
|
11
|
+
// Step 1: 校验范围端点。
|
|
12
|
+
//
|
|
13
|
+
// randomNumber 只接受有限数值,避免把 Infinity 或 NaN 带入后续区间计算。
|
|
14
|
+
const internalAssertRandomNumberBound = (value: number, name: string): void => {
|
|
15
|
+
if (Number.isFinite(value) === false) {
|
|
16
|
+
throw new TypeError(`Expected ${name} to be a finite number, got: ${value}`)
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
// Step 2: 规范化随机区间。
|
|
21
|
+
//
|
|
22
|
+
// 支持三种调用方式:
|
|
23
|
+
// - randomNumber() -> [0, 1)
|
|
24
|
+
// - randomNumber(max) -> [0, max) 或 [max, 0)
|
|
25
|
+
// - randomNumber(a, b) -> [min(a, b), max(a, b))
|
|
26
|
+
const internalNormalizeRandomNumberRange = (a?: number | undefined, b?: number | undefined): [number, number] => {
|
|
27
|
+
if (a === undefined && b === undefined) {
|
|
28
|
+
return [internalDefaultRandomNumberMin, internalDefaultRandomNumberMax]
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
if (a !== undefined && b === undefined) {
|
|
32
|
+
internalAssertRandomNumberBound(a, "max")
|
|
33
|
+
return a >= 0 ? [0, a] : [a, 0]
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
if (a === undefined || b === undefined) {
|
|
37
|
+
throw new TypeError("Expected both range bounds when specifying two arguments")
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
internalAssertRandomNumberBound(a, "a")
|
|
41
|
+
internalAssertRandomNumberBound(b, "b")
|
|
42
|
+
|
|
43
|
+
return a <= b ? [a, b] : [b, a]
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* 生成指定区间内的随机数。
|
|
48
|
+
*
|
|
49
|
+
* 当未传入参数时,返回 `0` 到 `1` 之间的随机小数;当传入一个参数时,将其视为区间另一端点;当传入两个参数时,会按较小值到较大值规范化区间。
|
|
50
|
+
* 结果包含下界,不包含上界;若上下界相同,则直接返回该值。
|
|
51
|
+
*
|
|
52
|
+
* @example
|
|
53
|
+
* ```
|
|
54
|
+
* const sample1 = randomNumber()
|
|
55
|
+
* // Expect: true
|
|
56
|
+
* const example1 = sample1 >= 0 && sample1 < 1
|
|
57
|
+
* const sample2 = randomNumber(10, 3)
|
|
58
|
+
* // Expect: true
|
|
59
|
+
* const example2 = sample2 >= 3 && sample2 < 10
|
|
60
|
+
* // Expect: 4
|
|
61
|
+
* const example3 = randomNumber(4, 4)
|
|
62
|
+
* ```
|
|
63
|
+
*/
|
|
64
|
+
export const randomNumber = (a?: number | undefined, b?: number | undefined): number => {
|
|
65
|
+
const [min, max] = internalNormalizeRandomNumberRange(a, b)
|
|
66
|
+
|
|
67
|
+
if (min === max) {
|
|
68
|
+
return min
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
return getRandomUnitValue() * (max - min) + min
|
|
72
|
+
}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import { getRandomIndex } from "./base.ts"
|
|
2
|
+
|
|
3
|
+
// 默认字符集包含数字、小写字母与大写字母。
|
|
4
|
+
// 当调用方未显式传入 chars 时,randomString 会使用这组字符。
|
|
5
|
+
const internalDefaultRandomStringChars = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|
6
|
+
|
|
7
|
+
// Step 1: 统一处理长度参数。
|
|
8
|
+
//
|
|
9
|
+
// 把这一步单独放出来之后,主流程里就不需要反复出现相同的边界判断。
|
|
10
|
+
const internalAssertRandomStringLength = (length: number): void => {
|
|
11
|
+
if (Number.isInteger(length) === false || length < 0) {
|
|
12
|
+
throw new RangeError(`Expected length to be a non-negative integer, got: ${length}`)
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
// Step 2: 规范化字符集。
|
|
17
|
+
//
|
|
18
|
+
// 这里返回字符数组而不是原始字符串,后面就可以直接按索引取字符。
|
|
19
|
+
// 同时也顺手把空字符集的情况拦截掉。
|
|
20
|
+
const internalPrepareRandomStringAlphabet = (chars?: string | undefined): string[] => {
|
|
21
|
+
const alphabet = Array.from(chars ?? internalDefaultRandomStringChars)
|
|
22
|
+
|
|
23
|
+
if (alphabet.length === 0) {
|
|
24
|
+
throw new RangeError("Expected chars to contain at least one character")
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
return alphabet
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
// Step 3: 逐个位置生成随机字符并拼接结果。
|
|
31
|
+
//
|
|
32
|
+
// 这里的流程很直接:循环 length 次,每次取一个随机索引,再取出对应字符。
|
|
33
|
+
const internalRandomStringFromAlphabet = (length: number, alphabet: string[]): string => {
|
|
34
|
+
let result = ""
|
|
35
|
+
|
|
36
|
+
for (let index = 0; index < length; index = index + 1) {
|
|
37
|
+
const alphabetIndex = getRandomIndex(alphabet.length)
|
|
38
|
+
result = result + alphabet[alphabetIndex]!
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
return result
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* 生成多环境可用的随机字符串,并可选限制字符集。
|
|
46
|
+
*
|
|
47
|
+
* 当运行时提供 `crypto.getRandomValues` 时,会使用统一的随机源逻辑生成字符索引;否则回退到 `Math.random`,以保持在常见浏览器、Node.js 与 Bun 环境中的可用性。
|
|
48
|
+
*
|
|
49
|
+
* @example
|
|
50
|
+
* ```
|
|
51
|
+
* // Expect: 12
|
|
52
|
+
* const example1 = randomString(12).length
|
|
53
|
+
* // Expect: true
|
|
54
|
+
* const example2 = randomString(6, "ABC").split("").every(char => char === "A" || char === "B" || char === "C")
|
|
55
|
+
* ```
|
|
56
|
+
*/
|
|
57
|
+
export const randomString = (length: number, chars?: string | undefined): string => {
|
|
58
|
+
internalAssertRandomStringLength(length)
|
|
59
|
+
|
|
60
|
+
if (length === 0) {
|
|
61
|
+
return ""
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
const alphabet = internalPrepareRandomStringAlphabet(chars)
|
|
65
|
+
return internalRandomStringFromAlphabet(length, alphabet)
|
|
66
|
+
}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
# Request
|
|
2
|
+
|
|
3
|
+
## Description
|
|
4
|
+
|
|
5
|
+
Request 模块提供围绕远程资源(remote resource)声明、请求执行、响应分类与失败补丁(patch output)生成的基础建模能力,用于把一类可长期复用的请求语义组织为稳定公共边界。
|
|
6
|
+
|
|
7
|
+
它关注的不是把宿主环境里的 `fetch` API 再包一层临时工具函数,也不是直接提供某个具体后端服务的 SDK,而是把“某个资源如何被声明”“一次请求成功时输出什么”“失败时如何补出可继续消费的结果”“请求生命周期中哪些事件值得稳定暴露”这些问题收束成一套可复用模型。
|
|
8
|
+
|
|
9
|
+
因此,这个模块更适合被理解为请求模型(request model)与资源模型(resource model)的组合边界,而不是一个只负责拼 URL、塞 headers、发 HTTP 的杂项请求工具箱。
|
|
10
|
+
|
|
11
|
+
## For Understanding
|
|
12
|
+
|
|
13
|
+
理解 Request 模块时,首先要把它与“底层传输实现”分开。这里真正要表达的重点,不是网络库本身,而是围绕某类远程资源建立稳定语义:输入如何约束、输出如何分类、成功与失败如何继续被上层消费,以及请求过程里哪些阶段值得被观察。
|
|
14
|
+
|
|
15
|
+
从这个角度看,这个模块主要由三层语义组成:
|
|
16
|
+
|
|
17
|
+
- 资源声明语义:通过 `Resource` 及其派生类型,把某一路径、方法、输入、成功输出、错误输出和补丁输出定义成一份可长期维护的资源契约。
|
|
18
|
+
- 执行适配语义:通过 `BaseFetch`、`BrowserFetch`、`NodejsFetch` 与 `generalFetch`,把不同运行时下的请求执行方式收束到统一读取接口,而不是把环境差异直接暴露给上层。
|
|
19
|
+
- 请求协调语义:通过 `BaseRequest`,把资源契约、请求选项、回调、事件派发与失败补丁生成组织到一个稳定的公共入口。
|
|
20
|
+
|
|
21
|
+
它适合放在这样的边界里:
|
|
22
|
+
|
|
23
|
+
- 你需要围绕某组远程资源建立长期可维护的输入输出契约,而不是在业务里直接散落请求实现。
|
|
24
|
+
- 你需要把“请求执行失败时如何回退为可消费结果”建模出来,例如补丁输出、占位输出、错误态输出或可恢复状态。
|
|
25
|
+
- 你需要把请求成功、失败、成功数据提取、错误数据提取等生命周期阶段稳定暴露给上层订阅者或组合逻辑。
|
|
26
|
+
|
|
27
|
+
同时也要守住几个边界:
|
|
28
|
+
|
|
29
|
+
- Request 模块表达的是请求与资源的公共语义,不负责替代完整的 API SDK 生成器、认证系统、重试策略中心或缓存一致性框架。
|
|
30
|
+
- `fetch` 子模块可以承载运行时适配,但它只是这个模块的执行基础,而不是整个模块存在的理由。
|
|
31
|
+
- 如果某项能力只是某个业务接口的临时参数拼接技巧、某个项目专属响应格式假设或某个服务端产品的耦合封装,通常不适合直接进入这个模块的顶层公共边界。
|
|
32
|
+
|
|
33
|
+
## For Using
|
|
34
|
+
|
|
35
|
+
当你希望把“远程资源”作为一份清楚、稳定、可组合的模型接入,而不是每次在业务代码里临时写一段 URL、请求方法、响应判断和错误兜底逻辑时,可以使用 Request 模块。
|
|
36
|
+
|
|
37
|
+
从使用角度看,这个模块大致可以分为三类能力:
|
|
38
|
+
|
|
39
|
+
- 资源声明能力:用于描述某类资源的基础 URL、路径、方法、输入结构、成功输出、错误输出与补丁输出,让请求契约先于具体实现被稳定定义。
|
|
40
|
+
- 请求执行能力:用于在浏览器或 Node.js 等运行时中发起请求,并以 `json`、`text`、`blob`、`arrayBuffer` 或 `stream` 等方式读取结果。
|
|
41
|
+
- 请求协调能力:用于把资源声明、执行器选择、请求前选项调整、生命周期回调以及事件订阅统一收束到一个请求入口,便于上层复用和继续组合。
|
|
42
|
+
|
|
43
|
+
更合理的接入方式通常是:先定义你的资源契约,再决定是否需要自定义 fetch 执行器,最后再通过继承 `BaseRequest` 或组合 `BaseRequest` 的方式把请求行为接入到上层流程中。这样做的重点不在于“少写几行发请求代码”,而在于把请求相关语义稳定下来,避免让业务代码直接依赖一次性实现细节。
|
|
44
|
+
|
|
45
|
+
默认情况下,可以直接使用 `generalFetch` 让模块在浏览器(browser)与 Node.js 运行时之间选择合适执行器;只有当你确实需要接入自定义传输层、测试替身(test double)或额外的请求前处理逻辑时,才更适合通过 `BaseRequestOptions.fetchFactory` 与 `modifyRequestOptions` 注入自己的适配方式。这里的重点仍然是复用稳定的请求语义,而不是把任意网络细节都暴露为公共入口。
|
|
46
|
+
|
|
47
|
+
当前 `BaseRequest` 的默认实现直接围绕 JSON 响应以及 `{ status, data }` 这类成功/错误分类结构工作:它会按 `status === "success"` 与 `status === "error"` 分流输出,并从 `data` 字段提取成功数据或错误数据。如果你的资源输出不满足这一约定,那么更合理的做法通常不是勉强沿用默认行为,而是通过继承 `BaseRequest` 并覆盖相关分类与提取逻辑,或直接在更外层定义更契合该资源语义的请求模型。
|
|
48
|
+
|
|
49
|
+
如果你的需求只是直接调用某个宿主环境里的原始 `fetch`,并没有资源契约、输出分类、补丁输出或生命周期事件这些长期语义诉求,那么直接使用宿主能力通常更合适,不一定需要进入这个模块边界。
|
|
50
|
+
|
|
51
|
+
## For Contributing
|
|
52
|
+
|
|
53
|
+
贡献 Request 模块时,优先判断新增内容是不是在澄清“远程资源与请求行为如何被建模为稳定公共语义”,而不是只是在补某个当前项目恰好需要的网络工具技巧。这个模块长期要维护的是资源契约、请求选项、响应分类、失败补丁与生命周期事件这些边界,而不是某个具体服务端接口的短期便利封装。
|
|
54
|
+
|
|
55
|
+
扩展时尤其要避免两类偏差:一类是把模块收缩成只剩 `fetch` 包装,导致资源语义与请求协调语义被忽略;另一类是把认证、缓存失效、重试编排、轮询、领域错误码映射、业务级状态机等所有与请求沾边的内容都混进来,最终让模块失去清楚边界。更稳妥的做法,是先判断新增能力表达的是资源契约、请求执行适配、还是请求生命周期协调;只有这些方向里的稳定语义,才适合成为该模块的长期公共能力。
|
|
56
|
+
|
|
57
|
+
其中 `resource.ts` 主要承载类型层契约,本身没有可单独执行的运行时代码;因此它通常不需要单独编写运行时单元测试,但它的语义必须通过 `BaseRequest` 以及具体 fetch 执行器的测试间接得到覆盖与约束。
|
|
58
|
+
|
|
59
|
+
如果后续要扩展 `fetch` 子模块,优先判断新增内容是在补充“标准化请求输入如何映射到宿主请求能力”这一稳定问题域,还是只是在引入某个具体框架或平台的偶发细节。前者可以作为请求执行适配的一部分继续演进,后者通常应留在更外层的集成代码里。
|
|
60
|
+
|
|
61
|
+
### JSDoc 注释格式要求
|
|
62
|
+
|
|
63
|
+
- 每个公开导出的目标(类型、函数、变量、类等)都应包含 JSDoc 注释,让人在不跳转实现的情况下就能理解用途。
|
|
64
|
+
- JSDoc 注释第一行应为清晰且简洁的描述,该描述优先使用中文(英文也可以)。
|
|
65
|
+
- 如果描述后还有其他内容,应在描述后加一个空行。
|
|
66
|
+
- 如果有示例,应使用 `@example` 标签,后接三重反引号代码块(不带语言标识)。
|
|
67
|
+
- 如果有示例,应包含多个场景,展示不同用法,尤其要覆盖常见组合方式或边界输入。
|
|
68
|
+
- 如果有示例,应使用注释格式说明每个场景:`// Expect: <result>`。
|
|
69
|
+
- 如果有示例,应将结果赋值给 `example1`、`example2` 之类的变量,以保持示例易读。
|
|
70
|
+
- 如果有示例,`// Expect: <result>` 应该位于 `example1`、`example2` 之前,以保持示例的逻辑清晰。
|
|
71
|
+
- 如果有示例,应优先使用确定性示例;避免断言精确的随机输出。
|
|
72
|
+
- 如果函数返回结构化字符串,应展示其预期格式特征。
|
|
73
|
+
- 如果有参考资料,应将 `@see` 放在 `@example` 代码块之后,并用一个空行分隔。
|
|
74
|
+
|
|
75
|
+
### 实现规范要求
|
|
76
|
+
|
|
77
|
+
- 不同程序元素之间使用一个空行分隔,保持结构清楚。这里的程序元素,通常指函数、类型、常量,以及直接服务于它们的辅助元素。
|
|
78
|
+
- 某程序元素独占的辅助元素与该程序元素本身视为一个整体,不要在它们之间添加空行。
|
|
79
|
+
- 程序元素的辅助元素应该放置在该程序元素的上方,以保持阅读时的逻辑顺序。
|
|
80
|
+
- 若辅助元素被多个程序元素共享,则应将其视为独立的程序元素,放在这些程序元素中第一个相关目标的上方,并与后续程序元素之间保留一个空行。
|
|
81
|
+
- 辅助元素也应该像其它程序元素一样,保持清晰的命名和适当的注释,以便在需要阅读实现细节时能够快速理解它们的作用和使用方式。
|
|
82
|
+
- 辅助元素的命名必须以前缀 `internal` 开头(或 `Internal`,大小写不敏感)。
|
|
83
|
+
- 辅助元素永远不要公开导出。
|
|
84
|
+
- 被模块内多个不同文件中的程序元素共享的辅助元素,应该放在一个单独的文件中,例如 `./src/request/internal.ts`。
|
|
85
|
+
- 模块内可以包含子模块。只有当某个子目录表达一个稳定、可单独理解、且可能被父模块重导出的子问题域时,才应将其视为子模块。
|
|
86
|
+
- 子模块包含多个文件时,应该为其单独创建子文件夹,并为其创建单独的 Barrel 文件;父模块的 Barrel 文件再重导出子模块的 Barrel 文件。
|
|
87
|
+
- 子模块不需要有自己的 `README.md`。
|
|
88
|
+
- 子模块可以有自己的 `internal.ts` 文件,多个子模块共享的辅助元素应该放在父模块的 `internal.ts` 文件中,单个子模块共享的辅助元素应该放在该子模块的 `internal.ts` 文件中。
|
|
89
|
+
- 对模块依赖关系的要求(通常是不循环依赖或不反向依赖)与对 DRY 的要求可能产生冲突。此时,若复用的代码数量不大,可以适当牺牲 DRY,复制粘贴并保留必要的注释说明;若复用的代码数量较大,则可以将其抽象到新的文件或子模块中,如 `common.ts`,并在需要的地方导入使用。
|
|
90
|
+
- 与请求相关的实现应优先围绕资源契约、请求选项规整、响应分类、补丁输出生成与生命周期事件组织,避免把业务专属认证、缓存策略或服务端接口细节直接提升为通用公共语义。
|
|
91
|
+
|
|
92
|
+
### 导出策略要求
|
|
93
|
+
|
|
94
|
+
- 保持内部辅助项和内部符号为私有,不要让外部接入依赖临时性的内部结构。
|
|
95
|
+
- 每个模块都应有一个用于重导出所有公共 API 的 Barrel 文件。
|
|
96
|
+
- Barrel 文件应命名为 `index.ts`,放在模块目录根部,并且所有公共 API 都应从该文件导出。
|
|
97
|
+
- 新增公共能力时,应优先检查它是否表达稳定、清楚且值得长期维护的请求或资源语义,而不是某段实现细节的便捷暴露;仅在确认需要长期对外承诺时再加入 Barrel 导出。
|
|
98
|
+
|
|
99
|
+
### 测试要求
|
|
100
|
+
|
|
101
|
+
- 若程序元素是函数,则只为该函数编写一个测试,如果该函数需要测试多个用例,应放在同一个测试中。
|
|
102
|
+
- 若程序元素是类,则至少要为该类的每一个方法编写一个测试,如果该方法需要测试多个用例,应放在同一个测试中。
|
|
103
|
+
- 若程序元素是类,除了为该类的每一个方法编写至少一个测试之外,还可以为该类编写任意多个测试,以覆盖该类的不同使用场景或边界情况。
|
|
104
|
+
- 若编写测试时需要用到辅助元素(Mock 或 Spy 等),可以在测试文件中直接定义这些辅助元素。若辅助元素较为简单,则可以直接放在每一个测试内部,优先保证每个测试的独立性,而不是追求极致 DRY;若辅助元素较为复杂或需要在多个测试中复用,则可以放在测试文件顶部,供该测试文件中的所有测试使用。
|
|
105
|
+
- 测试顺序应与源文件中被测试目标的原始顺序保持一致。
|
|
106
|
+
- 若该模块不需要测试,必须在说明文件中明确说明该模块不需要测试,并说明理由。一般来说,只有在该模块没有可执行的公共函数、只承载类型层表达,或其语义已被上层模块的测试完整覆盖且重复测试几乎不再带来额外价值时,才适合这样处理。
|
|
107
|
+
- 模块的单元测试文件目录是 `./tests/unit/request`,若模块包含子模块,则子模块的单元测试文件目录为 `./tests/unit/request/<sub-module-name>`。
|
|
108
|
+
- 对这个模块来说,测试应优先覆盖请求 URL 与 body 的组装、不同响应读取方式、运行时执行器分派、成功与错误输出分流、补丁输出生成、生命周期回调与事件派发,以及超时和外部取消信号的行为。
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 描述 fetch 层支持的响应读取方式。
|
|
3
|
+
*/
|
|
4
|
+
export type FetchResponseType = "json" | "text" | "blob" | "arrayBuffer" | "stream"
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* 描述一次请求在输入侧允许携带的 query 与 body 结构。
|
|
8
|
+
*/
|
|
9
|
+
export interface FetchInput {
|
|
10
|
+
query?: unknown | undefined
|
|
11
|
+
body?: unknown | undefined
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* 描述一次请求在输出侧暴露的响应类型与数据。
|
|
16
|
+
*/
|
|
17
|
+
export interface FetchOutput {
|
|
18
|
+
type: FetchResponseType
|
|
19
|
+
data: unknown
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* 从输出声明中提取 JSON 响应数据类型。
|
|
24
|
+
*/
|
|
25
|
+
export type FetchOutputJsonData<Output extends FetchOutput> = Output extends { type: "json"; data: infer T } ? T : never
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* 从输出声明中提取文本响应数据类型。
|
|
29
|
+
*/
|
|
30
|
+
export type FetchOutputTextData<Output extends FetchOutput> = Output extends { type: "text"; data: infer T } ? T : never
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* 从输出声明中提取 Blob 响应数据类型。
|
|
34
|
+
*/
|
|
35
|
+
export type FetchOutputBlobData<Output extends FetchOutput> = Output extends { type: "blob"; data: infer T } ? T : never
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* 从输出声明中提取 ArrayBuffer 响应数据类型。
|
|
39
|
+
*/
|
|
40
|
+
export type FetchOutputArrayBufferData<Output extends FetchOutput> = Output extends { type: "arrayBuffer"; data: infer T } ? T : never
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* 从输出声明中提取流式响应数据类型。
|
|
44
|
+
*/
|
|
45
|
+
export type FetchOutputStreamData<Output extends FetchOutput> = Output extends { type: "stream"; data: infer T } ? T : never
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* 描述构造 fetch 执行器时使用的标准化选项。
|
|
49
|
+
*
|
|
50
|
+
* 这些选项负责把资源契约中的 URL、方法、输入、期望响应类型以及中止控制
|
|
51
|
+
* 规整到统一结构里,便于不同运行时的执行器共享同一套输入语义。
|
|
52
|
+
*/
|
|
53
|
+
export interface BaseFetchOptions<Input extends FetchInput = FetchInput, Output extends FetchOutput = FetchOutput> {
|
|
54
|
+
baseUrl: string
|
|
55
|
+
path: string
|
|
56
|
+
method: string
|
|
57
|
+
headers?: Record<string, string> | undefined
|
|
58
|
+
query?: Input["query"] | undefined
|
|
59
|
+
body?: Input["body"] | undefined
|
|
60
|
+
responseType?: Output["type"] | undefined
|
|
61
|
+
timeout?: number | undefined
|
|
62
|
+
abortSignal?: AbortSignal | undefined
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* 表示一个可按多种响应类型读取结果的抽象 fetch 执行器。
|
|
67
|
+
*/
|
|
68
|
+
export abstract class BaseFetch<
|
|
69
|
+
Input extends FetchInput = FetchInput,
|
|
70
|
+
Output extends FetchOutput = FetchOutput
|
|
71
|
+
> {
|
|
72
|
+
protected options: BaseFetchOptions<Input, Output>
|
|
73
|
+
|
|
74
|
+
constructor(options: BaseFetchOptions<Input, Output>) {
|
|
75
|
+
this.options = options
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* 以 JSON 形式读取响应数据。
|
|
80
|
+
*/
|
|
81
|
+
abstract getJson(): Promise<FetchOutputJsonData<Output>>
|
|
82
|
+
|
|
83
|
+
/**
|
|
84
|
+
* 以文本形式读取响应数据。
|
|
85
|
+
*/
|
|
86
|
+
abstract getText(): Promise<FetchOutputTextData<Output>>
|
|
87
|
+
|
|
88
|
+
/**
|
|
89
|
+
* 以 Blob 形式读取响应数据。
|
|
90
|
+
*/
|
|
91
|
+
abstract getBlob(): Promise<FetchOutputBlobData<Output>>
|
|
92
|
+
|
|
93
|
+
/**
|
|
94
|
+
* 以 ArrayBuffer 形式读取响应数据。
|
|
95
|
+
*/
|
|
96
|
+
abstract getArrayBuffer(): Promise<FetchOutputArrayBufferData<Output>>
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* 以流形式读取响应数据。
|
|
100
|
+
*/
|
|
101
|
+
abstract getStream(): Promise<FetchOutputStreamData<Output>>
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
/**
|
|
105
|
+
* 根据标准化请求选项创建具体 fetch 执行器的工厂函数类型。
|
|
106
|
+
*/
|
|
107
|
+
export type FetchFactory<Input extends FetchInput = FetchInput, Output extends FetchOutput = FetchOutput> =
|
|
108
|
+
(options: BaseFetchOptions<Input, Output>) => BaseFetch<Input, Output>
|