@matter/general 0.16.7 → 0.16.8-alpha.0-20260125-38e62bc3e
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/dist/cjs/MatterError.d.ts +12 -1
- package/dist/cjs/MatterError.d.ts.map +1 -1
- package/dist/cjs/MatterError.js +26 -1
- package/dist/cjs/MatterError.js.map +1 -1
- package/dist/cjs/environment/SharedEnvironmentServices.d.ts +2 -2
- package/dist/cjs/environment/SharedEnvironmentServices.js +2 -2
- package/dist/cjs/storage/StringifyTools.d.ts +1 -1
- package/dist/cjs/storage/StringifyTools.d.ts.map +1 -1
- package/dist/cjs/storage/StringifyTools.js.map +1 -1
- package/dist/cjs/util/Abort.d.ts +11 -5
- package/dist/cjs/util/Abort.d.ts.map +1 -1
- package/dist/cjs/util/Abort.js +71 -29
- package/dist/cjs/util/Abort.js.map +1 -1
- package/dist/cjs/util/AsyncIterator.d.ts +21 -0
- package/dist/cjs/util/AsyncIterator.d.ts.map +1 -0
- package/dist/cjs/util/AsyncIterator.js +71 -0
- package/dist/cjs/util/AsyncIterator.js.map +6 -0
- package/dist/cjs/util/DataReadQueue.d.ts +4 -1
- package/dist/cjs/util/DataReadQueue.d.ts.map +1 -1
- package/dist/cjs/util/DataReadQueue.js +27 -5
- package/dist/cjs/util/DataReadQueue.js.map +1 -1
- package/dist/cjs/util/Semaphore.d.ts.map +1 -1
- package/dist/cjs/util/Semaphore.js +5 -5
- package/dist/cjs/util/Semaphore.js.map +1 -1
- package/dist/cjs/util/index.d.ts +1 -0
- package/dist/cjs/util/index.d.ts.map +1 -1
- package/dist/cjs/util/index.js +1 -0
- package/dist/cjs/util/index.js.map +1 -1
- package/dist/esm/MatterError.d.ts +12 -1
- package/dist/esm/MatterError.d.ts.map +1 -1
- package/dist/esm/MatterError.js +26 -1
- package/dist/esm/MatterError.js.map +1 -1
- package/dist/esm/environment/SharedEnvironmentServices.d.ts +2 -2
- package/dist/esm/environment/SharedEnvironmentServices.js +2 -2
- package/dist/esm/storage/StringifyTools.d.ts +1 -1
- package/dist/esm/storage/StringifyTools.d.ts.map +1 -1
- package/dist/esm/storage/StringifyTools.js.map +1 -1
- package/dist/esm/util/Abort.d.ts +11 -5
- package/dist/esm/util/Abort.d.ts.map +1 -1
- package/dist/esm/util/Abort.js +71 -29
- package/dist/esm/util/Abort.js.map +1 -1
- package/dist/esm/util/AsyncIterator.d.ts +21 -0
- package/dist/esm/util/AsyncIterator.d.ts.map +1 -0
- package/dist/esm/util/AsyncIterator.js +51 -0
- package/dist/esm/util/AsyncIterator.js.map +6 -0
- package/dist/esm/util/DataReadQueue.d.ts +4 -1
- package/dist/esm/util/DataReadQueue.d.ts.map +1 -1
- package/dist/esm/util/DataReadQueue.js +28 -6
- package/dist/esm/util/DataReadQueue.js.map +1 -1
- package/dist/esm/util/Semaphore.d.ts.map +1 -1
- package/dist/esm/util/Semaphore.js +6 -6
- package/dist/esm/util/Semaphore.js.map +1 -1
- package/dist/esm/util/index.d.ts +1 -0
- package/dist/esm/util/index.d.ts.map +1 -1
- package/dist/esm/util/index.js +1 -0
- package/dist/esm/util/index.js.map +1 -1
- package/package.json +2 -2
- package/src/MatterError.ts +31 -2
- package/src/environment/SharedEnvironmentServices.ts +2 -2
- package/src/storage/StringifyTools.ts +2 -1
- package/src/util/Abort.ts +94 -32
- package/src/util/AsyncIterator.ts +70 -0
- package/src/util/DataReadQueue.ts +33 -6
- package/src/util/Semaphore.ts +8 -7
- package/src/util/index.ts +1 -0
|
@@ -7,8 +7,9 @@
|
|
|
7
7
|
*/
|
|
8
8
|
import { Duration } from "#time/Duration.js";
|
|
9
9
|
import { Minutes } from "#time/TimeUnit.js";
|
|
10
|
-
import { MatterFlowError } from "../MatterError.js";
|
|
10
|
+
import { AbortedError, MatterFlowError } from "../MatterError.js";
|
|
11
11
|
import { Time } from "../time/Time.js";
|
|
12
|
+
import { Abort } from "./Abort.js";
|
|
12
13
|
import { asError } from "./Error.js";
|
|
13
14
|
import { createPromise } from "./Promises.js";
|
|
14
15
|
import { EndOfStreamError, NoResponseTimeoutError } from "./Streams.js";
|
|
@@ -16,7 +17,7 @@ class DataReadQueue {
|
|
|
16
17
|
#queue = new Array();
|
|
17
18
|
#pendingRead;
|
|
18
19
|
#closed = false;
|
|
19
|
-
async read(timeout = Minutes.one) {
|
|
20
|
+
async read({ timeout = Minutes.one, abort } = {}) {
|
|
20
21
|
const { promise, resolver, rejecter } = createPromise();
|
|
21
22
|
if (this.#closed) throw new EndOfStreamError();
|
|
22
23
|
const data = this.#queue.shift();
|
|
@@ -37,20 +38,34 @@ class DataReadQueue {
|
|
|
37
38
|
)
|
|
38
39
|
).start()
|
|
39
40
|
};
|
|
41
|
+
let localAbort;
|
|
40
42
|
try {
|
|
43
|
+
if (abort) {
|
|
44
|
+
localAbort = new Abort({
|
|
45
|
+
abort,
|
|
46
|
+
handler: (reason) => {
|
|
47
|
+
this.#clearPendingRead();
|
|
48
|
+
rejecter(reason);
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
}
|
|
41
52
|
return await promise;
|
|
42
53
|
} catch (e) {
|
|
54
|
+
if (e instanceof AbortedError) {
|
|
55
|
+
throw e;
|
|
56
|
+
}
|
|
43
57
|
const error = asError(e);
|
|
44
58
|
error.stack = new Error().stack;
|
|
45
59
|
throw error;
|
|
60
|
+
} finally {
|
|
61
|
+
localAbort?.close();
|
|
46
62
|
}
|
|
47
63
|
}
|
|
48
64
|
write(data) {
|
|
49
65
|
if (this.#closed) throw new EndOfStreamError();
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
this.#pendingRead = void 0;
|
|
66
|
+
const pendingRead = this.#pendingRead;
|
|
67
|
+
this.#clearPendingRead();
|
|
68
|
+
if (pendingRead) {
|
|
54
69
|
pendingRead.resolver(data);
|
|
55
70
|
return;
|
|
56
71
|
}
|
|
@@ -66,6 +81,13 @@ class DataReadQueue {
|
|
|
66
81
|
this.#pendingRead.timeoutTimer?.stop();
|
|
67
82
|
this.#pendingRead.rejecter(new EndOfStreamError());
|
|
68
83
|
}
|
|
84
|
+
#clearPendingRead() {
|
|
85
|
+
if (this.#pendingRead === void 0) {
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
this.#pendingRead.timeoutTimer?.stop();
|
|
89
|
+
this.#pendingRead = void 0;
|
|
90
|
+
}
|
|
69
91
|
}
|
|
70
92
|
export {
|
|
71
93
|
DataReadQueue
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/util/DataReadQueue.ts"],
|
|
4
|
-
"mappings": "AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAQA,SAAS,gBAAgB;AACzB,SAAS,eAAe;AACxB,SAAS,uBAAuB;
|
|
4
|
+
"mappings": "AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAQA,SAAS,gBAAgB;AACzB,SAAS,eAAe;AACxB,SAAS,cAAc,uBAAuB;AAC9C,SAAS,YAAmB;AAC5B,SAAS,aAAa;AACtB,SAAS,eAAe;AACxB,SAAS,qBAAqB;AAC9B,SAAS,kBAAkB,8BAA8B;AAElD,MAAM,cAAiB;AAAA,EACjB,SAAS,IAAI,MAAS;AAAA,EAC/B;AAAA,EACA,UAAU;AAAA,EAEV,MAAM,KAAK,EAAE,UAAU,QAAQ,KAAK,MAAM,IAAiD,CAAC,GAAe;AACvG,UAAM,EAAE,SAAS,UAAU,SAAS,IAAI,cAAiB;AACzD,QAAI,KAAK,QAAS,OAAM,IAAI,iBAAiB;AAC7C,UAAM,OAAO,KAAK,OAAO,MAAM;AAC/B,QAAI,SAAS,QAAW;AACpB,aAAO;AAAA,IACX;AACA,QAAI,KAAK,iBAAiB,OAAW,OAAM,IAAI,gBAAgB,oCAAoC;AACnG,SAAK,eAAe;AAAA,MAChB;AAAA,MACA;AAAA,MACA,cAAc,KAAK;AAAA,QAAS;AAAA,QAAiB;AAAA,QAAS,MAClD;AAAA,UACI,IAAI;AAAA,YACA,oDAAoD,SAAS,OAAO,OAAO,CAAC;AAAA,UAChF;AAAA,QACJ;AAAA,MACJ,EAAE,MAAM;AAAA,IACZ;AAEA,QAAI;AACJ,QAAI;AACA,UAAI,OAAO;AACP,qBAAa,IAAI,MAAM;AAAA,UACnB;AAAA,UAEA,SAAS,YAAU;AACf,iBAAK,kBAAkB;AACvB,qBAAS,MAAM;AAAA,UACnB;AAAA,QACJ,CAAC;AAAA,MACL;AAEA,aAAO,MAAM;AAAA,IACjB,SAAS,GAAG;AACR,UAAI,aAAa,cAAc;AAC3B,cAAM;AAAA,MACV;AAGA,YAAM,QAAQ,QAAQ,CAAC;AACvB,YAAM,QAAQ,IAAI,MAAM,EAAE;AAC1B,YAAM;AAAA,IACV,UAAE;AACE,kBAAY,MAAM;AAAA,IACtB;AAAA,EACJ;AAAA,EAEA,MAAM,MAAS;AACX,QAAI,KAAK,QAAS,OAAM,IAAI,iBAAiB;AAC7C,UAAM,cAAc,KAAK;AACzB,SAAK,kBAAkB;AACvB,QAAI,aAAa;AACb,kBAAY,SAAS,IAAI;AACzB;AAAA,IACJ;AACA,SAAK,OAAO,KAAK,IAAI;AAAA,EACzB;AAAA,EAEA,IAAI,OAAO;AACP,WAAO,KAAK,OAAO;AAAA,EACvB;AAAA,EAEA,QAAQ;AACJ,QAAI,KAAK,QAAS;AAClB,SAAK,UAAU;AACf,QAAI,KAAK,iBAAiB,OAAW;AACrC,SAAK,aAAa,cAAc,KAAK;AACrC,SAAK,aAAa,SAAS,IAAI,iBAAiB,CAAC;AAAA,EACrD;AAAA,EAEA,oBAAoB;AAChB,QAAI,KAAK,iBAAiB,QAAW;AACjC;AAAA,IACJ;AAEA,SAAK,aAAa,cAAc,KAAK;AACrC,SAAK,eAAe;AAAA,EACxB;AACJ;",
|
|
5
5
|
"names": []
|
|
6
6
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Semaphore.d.ts","sourceRoot":"","sources":["../../../src/util/Semaphore.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAGH,OAAO,EAAE,QAAQ,EAAE,MAAM,mBAAmB,CAAC;AAI7C,OAAO,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AAKnC;;GAEG;AACH,MAAM,WAAW,QAAS,SAAQ,UAAU;IACxC;;;OAGG;IACH,KAAK,IAAI,IAAI,CAAC;CACjB;AAED;;;;;GAKG;AACH,qBAAa,SAAS;;gBAYN,KAAK,EAAE,MAAM,EAAE,WAAW,SAAI,EAAE,KAAK,WAAU;IAO3D;;;;;;;;;;OAUG;IACG,UAAU,CAAC,KAAK,CAAC,EAAE,KAAK,CAAC,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC;
|
|
1
|
+
{"version":3,"file":"Semaphore.d.ts","sourceRoot":"","sources":["../../../src/util/Semaphore.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAGH,OAAO,EAAE,QAAQ,EAAE,MAAM,mBAAmB,CAAC;AAI7C,OAAO,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AAKnC;;GAEG;AACH,MAAM,WAAW,QAAS,SAAQ,UAAU;IACxC;;;OAGG;IACH,KAAK,IAAI,IAAI,CAAC;CACjB;AAED;;;;;GAKG;AACH,qBAAa,SAAS;;gBAYN,KAAK,EAAE,MAAM,EAAE,WAAW,SAAI,EAAE,KAAK,WAAU;IAO3D;;;;;;;;;;OAUG;IACG,UAAU,CAAC,KAAK,CAAC,EAAE,KAAK,CAAC,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC;IAuIzD;;OAEG;IACH,KAAK,IAAI,IAAI;IASb;;OAEG;IACH,IAAI,KAAK,WAER;IAED;;OAEG;IACH,IAAI,OAAO,WAEV;IAED;;OAEG;IACH,KAAK,IAAI,IAAI;CAMhB"}
|
|
@@ -48,7 +48,7 @@ var __callDispose = (stack, error, hasError) => {
|
|
|
48
48
|
* Copyright 2022-2026 Matter.js Authors
|
|
49
49
|
* SPDX-License-Identifier: Apache-2.0
|
|
50
50
|
*/
|
|
51
|
-
import { AbortedError } from "#MatterError.js";
|
|
51
|
+
import { AbortedError, ClosedError } from "#MatterError.js";
|
|
52
52
|
import { Time } from "#time/Time.js";
|
|
53
53
|
import { Instant } from "#time/TimeUnit.js";
|
|
54
54
|
import { Logger } from "../log/Logger.js";
|
|
@@ -85,7 +85,7 @@ class Semaphore {
|
|
|
85
85
|
var _stack = [];
|
|
86
86
|
try {
|
|
87
87
|
if (this.#closed) {
|
|
88
|
-
throw new
|
|
88
|
+
throw new ClosedError("Queue is closed");
|
|
89
89
|
}
|
|
90
90
|
if (abort) {
|
|
91
91
|
const signal = "signal" in abort ? abort.signal : abort;
|
|
@@ -111,8 +111,8 @@ class Semaphore {
|
|
|
111
111
|
this.#queue.length
|
|
112
112
|
);
|
|
113
113
|
}
|
|
114
|
-
|
|
115
|
-
throw
|
|
114
|
+
combinedAbort.throwIfAborted();
|
|
115
|
+
throw new AbortedError("Aborted without reason");
|
|
116
116
|
}
|
|
117
117
|
return result;
|
|
118
118
|
} catch (_) {
|
|
@@ -185,7 +185,7 @@ class Semaphore {
|
|
|
185
185
|
*/
|
|
186
186
|
clear() {
|
|
187
187
|
if (this.#queue.length > 0) {
|
|
188
|
-
this.#abort.abort(new
|
|
188
|
+
this.#abort.abort(new ClosedError("Queue cleared"));
|
|
189
189
|
this.#abort = new Abort();
|
|
190
190
|
}
|
|
191
191
|
this.#queue.length = 0;
|
|
@@ -207,7 +207,7 @@ class Semaphore {
|
|
|
207
207
|
*/
|
|
208
208
|
close() {
|
|
209
209
|
this.#closed = true;
|
|
210
|
-
this.#abort.abort(new
|
|
210
|
+
this.#abort.abort(new ClosedError("Queue is closed"));
|
|
211
211
|
this.clear();
|
|
212
212
|
this.#delayTimer.stop();
|
|
213
213
|
}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/util/Semaphore.ts"],
|
|
4
|
-
"mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA,SAAS,
|
|
4
|
+
"mappings": "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",
|
|
5
5
|
"names": []
|
|
6
6
|
}
|
package/dist/esm/util/index.d.ts
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/util/index.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,cAAc,YAAY,CAAC;AAC3B,cAAc,YAAY,CAAC;AAC3B,cAAc,WAAW,CAAC;AAC1B,cAAc,YAAY,CAAC;AAC3B,cAAc,YAAY,CAAC;AAC3B,cAAc,iBAAiB,CAAC;AAChC,cAAc,mBAAmB,CAAC;AAClC,cAAc,iBAAiB,CAAC;AAChC,cAAc,oBAAoB,CAAC;AACnC,cAAc,iBAAiB,CAAC;AAChC,cAAc,gBAAgB,CAAC;AAC/B,cAAc,eAAe,CAAC;AAC9B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,cAAc,CAAC;AAC7B,cAAc,YAAY,CAAC;AAC3B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC;AAC9B,cAAc,qBAAqB,CAAC;AACpC,cAAc,aAAa,CAAC;AAC5B,cAAc,WAAW,CAAC;AAC1B,cAAc,sBAAsB,CAAC;AACrC,cAAc,SAAS,CAAC;AACxB,cAAc,gBAAgB,CAAC;AAC/B,cAAc,eAAe,CAAC;AAC9B,cAAc,UAAU,CAAC;AACzB,cAAc,gBAAgB,CAAC;AAC/B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,YAAY,CAAC;AAC3B,cAAc,mBAAmB,CAAC;AAClC,cAAc,aAAa,CAAC;AAC5B,cAAc,iBAAiB,CAAC;AAChC,cAAc,eAAe,CAAC;AAC9B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,UAAU,CAAC;AACzB,cAAc,gBAAgB,CAAC;AAC/B,cAAc,cAAc,CAAC;AAC7B,cAAc,aAAa,CAAC;AAC5B,cAAc,WAAW,CAAC"}
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/util/index.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,cAAc,YAAY,CAAC;AAC3B,cAAc,YAAY,CAAC;AAC3B,cAAc,oBAAoB,CAAC;AACnC,cAAc,WAAW,CAAC;AAC1B,cAAc,YAAY,CAAC;AAC3B,cAAc,YAAY,CAAC;AAC3B,cAAc,iBAAiB,CAAC;AAChC,cAAc,mBAAmB,CAAC;AAClC,cAAc,iBAAiB,CAAC;AAChC,cAAc,oBAAoB,CAAC;AACnC,cAAc,iBAAiB,CAAC;AAChC,cAAc,gBAAgB,CAAC;AAC/B,cAAc,eAAe,CAAC;AAC9B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,cAAc,CAAC;AAC7B,cAAc,YAAY,CAAC;AAC3B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC;AAC9B,cAAc,qBAAqB,CAAC;AACpC,cAAc,aAAa,CAAC;AAC5B,cAAc,WAAW,CAAC;AAC1B,cAAc,sBAAsB,CAAC;AACrC,cAAc,SAAS,CAAC;AACxB,cAAc,gBAAgB,CAAC;AAC/B,cAAc,eAAe,CAAC;AAC9B,cAAc,UAAU,CAAC;AACzB,cAAc,gBAAgB,CAAC;AAC/B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,YAAY,CAAC;AAC3B,cAAc,mBAAmB,CAAC;AAClC,cAAc,aAAa,CAAC;AAC5B,cAAc,iBAAiB,CAAC;AAChC,cAAc,eAAe,CAAC;AAC9B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,UAAU,CAAC;AACzB,cAAc,gBAAgB,CAAC;AAC/B,cAAc,cAAc,CAAC;AAC7B,cAAc,aAAa,CAAC;AAC5B,cAAc,WAAW,CAAC"}
|
package/dist/esm/util/index.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/util/index.ts"],
|
|
4
|
-
"mappings": "AAAA;AAAA;AAAA;AAAA;AAAA;AAMA,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;",
|
|
4
|
+
"mappings": "AAAA;AAAA;AAAA;AAAA;AAAA;AAMA,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;AACd,cAAc;",
|
|
5
5
|
"names": []
|
|
6
6
|
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@matter/general",
|
|
3
|
-
"version": "0.16.
|
|
3
|
+
"version": "0.16.8-alpha.0-20260125-38e62bc3e",
|
|
4
4
|
"description": "Non-Matter support for Matter.js",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"iot",
|
|
@@ -36,7 +36,7 @@
|
|
|
36
36
|
"@noble/curves": "^2.0.1"
|
|
37
37
|
},
|
|
38
38
|
"devDependencies": {
|
|
39
|
-
"@matter/testing": "0.16.
|
|
39
|
+
"@matter/testing": "0.16.8-alpha.0-20260125-38e62bc3e"
|
|
40
40
|
},
|
|
41
41
|
"files": [
|
|
42
42
|
"dist/**/*",
|
package/src/MatterError.ts
CHANGED
|
@@ -100,6 +100,30 @@ export class MatterError extends Error {
|
|
|
100
100
|
}
|
|
101
101
|
}
|
|
102
102
|
|
|
103
|
+
/**
|
|
104
|
+
* Replace the message in an error.
|
|
105
|
+
*
|
|
106
|
+
* In addition to setting the message, updates the message in the stack.
|
|
107
|
+
*/
|
|
108
|
+
static replaceMessage(error: Error, message: string) {
|
|
109
|
+
const oldMessage = error.message;
|
|
110
|
+
error.message = message;
|
|
111
|
+
|
|
112
|
+
const stack = error.stack?.split("\n");
|
|
113
|
+
const messagePos = stack?.findIndex(line => {
|
|
114
|
+
if (line.startsWith("Error: ")) {
|
|
115
|
+
line = line.slice(7);
|
|
116
|
+
}
|
|
117
|
+
if (line === oldMessage) {
|
|
118
|
+
return true;
|
|
119
|
+
}
|
|
120
|
+
});
|
|
121
|
+
if (messagePos !== undefined && messagePos !== -1) {
|
|
122
|
+
stack![messagePos] = message;
|
|
123
|
+
error.stack = stack!.join("\n");
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
103
127
|
/**
|
|
104
128
|
* The fallback formatter factory. This produces a limited plaintext formatter.
|
|
105
129
|
*/
|
|
@@ -291,10 +315,10 @@ export class TimeoutError extends MatterError {
|
|
|
291
315
|
}
|
|
292
316
|
|
|
293
317
|
/**
|
|
294
|
-
* Thrown
|
|
318
|
+
* Thrown as the primary cause when an {@link AbortController} aborts.
|
|
295
319
|
*/
|
|
296
320
|
export class AbortedError extends CanceledError {
|
|
297
|
-
constructor(message = "
|
|
321
|
+
constructor(message = "Operation aborted", options?: ErrorOptions) {
|
|
298
322
|
super(message, options);
|
|
299
323
|
}
|
|
300
324
|
|
|
@@ -320,6 +344,11 @@ export class AbortedError extends CanceledError {
|
|
|
320
344
|
}
|
|
321
345
|
}
|
|
322
346
|
|
|
347
|
+
/**
|
|
348
|
+
* Thrown when an operation can't complete because a resource is closed.
|
|
349
|
+
*/
|
|
350
|
+
export class ClosedError extends CanceledError {}
|
|
351
|
+
|
|
323
352
|
/**
|
|
324
353
|
* Node.js-style object inspection.
|
|
325
354
|
*
|
|
@@ -85,8 +85,8 @@ export class SharedEnvironmentServices implements ServiceProvider {
|
|
|
85
85
|
/**
|
|
86
86
|
* Load an environmental service asynchronously and register this instance as a consumer.
|
|
87
87
|
*
|
|
88
|
-
* Waits for the service's construction promise to resolve if present, then tracks
|
|
89
|
-
*
|
|
88
|
+
* Waits for the service's construction promise to resolve if present, then tracks the service for lifecycle
|
|
89
|
+
* management.
|
|
90
90
|
*/
|
|
91
91
|
async load<T extends Environmental.Service>(type: Environmental.Factory<T>): Promise<T> {
|
|
92
92
|
this.#assertClosed();
|
|
@@ -15,7 +15,8 @@ type SupportedStorageBaseTypes = string | number | boolean | bigint | Bytes;
|
|
|
15
15
|
|
|
16
16
|
/** Supported combined types to stringify the data for the storage that can be used as values. */
|
|
17
17
|
type SupportedComplexStorageTypes =
|
|
18
|
-
|
|
|
18
|
+
| (SupportedStorageBaseTypes | SupportedComplexStorageTypes)[] // Arrays
|
|
19
|
+
| readonly (SupportedStorageBaseTypes | SupportedComplexStorageTypes)[] // Arrays
|
|
19
20
|
| { [key: string]: SupportedStorageBaseTypes | SupportedComplexStorageTypes | null | undefined } // Objects
|
|
20
21
|
| Array<[SupportedStorageBaseTypes, SupportedStorageBaseTypes | SupportedComplexStorageTypes | null | undefined]> // Map style arrays
|
|
21
22
|
| Map<SupportedStorageBaseTypes, SupportedStorageBaseTypes | SupportedComplexStorageTypes>
|
package/src/util/Abort.ts
CHANGED
|
@@ -23,13 +23,15 @@ import { SafePromise } from "./Promises.js";
|
|
|
23
23
|
* Optionally will register for abort with an outer {@link AbortController} and/or add a timeout. You must abort or
|
|
24
24
|
* invoke {@link close} if you use either of these options.
|
|
25
25
|
*/
|
|
26
|
-
export class Abort
|
|
26
|
+
export class Abort
|
|
27
|
+
extends Callable<[reason?: string | Error]>
|
|
28
|
+
implements AbortController, AbortSignal, PromiseLike<Error>
|
|
29
|
+
{
|
|
27
30
|
// The native controller implementation
|
|
28
31
|
#controller: AbortController;
|
|
29
32
|
|
|
30
33
|
// Optional abort chaining
|
|
31
|
-
#
|
|
32
|
-
#listener?: (reason: any) => void;
|
|
34
|
+
#unregisterDependencies?: () => void;
|
|
33
35
|
|
|
34
36
|
// Optional PromiseLike behavior
|
|
35
37
|
#aborted?: Promise<Error>;
|
|
@@ -38,49 +40,106 @@ export class Abort extends Callable<[reason?: Error]> implements AbortController
|
|
|
38
40
|
// Optional timeout
|
|
39
41
|
#timeout?: Timer;
|
|
40
42
|
|
|
41
|
-
constructor({ abort, timeout, handler }: Abort.Options = {}) {
|
|
42
|
-
|
|
43
|
+
constructor({ abort: aborts, timeout, handler, timeoutHandler }: Abort.Options = {}) {
|
|
44
|
+
const abort = (reason?: Error | string) => {
|
|
45
|
+
if (typeof reason === "string") {
|
|
46
|
+
reason = new AbortedError(reason);
|
|
47
|
+
}
|
|
48
|
+
this.abort(reason);
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
super(abort);
|
|
43
52
|
|
|
44
53
|
this.#controller = new AbortController();
|
|
45
54
|
|
|
55
|
+
const throwIfAborted = this.#controller.signal.throwIfAborted.bind(this.#controller.signal);
|
|
56
|
+
this.#controller.signal.throwIfAborted = () => {
|
|
57
|
+
try {
|
|
58
|
+
throwIfAborted();
|
|
59
|
+
} catch (e) {
|
|
60
|
+
const error = new AbortedError();
|
|
61
|
+
|
|
62
|
+
// Remove stack lines for this abort logic
|
|
63
|
+
error.stack = error.stack
|
|
64
|
+
?.split("\n")
|
|
65
|
+
.filter(line => !line.match(/\.throwIfAborted/))
|
|
66
|
+
.join("\n");
|
|
67
|
+
|
|
68
|
+
error.cause = e;
|
|
69
|
+
throw error;
|
|
70
|
+
}
|
|
71
|
+
};
|
|
72
|
+
|
|
46
73
|
const self = (reason?: any) => {
|
|
47
74
|
this.abort(reason);
|
|
48
75
|
};
|
|
49
76
|
Object.setPrototypeOf(self, Object.getPrototypeOf(this));
|
|
50
77
|
|
|
51
|
-
if (
|
|
52
|
-
|
|
78
|
+
if (aborts && !Array.isArray(aborts)) {
|
|
79
|
+
aborts = [aborts];
|
|
53
80
|
}
|
|
54
81
|
|
|
55
|
-
if (
|
|
56
|
-
const
|
|
57
|
-
|
|
82
|
+
if (aborts?.length) {
|
|
83
|
+
const dependencies = aborts.map(abort => abort && ("signal" in abort ? abort.signal : abort));
|
|
84
|
+
|
|
85
|
+
for (const dependency of dependencies) {
|
|
86
|
+
if (dependency === undefined) {
|
|
87
|
+
continue;
|
|
88
|
+
}
|
|
58
89
|
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
90
|
+
const listener = () => this.abort(asError(dependency.reason));
|
|
91
|
+
dependency.addEventListener("abort", listener);
|
|
92
|
+
const unregisterPrev = this.#unregisterDependencies;
|
|
93
|
+
this.#unregisterDependencies = () => {
|
|
94
|
+
unregisterPrev?.();
|
|
95
|
+
dependency.removeEventListener("abort", listener);
|
|
96
|
+
};
|
|
62
97
|
}
|
|
63
98
|
}
|
|
64
99
|
|
|
65
|
-
if (timeout) {
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
100
|
+
if (timeout !== undefined) {
|
|
101
|
+
if (timeoutHandler) {
|
|
102
|
+
const original = timeoutHandler;
|
|
103
|
+
timeoutHandler = () => {
|
|
104
|
+
try {
|
|
105
|
+
original.call(this);
|
|
106
|
+
} catch (e) {
|
|
107
|
+
this.abort(asError(e));
|
|
108
|
+
}
|
|
109
|
+
};
|
|
110
|
+
} else {
|
|
111
|
+
timeoutHandler = () => this.abort(new TimeoutError());
|
|
112
|
+
}
|
|
70
113
|
|
|
71
|
-
|
|
72
|
-
|
|
114
|
+
if (timeout <= 0) {
|
|
115
|
+
timeoutHandler.call(this);
|
|
116
|
+
} else {
|
|
117
|
+
this.#timeout = Time.getPeriodicTimer("subtask timeout", timeout, () => {
|
|
118
|
+
if (this.aborted) {
|
|
119
|
+
return;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
timeoutHandler!.call(this);
|
|
123
|
+
});
|
|
73
124
|
|
|
74
|
-
|
|
125
|
+
this.#timeout.start();
|
|
126
|
+
}
|
|
75
127
|
}
|
|
76
128
|
|
|
77
129
|
if (handler) {
|
|
78
|
-
|
|
130
|
+
if (this.aborted) {
|
|
131
|
+
handler.call(this, this.reason);
|
|
132
|
+
} else {
|
|
133
|
+
this.addEventListener("abort", () => handler.call(this, this.reason));
|
|
134
|
+
}
|
|
79
135
|
}
|
|
80
136
|
}
|
|
81
137
|
|
|
82
|
-
abort(reason?:
|
|
83
|
-
|
|
138
|
+
abort(reason?: Error | string) {
|
|
139
|
+
if (typeof reason === "string") {
|
|
140
|
+
reason = new AbortedError(reason);
|
|
141
|
+
}
|
|
142
|
+
this.#controller.abort(reason ?? new AbortedError("Operation aborted with no reason given"));
|
|
84
143
|
}
|
|
85
144
|
|
|
86
145
|
get signal() {
|
|
@@ -100,7 +159,7 @@ export class Abort extends Callable<[reason?: Error]> implements AbortController
|
|
|
100
159
|
* Race with throw on abort.
|
|
101
160
|
*/
|
|
102
161
|
async attempt<T>(...promises: Array<T | PromiseLike<T>>) {
|
|
103
|
-
return Abort.attempt(this, ...promises);
|
|
162
|
+
return await Abort.attempt(this, ...promises);
|
|
104
163
|
}
|
|
105
164
|
|
|
106
165
|
/**
|
|
@@ -111,11 +170,7 @@ export class Abort extends Callable<[reason?: Error]> implements AbortController
|
|
|
111
170
|
*/
|
|
112
171
|
close() {
|
|
113
172
|
this.#timeout?.stop();
|
|
114
|
-
|
|
115
|
-
for (const dependent of this.#dependents) {
|
|
116
|
-
dependent.removeEventListener("abort", this.#listener);
|
|
117
|
-
}
|
|
118
|
-
}
|
|
173
|
+
this.#unregisterDependencies?.();
|
|
119
174
|
}
|
|
120
175
|
|
|
121
176
|
[Symbol.dispose]() {
|
|
@@ -208,7 +263,7 @@ export namespace Abort {
|
|
|
208
263
|
*
|
|
209
264
|
* This functions similarly to {@link AbortSignal.any} but has additional protection against memory leaks.
|
|
210
265
|
*/
|
|
211
|
-
abort?: Signal | Signal[];
|
|
266
|
+
abort?: Signal | (Signal | undefined)[];
|
|
212
267
|
|
|
213
268
|
/**
|
|
214
269
|
* An abort timeout.
|
|
@@ -220,7 +275,14 @@ export namespace Abort {
|
|
|
220
275
|
/**
|
|
221
276
|
* Adds a default abort handler.
|
|
222
277
|
*/
|
|
223
|
-
handler?: (reason?: Error) => void;
|
|
278
|
+
handler?: (this: Abort, reason?: Error) => void;
|
|
279
|
+
|
|
280
|
+
/**
|
|
281
|
+
* Replaces the default timeout handler.
|
|
282
|
+
*
|
|
283
|
+
* The default implementation aborts with {@link TimeoutError}.
|
|
284
|
+
*/
|
|
285
|
+
timeoutHandler?: (this: Abort) => void;
|
|
224
286
|
}
|
|
225
287
|
|
|
226
288
|
/**
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright 2022-2026 Matter.js Authors
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Utilities for working with async iterators.
|
|
9
|
+
*/
|
|
10
|
+
export namespace AsyncIterator {
|
|
11
|
+
/**
|
|
12
|
+
* Merge multiple async iterators, yielding results as they become available from any iterator.
|
|
13
|
+
*
|
|
14
|
+
* Results are yielded in the order they resolve, not the order of iterators.
|
|
15
|
+
* Errors from individual iterators are collected and thrown as an aggregate error after all iterators complete.
|
|
16
|
+
*
|
|
17
|
+
* @param iterators The async iterables to merge
|
|
18
|
+
* @param errorMessage Optional message for the aggregate error if any iterators fail
|
|
19
|
+
*/
|
|
20
|
+
export async function* merge<T>(
|
|
21
|
+
iterators: AsyncIterable<T>[],
|
|
22
|
+
errorMessage = "One or more async iterators failed",
|
|
23
|
+
): AsyncGenerator<T> {
|
|
24
|
+
const asyncIterators = iterators.map(iter => iter[Symbol.asyncIterator]());
|
|
25
|
+
const pending = new Map<number, Promise<{ index: number; result: IteratorResult<T> }>>();
|
|
26
|
+
const errors: Error[] = [];
|
|
27
|
+
|
|
28
|
+
// Initialize with first .next() call for each iterator
|
|
29
|
+
for (let i = 0; i < asyncIterators.length; i++) {
|
|
30
|
+
pending.set(
|
|
31
|
+
i,
|
|
32
|
+
asyncIterators[i].next().then(
|
|
33
|
+
result => ({ index: i, result }),
|
|
34
|
+
error => {
|
|
35
|
+
// On error, mark as done and collect error
|
|
36
|
+
errors.push(error);
|
|
37
|
+
return { index: i, result: { done: true, value: undefined } as IteratorResult<T> };
|
|
38
|
+
},
|
|
39
|
+
),
|
|
40
|
+
);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
while (pending.size > 0) {
|
|
44
|
+
// Race all pending promises
|
|
45
|
+
const { index, result } = await Promise.race(pending.values());
|
|
46
|
+
|
|
47
|
+
if (result.done) {
|
|
48
|
+
pending.delete(index);
|
|
49
|
+
} else {
|
|
50
|
+
yield result.value;
|
|
51
|
+
// Queue next value from this iterator
|
|
52
|
+
pending.set(
|
|
53
|
+
index,
|
|
54
|
+
asyncIterators[index].next().then(
|
|
55
|
+
result => ({ index, result }),
|
|
56
|
+
error => {
|
|
57
|
+
errors.push(error);
|
|
58
|
+
return { index, result: { done: true, value: undefined } as IteratorResult<T> };
|
|
59
|
+
},
|
|
60
|
+
),
|
|
61
|
+
);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
// After all iterators complete, throw aggregate error if any occurred
|
|
66
|
+
if (errors.length > 0) {
|
|
67
|
+
throw new AggregateError(errors, errorMessage);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
}
|
|
@@ -8,8 +8,9 @@
|
|
|
8
8
|
|
|
9
9
|
import { Duration } from "#time/Duration.js";
|
|
10
10
|
import { Minutes } from "#time/TimeUnit.js";
|
|
11
|
-
import { MatterFlowError } from "../MatterError.js";
|
|
11
|
+
import { AbortedError, MatterFlowError } from "../MatterError.js";
|
|
12
12
|
import { Time, Timer } from "../time/Time.js";
|
|
13
|
+
import { Abort } from "./Abort.js";
|
|
13
14
|
import { asError } from "./Error.js";
|
|
14
15
|
import { createPromise } from "./Promises.js";
|
|
15
16
|
import { EndOfStreamError, NoResponseTimeoutError } from "./Streams.js";
|
|
@@ -19,7 +20,7 @@ export class DataReadQueue<T> {
|
|
|
19
20
|
#pendingRead?: { resolver: (data: T) => void; rejecter: (reason: any) => void; timeoutTimer?: Timer };
|
|
20
21
|
#closed = false;
|
|
21
22
|
|
|
22
|
-
async read(timeout = Minutes.one): Promise<T> {
|
|
23
|
+
async read({ timeout = Minutes.one, abort }: { timeout?: Duration; abort?: AbortSignal } = {}): Promise<T> {
|
|
23
24
|
const { promise, resolver, rejecter } = createPromise<T>();
|
|
24
25
|
if (this.#closed) throw new EndOfStreamError();
|
|
25
26
|
const data = this.#queue.shift();
|
|
@@ -39,22 +40,39 @@ export class DataReadQueue<T> {
|
|
|
39
40
|
).start(),
|
|
40
41
|
};
|
|
41
42
|
|
|
43
|
+
let localAbort: Abort | undefined;
|
|
42
44
|
try {
|
|
45
|
+
if (abort) {
|
|
46
|
+
localAbort = new Abort({
|
|
47
|
+
abort,
|
|
48
|
+
|
|
49
|
+
handler: reason => {
|
|
50
|
+
this.#clearPendingRead();
|
|
51
|
+
rejecter(reason);
|
|
52
|
+
},
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
|
|
43
56
|
return await promise;
|
|
44
57
|
} catch (e) {
|
|
58
|
+
if (e instanceof AbortedError) {
|
|
59
|
+
throw e;
|
|
60
|
+
}
|
|
61
|
+
|
|
45
62
|
// The stack trace where we created the error is useless (either a timer or close()) so replace here
|
|
46
63
|
const error = asError(e);
|
|
47
64
|
error.stack = new Error().stack;
|
|
48
65
|
throw error;
|
|
66
|
+
} finally {
|
|
67
|
+
localAbort?.close();
|
|
49
68
|
}
|
|
50
69
|
}
|
|
51
70
|
|
|
52
71
|
write(data: T) {
|
|
53
72
|
if (this.#closed) throw new EndOfStreamError();
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
this.#pendingRead = undefined;
|
|
73
|
+
const pendingRead = this.#pendingRead;
|
|
74
|
+
this.#clearPendingRead();
|
|
75
|
+
if (pendingRead) {
|
|
58
76
|
pendingRead.resolver(data);
|
|
59
77
|
return;
|
|
60
78
|
}
|
|
@@ -72,4 +90,13 @@ export class DataReadQueue<T> {
|
|
|
72
90
|
this.#pendingRead.timeoutTimer?.stop();
|
|
73
91
|
this.#pendingRead.rejecter(new EndOfStreamError());
|
|
74
92
|
}
|
|
93
|
+
|
|
94
|
+
#clearPendingRead() {
|
|
95
|
+
if (this.#pendingRead === undefined) {
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
this.#pendingRead.timeoutTimer?.stop();
|
|
100
|
+
this.#pendingRead = undefined;
|
|
101
|
+
}
|
|
75
102
|
}
|