@aztec/foundation 5.0.0-private.20260319 → 5.0.0-rc.2

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 (246) hide show
  1. package/dest/bigint-buffer/index.d.ts +1 -1
  2. package/dest/bigint-buffer/index.d.ts.map +1 -1
  3. package/dest/bigint-buffer/index.js +6 -1
  4. package/dest/branded-types/block_number.d.ts +3 -3
  5. package/dest/branded-types/block_number.d.ts.map +1 -1
  6. package/dest/branded-types/block_number.js +15 -4
  7. package/dest/branded-types/buffer32_hash.d.ts +27 -0
  8. package/dest/branded-types/buffer32_hash.d.ts.map +1 -0
  9. package/dest/branded-types/buffer32_hash.js +12 -0
  10. package/dest/branded-types/checkpoint_number.d.ts +9 -3
  11. package/dest/branded-types/checkpoint_number.d.ts.map +1 -1
  12. package/dest/branded-types/checkpoint_number.js +11 -0
  13. package/dest/branded-types/epoch.d.ts +2 -2
  14. package/dest/branded-types/epoch.d.ts.map +1 -1
  15. package/dest/branded-types/index.d.ts +2 -1
  16. package/dest/branded-types/index.d.ts.map +1 -1
  17. package/dest/branded-types/index.js +1 -0
  18. package/dest/branded-types/index_within_checkpoint.d.ts +2 -2
  19. package/dest/branded-types/index_within_checkpoint.d.ts.map +1 -1
  20. package/dest/branded-types/slot.d.ts +5 -2
  21. package/dest/branded-types/slot.d.ts.map +1 -1
  22. package/dest/branded-types/slot.js +3 -0
  23. package/dest/buffer/buffer16.d.ts +3 -1
  24. package/dest/buffer/buffer16.d.ts.map +1 -1
  25. package/dest/buffer/buffer32.d.ts +28 -61
  26. package/dest/buffer/buffer32.d.ts.map +1 -1
  27. package/dest/buffer/buffer32.js +29 -63
  28. package/dest/collection/array.d.ts +11 -1
  29. package/dest/collection/array.d.ts.map +1 -1
  30. package/dest/collection/array.js +33 -0
  31. package/dest/collection/index.d.ts +3 -1
  32. package/dest/collection/index.d.ts.map +1 -1
  33. package/dest/collection/index.js +2 -0
  34. package/dest/collection/lru_map.d.ts +41 -0
  35. package/dest/collection/lru_map.d.ts.map +1 -0
  36. package/dest/collection/lru_map.js +119 -0
  37. package/dest/collection/lru_set.d.ts +35 -0
  38. package/dest/collection/lru_set.d.ts.map +1 -0
  39. package/dest/collection/lru_set.js +95 -0
  40. package/dest/committable/committable.js +1 -1
  41. package/dest/config/env_var.d.ts +2 -2
  42. package/dest/config/env_var.d.ts.map +1 -1
  43. package/dest/config/index.d.ts +47 -30
  44. package/dest/config/index.d.ts.map +1 -1
  45. package/dest/config/index.js +30 -45
  46. package/dest/config/network_config.d.ts +11 -55
  47. package/dest/config/network_config.d.ts.map +1 -1
  48. package/dest/config/network_name.d.ts +2 -2
  49. package/dest/config/network_name.d.ts.map +1 -1
  50. package/dest/config/network_name.js +1 -3
  51. package/dest/config/secret_value.d.ts +2 -2
  52. package/dest/config/secret_value.d.ts.map +1 -1
  53. package/dest/crypto/aes128/index.d.ts +2 -1
  54. package/dest/crypto/aes128/index.d.ts.map +1 -1
  55. package/dest/crypto/aes128/index.js +11 -2
  56. package/dest/crypto/bls/bn254_keystore.d.ts +9 -176
  57. package/dest/crypto/bls/bn254_keystore.d.ts.map +1 -1
  58. package/dest/crypto/bls/bn254_keystore.js +0 -17
  59. package/dest/crypto/poseidon/index.d.ts +1 -1
  60. package/dest/crypto/poseidon/index.d.ts.map +1 -1
  61. package/dest/crypto/poseidon/index.js +40 -33
  62. package/dest/crypto/schnorr/index.d.ts +8 -7
  63. package/dest/crypto/schnorr/index.d.ts.map +1 -1
  64. package/dest/crypto/schnorr/index.js +6 -6
  65. package/dest/crypto/schnorr/signature.d.ts +9 -29
  66. package/dest/crypto/schnorr/signature.d.ts.map +1 -1
  67. package/dest/crypto/schnorr/signature.js +20 -36
  68. package/dest/curves/bls12/field.d.ts +3 -3
  69. package/dest/curves/bls12/field.d.ts.map +1 -1
  70. package/dest/curves/bls12/field.js +0 -5
  71. package/dest/curves/bls12/point.d.ts +2 -2
  72. package/dest/curves/bls12/point.d.ts.map +1 -1
  73. package/dest/curves/bn254/field.d.ts +23 -19
  74. package/dest/curves/bn254/field.d.ts.map +1 -1
  75. package/dest/curves/bn254/field.js +23 -17
  76. package/dest/curves/grumpkin/point.d.ts +14 -28
  77. package/dest/curves/grumpkin/point.d.ts.map +1 -1
  78. package/dest/curves/grumpkin/point.js +29 -43
  79. package/dest/eth-address/index.d.ts +2 -2
  80. package/dest/eth-address/index.d.ts.map +1 -1
  81. package/dest/eth-address/index.js +0 -3
  82. package/dest/eth-signature/eth_signature.d.ts +2 -2
  83. package/dest/eth-signature/eth_signature.d.ts.map +1 -1
  84. package/dest/fifo/fifo_frame_reader.d.ts +41 -0
  85. package/dest/fifo/fifo_frame_reader.d.ts.map +1 -0
  86. package/dest/fifo/fifo_frame_reader.js +74 -0
  87. package/dest/fifo/index.d.ts +2 -0
  88. package/dest/fifo/index.d.ts.map +1 -0
  89. package/dest/fifo/index.js +1 -0
  90. package/dest/fifo_set/fifo_set.d.ts +15 -0
  91. package/dest/fifo_set/fifo_set.d.ts.map +1 -0
  92. package/dest/fifo_set/fifo_set.js +39 -0
  93. package/dest/fifo_set/index.d.ts +2 -0
  94. package/dest/fifo_set/index.d.ts.map +1 -0
  95. package/dest/fifo_set/index.js +1 -0
  96. package/dest/json-rpc/client/fetch.d.ts +1 -1
  97. package/dest/json-rpc/client/fetch.d.ts.map +1 -1
  98. package/dest/json-rpc/client/fetch.js +4 -3
  99. package/dest/json-rpc/client/safe_json_rpc_client.d.ts +1 -1
  100. package/dest/json-rpc/client/safe_json_rpc_client.d.ts.map +1 -1
  101. package/dest/json-rpc/client/safe_json_rpc_client.js +2 -2
  102. package/dest/json-rpc/client/undici.d.ts +1 -1
  103. package/dest/json-rpc/client/undici.d.ts.map +1 -1
  104. package/dest/json-rpc/client/undici.js +5 -3
  105. package/dest/json-rpc/fixtures/test_state.d.ts +4 -8
  106. package/dest/json-rpc/fixtures/test_state.d.ts.map +1 -1
  107. package/dest/json-rpc/fixtures/test_state.js +57 -17
  108. package/dest/json-rpc/server/safe_json_rpc_server.d.ts +8 -3
  109. package/dest/json-rpc/server/safe_json_rpc_server.d.ts.map +1 -1
  110. package/dest/json-rpc/server/safe_json_rpc_server.js +27 -13
  111. package/dest/log/pino-logger.d.ts +1 -1
  112. package/dest/log/pino-logger.d.ts.map +1 -1
  113. package/dest/log/pino-logger.js +17 -2
  114. package/dest/noir/noir_package_config.d.ts +19 -104
  115. package/dest/noir/noir_package_config.d.ts.map +1 -1
  116. package/dest/noir/noir_package_config.js +1 -1
  117. package/dest/queue/batch_queue.d.ts +1 -1
  118. package/dest/queue/batch_queue.d.ts.map +1 -1
  119. package/dest/queue/batch_queue.js +1 -0
  120. package/dest/retry/index.d.ts +31 -3
  121. package/dest/retry/index.d.ts.map +1 -1
  122. package/dest/retry/index.js +44 -2
  123. package/dest/schemas/api.d.ts +12 -10
  124. package/dest/schemas/api.d.ts.map +1 -1
  125. package/dest/schemas/api.js +7 -1
  126. package/dest/schemas/parse.d.ts +4 -4
  127. package/dest/schemas/parse.d.ts.map +1 -1
  128. package/dest/schemas/parse.js +6 -5
  129. package/dest/schemas/schemas.d.ts +15 -15
  130. package/dest/schemas/types.d.ts +3 -3
  131. package/dest/schemas/types.d.ts.map +1 -1
  132. package/dest/schemas/utils.d.ts +7 -11
  133. package/dest/schemas/utils.d.ts.map +1 -1
  134. package/dest/schemas/utils.js +2 -18
  135. package/dest/serialize/buffer_sink.d.ts +134 -0
  136. package/dest/serialize/buffer_sink.d.ts.map +1 -0
  137. package/dest/serialize/buffer_sink.js +297 -0
  138. package/dest/serialize/index.d.ts +2 -1
  139. package/dest/serialize/index.d.ts.map +1 -1
  140. package/dest/serialize/index.js +1 -0
  141. package/dest/string/index.js +1 -1
  142. package/dest/timer/index.d.ts +2 -2
  143. package/dest/timer/index.d.ts.map +1 -1
  144. package/dest/timer/index.js +1 -1
  145. package/dest/timer/timeout.d.ts +3 -1
  146. package/dest/timer/timeout.d.ts.map +1 -1
  147. package/dest/timer/timeout.js +26 -0
  148. package/dest/transport/dispatch/create_dispatch_proxy.d.ts +11 -2
  149. package/dest/transport/dispatch/create_dispatch_proxy.d.ts.map +1 -1
  150. package/dest/transport/index.d.ts +1 -2
  151. package/dest/transport/index.d.ts.map +1 -1
  152. package/dest/transport/index.js +0 -1
  153. package/dest/trees/balanced_merkle_tree_root.d.ts +2 -3
  154. package/dest/trees/balanced_merkle_tree_root.d.ts.map +1 -1
  155. package/dest/trees/balanced_merkle_tree_root.js +2 -3
  156. package/dest/trees/hasher.d.ts +3 -2
  157. package/dest/trees/hasher.d.ts.map +1 -1
  158. package/dest/trees/hasher.js +5 -5
  159. package/dest/trees/indexed_merkle_tree_calculator.d.ts +1 -1
  160. package/dest/trees/indexed_merkle_tree_calculator.d.ts.map +1 -1
  161. package/dest/trees/indexed_merkle_tree_calculator.js +5 -1
  162. package/dest/trees/membership_witness.d.ts +2 -6
  163. package/dest/trees/membership_witness.d.ts.map +1 -1
  164. package/dest/trees/membership_witness.js +0 -9
  165. package/dest/trees/merkle_tree_calculator.d.ts +4 -2
  166. package/dest/trees/merkle_tree_calculator.d.ts.map +1 -1
  167. package/dest/trees/merkle_tree_calculator.js +1 -5
  168. package/dest/trees/sibling_path.d.ts +5 -4
  169. package/dest/trees/sibling_path.d.ts.map +1 -1
  170. package/dest/trees/sibling_path.js +10 -8
  171. package/dest/trees/unbalanced_merkle_tree_root.d.ts +2 -3
  172. package/dest/trees/unbalanced_merkle_tree_root.d.ts.map +1 -1
  173. package/dest/trees/unbalanced_merkle_tree_root.js +2 -3
  174. package/dest/types/index.d.ts +23 -1
  175. package/dest/types/index.d.ts.map +1 -1
  176. package/dest/types/index.js +15 -0
  177. package/package.json +6 -4
  178. package/src/bigint-buffer/index.ts +6 -1
  179. package/src/branded-types/block_number.ts +20 -2
  180. package/src/branded-types/buffer32_hash.ts +38 -0
  181. package/src/branded-types/checkpoint_number.ts +16 -1
  182. package/src/branded-types/epoch.ts +1 -1
  183. package/src/branded-types/index.ts +1 -0
  184. package/src/branded-types/index_within_checkpoint.ts +1 -1
  185. package/src/branded-types/slot.ts +6 -1
  186. package/src/buffer/buffer16.ts +3 -0
  187. package/src/buffer/buffer32.ts +39 -68
  188. package/src/collection/array.ts +34 -0
  189. package/src/collection/index.ts +2 -0
  190. package/src/collection/lru_map.ts +143 -0
  191. package/src/collection/lru_set.ts +115 -0
  192. package/src/committable/committable.ts +1 -1
  193. package/src/config/env_var.ts +39 -24
  194. package/src/config/index.ts +102 -97
  195. package/src/config/network_name.ts +2 -5
  196. package/src/config/secret_value.ts +1 -1
  197. package/src/crypto/aes128/index.ts +11 -2
  198. package/src/crypto/bls/bn254_keystore.ts +0 -23
  199. package/src/crypto/poseidon/index.ts +42 -34
  200. package/src/crypto/schnorr/index.ts +9 -8
  201. package/src/crypto/schnorr/signature.ts +17 -46
  202. package/src/curves/bls12/field.ts +0 -7
  203. package/src/curves/bn254/field.ts +35 -34
  204. package/src/curves/grumpkin/point.ts +23 -40
  205. package/src/eth-address/index.ts +0 -4
  206. package/src/fifo/fifo_frame_reader.ts +98 -0
  207. package/src/fifo/index.ts +1 -0
  208. package/src/fifo_set/fifo_set.ts +52 -0
  209. package/src/fifo_set/index.ts +1 -0
  210. package/src/json-rpc/client/fetch.ts +4 -3
  211. package/src/json-rpc/client/safe_json_rpc_client.ts +2 -2
  212. package/src/json-rpc/client/undici.ts +5 -3
  213. package/src/json-rpc/fixtures/test_state.ts +10 -10
  214. package/src/json-rpc/server/safe_json_rpc_server.ts +31 -13
  215. package/src/log/pino-logger.ts +19 -2
  216. package/src/noir/noir_package_config.ts +32 -20
  217. package/src/queue/batch_queue.ts +1 -0
  218. package/src/retry/index.ts +66 -4
  219. package/src/schemas/api.ts +25 -25
  220. package/src/schemas/parse.ts +9 -6
  221. package/src/schemas/schemas.ts +4 -4
  222. package/src/schemas/types.ts +2 -2
  223. package/src/schemas/utils.ts +4 -38
  224. package/src/serialize/buffer_sink.ts +350 -0
  225. package/src/serialize/index.ts +1 -0
  226. package/src/string/index.ts +1 -1
  227. package/src/timer/index.ts +1 -1
  228. package/src/timer/timeout.ts +31 -0
  229. package/src/transport/dispatch/create_dispatch_proxy.ts +11 -1
  230. package/src/transport/index.ts +0 -1
  231. package/src/trees/balanced_merkle_tree_root.ts +2 -5
  232. package/src/trees/hasher.ts +6 -3
  233. package/src/trees/indexed_merkle_tree_calculator.ts +5 -1
  234. package/src/trees/membership_witness.ts +0 -8
  235. package/src/trees/merkle_tree_calculator.ts +4 -10
  236. package/src/trees/sibling_path.ts +11 -6
  237. package/src/trees/unbalanced_merkle_tree_root.ts +2 -5
  238. package/src/types/index.ts +47 -0
  239. package/dest/crypto/serialize.d.ts +0 -51
  240. package/dest/crypto/serialize.d.ts.map +0 -1
  241. package/dest/crypto/serialize.js +0 -68
  242. package/dest/transport/dispatch/create_dispatch_fn.d.ts +0 -25
  243. package/dest/transport/dispatch/create_dispatch_fn.d.ts.map +0 -1
  244. package/dest/transport/dispatch/create_dispatch_fn.js +0 -17
  245. package/src/crypto/serialize.ts +0 -85
  246. package/src/transport/dispatch/create_dispatch_fn.ts +0 -35
