@planet-matrix/mobius-model 0.5.0 → 0.6.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 +24 -0
- package/README.md +123 -36
- package/dist/index.js +45 -4
- package/dist/index.js.map +183 -11
- package/oxlint.config.ts +6 -0
- package/package.json +16 -10
- package/src/abort/README.md +92 -0
- package/src/abort/abort-manager.ts +278 -0
- package/src/abort/abort-signal-listener-manager.ts +81 -0
- package/src/abort/index.ts +2 -0
- package/src/basic/README.md +69 -118
- package/src/basic/function.ts +81 -62
- package/src/basic/is.ts +152 -71
- package/src/basic/promise.ts +29 -8
- package/src/basic/string.ts +2 -33
- package/src/color/README.md +105 -0
- package/src/color/index.ts +3 -0
- package/src/color/internal.ts +42 -0
- package/src/color/rgb/analyze.ts +236 -0
- package/src/color/rgb/construct.ts +130 -0
- package/src/color/rgb/convert.ts +227 -0
- package/src/color/rgb/derive.ts +303 -0
- package/src/color/rgb/index.ts +6 -0
- package/src/color/rgb/internal.ts +208 -0
- package/src/color/rgb/parse.ts +302 -0
- package/src/color/rgb/serialize.ts +144 -0
- package/src/color/types.ts +57 -0
- package/src/color/xyz/analyze.ts +80 -0
- package/src/color/xyz/construct.ts +19 -0
- package/src/color/xyz/convert.ts +71 -0
- package/src/color/xyz/index.ts +3 -0
- package/src/color/xyz/internal.ts +23 -0
- package/src/css/README.md +93 -0
- package/src/css/class.ts +559 -0
- package/src/css/index.ts +1 -0
- package/src/encoding/README.md +66 -79
- package/src/encoding/base64.ts +13 -4
- package/src/environment/README.md +97 -0
- package/src/environment/basic.ts +26 -0
- package/src/environment/device.ts +311 -0
- package/src/environment/feature.ts +285 -0
- package/src/environment/geo.ts +337 -0
- package/src/environment/index.ts +7 -0
- package/src/environment/runtime.ts +400 -0
- package/src/environment/snapshot.ts +60 -0
- package/src/environment/variable.ts +239 -0
- package/src/event/README.md +90 -0
- package/src/event/class-event-proxy.ts +228 -0
- package/src/event/common.ts +19 -0
- package/src/event/event-manager.ts +203 -0
- package/src/event/index.ts +4 -0
- package/src/event/instance-event-proxy.ts +186 -0
- package/src/event/internal.ts +24 -0
- package/src/exception/README.md +96 -0
- package/src/exception/browser.ts +219 -0
- package/src/exception/index.ts +4 -0
- package/src/exception/nodejs.ts +169 -0
- package/src/exception/normalize.ts +106 -0
- package/src/exception/types.ts +99 -0
- package/src/identifier/README.md +92 -0
- package/src/identifier/id.ts +119 -0
- package/src/identifier/index.ts +2 -0
- package/src/identifier/uuid.ts +187 -0
- package/src/index.ts +16 -1
- package/src/log/README.md +79 -0
- package/src/log/index.ts +5 -0
- package/src/log/log-emitter.ts +72 -0
- package/src/log/log-record.ts +10 -0
- package/src/log/log-scheduler.ts +74 -0
- package/src/log/log-type.ts +8 -0
- package/src/log/logger.ts +543 -0
- package/src/orchestration/README.md +89 -0
- package/src/orchestration/coordination/barrier.ts +214 -0
- package/src/orchestration/coordination/count-down-latch.ts +215 -0
- package/src/orchestration/coordination/errors.ts +98 -0
- package/src/orchestration/coordination/index.ts +16 -0
- package/src/orchestration/coordination/internal/wait-constraints.ts +95 -0
- package/src/orchestration/coordination/internal/wait-queue.ts +109 -0
- package/src/orchestration/coordination/keyed-lock.ts +168 -0
- package/src/orchestration/coordination/mutex.ts +257 -0
- package/src/orchestration/coordination/permit.ts +127 -0
- package/src/orchestration/coordination/read-write-lock.ts +444 -0
- package/src/orchestration/coordination/semaphore.ts +280 -0
- package/src/orchestration/index.ts +1 -0
- package/src/random/README.md +55 -86
- package/src/random/index.ts +1 -1
- package/src/random/string.ts +35 -0
- package/src/reactor/README.md +4 -0
- package/src/reactor/reactor-core/primitive.ts +9 -9
- package/src/reactor/reactor-core/reactive-system.ts +5 -5
- package/src/singleton/README.md +79 -0
- package/src/singleton/factory.ts +55 -0
- package/src/singleton/index.ts +2 -0
- package/src/singleton/manager.ts +204 -0
- package/src/storage/README.md +107 -0
- package/src/storage/index.ts +1 -0
- package/src/storage/table.ts +449 -0
- package/src/timer/README.md +86 -0
- package/src/timer/expiration/expiration-manager.ts +594 -0
- package/src/timer/expiration/index.ts +3 -0
- package/src/timer/expiration/min-heap.ts +208 -0
- package/src/timer/expiration/remaining-manager.ts +241 -0
- package/src/timer/index.ts +1 -0
- package/src/type/README.md +54 -307
- package/src/type/class.ts +2 -2
- package/src/type/index.ts +14 -14
- package/src/type/is.ts +265 -2
- package/src/type/object.ts +37 -0
- package/src/type/string.ts +7 -2
- package/src/type/tuple.ts +6 -6
- package/src/type/union.ts +16 -0
- package/src/web/README.md +77 -0
- package/src/web/capture.ts +35 -0
- package/src/web/clipboard.ts +97 -0
- package/src/web/dom.ts +117 -0
- package/src/web/download.ts +16 -0
- package/src/web/event.ts +46 -0
- package/src/web/index.ts +10 -0
- package/src/web/local-storage.ts +113 -0
- package/src/web/location.ts +28 -0
- package/src/web/permission.ts +172 -0
- package/src/web/script-loader.ts +432 -0
- package/tests/unit/abort/abort-manager.spec.ts +225 -0
- package/tests/unit/abort/abort-signal-listener-manager.spec.ts +62 -0
- package/tests/unit/basic/array.spec.ts +1 -1
- package/tests/unit/basic/stream.spec.ts +1 -1
- package/tests/unit/basic/string.spec.ts +0 -9
- package/tests/unit/color/rgb/analyze.spec.ts +110 -0
- package/tests/unit/color/rgb/construct.spec.ts +56 -0
- package/tests/unit/color/rgb/convert.spec.ts +60 -0
- package/tests/unit/color/rgb/derive.spec.ts +103 -0
- package/tests/unit/color/rgb/parse.spec.ts +66 -0
- package/tests/unit/color/rgb/serialize.spec.ts +46 -0
- package/tests/unit/color/xyz/analyze.spec.ts +33 -0
- package/tests/unit/color/xyz/construct.spec.ts +10 -0
- package/tests/unit/color/xyz/convert.spec.ts +18 -0
- package/tests/unit/css/class.spec.ts +157 -0
- package/tests/unit/environment/basic.spec.ts +20 -0
- package/tests/unit/environment/device.spec.ts +146 -0
- package/tests/unit/environment/feature.spec.ts +388 -0
- package/tests/unit/environment/geo.spec.ts +111 -0
- package/tests/unit/environment/runtime.spec.ts +364 -0
- package/tests/unit/environment/snapshot.spec.ts +4 -0
- package/tests/unit/environment/variable.spec.ts +190 -0
- package/tests/unit/event/class-event-proxy.spec.ts +225 -0
- package/tests/unit/event/event-manager.spec.ts +246 -0
- package/tests/unit/event/instance-event-proxy.spec.ts +187 -0
- package/tests/unit/exception/browser.spec.ts +213 -0
- package/tests/unit/exception/nodejs.spec.ts +144 -0
- package/tests/unit/exception/normalize.spec.ts +57 -0
- package/tests/unit/identifier/id.spec.ts +71 -0
- package/tests/unit/identifier/uuid.spec.ts +85 -0
- package/tests/unit/log/log-emitter.spec.ts +33 -0
- package/tests/unit/log/log-scheduler.spec.ts +40 -0
- package/tests/unit/log/log-type.spec.ts +7 -0
- package/tests/unit/log/logger.spec.ts +222 -0
- package/tests/unit/orchestration/coordination/barrier.spec.ts +96 -0
- package/tests/unit/orchestration/coordination/count-down-latch.spec.ts +63 -0
- package/tests/unit/orchestration/coordination/errors.spec.ts +29 -0
- package/tests/unit/orchestration/coordination/keyed-lock.spec.ts +109 -0
- package/tests/unit/orchestration/coordination/mutex.spec.ts +132 -0
- package/tests/unit/orchestration/coordination/permit.spec.ts +43 -0
- package/tests/unit/orchestration/coordination/read-write-lock.spec.ts +154 -0
- package/tests/unit/orchestration/coordination/semaphore.spec.ts +135 -0
- package/tests/unit/random/string.spec.ts +11 -0
- package/tests/unit/reactor/alien-signals-effect.spec.ts +11 -10
- package/tests/unit/reactor/preact-signal.spec.ts +1 -2
- package/tests/unit/singleton/singleton.spec.ts +49 -0
- package/tests/unit/storage/table.spec.ts +620 -0
- package/tests/unit/timer/expiration/expiration-manager.spec.ts +464 -0
- package/tests/unit/timer/expiration/min-heap.spec.ts +71 -0
- package/tests/unit/timer/expiration/remaining-manager.spec.ts +234 -0
- package/.oxlintrc.json +0 -5
- package/src/random/uuid.ts +0 -103
- package/tests/unit/random/uuid.spec.ts +0 -37
|
@@ -0,0 +1,204 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 定义创建具名单例项时使用的选项。
|
|
3
|
+
*/
|
|
4
|
+
export interface SingletonOptions<Name extends string, Value> {
|
|
5
|
+
name: Name
|
|
6
|
+
value: () => Value
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* 表示一个具名且按需初始化的单例项。
|
|
11
|
+
*/
|
|
12
|
+
export class SingletonItem<Name extends string, Value> {
|
|
13
|
+
private name: Name
|
|
14
|
+
private value: () => Value
|
|
15
|
+
|
|
16
|
+
private cache: Value | null = null
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* 使用名称和值工厂创建单例项。
|
|
20
|
+
*/
|
|
21
|
+
constructor(options: SingletonOptions<Name, Value>) {
|
|
22
|
+
this.name = options.name
|
|
23
|
+
this.value = options.value
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* 获取该单例项的名称。
|
|
28
|
+
*/
|
|
29
|
+
getName(): Name {
|
|
30
|
+
return this.name
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* 获取该单例项的值,并在首次访问时完成初始化。
|
|
35
|
+
*/
|
|
36
|
+
getValue(): Value {
|
|
37
|
+
if (this.cache === null) {
|
|
38
|
+
this.cache = this.value()
|
|
39
|
+
}
|
|
40
|
+
return this.cache
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* 创建一个具名单例项实例。
|
|
46
|
+
*
|
|
47
|
+
* @example
|
|
48
|
+
* ```
|
|
49
|
+
* // Scenario 1: the item keeps a stable name and caches the computed value.
|
|
50
|
+
* let calls = 0
|
|
51
|
+
* const item = singletonItem("token", () => {
|
|
52
|
+
* calls += 1
|
|
53
|
+
* return "secret"
|
|
54
|
+
* })
|
|
55
|
+
*
|
|
56
|
+
* const example1 = item.getValue()
|
|
57
|
+
* const example2 = item.getValue()
|
|
58
|
+
* const example3 = item.getName()
|
|
59
|
+
* const example4 = example1 === example2
|
|
60
|
+
* const example5 = calls
|
|
61
|
+
*
|
|
62
|
+
* // Expect: "token"
|
|
63
|
+
* example3
|
|
64
|
+
* // Expect: true
|
|
65
|
+
* example4
|
|
66
|
+
* // Expect: 1
|
|
67
|
+
* example5
|
|
68
|
+
*
|
|
69
|
+
* // Scenario 2: different items keep independent caches.
|
|
70
|
+
* let leftCalls = 0
|
|
71
|
+
* let rightCalls = 0
|
|
72
|
+
* const left = singletonItem("left", () => {
|
|
73
|
+
* leftCalls += 1
|
|
74
|
+
* return 1
|
|
75
|
+
* })
|
|
76
|
+
* const right = singletonItem("right", () => {
|
|
77
|
+
* rightCalls += 1
|
|
78
|
+
* return 2
|
|
79
|
+
* })
|
|
80
|
+
*
|
|
81
|
+
* const example6 = left.getValue()
|
|
82
|
+
* const example7 = right.getValue()
|
|
83
|
+
* const example8 = leftCalls
|
|
84
|
+
* const example9 = rightCalls
|
|
85
|
+
*
|
|
86
|
+
* // Expect: 1
|
|
87
|
+
* example6
|
|
88
|
+
* // Expect: 2
|
|
89
|
+
* example7
|
|
90
|
+
* // Expect: 1
|
|
91
|
+
* example8
|
|
92
|
+
* // Expect: 1
|
|
93
|
+
* example9
|
|
94
|
+
* ```
|
|
95
|
+
*/
|
|
96
|
+
export const singletonItem = <Name extends string, Value>(
|
|
97
|
+
name: Name,
|
|
98
|
+
value: () => Value
|
|
99
|
+
): SingletonItem<Name, Value> => {
|
|
100
|
+
return new SingletonItem({ name, value })
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
type TupleToUnion<T> = T extends unknown[] ? T[number] : never
|
|
104
|
+
type GetName<Items> = Items extends []
|
|
105
|
+
? []
|
|
106
|
+
: (
|
|
107
|
+
Items extends [infer x, ...infer xs]
|
|
108
|
+
? (
|
|
109
|
+
x extends SingletonItem<infer Name, unknown>
|
|
110
|
+
? [Name, ...GetName<xs>]
|
|
111
|
+
: never
|
|
112
|
+
)
|
|
113
|
+
: never
|
|
114
|
+
)
|
|
115
|
+
type GetValue<Items, Name> = Items extends []
|
|
116
|
+
? never
|
|
117
|
+
: (
|
|
118
|
+
Items extends [infer x, ...infer xs]
|
|
119
|
+
? (
|
|
120
|
+
x extends SingletonItem<infer name, infer value>
|
|
121
|
+
? (
|
|
122
|
+
name extends Name
|
|
123
|
+
? (Name extends name
|
|
124
|
+
? value
|
|
125
|
+
: GetValue<xs, Name>
|
|
126
|
+
)
|
|
127
|
+
: GetValue<xs, Name>
|
|
128
|
+
)
|
|
129
|
+
: never
|
|
130
|
+
)
|
|
131
|
+
: never
|
|
132
|
+
)
|
|
133
|
+
|
|
134
|
+
/**
|
|
135
|
+
* 表示一个可按名称检索单例值的集合。
|
|
136
|
+
*/
|
|
137
|
+
export class SingletonCollection<Item extends Array<SingletonItem<string, unknown>>> {
|
|
138
|
+
private map: Map<string, SingletonItem<string, unknown>>
|
|
139
|
+
|
|
140
|
+
/**
|
|
141
|
+
* 使用一组单例项创建集合。
|
|
142
|
+
*/
|
|
143
|
+
constructor(items: [...Item]) {
|
|
144
|
+
this.map = new Map()
|
|
145
|
+
items.forEach(item => this.map.set(item.getName(), item))
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
/**
|
|
149
|
+
* 按名称获取集合中的单例值。
|
|
150
|
+
*
|
|
151
|
+
* @throws 当名称不存在于集合中时抛出错误。
|
|
152
|
+
*/
|
|
153
|
+
getItem<Name extends TupleToUnion<GetName<Item>>>(name: Name): GetValue<Item, Name> {
|
|
154
|
+
const item = this.map.get(name)
|
|
155
|
+
if (item === undefined) {
|
|
156
|
+
throw new Error(`GlobalService: item ${String(name)} not found`)
|
|
157
|
+
}
|
|
158
|
+
// oxlint-disable-next-line no-unsafe-type-assertion
|
|
159
|
+
return item.getValue() as GetValue<Item, Name>
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* 从多个单例项创建单例集合。
|
|
165
|
+
*
|
|
166
|
+
* @example
|
|
167
|
+
* ```
|
|
168
|
+
* // Scenario 1: different names resolve to their corresponding cached values.
|
|
169
|
+
* const item1 = singletonItem("count", () => 1)
|
|
170
|
+
* const item2 = singletonItem("label", () => "ok")
|
|
171
|
+
* const collection = singletonCollection([item1, item2])
|
|
172
|
+
*
|
|
173
|
+
* const example1 = collection.getItem("count")
|
|
174
|
+
* const example2 = collection.getItem("label")
|
|
175
|
+
*
|
|
176
|
+
* // Expect: 1
|
|
177
|
+
* example1
|
|
178
|
+
* // Expect: "ok"
|
|
179
|
+
* example2
|
|
180
|
+
*
|
|
181
|
+
* // Scenario 2: reading the same name repeatedly returns the same cached value.
|
|
182
|
+
* let calls = 0
|
|
183
|
+
* const item3 = singletonItem("config", () => {
|
|
184
|
+
* calls += 1
|
|
185
|
+
* return { enabled: true }
|
|
186
|
+
* })
|
|
187
|
+
* const collection2 = singletonCollection([item3])
|
|
188
|
+
*
|
|
189
|
+
* const example3 = collection2.getItem("config")
|
|
190
|
+
* const example4 = collection2.getItem("config")
|
|
191
|
+
* const example5 = example3 === example4
|
|
192
|
+
* const example6 = calls
|
|
193
|
+
*
|
|
194
|
+
* // Expect: true
|
|
195
|
+
* example5
|
|
196
|
+
* // Expect: 1
|
|
197
|
+
* example6
|
|
198
|
+
* ```
|
|
199
|
+
*/
|
|
200
|
+
export const singletonCollection = <Item extends Array<SingletonItem<string, unknown>>>(
|
|
201
|
+
items: [...Item]
|
|
202
|
+
): SingletonCollection<Item> => {
|
|
203
|
+
return new SingletonCollection(items)
|
|
204
|
+
}
|
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
# Storage
|
|
2
|
+
|
|
3
|
+
## Description
|
|
4
|
+
|
|
5
|
+
Storage 模块提供围绕运行时受控数据驻留(controlled runtime data residency)的通用建模能力,用于把一组需要在应用运行过程中被稳定放置、命名、查询、更新、删除与观察的数据,组织为清楚且可复用的公共语义。
|
|
6
|
+
|
|
7
|
+
它关注的不是某个具体宿主提供的存储 API,也不是某个数据库产品、浏览器介质或网络协议本身,而是“数据在应用运行时如何以某种稳定结构驻留、如何成为多个上层消费者共同依赖的受控对象、以及这些结构本身该暴露什么样的长期语义”这一类更一般的问题。因此,这个模块更适合被理解为一组运行时存储模型,而不是浏览器存储工具箱、数据库客户端封装层或临时缓存函数集合。
|
|
8
|
+
|
|
9
|
+
## For Understanding
|
|
10
|
+
|
|
11
|
+
理解 Storage 模块时,应先把它看作“运行时受控数据驻留”的建模层,而不是把任意能把值放起来的能力都塞进来的杂项容器。它要解决的问题,不只是某条数据该如何增删改查,也包括一组数据应如何在运行时被稳定地放置为某种结构、如何通过清楚边界对外暴露、以及如何成为 UI、业务逻辑或其它运行时协作方可以共同消费的对象。
|
|
12
|
+
|
|
13
|
+
理解这个模块时,不适合把它简单归入“状态管理”一类过于宽泛的讨论。几乎所有应用都可以被描述为数据流动与状态变化,因此“状态管理”本身很难为模块提供足够稳定的立足点。相比之下,Storage 模块真正的立足点更具体:数据以什么结构驻留在运行时,这些结构如何被查询、更新、删除与观察,以及它们如何成为应用内部长期可依赖的消费面。
|
|
14
|
+
|
|
15
|
+
在很多实际场景里,应用不会直接消费一次请求的返回结果,而是先把这些结果收束进运行时存储结构,再由界面层和业务层读取这份经过统一管理的数据驻留对象。这样做的价值,不在于单纯“缓存一下结果”,而在于把查询结果、本地可见状态、用户操作产生的临时变更以及后续可能发生的协调过程,统一纳入一个稳定的运行时存储模型。像乐观更新、局部回滚、统一订阅、跨视图共享、局部合并、以及与外部数据来源或持久化介质对接,也因此更容易被组织成清楚的公共语义。
|
|
16
|
+
|
|
17
|
+
从这个角度看,当前已实现的 `table` 更适合被理解为 Storage 模块下的一个子形态:它表达的是“以唯一标识组织行数据集合”的表式运行时存储模型,而不是 Storage 模块全部问题域本身。未来若出现 `KeyValueStore`、命名空间存储或其它存储结构,它们也应与 `table` 一样,被理解为 Storage 父边界下的不同子模型,而不是彼此互不相干的零散工具。
|
|
18
|
+
|
|
19
|
+
理解这个模块时,还应守住几条边界原则:
|
|
20
|
+
|
|
21
|
+
- Storage 模块表达的是运行时存储结构、驻留边界、变更管理与可观察语义,而不是某个宿主存储 API 的直接包装。
|
|
22
|
+
- 浏览器的 `localStorage`、`indexedDB`,或服务端数据库、远程接口、文件系统等具体介质与数据来源,通常属于宿主环境能力或外部系统;Storage 可以与它们对接,但不应把它们的临时接口细节直接上升为模块公共语义。
|
|
23
|
+
- 它适合表达“数据如何在运行时被组织”“结构内部如何查询、更新、删除、观察”“模型如何与外部系统形成接点”,但不应直接承担传输协议、连接管理、数据库迁移、分布式一致性、权限体系、查询规划器或业务专属缓存规则。
|
|
24
|
+
- 它可以天然承载运行时副本、乐观更新中介、多来源合流或本地持久化桥接等场景,但这些都应被理解为该问题域上的典型使用方式,而不是 Storage 模块得以成立的唯一理由。
|
|
25
|
+
- 它未来也可以自然吸收数据对齐、传播与副本协同之类的能力;像 `sync` 或 `replication` 这样的方向,可以成为 Storage 模块的一部分功能,但不应反过来成为当前模块定义本身的前提。
|
|
26
|
+
- 它关心的是“存储模型如何在运行时稳定存在”,而不是完整替代服务端数据库、ORM、状态管理框架或同步引擎的全部职责。
|
|
27
|
+
|
|
28
|
+
## For Using
|
|
29
|
+
|
|
30
|
+
当你希望在应用内部维护一组真正可被持续消费的受控数据驻留结构,而不是让 UI 和业务逻辑直接依赖零散变量、一次性请求结果或宿主介质的原始接口时,可以使用 Storage 模块。
|
|
31
|
+
|
|
32
|
+
从使用角度看,这个模块大致可以分为三类能力:
|
|
33
|
+
|
|
34
|
+
- 运行时存储结构能力:用于表达数据在应用内如何被组织、命名、查询、更新、删除、批量处理以及如何保持结构本身的稳定边界。`table` 属于这一类,它适合承载具备唯一标识的一组记录。
|
|
35
|
+
- 变更管理能力:用于表达插入、更新、删除、覆盖、合并、乐观修改、回滚以及变更通知等语义,使运行时驻留数据能够成为应用内部真正可信的消费对象,而不是一次性响应结果的临时中转站。
|
|
36
|
+
- 集成接点能力:用于让运行时存储结构与外部系统形成清楚边界,例如吸收来自远端或其它来源的数据,或把内部数据继续写入本地持久化介质、离线缓存层或其它端侧环境中。
|
|
37
|
+
|
|
38
|
+
更合适的接入方式,通常是先判断你的核心对象是不是“运行时中的一份受控数据驻留结构”。如果你需要的是:
|
|
39
|
+
|
|
40
|
+
- 请求结果先进入一份本地副本,再由多个界面和业务层共同消费;
|
|
41
|
+
- 某组数据需要在运行时被稳定放置为表、键值集合或其它结构,而不是散落在临时变量里;
|
|
42
|
+
- 用户操作要先体现在本地,再异步与远端对齐;
|
|
43
|
+
- 同一份数据需要同时连接上游远端与下游本地持久化;
|
|
44
|
+
- 你希望把查询、更新、订阅和外部集成入口都收束到一份统一模型中;
|
|
45
|
+
|
|
46
|
+
那么这个模块就是合适的边界。
|
|
47
|
+
|
|
48
|
+
相反,如果你的需求只是某个宿主环境里对具体介质进行直接读写,例如浏览器 `localStorage` 的原生操作、IndexedDB 事务细节、某个数据库驱动的连接与 SQL 交互,那么这些通常不应直接落在 Storage 模块本身,而应放在对应宿主模块、适配层或更外层系统中。
|
|
49
|
+
|
|
50
|
+
## For Contributing
|
|
51
|
+
|
|
52
|
+
为 Storage 模块贡献内容时,优先判断新增能力是否真的在澄清“运行时受控数据如何被建模为稳定存储结构”这一问题,而不是只是在补一个看起来方便的宿主 API 包装或某个业务缓存技巧。这个模块应长期服务于“数据如何在运行时稳定驻留并持续消费”“变更如何被统一管理”“不同存储结构如何表达各自稳定语义”“模型如何与外部系统形成清楚接点”这几类问题。
|
|
53
|
+
|
|
54
|
+
扩展这个模块时,应特别警惕以下倾向:把 Storage 退化成浏览器存储工具集合;把某个数据库产品的接口直接平移成公共 API;把网络请求、同步协议、重试策略、权限判断或业务缓存失效规则无差别地下沉到存储模型内部;或者为了某个当前实现方便而暴露临时内部结构。文档应说明哪些运行时存储边界是长期成立的,以及为什么这些边界成立,而不是复述某个版本里的局部实现技巧。
|
|
55
|
+
|
|
56
|
+
当前已经落地的 `table` 应被理解为 Storage 模块下的一个具体子模型,而不是整个模块的全部定义。后续若新增 `KeyValueStore`、命名空间存储、快照模型或其它存储结构,应优先判断它们是否都属于“运行时受控数据驻留结构”这一父问题域;如果只是某个特定宿主介质的读写封装,通常不应直接成为 Storage 模块的顶级公共能力。
|
|
57
|
+
|
|
58
|
+
如果后续在这个模块中继续生长出数据对齐、传播、远端回灌、副本协同或持久化桥接能力,应优先把它们理解为“围绕运行时存储结构发生的后续语义”,而不是让这些能力反过来改写 Storage 的父边界。换句话说,`sync`、`replication` 或类似方向可以成为这个模块未来的组成部分,但它们应建立在清楚的存储结构语义之上,而不是取代这些结构语义本身。
|
|
59
|
+
|
|
60
|
+
### JSDoc 注释格式要求
|
|
61
|
+
|
|
62
|
+
- 每个公开导出的目标(类型、函数、变量、类等)都应包含 JSDoc 注释,让人在不跳转实现的情况下就能理解用途。
|
|
63
|
+
- JSDoc 注释第一行应为清晰且简洁的描述,该描述优先使用中文(英文也可以)。
|
|
64
|
+
- 如果描述后还有其他内容,应在描述后加一个空行。
|
|
65
|
+
- 如果有示例,应使用 `@example` 标签,后接三重反引号代码块(不带语言标识)。
|
|
66
|
+
- 如果有示例,应包含多个场景,展示不同用法,尤其要覆盖常见组合方式或边界输入。
|
|
67
|
+
- 如果有示例,应使用注释格式说明每个场景:`// Expect: <result>`。
|
|
68
|
+
- 如果有示例,应将结果赋值给 `example1`、`example2` 之类的变量,以保持示例易读。
|
|
69
|
+
- 如果有示例,`// Expect: <result>` 应该位于 `example1`、`example2` 之前,以保持示例的逻辑清晰。
|
|
70
|
+
- 如果有示例,应优先使用确定性示例;避免断言精确的随机输出。
|
|
71
|
+
- 如果函数返回结构化字符串,应展示其预期格式特征。
|
|
72
|
+
- 如果有参考资料,应将 `@see` 放在 `@example` 代码块之后,并用一个空行分隔。
|
|
73
|
+
|
|
74
|
+
### 实现规范要求
|
|
75
|
+
|
|
76
|
+
- 不同程序元素之间使用一个空行分隔,保持结构清楚。这里的程序元素,通常指函数、类型、常量,以及直接服务于它们的辅助元素。
|
|
77
|
+
- 某程序元素独占的辅助元素与该程序元素本身视为一个整体,不要在它们之间添加空行。
|
|
78
|
+
- 程序元素的辅助元素应该放置在该程序元素的上方,以保持阅读时的逻辑顺序。
|
|
79
|
+
- 若辅助元素被多个程序元素共享,则应将其视为独立的程序元素,放在这些程序元素中第一个相关目标的上方,并与后续程序元素之间保留一个空行。
|
|
80
|
+
- 辅助元素也应该像其它程序元素一样,保持清晰的命名和适当的注释,以便在需要阅读实现细节时能够快速理解它们的作用和使用方式。
|
|
81
|
+
- 辅助元素的命名必须以前缀 `internal` 开头(或 `Internal`,大小写不敏感)。
|
|
82
|
+
- 辅助元素永远不要公开导出。
|
|
83
|
+
- 被模块内多个不同文件中的程序元素共享的辅助元素,应该放在一个单独的文件中,例如 `./src/storage/internal.ts`。
|
|
84
|
+
- 模块内可以包含子模块。只有当某个子目录表达一个稳定、可单独理解、且可能被父模块重导出的子问题域时,才应将其视为子模块。
|
|
85
|
+
- 子模块包含多个文件时,应该为其单独创建子文件夹,并为其创建单独的 Barrel 文件;父模块的 Barrel 文件再重导出子模块的 Barrel 文件。
|
|
86
|
+
- 子模块不需要有自己的 `README.md`。
|
|
87
|
+
- 子模块可以有自己的 `internal.ts` 文件,多个子模块共享的辅助元素应该放在父模块的 `internal.ts` 文件中,单个子模块共享的辅助元素应该放在该子模块的 `internal.ts` 文件中。
|
|
88
|
+
- 对模块依赖关系的要求(通常是不循环依赖或不反向依赖)与对 DRY 的要求可能产生冲突。此时,若复用的代码数量不大,可以适当牺牲 DRY,复制粘贴并保留必要的注释说明;若复用的代码数量较大,则可以将其抽象到新的文件或子模块中,如 `common.ts`,并在需要的地方导入使用。
|
|
89
|
+
- 与 storage 相关的实现应优先围绕存储结构边界、唯一标识或命名语义、变更管理、查询与订阅、快照隔离、以及与外部系统的集成接点组织,避免把具体宿主介质细节或业务缓存策略混入模块边界。
|
|
90
|
+
|
|
91
|
+
### 导出策略要求
|
|
92
|
+
|
|
93
|
+
- 保持内部辅助项和内部符号为私有,不要让外部接入依赖临时性的内部结构。
|
|
94
|
+
- 每个模块都应有一个用于重导出所有公共 API 的 Barrel 文件。
|
|
95
|
+
- Barrel 文件应命名为 `index.ts`,放在模块目录根部,并且所有公共 API 都应从该文件导出。
|
|
96
|
+
- 新增公共能力时,应优先检查它是否表达稳定、清楚且值得长期维护的运行时存储语义,而不是某段宿主实现细节的便捷暴露;仅在确认需要长期对外承诺时再加入 Barrel 导出。
|
|
97
|
+
|
|
98
|
+
### 测试要求
|
|
99
|
+
|
|
100
|
+
- 若程序元素是函数,则只为该函数编写一个测试,如果该函数需要测试多个用例,应放在同一个测试中。
|
|
101
|
+
- 若程序元素是类,则至少要为该类的每一个方法编写一个测试,如果该方法需要测试多个用例,应放在同一个测试中。
|
|
102
|
+
- 若程序元素是类,除了为该类的每一个方法编写至少一个测试之外,还可以为该类编写任意多个测试,以覆盖该类的不同使用场景或边界情况。
|
|
103
|
+
- 若编写测试时需要用到辅助元素(Mock 或 Spy 等),可以在测试文件中直接定义这些辅助元素。若辅助元素较为简单,则可以直接放在每一个测试内部,优先保证每个测试的独立性,而不是追求极致 DRY;若辅助元素较为复杂或需要在多个测试中复用,则可以放在测试文件顶部,供该测试文件中的所有测试使用。
|
|
104
|
+
- 测试顺序应与源文件中被测试目标的原始顺序保持一致。
|
|
105
|
+
- 若该模块不需要测试,必须在说明文件中明确说明该模块不需要测试,并说明理由。一般来说,只有在该模块没有可执行的公共函数、只承载类型层表达,或其语义已被上层模块的测试完整覆盖且重复测试几乎不再带来额外价值时,才适合这样处理。
|
|
106
|
+
- 模块的单元测试文件目录是 `./tests/unit/storage`,若模块包含子模块,则子模块的单元测试文件目录为 `./tests/unit/storage/<sub-module-name>`。
|
|
107
|
+
- 对这个模块来说,测试应优先覆盖唯一标识约束、查询与批量操作、变更事件通知、快照隔离、乐观更新或合并更新的语义边界、删除与回滚路径,以及运行时存储结构与外部系统接点之间的协同行为。
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from "./table.ts"
|