@aztec/p2p 0.0.1-commit.a072138 → 0.0.1-commit.aada20e3

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 (106) hide show
  1. package/dest/client/factory.d.ts +1 -1
  2. package/dest/client/factory.d.ts.map +1 -1
  3. package/dest/client/factory.js +3 -1
  4. package/dest/client/p2p_client.d.ts +4 -2
  5. package/dest/client/p2p_client.d.ts.map +1 -1
  6. package/dest/client/p2p_client.js +6 -2
  7. package/dest/config.d.ts +8 -2
  8. package/dest/config.d.ts.map +1 -1
  9. package/dest/config.js +3 -1
  10. package/dest/mem_pools/tx_pool_v2/archive/index.d.ts +2 -0
  11. package/dest/mem_pools/tx_pool_v2/archive/index.d.ts.map +1 -0
  12. package/dest/mem_pools/tx_pool_v2/archive/index.js +1 -0
  13. package/dest/mem_pools/tx_pool_v2/archive/tx_archive.d.ts +43 -0
  14. package/dest/mem_pools/tx_pool_v2/archive/tx_archive.d.ts.map +1 -0
  15. package/dest/mem_pools/tx_pool_v2/archive/tx_archive.js +103 -0
  16. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts +47 -0
  17. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts.map +1 -0
  18. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.js +119 -0
  19. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts +17 -0
  20. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -0
  21. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.js +90 -0
  22. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts +19 -0
  23. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts.map +1 -0
  24. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.js +89 -0
  25. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts +10 -0
  26. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts.map +1 -0
  27. package/dest/mem_pools/tx_pool_v2/eviction/index.js +11 -0
  28. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts +131 -0
  29. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts.map +1 -0
  30. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.js +17 -0
  31. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.d.ts +15 -0
  32. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.d.ts.map +1 -0
  33. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.js +63 -0
  34. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.d.ts +17 -0
  35. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.d.ts.map +1 -0
  36. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.js +91 -0
  37. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts +16 -0
  38. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts.map +1 -0
  39. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.js +70 -0
  40. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts +20 -0
  41. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts.map +1 -0
  42. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.js +63 -0
  43. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts +15 -0
  44. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts.map +1 -0
  45. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.js +19 -0
  46. package/dest/mem_pools/tx_pool_v2/index.d.ts +5 -0
  47. package/dest/mem_pools/tx_pool_v2/index.d.ts.map +1 -0
  48. package/dest/mem_pools/tx_pool_v2/index.js +4 -0
  49. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts +193 -0
  50. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts.map +1 -0
  51. package/dest/mem_pools/tx_pool_v2/interfaces.js +6 -0
  52. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts +71 -0
  53. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts.map +1 -0
  54. package/dest/mem_pools/tx_pool_v2/tx_metadata.js +94 -0
  55. package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.d.ts +26 -0
  56. package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.d.ts.map +1 -0
  57. package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.js +70 -0
  58. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts +55 -0
  59. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts.map +1 -0
  60. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.js +150 -0
  61. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts +69 -0
  62. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts.map +1 -0
  63. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.js +1041 -0
  64. package/dest/services/index.d.ts +2 -1
  65. package/dest/services/index.d.ts.map +1 -1
  66. package/dest/services/index.js +1 -0
  67. package/dest/services/tx_file_store/config.d.ts +18 -0
  68. package/dest/services/tx_file_store/config.d.ts.map +1 -0
  69. package/dest/services/tx_file_store/config.js +26 -0
  70. package/dest/services/tx_file_store/index.d.ts +4 -0
  71. package/dest/services/tx_file_store/index.d.ts.map +1 -0
  72. package/dest/services/tx_file_store/index.js +3 -0
  73. package/dest/services/tx_file_store/instrumentation.d.ts +15 -0
  74. package/dest/services/tx_file_store/instrumentation.d.ts.map +1 -0
  75. package/dest/services/tx_file_store/instrumentation.js +29 -0
  76. package/dest/services/tx_file_store/tx_file_store.d.ts +47 -0
  77. package/dest/services/tx_file_store/tx_file_store.d.ts.map +1 -0
  78. package/dest/services/tx_file_store/tx_file_store.js +149 -0
  79. package/package.json +14 -14
  80. package/src/client/factory.ts +4 -0
  81. package/src/client/p2p_client.ts +5 -0
  82. package/src/config.ts +8 -1
  83. package/src/mem_pools/tx_pool_v2/README.md +188 -0
  84. package/src/mem_pools/tx_pool_v2/archive/index.ts +1 -0
  85. package/src/mem_pools/tx_pool_v2/archive/tx_archive.ts +120 -0
  86. package/src/mem_pools/tx_pool_v2/eviction/eviction_manager.ts +147 -0
  87. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.ts +118 -0
  88. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.ts +111 -0
  89. package/src/mem_pools/tx_pool_v2/eviction/index.ts +23 -0
  90. package/src/mem_pools/tx_pool_v2/eviction/interfaces.ts +164 -0
  91. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.ts +74 -0
  92. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.ts +101 -0
  93. package/src/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.ts +86 -0
  94. package/src/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.ts +72 -0
  95. package/src/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.ts +31 -0
  96. package/src/mem_pools/tx_pool_v2/index.ts +11 -0
  97. package/src/mem_pools/tx_pool_v2/interfaces.ts +225 -0
  98. package/src/mem_pools/tx_pool_v2/tx_metadata.ts +160 -0
  99. package/src/mem_pools/tx_pool_v2/tx_pool_bench_metrics.ts +77 -0
  100. package/src/mem_pools/tx_pool_v2/tx_pool_v2.ts +209 -0
  101. package/src/mem_pools/tx_pool_v2/tx_pool_v2_impl.ts +1265 -0
  102. package/src/services/index.ts +1 -0
  103. package/src/services/tx_file_store/config.ts +43 -0
  104. package/src/services/tx_file_store/index.ts +3 -0
  105. package/src/services/tx_file_store/instrumentation.ts +36 -0
  106. package/src/services/tx_file_store/tx_file_store.ts +173 -0
