@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.
Files changed (65) hide show
  1. package/dist/cjs/MatterError.d.ts +12 -1
  2. package/dist/cjs/MatterError.d.ts.map +1 -1
  3. package/dist/cjs/MatterError.js +26 -1
  4. package/dist/cjs/MatterError.js.map +1 -1
  5. package/dist/cjs/environment/SharedEnvironmentServices.d.ts +2 -2
  6. package/dist/cjs/environment/SharedEnvironmentServices.js +2 -2
  7. package/dist/cjs/storage/StringifyTools.d.ts +1 -1
  8. package/dist/cjs/storage/StringifyTools.d.ts.map +1 -1
  9. package/dist/cjs/storage/StringifyTools.js.map +1 -1
  10. package/dist/cjs/util/Abort.d.ts +11 -5
  11. package/dist/cjs/util/Abort.d.ts.map +1 -1
  12. package/dist/cjs/util/Abort.js +71 -29
  13. package/dist/cjs/util/Abort.js.map +1 -1
  14. package/dist/cjs/util/AsyncIterator.d.ts +21 -0
  15. package/dist/cjs/util/AsyncIterator.d.ts.map +1 -0
  16. package/dist/cjs/util/AsyncIterator.js +71 -0
  17. package/dist/cjs/util/AsyncIterator.js.map +6 -0
  18. package/dist/cjs/util/DataReadQueue.d.ts +4 -1
  19. package/dist/cjs/util/DataReadQueue.d.ts.map +1 -1
  20. package/dist/cjs/util/DataReadQueue.js +27 -5
  21. package/dist/cjs/util/DataReadQueue.js.map +1 -1
  22. package/dist/cjs/util/Semaphore.d.ts.map +1 -1
  23. package/dist/cjs/util/Semaphore.js +5 -5
  24. package/dist/cjs/util/Semaphore.js.map +1 -1
  25. package/dist/cjs/util/index.d.ts +1 -0
  26. package/dist/cjs/util/index.d.ts.map +1 -1
  27. package/dist/cjs/util/index.js +1 -0
  28. package/dist/cjs/util/index.js.map +1 -1
  29. package/dist/esm/MatterError.d.ts +12 -1
  30. package/dist/esm/MatterError.d.ts.map +1 -1
  31. package/dist/esm/MatterError.js +26 -1
  32. package/dist/esm/MatterError.js.map +1 -1
  33. package/dist/esm/environment/SharedEnvironmentServices.d.ts +2 -2
  34. package/dist/esm/environment/SharedEnvironmentServices.js +2 -2
  35. package/dist/esm/storage/StringifyTools.d.ts +1 -1
  36. package/dist/esm/storage/StringifyTools.d.ts.map +1 -1
  37. package/dist/esm/storage/StringifyTools.js.map +1 -1
  38. package/dist/esm/util/Abort.d.ts +11 -5
  39. package/dist/esm/util/Abort.d.ts.map +1 -1
  40. package/dist/esm/util/Abort.js +71 -29
  41. package/dist/esm/util/Abort.js.map +1 -1
  42. package/dist/esm/util/AsyncIterator.d.ts +21 -0
  43. package/dist/esm/util/AsyncIterator.d.ts.map +1 -0
  44. package/dist/esm/util/AsyncIterator.js +51 -0
  45. package/dist/esm/util/AsyncIterator.js.map +6 -0
  46. package/dist/esm/util/DataReadQueue.d.ts +4 -1
  47. package/dist/esm/util/DataReadQueue.d.ts.map +1 -1
  48. package/dist/esm/util/DataReadQueue.js +28 -6
  49. package/dist/esm/util/DataReadQueue.js.map +1 -1
  50. package/dist/esm/util/Semaphore.d.ts.map +1 -1
  51. package/dist/esm/util/Semaphore.js +6 -6
  52. package/dist/esm/util/Semaphore.js.map +1 -1
  53. package/dist/esm/util/index.d.ts +1 -0
  54. package/dist/esm/util/index.d.ts.map +1 -1
  55. package/dist/esm/util/index.js +1 -0
  56. package/dist/esm/util/index.js.map +1 -1
  57. package/package.json +2 -2
  58. package/src/MatterError.ts +31 -2
  59. package/src/environment/SharedEnvironmentServices.ts +2 -2
  60. package/src/storage/StringifyTools.ts +2 -1
  61. package/src/util/Abort.ts +94 -32
  62. package/src/util/AsyncIterator.ts +70 -0
  63. package/src/util/DataReadQueue.ts +33 -6
  64. package/src/util/Semaphore.ts +8 -7
  65. 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