@@ -7,69 +7,39 @@ import { inspect } from 'util';
7
7
  import { bufferToHex } from '../string/index.js';
8
8
 
9
9
  /**
10
- * A class representing a 32 byte Buffer.
10
+ * Abstract unbranded base class for 32-byte buffers.
11
+ * Extend this instead of Buffer32 when defining a branded subtype
12
+ * to avoid inheriting Buffer32's `_branding`.
11
13
  */
12
- export class Buffer32 {
13
- /**
14
- * The size of the hash in bytes.
15
- */
14
+ export abstract class BaseBuffer32 {
15
+ /** The size of the buffer in bytes. */
16
16
  public static SIZE = 32;
17
17
 
18
- /**
19
- * Buffer32 with value zero.
20
- */
21
- public static ZERO = new Buffer32(Buffer.alloc(Buffer32.SIZE));
22
-
23
18
  constructor(
24
- /**
25
- * The buffer containing the hash.
26
- */
19
+ /** The buffer containing the data. */
27
20
  public buffer: Buffer,
28
21
  ) {
29
- if (buffer.length !== Buffer32.SIZE) {
30
- throw new Error(`Expected buffer to have length ${Buffer32.SIZE} but was ${buffer.length}`);
22
+ if (buffer.length !== BaseBuffer32.SIZE) {
23
+ throw new Error(`Expected buffer to have length ${BaseBuffer32.SIZE} but was ${buffer.length}`);
31
24
  }
32
25
  }
33
26
 
34
- /**
35
- * Returns the raw buffer of the hash.
36
- * @returns The buffer containing the hash.
37
- */
27
+ /** Returns the raw buffer. */
38
28
  public toBuffer() {
39
29
  return this.buffer;
40
30
  }
41
31
 
42
- /**
43
- * Creates a Buffer32 from a buffer.
44
- * @param buffer - The buffer to create from.
45
- * @returns A new Buffer32 object.
46
- */
47
- public static fromBuffer(buffer: Buffer | BufferReader) {
48
- const reader = BufferReader.asReader(buffer);
49
- return new Buffer32(reader.readBytes(Buffer32.SIZE));
50
- }
51
-
52
- /**
53
- * Checks if this hash and another hash are equal.
54
- * @param hash - A hash to compare with.
55
- * @returns True if the hashes are equal, false otherwise.
56
- */
57
- public equals(hash: Buffer32): boolean {
32
+ /** Checks if this and another buffer are equal. */
33
+ public equals(hash: BaseBuffer32): boolean {
58
34
  return this.buffer.equals(hash.buffer);
59
35
  }
60
36
 
61
- /**
62
- * Returns true if this hash is zero.
63
- * @returns True if this hash is zero.
64
- */
37
+ /** Returns true if all bytes are zero. */
65
38
  public isZero(): boolean {
66
39
  return this.buffer.equals(Buffer.alloc(32, 0));
67
40
  }
68
41
 
69
- /**
70
- * Convert this hash to a hex string.
71
- * @returns The hex string.
72
- */
42
+ /** Convert to a hex string. */
73
43
  public toString() {
74
44
  return bufferToHex(this.buffer);
75
45
  }
@@ -82,16 +52,27 @@ export class Buffer32 {
82
52
  return this.toString();
83
53
  }
84
54
 
85
- /**
86
- * Convert this hash to a big int.
87
- * @returns The big int.
88
- */
55
+ /** Convert to a big int. */
89
56
  public toBigInt() {
90
- return deserializeBigInt(this.buffer, 0, Buffer32.SIZE).elem;
57
+ return deserializeBigInt(this.buffer, 0, BaseBuffer32.SIZE).elem;
91
58
  }
92
- /**
93
- * Creates a Buffer32 from a bigint.
94
- */
59
+ }
60
+
61
+ /** A branded 32-byte buffer. */
62
+ export class Buffer32 extends BaseBuffer32 {
63
+ /** Branding for nominal typing. */
64
+ declare private readonly _branding: 'Buffer32';
65
+
66
+ /** Buffer32 with value zero. */
67
+ public static ZERO = new Buffer32(Buffer.alloc(Buffer32.SIZE));
68
+
69
+ /** Creates a Buffer32 from a buffer. */
70
+ public static fromBuffer(buffer: Buffer | BufferReader) {
71
+ const reader = BufferReader.asReader(buffer);
72
+ return new Buffer32(reader.readBytes(Buffer32.SIZE));
73
+ }
74
+
75
+ /** Creates a Buffer32 from a bigint. */
95
76
  public static fromBigInt(hash: bigint) {
96
77
  return new Buffer32(serializeBigInt(hash, Buffer32.SIZE));
97
78
  }
@@ -101,8 +82,7 @@ export class Buffer32 {
101
82
  }
102
83
 
103
84
  /**
104
- * Converts this hash from a buffer of 28 bytes.
105
- * Verifies the input is 28 bytes.
85
+ * Converts from a buffer of 28 bytes.
106
86
  * @param buffer - The 28 byte buffer to construct from.
107
87
  * @returns A Buffer32 created from the input buffer with 4 bytes 0 padding at the front.
108
88
  */
@@ -110,36 +90,27 @@ export class Buffer32 {
110
90
  if (buffer.length != 28) {
111
91
  throw new Error(`Expected Buffer32 input buffer to be 28 bytes`);
112
92
  }
113
- const padded = Buffer.concat([Buffer.alloc(this.SIZE - 28), buffer]);
93
+ const padded = Buffer.concat([Buffer.alloc(Buffer32.SIZE - 28), buffer]);
114
94
  return new Buffer32(padded);
115
95
  }
116
96
 
117
- /**
118
- * Converts a string into a Buffer32 object.
119
- */
97
+ /** Converts a string into a Buffer32 object. */
120
98
  public static fromString(str: string): Buffer32 {
121
99
  if (str.startsWith('0x')) {
122
100
  str = str.slice(2);
123
101
  }
124
- if (str.length !== this.SIZE * 2) {
125
- throw new Error(`Expected string to be ${this.SIZE * 2} characters long, but was ${str.length}`);
102
+ if (str.length !== Buffer32.SIZE * 2) {
103
+ throw new Error(`Expected string to be ${Buffer32.SIZE * 2} characters long, but was ${str.length}`);
126
104
  }
127
105
  return new Buffer32(Buffer.from(str, 'hex'));
128
106
  }
129
107
 
130
- /**
131
- * Converts a number into a Buffer32 object.
132
- * @param num - The number to convert.
133
- * @returns A new Buffer32 object.
134
- */
108
+ /** Converts a number into a Buffer32 object. */
135
109
  public static fromNumber(num: number): Buffer32 {
136
110
  return new Buffer32(serializeBigInt(BigInt(num), Buffer32.SIZE));
137
111
  }
138
112
 
139
- /**
140
- * Generates a random Buffer32.
141
- * @returns A new Buffer32 object.
142
- */
113
+ /** Generates a random Buffer32. */
143
114
  public static random(): Buffer32 {
144
115
  return new Buffer32(Buffer.from(randomBytes(Buffer32.SIZE)));
145
116
  }
@@ -138,6 +138,26 @@ export function unique<T>(arr: T[]): T[] {
138
138
  return [...new Set(arr)];
139
139
  }
140
140
 
141
+ /**
142
+ * Removes duplicates from the given array using a key function. The first occurrence of each key is kept.
143
+ * @param arr - The array.
144
+ * @param keyFn - A function that returns a primitive key for each element. Elements with the same key are
145
+ * considered duplicates.
146
+ * @returns A new array.
147
+ */
148
+ export function uniqueBy<T, K extends string | number | bigint>(arr: T[], keyFn: (item: T) => K): T[] {
149
+ const seen = new Set<K>();
150
+ const result: T[] = [];
151
+ for (const item of arr) {
152
+ const key = keyFn(item);
153
+ if (!seen.has(key)) {
154
+ seen.add(key);
155
+ result.push(item);
156
+ }
157
+ }
158
+ return result;
159
+ }
160
+
141
161
  /**
142
162
  * Removes all undefined elements from the array.
143
163
  * @param arr - The array.
@@ -315,3 +335,17 @@ export function partition<T>(items: T[], predicate: (item: T) => boolean): [T[],
315
335
  }
316
336
  return [pass, fail];
317
337
  }
338
+
339
+ /** Partitions the given iterable into two arrays based on the predicate. */
340
+ export async function partitionAsync<T>(items: T[], predicate: (item: T) => Promise<boolean>): Promise<[T[], T[]]> {
341
+ const pass: T[] = [];
342
+ const fail: T[] = [];
343
+ for (const item of items) {
344
+ if (await predicate(item)) {
345
+ pass.push(item);
346
+ } else {
347
+ fail.push(item);
348
+ }
349
+ }
350
+ return [pass, fail];
351
+ }
@@ -1,2 +1,4 @@
1
1
  export * from './array.js';
2
+ export * from './lru_map.js';
3
+ export * from './lru_set.js';
2
4
  export * from './object.js';
@@ -0,0 +1,143 @@
1
+ /** Node in a doubly-linked list used by {@link LruMap}. */
2
+ type LruNode<K, V> = {
3
+ key: K;
4
+ value: V;
5
+ prev: LruNode<K, V> | undefined;
6
+ next: LruNode<K, V> | undefined;
7
+ };
8
+
9
+ /**
10
+ * A bounded key-value map with Least Recently Used (LRU) eviction.
11
+ * Both {@link get} and {@link set} count as an access and refresh the entry's
12
+ * recency, so entries that are actively used stay in the map longest.
13
+ *
14
+ * Uses a doubly-linked list for O(1) ordering and a Map for O(1) lookup.
15
+ * Head = least recent, tail = most recent.
16
+ */
17
+ export class LruMap<K, V> {
18
+ /** Map from key to its linked-list node for O(1) lookup. */
19
+ private readonly map = new Map<K, LruNode<K, V>>();
20
+ private head: LruNode<K, V> | undefined;
21
+ private tail: LruNode<K, V> | undefined;
22
+
23
+ constructor(private readonly maxSize: number) {
24
+ if (maxSize < 1) {
25
+ throw new Error('LruMap maxSize must be at least 1');
26
+ }
27
+ }
28
+
29
+ /** Number of entries in the map. */
30
+ get size(): number {
31
+ return this.map.size;
32
+ }
33
+
34
+ /** Returns true if the key is present, without refreshing its recency. */
35
+ has(key: K): boolean {
36
+ return this.map.has(key);
37
+ }
38
+
39
+ /**
40
+ * Returns the value for the key, or undefined if absent.
41
+ * Refreshes the entry's recency so it becomes the most recently used.
42
+ */
43
+ get(key: K): V | undefined {
44
+ const node = this.map.get(key);
45
+ if (!node) {
46
+ return undefined;
47
+ }
48
+ this.moveToTail(node);
49
+ return node.value;
50
+ }
51
+
52
+ /**
53
+ * Stores a value for the key, refreshing its recency. If the key already exists, overwrites the value.
54
+ * If the map is at capacity, evicts the least recently used entry.
55
+ */
56
+ set(key: K, value: V): void {
57
+ const existing = this.map.get(key);
58
+ if (existing) {
59
+ existing.value = value;
60
+ this.moveToTail(existing);
61
+ return;
62
+ }
63
+
64
+ if (this.map.size >= this.maxSize) {
65
+ this.evictHead();
66
+ }
67
+
68
+ const node: LruNode<K, V> = { key, value, prev: this.tail, next: undefined };
69
+ if (this.tail) {
70
+ this.tail.next = node;
71
+ } else {
72
+ this.head = node;
73
+ }
74
+ this.tail = node;
75
+ this.map.set(key, node);
76
+ }
77
+
78
+ /** Removes the entry for the key, returning true if it was present. */
79
+ delete(key: K): boolean {
80
+ const node = this.map.get(key);
81
+ if (!node) {
82
+ return false;
83
+ }
84
+ this.unlink(node);
85
+ this.map.delete(key);
86
+ return true;
87
+ }
88
+
89
+ /** Removes all entries from the map. */
90
+ clear(): void {
91
+ this.map.clear();
92
+ this.head = undefined;
93
+ this.tail = undefined;
94
+ }
95
+
96
+ /** Unlinks a node from its current position and relinks it at the tail. */
97
+ private moveToTail(node: LruNode<K, V>): void {
98
+ if (node === this.tail) {
99
+ return;
100
+ }
101
+ this.unlink(node);
102
+
103
+ node.prev = this.tail;
104
+ node.next = undefined;
105
+ if (this.tail) {
106
+ this.tail.next = node;
107
+ } else {
108
+ this.head = node;
109
+ }
110
+ this.tail = node;
111
+ }
112
+
113
+ /** Detaches a node from the linked list, fixing up its neighbours and the head/tail pointers. */
114
+ private unlink(node: LruNode<K, V>): void {
115
+ if (node.prev) {
116
+ node.prev.next = node.next;
117
+ } else {
118
+ this.head = node.next;
119
+ }
120
+ if (node.next) {
121
+ node.next.prev = node.prev;
122
+ } else {
123
+ this.tail = node.prev;
124
+ }
125
+ node.prev = undefined;
126
+ node.next = undefined;
127
+ }
128
+
129
+ /** Evicts the head (least recently used) node. */
130
+ private evictHead(): void {
131
+ const oldHead = this.head;
132
+ if (!oldHead) {
133
+ return;
134
+ }
135
+ this.head = oldHead.next;
136
+ if (this.head) {
137
+ this.head.prev = undefined;
138
+ } else {
139
+ this.tail = undefined;
140
+ }
141
+ this.map.delete(oldHead.key);
142
+ }
143
+ }
@@ -0,0 +1,115 @@
1
+ /** Node in a doubly-linked list used by {@link LruSet}. */
2
+ type LruNode<T> = {
3
+ value: T;
4
+ prev: LruNode<T> | undefined;
5
+ next: LruNode<T> | undefined;
6
+ };
7
+
8
+ /**
9
+ * A bounded set with Least Recently Used (LRU) eviction.
10
+ * Both {@link has} and {@link add} count as an access and refresh the entry's
11
+ * recency, so items that are actively checked stay in the set longest.
12
+ *
13
+ * Uses a doubly-linked list for O(1) ordering and a Map for O(1) lookup.
14
+ * Head = least recent, tail = most recent.
15
+ */
16
+ export class LruSet<T> {
17
+ /** Map from value to its linked-list node for O(1) lookup. */
18
+ private readonly map = new Map<T, LruNode<T>>();
19
+ private head: LruNode<T> | undefined;
20
+ private tail: LruNode<T> | undefined;
21
+
22
+ constructor(private readonly maxSize: number) {
23
+ if (maxSize < 1) {
24
+ throw new Error('LruSet maxSize must be at least 1');
25
+ }
26
+ }
27
+
28
+ /** Number of entries in the set. */
29
+ get size(): number {
30
+ return this.map.size;
31
+ }
32
+
33
+ /**
34
+ * Returns true if the item is in the set.
35
+ * Refreshes the item's recency so it becomes the most recently used.
36
+ */
37
+ has(item: T): boolean {
38
+ const node = this.map.get(item);
39
+ if (!node) {
40
+ return false;
41
+ }
42
+ this.moveToTail(node);
43
+ return true;
44
+ }
45
+
46
+ /**
47
+ * Adds an item to the set. If the item already exists, refreshes its recency.
48
+ * If the set is at capacity, evicts the least recently used item.
49
+ */
50
+ add(item: T): void {
51
+ const existing = this.map.get(item);
52
+ if (existing) {
53
+ this.moveToTail(existing);
54
+ return;
55
+ }
56
+
57
+ if (this.map.size >= this.maxSize) {
58
+ this.evictHead();
59
+ }
60
+
61
+ const node: LruNode<T> = { value: item, prev: this.tail, next: undefined };
62
+ if (this.tail) {
63
+ this.tail.next = node;
64
+ } else {
65
+ this.head = node;
66
+ }
67
+ this.tail = node;
68
+ this.map.set(item, node);
69
+ }
70
+
71
+ /** Removes all entries from the set. */
72
+ clear(): void {
73
+ this.map.clear();
74
+ this.head = undefined;
75
+ this.tail = undefined;
76
+ }
77
+
78
+ /** Unlinks a node from its current position and relinks it at the tail. */
79
+ private moveToTail(node: LruNode<T>): void {
80
+ if (node === this.tail) {
81
+ return;
82
+ }
83
+
84
+ // Unlink
85
+ if (node.prev) {
86
+ node.prev.next = node.next;
87
+ } else {
88
+ this.head = node.next;
89
+ }
90
+ if (node.next) {
91
+ node.next.prev = node.prev;
92
+ }
93
+
94
+ // Relink at tail
95
+ node.prev = this.tail;
96
+ node.next = undefined;
97
+ if (this.tail) {
98
+ this.tail.next = node;
99
+ }
100
+ this.tail = node;
101
+ }
102
+
103
+ /** Evicts the head (least recently used) node. */
104
+ private evictHead(): void {
105
+ const oldHead = this.head!;
106
+ this.map.delete(oldHead.value);
107
+
108
+ this.head = oldHead.next;
109
+ if (this.head) {
110
+ this.head.prev = undefined;
111
+ } else {
112
+ this.tail = undefined;
113
+ }
114
+ }
115
+ }
@@ -24,7 +24,7 @@ export class Committable<T> {
24
24
  * Rolls back the uncommitted value.
25
25
  */
26
26
  public rollback() {
27
- this.nextValue === undefined;
27
+ this.nextValue = undefined;
28
28
  }
29
29
 
30
30
  /**