@@ -3,4 +3,5 @@ export * from './libp2p/libp2p_service.js';
3
3
  export * from './tx_provider.js';
4
4
  export * from './dummy_service.js';
5
5
  export * from './reqresp/index.js';
6
+ export * from './tx_file_store/index.js';
6
7
  export * from './tx_collection/index.js';
@@ -0,0 +1,43 @@
1
+ import { type ConfigMappingsType, booleanConfigHelper, numberConfigHelper } from '@aztec/foundation/config';
2
+
3
+ /**
4
+ * Configuration for the TxFileStore service.
5
+ */
6
+ export type TxFileStoreConfig = {
7
+ /** URL for uploading txs to file storage (s3://, gs://, file://) */
8
+ txFileStoreUrl?: string;
9
+ /** URL for downloading txs from file storage */
10
+ txFileStoreDownloadUrl?: string;
11
+ /** Max concurrent uploads */
12
+ txFileStoreUploadConcurrency: number;
13
+ /** Max queue size to prevent unbounded memory growth */
14
+ txFileStoreMaxQueueSize: number;
15
+ /** Enable tx file store upload */
16
+ txFileStoreEnabled: boolean;
17
+ };
18
+
19
+ export const txFileStoreConfigMappings: ConfigMappingsType<TxFileStoreConfig> = {
20
+ txFileStoreUrl: {
21
+ env: 'TX_FILE_STORE_URL',
22
+ description: 'URL for uploading txs to file storage (s3://, gs://, file://)',
23
+ },
24
+ txFileStoreDownloadUrl: {
25
+ env: 'TX_FILE_STORE_DOWNLOAD_URL',
26
+ description: 'URL for downloading txs from file storage',
27
+ },
28
+ txFileStoreUploadConcurrency: {
29
+ env: 'TX_FILE_STORE_UPLOAD_CONCURRENCY',
30
+ description: 'Maximum number of concurrent tx uploads',
31
+ ...numberConfigHelper(10),
32
+ },
33
+ txFileStoreMaxQueueSize: {
34
+ env: 'TX_FILE_STORE_MAX_QUEUE_SIZE',
35
+ description: 'Maximum queue size for pending uploads (oldest dropped when exceeded)',
36
+ ...numberConfigHelper(1000),
37
+ },
38
+ txFileStoreEnabled: {
39
+ env: 'TX_FILE_STORE_ENABLED',
40
+ description: 'Enable uploading transactions to file storage',
41
+ ...booleanConfigHelper(false),
42
+ },
43
+ };
@@ -0,0 +1,3 @@
1
+ export { TxFileStore } from './tx_file_store.js';
2
+ export { TxFileStoreInstrumentation } from './instrumentation.js';
3
+ export { type TxFileStoreConfig, txFileStoreConfigMappings } from './config.js';
@@ -0,0 +1,36 @@
1
+ import { type Gauge, type Histogram, Metrics, type TelemetryClient, type UpDownCounter } from '@aztec/telemetry-client';
2
+
3
+ /** Instrumentation for the TxFileStore service. */
4
+ export class TxFileStoreInstrumentation {
5
+ private uploadsSuccess: UpDownCounter;
6
+ private uploadsFailed: UpDownCounter;
7
+ private uploadsSkipped: UpDownCounter;
8
+ private uploadDuration: Histogram;
9
+ private queueSize: Gauge;
10
+
11
+ constructor(client: TelemetryClient, name: string) {
12
+ const meter = client.getMeter(name);
13
+ this.uploadsSuccess = meter.createUpDownCounter(Metrics.TX_FILE_STORE_UPLOADS_SUCCESS);
14
+ this.uploadsFailed = meter.createUpDownCounter(Metrics.TX_FILE_STORE_UPLOADS_FAILED);
15
+ this.uploadsSkipped = meter.createUpDownCounter(Metrics.TX_FILE_STORE_UPLOADS_SKIPPED);
16
+ this.uploadDuration = meter.createHistogram(Metrics.TX_FILE_STORE_UPLOAD_DURATION);
17
+ this.queueSize = meter.createGauge(Metrics.TX_FILE_STORE_QUEUE_SIZE);
18
+ }
19
+
20
+ recordUploadSuccess(durationMs: number) {
21
+ this.uploadsSuccess.add(1);
22
+ this.uploadDuration.record(durationMs);
23
+ }
24
+
25
+ recordUploadFailed() {
26
+ this.uploadsFailed.add(1);
27
+ }
28
+
29
+ recordUploadSkipped() {
30
+ this.uploadsSkipped.add(1);
31
+ }
32
+
33
+ recordQueueSize(size: number) {
34
+ this.queueSize.record(size);
35
+ }
36
+ }
@@ -0,0 +1,173 @@
1
+ import { type Logger, createLogger } from '@aztec/foundation/log';
2
+ import { RunningPromise } from '@aztec/foundation/promise';
3
+ import { makeBackoff, retry } from '@aztec/foundation/retry';
4
+ import { Timer } from '@aztec/foundation/timer';
5
+ import { type FileStore, createFileStore } from '@aztec/stdlib/file-store';
6
+ import type { Tx } from '@aztec/stdlib/tx';
7
+ import { type TelemetryClient, getTelemetryClient } from '@aztec/telemetry-client';
8
+
9
+ import type { TxPool, TxPoolEvents } from '../../mem_pools/tx_pool/index.js';
10
+ import type { TxFileStoreConfig } from './config.js';
11
+ import { TxFileStoreInstrumentation } from './instrumentation.js';
12
+
13
+ /**
14
+ * Uploads validated transactions to a file store as a fallback retrieval mechanism.
15
+ * Listens to TxPool txs-added events and uploads txs asynchronously with bounded concurrency.
16
+ */
17
+ export class TxFileStore {
18
+ private uploadQueue: Tx[] = [];
19
+ private activeUploads = 0;
20
+ private readonly queueProcessor: RunningPromise;
21
+ private readonly handleTxsAdded: TxPoolEvents['txs-added'];
22
+
23
+ /** Recently uploaded tx hashes for deduplication. */
24
+ private recentUploads: Set<string> = new Set();
25
+ private recentUploadsOrder: string[] = [];
26
+ private readonly maxRecentUploads = 1000;
27
+
28
+ private constructor(
29
+ private readonly fileStore: FileStore,
30
+ private readonly txPool: TxPool,
31
+ private readonly config: TxFileStoreConfig,
32
+ private readonly instrumentation: TxFileStoreInstrumentation,
33
+ private readonly log: Logger,
34
+ ) {
35
+ this.handleTxsAdded = (args: { txs: Tx[]; source?: string }) => {
36
+ this.enqueueTxs(args.txs);
37
+ };
38
+ this.queueProcessor = new RunningPromise(() => this.processQueueBatch(), this.log, 100);
39
+ }
40
+
41
+ /**
42
+ * Creates and initializes the file store.
43
+ * @param txPool - The transaction pool to listen to.
44
+ * @param config - The file store configuration.
45
+ * @param log - Optional logger.
46
+ * @param telemetry - Optional telemetry client.
47
+ * @param fileStoreOverride - Optional FileStore for testing (bypasses createFileStore).
48
+ * @returns The file store instance, or undefined if not configured/enabled.
49
+ */
50
+ static async create(
51
+ txPool: TxPool,
52
+ config: TxFileStoreConfig,
53
+ log: Logger = createLogger('p2p:tx_file_store'),
54
+ telemetry: TelemetryClient = getTelemetryClient(),
55
+ fileStoreOverride?: FileStore,
56
+ ): Promise<TxFileStore | undefined> {
57
+ if (!config.txFileStoreEnabled) {
58
+ log.debug('Tx file store is disabled');
59
+ return undefined;
60
+ }
61
+
62
+ if (!config.txFileStoreUrl) {
63
+ log.warn('Tx file store is enabled but URL is not configured');
64
+ return undefined;
65
+ }
66
+
67
+ const fileStore = fileStoreOverride ?? (await createFileStore(config.txFileStoreUrl, log));
68
+ if (!fileStore) {
69
+ log.warn('Failed to create file store for tx file store');
70
+ return undefined;
71
+ }
72
+
73
+ const instrumentation = new TxFileStoreInstrumentation(telemetry, 'TxFileStore');
74
+ log.info('Created tx file store', { url: config.txFileStoreUrl });
75
+ return new TxFileStore(fileStore, txPool, config, instrumentation, log);
76
+ }
77
+
78
+ /** Starts listening to TxPool events and uploading txs. */
79
+ public start(): void {
80
+ this.queueProcessor.start();
81
+ this.txPool.on('txs-added', this.handleTxsAdded);
82
+ this.log.info('Started tx file store', {
83
+ concurrency: this.config.txFileStoreUploadConcurrency,
84
+ maxQueueSize: this.config.txFileStoreMaxQueueSize,
85
+ });
86
+ }
87
+
88
+ /** Stops listening and waits for pending uploads to complete. */
89
+ public async stop(): Promise<void> {
90
+ this.txPool.removeListener('txs-added', this.handleTxsAdded);
91
+ await this.queueProcessor.stop();
92
+ this.log.info('Stopped tx file store');
93
+ }
94
+
95
+ private enqueueTxs(txs: Tx[]): void {
96
+ this.uploadQueue.push(...txs);
97
+
98
+ // Enforce max queue size by dropping oldest entries
99
+ const overflow = this.uploadQueue.length - this.config.txFileStoreMaxQueueSize;
100
+ if (overflow > 0) {
101
+ this.log.warn(`Upload queue overflow, dropping ${overflow} oldest txs`);
102
+ this.uploadQueue.splice(0, overflow);
103
+ }
104
+
105
+ this.instrumentation.recordQueueSize(this.uploadQueue.length);
106
+
107
+ // Immediately start uploading txs
108
+ void this.queueProcessor.trigger();
109
+ }
110
+
111
+ private async processQueueBatch(): Promise<void> {
112
+ const batch = this.uploadQueue.splice(0, this.config.txFileStoreUploadConcurrency);
113
+ this.instrumentation.recordQueueSize(this.uploadQueue.length);
114
+
115
+ this.activeUploads += batch.length;
116
+ try {
117
+ await Promise.all(batch.map(tx => this.uploadTx(tx)));
118
+ } finally {
119
+ this.activeUploads -= batch.length;
120
+ }
121
+ }
122
+
123
+ private async uploadTx(tx: Tx): Promise<void> {
124
+ const txHash = tx.getTxHash().toString();
125
+ const path = `txs/${txHash}.bin`;
126
+ const timer = new Timer();
127
+
128
+ if (this.recentUploads.has(txHash)) {
129
+ return;
130
+ }
131
+
132
+ try {
133
+ this.recentUploads.add(txHash);
134
+ this.recentUploadsOrder.push(txHash);
135
+
136
+ if (this.recentUploadsOrder.length > this.maxRecentUploads) {
137
+ // delete old entries in recentUploads
138
+ for (const txHashToRemove of this.recentUploadsOrder.splice(
139
+ 0,
140
+ this.recentUploadsOrder.length - this.maxRecentUploads,
141
+ )) {
142
+ this.recentUploads.delete(txHashToRemove);
143
+ }
144
+ }
145
+
146
+ await retry(
147
+ () => this.fileStore.save(path, tx.toBuffer(), { compress: false }),
148
+ `Uploading tx ${txHash}`,
149
+ makeBackoff([0.1, 0.5, 2]),
150
+ this.log,
151
+ true, // failSilently - don't log errors during retries
152
+ );
153
+ const durationMs = Math.trunc(timer.ms());
154
+ this.log.debug(`Uploaded tx to file store`, { txHash, path, durationMs });
155
+ this.instrumentation.recordUploadSuccess(durationMs);
156
+ } catch (err) {
157
+ this.log.warn(`Failed to upload tx to file store after retries`, { txHash, error: err });
158
+ this.instrumentation.recordUploadFailed();
159
+ }
160
+ }
161
+
162
+ /** Waits for all queued and in-flight uploads to complete. For testing. */
163
+ public async flush(): Promise<void> {
164
+ while (this.uploadQueue.length > 0 || this.activeUploads > 0) {
165
+ await this.queueProcessor.trigger();
166
+ }
167
+ }
168
+
169
+ /** Returns the number of pending uploads (queued + in-flight). */
170
+ public getPendingUploadCount(): number {
171
+ return this.uploadQueue.length + this.activeUploads;
172
+ }
173
+ }