- if (this.#pendingRead !== void 0) {
51
- this.#pendingRead.timeoutTimer?.stop();
52
- const pendingRead = this.#pendingRead;
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;AAChC,SAAS,YAAmB;AAC5B,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,UAAU,QAAQ,KAAiB;AAC1C,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;AACA,aAAO,MAAM;AAAA,IACjB,SAAS,GAAG;AAER,YAAM,QAAQ,QAAQ,CAAC;AACvB,YAAM,QAAQ,IAAI,MAAM,EAAE;AAC1B,YAAM;AAAA,IACV;AAAA,EACJ;AAAA,EAEA,MAAM,MAAS;AACX,QAAI,KAAK,QAAS,OAAM,IAAI,iBAAiB;AAC7C,QAAI,KAAK,iBAAiB,QAAW;AACjC,WAAK,aAAa,cAAc,KAAK;AACrC,YAAM,cAAc,KAAK;AACzB,WAAK,eAAe;AACpB,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;AACJ;",
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;IAsIzD;;OAEG;IACH,KAAK,IAAI,IAAI;IASb;;OAEG;IACH,IAAI,KAAK,WAER;IAED;;OAEG;IACH,IAAI,OAAO,WAEV;IAED;;OAEG;IACH,KAAK,IAAI,IAAI;CAMhB"}
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 AbortedError("Queue is closed");
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
- const reason = combinedAbort.reason;
115
- throw reason instanceof AbortedError ? reason : new AbortedError();
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 AbortedError("Queue cleared"));
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 AbortedError("Queue is closed"));
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": "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",
4
+ "mappings": "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",
5
5
  "names": []
6
6
  }
@@ -5,6 +5,7 @@
5
5
  */
6
6
  export * from "./Abort.js";
7
7
  export * from "./Array.js";
8
+ export * from "./AsyncIterator.js";
8
9
  export * from "./Boot.js";
9
10
  export * from "./Bytes.js";
10
11
  export * from "./Cache.js";
@@ -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"}
@@ -5,6 +5,7 @@
5
5
  */
6
6
  export * from "./Abort.js";
7
7
  export * from "./Array.js";
8
+ export * from "./AsyncIterator.js";
8
9
  export * from "./Boot.js";
9
10
  export * from "./Bytes.js";
10
11
  export * from "./Cache.js";
@@ -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.7",
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.7"
39
+ "@matter/testing": "0.16.8-alpha.0-20260125-38e62bc3e"
40
40
  },
41
41
  "files": [
42
42
  "dist/**/*",
@@ -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 on abort when there is not an underlying error.
318
+ * Thrown as the primary cause when an {@link AbortController} aborts.
295
319
  */
296
320
  export class AbortedError extends CanceledError {
297
- constructor(message = "This operation was aborted", options?: ErrorOptions) {
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
- * the service for lifecycle management.
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
- | Array<SupportedStorageBaseTypes | SupportedComplexStorageTypes> // Arrays
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 extends Callable<[reason?: Error]> implements AbortController, AbortSignal, PromiseLike<Error> {
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
- #dependents?: AbortSignal[];
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
- super(() => this.abort());
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 (abort && !Array.isArray(abort)) {
52
- abort = [abort];
78
+ if (aborts && !Array.isArray(aborts)) {
79
+ aborts = [aborts];
53
80
  }
54
81
 
55
- if (abort?.length) {
56
- const dependents = abort.map(abort => ("signal" in abort ? abort.signal : abort));
57
- this.#dependents = dependents;
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
- this.#listener = (reason: any) => this.abort(reason);
60
- for (const dependent of dependents) {
61
- dependent.addEventListener("abort", this.#listener);
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
- this.#timeout = Time.getPeriodicTimer("subtask timeout", timeout, () => {
67
- if (this.aborted) {
68
- return;
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
- this.abort(new TimeoutError());
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
- this.#timeout.start();
125
+ this.#timeout.start();
126
+ }
75
127
  }
76
128
 
77
129
  if (handler) {
78
- this.addEventListener("abort", () => handler(this.reason));
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?: any) {
83
- this.#controller.abort(reason ?? new AbortedError());
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
- if (this.#listener && this.#dependents) {
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
- if (this.#pendingRead !== undefined) {
55
- this.#pendingRead.timeoutTimer?.stop();
56
- const pendingRead = this.#pendingRead;
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
  }