@aztec/validator-ha-signer 0.0.1-commit.96bb3f7 → 0.0.1-commit.96dac018d

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 (50) hide show
  1. package/README.md +52 -37
  2. package/dest/db/postgres.d.ts +34 -5
  3. package/dest/db/postgres.d.ts.map +1 -1
  4. package/dest/db/postgres.js +80 -22
  5. package/dest/db/schema.d.ts +21 -10
  6. package/dest/db/schema.d.ts.map +1 -1
  7. package/dest/db/schema.js +49 -20
  8. package/dest/db/types.d.ts +76 -31
  9. package/dest/db/types.d.ts.map +1 -1
  10. package/dest/db/types.js +32 -8
  11. package/dest/errors.d.ts +9 -5
  12. package/dest/errors.d.ts.map +1 -1
  13. package/dest/errors.js +7 -4
  14. package/dest/factory.d.ts +6 -14
  15. package/dest/factory.d.ts.map +1 -1
  16. package/dest/factory.js +17 -12
  17. package/dest/metrics.d.ts +51 -0
  18. package/dest/metrics.d.ts.map +1 -0
  19. package/dest/metrics.js +103 -0
  20. package/dest/migrations.d.ts +1 -1
  21. package/dest/migrations.d.ts.map +1 -1
  22. package/dest/migrations.js +13 -2
  23. package/dest/slashing_protection_service.d.ts +25 -6
  24. package/dest/slashing_protection_service.d.ts.map +1 -1
  25. package/dest/slashing_protection_service.js +72 -20
  26. package/dest/test/pglite_pool.d.ts +92 -0
  27. package/dest/test/pglite_pool.d.ts.map +1 -0
  28. package/dest/test/pglite_pool.js +210 -0
  29. package/dest/types.d.ts +38 -18
  30. package/dest/types.d.ts.map +1 -1
  31. package/dest/types.js +4 -1
  32. package/dest/validator_ha_signer.d.ts +18 -13
  33. package/dest/validator_ha_signer.d.ts.map +1 -1
  34. package/dest/validator_ha_signer.js +46 -33
  35. package/package.json +13 -10
  36. package/src/db/postgres.ts +101 -21
  37. package/src/db/schema.ts +51 -20
  38. package/src/db/types.ts +110 -31
  39. package/src/errors.ts +7 -2
  40. package/src/factory.ts +20 -14
  41. package/src/metrics.ts +138 -0
  42. package/src/migrations.ts +17 -1
  43. package/src/slashing_protection_service.ts +117 -25
  44. package/src/test/pglite_pool.ts +256 -0
  45. package/src/types.ts +63 -19
  46. package/src/validator_ha_signer.ts +66 -42
  47. package/dest/config.d.ts +0 -47
  48. package/dest/config.d.ts.map +0 -1
  49. package/dest/config.js +0 -64
  50. package/src/config.ts +0 -116
package/dest/db/schema.js CHANGED
@@ -11,11 +11,13 @@
11
11
  * SQL to create the validator_duties table
12
12
  */ export const CREATE_VALIDATOR_DUTIES_TABLE = `
13
13
  CREATE TABLE IF NOT EXISTS validator_duties (
14
+ rollup_address VARCHAR(42) NOT NULL,
14
15
  validator_address VARCHAR(42) NOT NULL,
15
16
  slot BIGINT NOT NULL,
16
17
  block_number BIGINT NOT NULL,
17
- duty_type VARCHAR(30) NOT NULL CHECK (duty_type IN ('BLOCK_PROPOSAL', 'ATTESTATION', 'ATTESTATIONS_AND_SIGNERS')),
18
- status VARCHAR(20) NOT NULL CHECK (status IN ('signing', 'signed', 'failed')),
18
+ block_index_within_checkpoint INTEGER NOT NULL DEFAULT 0,
19
+ duty_type VARCHAR(30) NOT NULL CHECK (duty_type IN ('BLOCK_PROPOSAL', 'CHECKPOINT_PROPOSAL', 'ATTESTATION', 'ATTESTATIONS_AND_SIGNERS', 'GOVERNANCE_VOTE', 'SLASHING_VOTE')),
20
+ status VARCHAR(20) NOT NULL CHECK (status IN ('signing', 'signed')),
19
21
  message_hash VARCHAR(66) NOT NULL,
20
22
  signature VARCHAR(132),
21
23
  node_id VARCHAR(255) NOT NULL,
@@ -24,7 +26,7 @@ CREATE TABLE IF NOT EXISTS validator_duties (
24
26
  completed_at TIMESTAMP,
25
27
  error_message TEXT,
26
28
 
27
- PRIMARY KEY (validator_address, slot, duty_type),
29
+ PRIMARY KEY (rollup_address, validator_address, slot, duty_type, block_index_within_checkpoint),
28
30
  CHECK (completed_at IS NULL OR completed_at >= started_at)
29
31
  );
30
32
  `;
@@ -74,24 +76,32 @@ SELECT version FROM schema_version ORDER BY version DESC LIMIT 1;
74
76
  * returns the existing record instead.
75
77
  *
76
78
  * Returns the record with an `is_new` flag indicating whether we inserted or got existing.
79
+ *
80
+ * Note: In high concurrency scenarios, if the INSERT conflicts and another transaction
81
+ * just committed the row, there's a small window where the SELECT might not see it yet.
82
+ * The application layer should retry if no rows are returned.
77
83
  */ export const INSERT_OR_GET_DUTY = `
78
84
  WITH inserted AS (
79
85
  INSERT INTO validator_duties (
86
+ rollup_address,
80
87
  validator_address,
81
88
  slot,
82
89
  block_number,
90
+ block_index_within_checkpoint,
83
91
  duty_type,
84
92
  status,
85
93
  message_hash,
86
94
  node_id,
87
95
  lock_token,
88
96
  started_at
89
- ) VALUES ($1, $2, $3, $4, 'signing', $5, $6, $7, CURRENT_TIMESTAMP)
90
- ON CONFLICT (validator_address, slot, duty_type) DO NOTHING
97
+ ) VALUES ($1, $2, $3, $4, $5, $6, 'signing', $7, $8, $9, CURRENT_TIMESTAMP)
98
+ ON CONFLICT (rollup_address, validator_address, slot, duty_type, block_index_within_checkpoint) DO NOTHING
91
99
  RETURNING
100
+ rollup_address,
92
101
  validator_address,
93
102
  slot,
94
103
  block_number,
104
+ block_index_within_checkpoint,
95
105
  duty_type,
96
106
  status,
97
107
  message_hash,
@@ -106,9 +116,11 @@ WITH inserted AS (
106
116
  SELECT * FROM inserted
107
117
  UNION ALL
108
118
  SELECT
119
+ rollup_address,
109
120
  validator_address,
110
121
  slot,
111
122
  block_number,
123
+ block_index_within_checkpoint,
112
124
  duty_type,
113
125
  status,
114
126
  message_hash,
@@ -120,9 +132,11 @@ SELECT
120
132
  error_message,
121
133
  FALSE as is_new
122
134
  FROM validator_duties
123
- WHERE validator_address = $1
124
- AND slot = $2
125
- AND duty_type = $4
135
+ WHERE rollup_address = $1
136
+ AND validator_address = $2
137
+ AND slot = $3
138
+ AND duty_type = $6
139
+ AND block_index_within_checkpoint = $5
126
140
  AND NOT EXISTS (SELECT 1 FROM inserted);
127
141
  `;
128
142
  /**
@@ -132,22 +146,26 @@ UPDATE validator_duties
132
146
  SET status = 'signed',
133
147
  signature = $1,
134
148
  completed_at = CURRENT_TIMESTAMP
135
- WHERE validator_address = $2
136
- AND slot = $3
137
- AND duty_type = $4
149
+ WHERE rollup_address = $2
150
+ AND validator_address = $3
151
+ AND slot = $4
152
+ AND duty_type = $5
153
+ AND block_index_within_checkpoint = $6
138
154
  AND status = 'signing'
139
- AND lock_token = $5;
155
+ AND lock_token = $7;
140
156
  `;
141
157
  /**
142
158
  * Query to delete a duty
143
159
  * Only deletes if the lockToken matches
144
160
  */ export const DELETE_DUTY = `
145
161
  DELETE FROM validator_duties
146
- WHERE validator_address = $1
147
- AND slot = $2
148
- AND duty_type = $3
162
+ WHERE rollup_address = $1
163
+ AND validator_address = $2
164
+ AND slot = $3
165
+ AND duty_type = $4
166
+ AND block_index_within_checkpoint = $5
149
167
  AND status = 'signing'
150
- AND lock_token = $4;
168
+ AND lock_token = $6;
151
169
  `;
152
170
  /**
153
171
  * Query to clean up old signed duties (for maintenance)
@@ -159,20 +177,29 @@ WHERE status = 'signed'
159
177
  `;
160
178
  /**
161
179
  * Query to clean up old duties (for maintenance)
162
- * Removes duties older than a specified timestamp
180
+ * Removes SIGNED duties older than a specified age (in milliseconds)
163
181
  */ export const CLEANUP_OLD_DUTIES = `
164
182
  DELETE FROM validator_duties
165
- WHERE status IN ('signing', 'signed', 'failed')
166
- AND started_at < $1;
183
+ WHERE status = 'signed'
184
+ AND started_at < CURRENT_TIMESTAMP - ($1 || ' milliseconds')::INTERVAL;
167
185
  `;
168
186
  /**
169
187
  * Query to cleanup own stuck duties
170
188
  * Removes duties in 'signing' status for a specific node that are older than maxAgeMs
189
+ * Uses DB's CURRENT_TIMESTAMP to avoid clock skew issues between nodes
171
190
  */ export const CLEANUP_OWN_STUCK_DUTIES = `
172
191
  DELETE FROM validator_duties
173
192
  WHERE node_id = $1
174
193
  AND status = 'signing'
175
- AND started_at < $2;
194
+ AND started_at < CURRENT_TIMESTAMP - ($2 || ' milliseconds')::INTERVAL;
195
+ `;
196
+ /**
197
+ * Query to cleanup duties with outdated rollup address
198
+ * Removes all duties where the rollup address doesn't match the current one
199
+ * Used after a rollup upgrade to clean up duties for the old rollup
200
+ */ export const CLEANUP_OUTDATED_ROLLUP_DUTIES = `
201
+ DELETE FROM validator_duties
202
+ WHERE rollup_address != $1;
176
203
  `;
177
204
  /**
178
205
  * SQL to drop the validator_duties table
@@ -185,9 +212,11 @@ WHERE node_id = $1
185
212
  * Returns duties in 'signing' status that have been stuck for too long
186
213
  */ export const GET_STUCK_DUTIES = `
187
214
  SELECT
215
+ rollup_address,
188
216
  validator_address,
189
217
  slot,
190
218
  block_number,
219
+ block_index_within_checkpoint,
191
220
  duty_type,
192
221
  status,
193
222
  message_hash,
@@ -1,12 +1,16 @@
1
+ import type { BlockNumber, CheckpointNumber, IndexWithinCheckpoint, SlotNumber } from '@aztec/foundation/branded-types';
1
2
  import type { EthAddress } from '@aztec/foundation/eth-address';
2
3
  import type { Signature } from '@aztec/foundation/eth-signature';
4
+ import { DutyType } from '@aztec/stdlib/ha-signing';
3
5
  /**
4
6
  * Row type from PostgreSQL query
5
7
  */
6
8
  export interface DutyRow {
9
+ rollup_address: string;
7
10
  validator_address: string;
8
11
  slot: string;
9
12
  block_number: string;
13
+ block_index_within_checkpoint: number;
10
14
  duty_type: DutyType;
11
15
  status: DutyStatus;
12
16
  message_hash: string;
@@ -23,14 +27,6 @@ export interface DutyRow {
23
27
  export interface InsertOrGetRow extends DutyRow {
24
28
  is_new: boolean;
25
29
  }
26
- /**
27
- * Type of validator duty being performed
28
- */
29
- export declare enum DutyType {
30
- BLOCK_PROPOSAL = "BLOCK_PROPOSAL",
31
- ATTESTATION = "ATTESTATION",
32
- ATTESTATIONS_AND_SIGNERS = "ATTESTATIONS_AND_SIGNERS"
33
- }
34
30
  /**
35
31
  * Status of a duty in the database
36
32
  */
@@ -38,16 +34,21 @@ export declare enum DutyStatus {
38
34
  SIGNING = "signing",
39
35
  SIGNED = "signed"
40
36
  }
37
+ export { DutyType };
41
38
  /**
42
39
  * Record of a validator duty in the database
43
40
  */
44
41
  export interface ValidatorDutyRecord {
42
+ /** Ethereum address of the rollup contract */
43
+ rollupAddress: EthAddress;
45
44
  /** Ethereum address of the validator */
46
45
  validatorAddress: EthAddress;
47
46
  /** Slot number for this duty */
48
- slot: bigint;
47
+ slot: SlotNumber;
49
48
  /** Block number for this duty */
50
- blockNumber: bigint;
49
+ blockNumber: BlockNumber;
50
+ /** Block index within checkpoint (0, 1, 2... for block proposals, -1 for other duty types) */
51
+ blockIndexWithinCheckpoint: number;
51
52
  /** Type of duty being performed */
52
53
  dutyType: DutyType;
53
54
  /** Current status of the duty */
@@ -64,46 +65,90 @@ export interface ValidatorDutyRecord {
64
65
  startedAt: Date;
65
66
  /** When the duty signing was completed (success or failure) */
66
67
  completedAt?: Date;
67
- /** Error message if status is 'failed' */
68
+ /** Error message (currently unused) */
68
69
  errorMessage?: string;
69
70
  }
70
71
  /**
71
- * Minimal info needed to identify a unique duty
72
+ * Duty identifier for block proposals.
73
+ * blockIndexWithinCheckpoint is REQUIRED and must be >= 0.
72
74
  */
73
- export interface DutyIdentifier {
75
+ export interface BlockProposalDutyIdentifier {
76
+ rollupAddress: EthAddress;
74
77
  validatorAddress: EthAddress;
75
- slot: bigint;
76
- dutyType: DutyType;
78
+ slot: SlotNumber;
79
+ /** Block index within checkpoint (0, 1, 2...). Required for block proposals. */
80
+ blockIndexWithinCheckpoint: IndexWithinCheckpoint;
81
+ dutyType: DutyType.BLOCK_PROPOSAL;
77
82
  }
78
83
  /**
79
- * Parameters for checking and recording a new duty
84
+ * Duty identifier for non-block-proposal duties.
85
+ * blockIndexWithinCheckpoint is not applicable (internally stored as -1).
80
86
  */
81
- export interface CheckAndRecordParams {
87
+ export interface OtherDutyIdentifier {
88
+ rollupAddress: EthAddress;
82
89
  validatorAddress: EthAddress;
83
- slot: bigint;
84
- blockNumber: bigint;
85
- dutyType: DutyType;
90
+ slot: SlotNumber;
91
+ dutyType: DutyType.CHECKPOINT_PROPOSAL | DutyType.ATTESTATION | DutyType.ATTESTATIONS_AND_SIGNERS | DutyType.GOVERNANCE_VOTE | DutyType.SLASHING_VOTE | DutyType.AUTH_REQUEST | DutyType.TXS;
92
+ }
93
+ /**
94
+ * Minimal info needed to identify a unique duty.
95
+ * Uses discriminated union to enforce type safety:
96
+ * - BLOCK_PROPOSAL duties MUST have blockIndexWithinCheckpoint >= 0
97
+ * - Other duty types do NOT have blockIndexWithinCheckpoint (internally -1)
98
+ */
99
+ export type DutyIdentifier = BlockProposalDutyIdentifier | OtherDutyIdentifier;
100
+ /**
101
+ * Validates and normalizes the block index for a duty.
102
+ * - BLOCK_PROPOSAL: validates blockIndexWithinCheckpoint is provided and >= 0
103
+ * - Other duty types: always returns -1
104
+ *
105
+ * @throws Error if BLOCK_PROPOSAL is missing blockIndexWithinCheckpoint or has invalid value
106
+ */
107
+ export declare function normalizeBlockIndex(dutyType: DutyType, blockIndexWithinCheckpoint: number | undefined): number;
108
+ /**
109
+ * Gets the block index from a DutyIdentifier.
110
+ * - BLOCK_PROPOSAL: returns the blockIndexWithinCheckpoint
111
+ * - Other duty types: returns -1
112
+ */
113
+ export declare function getBlockIndexFromDutyIdentifier(duty: DutyIdentifier): number;
114
+ /**
115
+ * Additional parameters for checking and recording a new duty
116
+ */
117
+ interface CheckAndRecordExtra {
118
+ /** Block number for this duty */
119
+ blockNumber: BlockNumber | CheckpointNumber;
120
+ /** The signing root (hash) for this duty */
86
121
  messageHash: string;
122
+ /** Identifier for the node that acquired the lock */
87
123
  nodeId: string;
88
124
  }
89
125
  /**
90
- * Parameters for recording a successful signing
126
+ * Parameters for checking and recording a new duty.
127
+ * Uses intersection with DutyIdentifier to preserve the discriminated union.
91
128
  */
92
- export interface RecordSuccessParams {
93
- validatorAddress: EthAddress;
94
- slot: bigint;
95
- dutyType: DutyType;
129
+ export type CheckAndRecordParams = DutyIdentifier & CheckAndRecordExtra;
130
+ /**
131
+ * Additional parameters for recording a successful signing
132
+ */
133
+ interface RecordSuccessExtra {
96
134
  signature: Signature;
97
135
  nodeId: string;
98
136
  lockToken: string;
99
137
  }
100
138
  /**
101
- * Parameters for deleting a duty
139
+ * Parameters for recording a successful signing.
140
+ * Uses intersection with DutyIdentifier to preserve the discriminated union.
102
141
  */
103
- export interface DeleteDutyParams {
104
- validatorAddress: EthAddress;
105
- slot: bigint;
106
- dutyType: DutyType;
142
+ export type RecordSuccessParams = DutyIdentifier & RecordSuccessExtra;
143
+ /**
144
+ * Additional parameters for deleting a duty
145
+ */
146
+ interface DeleteDutyExtra {
107
147
  lockToken: string;
108
148
  }
109
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuZC50cyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssRUFBRSxVQUFVLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUNoRSxPQUFPLEtBQUssRUFBRSxTQUFTLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUVqRTs7R0FFRztBQUNILE1BQU0sV0FBVyxPQUFPO0lBQ3RCLGlCQUFpQixFQUFFLE1BQU0sQ0FBQztJQUMxQixJQUFJLEVBQUUsTUFBTSxDQUFDO0lBQ2IsWUFBWSxFQUFFLE1BQU0sQ0FBQztJQUNyQixTQUFTLEVBQUUsUUFBUSxDQUFDO0lBQ3BCLE1BQU0sRUFBRSxVQUFVLENBQUM7SUFDbkIsWUFBWSxFQUFFLE1BQU0sQ0FBQztJQUNyQixTQUFTLEVBQUUsTUFBTSxHQUFHLElBQUksQ0FBQztJQUN6QixPQUFPLEVBQUUsTUFBTSxDQUFDO0lBQ2hCLFVBQVUsRUFBRSxNQUFNLENBQUM7SUFDbkIsVUFBVSxFQUFFLElBQUksQ0FBQztJQUNqQixZQUFZLEVBQUUsSUFBSSxHQUFHLElBQUksQ0FBQztJQUMxQixhQUFhLEVBQUUsTUFBTSxHQUFHLElBQUksQ0FBQztDQUM5QjtBQUVEOztHQUVHO0FBQ0gsTUFBTSxXQUFXLGNBQWUsU0FBUSxPQUFPO0lBQzdDLE1BQU0sRUFBRSxPQUFPLENBQUM7Q0FDakI7QUFFRDs7R0FFRztBQUNILG9CQUFZLFFBQVE7SUFDbEIsY0FBYyxtQkFBbUI7SUFDakMsV0FBVyxnQkFBZ0I7SUFDM0Isd0JBQXdCLDZCQUE2QjtDQUN0RDtBQUVEOztHQUVHO0FBQ0gsb0JBQVksVUFBVTtJQUNwQixPQUFPLFlBQVk7SUFDbkIsTUFBTSxXQUFXO0NBQ2xCO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLFdBQVcsbUJBQW1CO0lBQ2xDLHdDQUF3QztJQUN4QyxnQkFBZ0IsRUFBRSxVQUFVLENBQUM7SUFDN0IsZ0NBQWdDO0lBQ2hDLElBQUksRUFBRSxNQUFNLENBQUM7SUFDYixpQ0FBaUM7SUFDakMsV0FBVyxFQUFFLE1BQU0sQ0FBQztJQUNwQixtQ0FBbUM7SUFDbkMsUUFBUSxFQUFFLFFBQVEsQ0FBQztJQUNuQixpQ0FBaUM7SUFDakMsTUFBTSxFQUFFLFVBQVUsQ0FBQztJQUNuQiw0Q0FBNEM7SUFDNUMsV0FBVyxFQUFFLE1BQU0sQ0FBQztJQUNwQix5REFBeUQ7SUFDekQsU0FBUyxDQUFDLEVBQUUsTUFBTSxDQUFDO0lBQ25CLDREQUE0RDtJQUM1RCxNQUFNLEVBQUUsTUFBTSxDQUFDO0lBQ2YsNERBQTREO0lBQzVELFNBQVMsRUFBRSxNQUFNLENBQUM7SUFDbEIsd0NBQXdDO0lBQ3hDLFNBQVMsRUFBRSxJQUFJLENBQUM7SUFDaEIsK0RBQStEO0lBQy9ELFdBQVcsQ0FBQyxFQUFFLElBQUksQ0FBQztJQUNuQiwwQ0FBMEM7SUFDMUMsWUFBWSxDQUFDLEVBQUUsTUFBTSxDQUFDO0NBQ3ZCO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLFdBQVcsY0FBYztJQUM3QixnQkFBZ0IsRUFBRSxVQUFVLENBQUM7SUFDN0IsSUFBSSxFQUFFLE1BQU0sQ0FBQztJQUNiLFFBQVEsRUFBRSxRQUFRLENBQUM7Q0FDcEI7QUFFRDs7R0FFRztBQUNILE1BQU0sV0FBVyxvQkFBb0I7SUFDbkMsZ0JBQWdCLEVBQUUsVUFBVSxDQUFDO0lBQzdCLElBQUksRUFBRSxNQUFNLENBQUM7SUFDYixXQUFXLEVBQUUsTUFBTSxDQUFDO0lBQ3BCLFFBQVEsRUFBRSxRQUFRLENBQUM7SUFDbkIsV0FBVyxFQUFFLE1BQU0sQ0FBQztJQUNwQixNQUFNLEVBQUUsTUFBTSxDQUFDO0NBQ2hCO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLFdBQVcsbUJBQW1CO0lBQ2xDLGdCQUFnQixFQUFFLFVBQVUsQ0FBQztJQUM3QixJQUFJLEVBQUUsTUFBTSxDQUFDO0lBQ2IsUUFBUSxFQUFFLFFBQVEsQ0FBQztJQUNuQixTQUFTLEVBQUUsU0FBUyxDQUFDO0lBQ3JCLE1BQU0sRUFBRSxNQUFNLENBQUM7SUFDZixTQUFTLEVBQUUsTUFBTSxDQUFDO0NBQ25CO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLFdBQVcsZ0JBQWdCO0lBQy9CLGdCQUFnQixFQUFFLFVBQVUsQ0FBQztJQUM3QixJQUFJLEVBQUUsTUFBTSxDQUFDO0lBQ2IsUUFBUSxFQUFFLFFBQVEsQ0FBQztJQUNuQixTQUFTLEVBQUUsTUFBTSxDQUFDO0NBQ25CIn0=
149
+ /**
150
+ * Parameters for deleting a duty.
151
+ * Uses intersection with DutyIdentifier to preserve the discriminated union.
152
+ */
153
+ export type DeleteDutyParams = DutyIdentifier & DeleteDutyExtra;
154
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuZC50cyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYi90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssRUFBRSxXQUFXLEVBQUUsZ0JBQWdCLEVBQUUscUJBQXFCLEVBQUUsVUFBVSxFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFDeEgsT0FBTyxLQUFLLEVBQUUsVUFBVSxFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDaEUsT0FBTyxLQUFLLEVBQUUsU0FBUyxFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFDakUsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBRXBEOztHQUVHO0FBQ0gsTUFBTSxXQUFXLE9BQU87SUFDdEIsY0FBYyxFQUFFLE1BQU0sQ0FBQztJQUN2QixpQkFBaUIsRUFBRSxNQUFNLENBQUM7SUFDMUIsSUFBSSxFQUFFLE1BQU0sQ0FBQztJQUNiLFlBQVksRUFBRSxNQUFNLENBQUM7SUFDckIsNkJBQTZCLEVBQUUsTUFBTSxDQUFDO0lBQ3RDLFNBQVMsRUFBRSxRQUFRLENBQUM7SUFDcEIsTUFBTSxFQUFFLFVBQVUsQ0FBQztJQUNuQixZQUFZLEVBQUUsTUFBTSxDQUFDO0lBQ3JCLFNBQVMsRUFBRSxNQUFNLEdBQUcsSUFBSSxDQUFDO0lBQ3pCLE9BQU8sRUFBRSxNQUFNLENBQUM7SUFDaEIsVUFBVSxFQUFFLE1BQU0sQ0FBQztJQUNuQixVQUFVLEVBQUUsSUFBSSxDQUFDO0lBQ2pCLFlBQVksRUFBRSxJQUFJLEdBQUcsSUFBSSxDQUFDO0lBQzFCLGFBQWEsRUFBRSxNQUFNLEdBQUcsSUFBSSxDQUFDO0NBQzlCO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLFdBQVcsY0FBZSxTQUFRLE9BQU87SUFDN0MsTUFBTSxFQUFFLE9BQU8sQ0FBQztDQUNqQjtBQUVEOztHQUVHO0FBQ0gsb0JBQVksVUFBVTtJQUNwQixPQUFPLFlBQVk7SUFDbkIsTUFBTSxXQUFXO0NBQ2xC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
@@ -1 +1 @@
1
- {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/db/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,+BAA+B,CAAC;AAChE,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,iCAAiC,CAAC;AAEjE;;GAEG;AACH,MAAM,WAAW,OAAO;IACtB,iBAAiB,EAAE,MAAM,CAAC;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,QAAQ,CAAC;IACpB,MAAM,EAAE,UAAU,CAAC;IACnB,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,IAAI,CAAC;IACjB,YAAY,EAAE,IAAI,GAAG,IAAI,CAAC;IAC1B,aAAa,EAAE,MAAM,GAAG,IAAI,CAAC;CAC9B;AAED;;GAEG;AACH,MAAM,WAAW,cAAe,SAAQ,OAAO;IAC7C,MAAM,EAAE,OAAO,CAAC;CACjB;AAED;;GAEG;AACH,oBAAY,QAAQ;IAClB,cAAc,mBAAmB;IACjC,WAAW,gBAAgB;IAC3B,wBAAwB,6BAA6B;CACtD;AAED;;GAEG;AACH,oBAAY,UAAU;IACpB,OAAO,YAAY;IACnB,MAAM,WAAW;CAClB;AAED;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,wCAAwC;IACxC,gBAAgB,EAAE,UAAU,CAAC;IAC7B,gCAAgC;IAChC,IAAI,EAAE,MAAM,CAAC;IACb,iCAAiC;IACjC,WAAW,EAAE,MAAM,CAAC;IACpB,mCAAmC;IACnC,QAAQ,EAAE,QAAQ,CAAC;IACnB,iCAAiC;IACjC,MAAM,EAAE,UAAU,CAAC;IACnB,4CAA4C;IAC5C,WAAW,EAAE,MAAM,CAAC;IACpB,yDAAyD;IACzD,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,4DAA4D;IAC5D,MAAM,EAAE,MAAM,CAAC;IACf,4DAA4D;IAC5D,SAAS,EAAE,MAAM,CAAC;IAClB,wCAAwC;IACxC,SAAS,EAAE,IAAI,CAAC;IAChB,+DAA+D;IAC/D,WAAW,CAAC,EAAE,IAAI,CAAC;IACnB,0CAA0C;IAC1C,YAAY,CAAC,EAAE,MAAM,CAAC;CACvB;AAED;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,gBAAgB,EAAE,UAAU,CAAC;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,QAAQ,CAAC;CACpB;AAED;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC,gBAAgB,EAAE,UAAU,CAAC;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,EAAE,QAAQ,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,EAAE,MAAM,CAAC;CAChB;AAED;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,gBAAgB,EAAE,UAAU,CAAC;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,QAAQ,CAAC;IACnB,SAAS,EAAE,SAAS,CAAC;IACrB,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,EAAE,MAAM,CAAC;CACnB;AAED;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,gBAAgB,EAAE,UAAU,CAAC;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,QAAQ,CAAC;IACnB,SAAS,EAAE,MAAM,CAAC;CACnB"}
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/db/types.ts"],"names":[],"mappings":"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"}
package/dest/db/types.js CHANGED
@@ -1,11 +1,4 @@
1
- /**
2
- * Type of validator duty being performed
3
- */ export var DutyType = /*#__PURE__*/ function(DutyType) {
4
- DutyType["BLOCK_PROPOSAL"] = "BLOCK_PROPOSAL";
5
- DutyType["ATTESTATION"] = "ATTESTATION";
6
- DutyType["ATTESTATIONS_AND_SIGNERS"] = "ATTESTATIONS_AND_SIGNERS";
7
- return DutyType;
8
- }({});
1
+ import { DutyType } from '@aztec/stdlib/ha-signing';
9
2
  /**
10
3
  * Status of a duty in the database
11
4
  */ export var DutyStatus = /*#__PURE__*/ function(DutyStatus) {
@@ -13,3 +6,34 @@
13
6
  DutyStatus["SIGNED"] = "signed";
14
7
  return DutyStatus;
15
8
  }({});
9
+ // Re-export DutyType from stdlib
10
+ export { DutyType };
11
+ /**
12
+ * Validates and normalizes the block index for a duty.
13
+ * - BLOCK_PROPOSAL: validates blockIndexWithinCheckpoint is provided and >= 0
14
+ * - Other duty types: always returns -1
15
+ *
16
+ * @throws Error if BLOCK_PROPOSAL is missing blockIndexWithinCheckpoint or has invalid value
17
+ */ export function normalizeBlockIndex(dutyType, blockIndexWithinCheckpoint) {
18
+ if (dutyType === DutyType.BLOCK_PROPOSAL) {
19
+ if (blockIndexWithinCheckpoint === undefined) {
20
+ throw new Error('BLOCK_PROPOSAL duties require blockIndexWithinCheckpoint to be specified');
21
+ }
22
+ if (blockIndexWithinCheckpoint < 0) {
23
+ throw new Error(`BLOCK_PROPOSAL duties require blockIndexWithinCheckpoint >= 0, got ${blockIndexWithinCheckpoint}`);
24
+ }
25
+ return blockIndexWithinCheckpoint;
26
+ }
27
+ // For all other duty types, always use -1
28
+ return -1;
29
+ }
30
+ /**
31
+ * Gets the block index from a DutyIdentifier.
32
+ * - BLOCK_PROPOSAL: returns the blockIndexWithinCheckpoint
33
+ * - Other duty types: returns -1
34
+ */ export function getBlockIndexFromDutyIdentifier(duty) {
35
+ if (duty.dutyType === DutyType.BLOCK_PROPOSAL) {
36
+ return duty.blockIndexWithinCheckpoint;
37
+ }
38
+ return -1;
39
+ }
package/dest/errors.d.ts CHANGED
@@ -1,6 +1,7 @@
1
1
  /**
2
2
  * Custom errors for the validator HA signer
3
3
  */
4
+ import type { SlotNumber } from '@aztec/foundation/branded-types';
4
5
  import type { DutyType } from './db/types.js';
5
6
  /**
6
7
  * Thrown when a duty has already been signed (by any node).
@@ -8,10 +9,11 @@ import type { DutyType } from './db/types.js';
8
9
  * the first one wins, and subsequent attempts get this error.
9
10
  */
10
11
  export declare class DutyAlreadySignedError extends Error {
11
- readonly slot: bigint;
12
+ readonly slot: SlotNumber;
12
13
  readonly dutyType: DutyType;
14
+ readonly blockIndexWithinCheckpoint: number;
13
15
  readonly signedByNode: string;
14
- constructor(slot: bigint, dutyType: DutyType, signedByNode: string);
16
+ constructor(slot: SlotNumber, dutyType: DutyType, blockIndexWithinCheckpoint: number, signedByNode: string);
15
17
  }
16
18
  /**
17
19
  * Thrown when attempting to sign data that conflicts with an already-signed duty.
@@ -21,10 +23,12 @@ export declare class DutyAlreadySignedError extends Error {
21
23
  * (e.g., different transaction ordering) - the protection prevents double-signing.
22
24
  */
23
25
  export declare class SlashingProtectionError extends Error {
24
- readonly slot: bigint;
26
+ readonly slot: SlotNumber;
25
27
  readonly dutyType: DutyType;
28
+ readonly blockIndexWithinCheckpoint: number;
26
29
  readonly existingMessageHash: string;
27
30
  readonly attemptedMessageHash: string;
28
- constructor(slot: bigint, dutyType: DutyType, existingMessageHash: string, attemptedMessageHash: string);
31
+ readonly signedByNode: string;
32
+ constructor(slot: SlotNumber, dutyType: DutyType, blockIndexWithinCheckpoint: number, existingMessageHash: string, attemptedMessageHash: string, signedByNode: string);
29
33
  }
30
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmQudHMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvZXJyb3JzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBQ0gsT0FBTyxLQUFLLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRTlDOzs7O0dBSUc7QUFDSCxxQkFBYSxzQkFBdUIsU0FBUSxLQUFLO2FBRTdCLElBQUksRUFBRSxNQUFNO2FBQ1osUUFBUSxFQUFFLFFBQVE7YUFDbEIsWUFBWSxFQUFFLE1BQU07SUFIdEMsWUFDa0IsSUFBSSxFQUFFLE1BQU0sRUFDWixRQUFRLEVBQUUsUUFBUSxFQUNsQixZQUFZLEVBQUUsTUFBTSxFQUlyQztDQUNGO0FBRUQ7Ozs7OztHQU1HO0FBQ0gscUJBQWEsdUJBQXdCLFNBQVEsS0FBSzthQUU5QixJQUFJLEVBQUUsTUFBTTthQUNaLFFBQVEsRUFBRSxRQUFRO2FBQ2xCLG1CQUFtQixFQUFFLE1BQU07YUFDM0Isb0JBQW9CLEVBQUUsTUFBTTtJQUo5QyxZQUNrQixJQUFJLEVBQUUsTUFBTSxFQUNaLFFBQVEsRUFBRSxRQUFRLEVBQ2xCLG1CQUFtQixFQUFFLE1BQU0sRUFDM0Isb0JBQW9CLEVBQUUsTUFBTSxFQU83QztDQUNGIn0=
34
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmQudHMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvZXJyb3JzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBQ0gsT0FBTyxLQUFLLEVBQUUsVUFBVSxFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFFbEUsT0FBTyxLQUFLLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRTlDOzs7O0dBSUc7QUFDSCxxQkFBYSxzQkFBdUIsU0FBUSxLQUFLO2FBRTdCLElBQUksRUFBRSxVQUFVO2FBQ2hCLFFBQVEsRUFBRSxRQUFRO2FBQ2xCLDBCQUEwQixFQUFFLE1BQU07YUFDbEMsWUFBWSxFQUFFLE1BQU07SUFKdEMsWUFDa0IsSUFBSSxFQUFFLFVBQVUsRUFDaEIsUUFBUSxFQUFFLFFBQVEsRUFDbEIsMEJBQTBCLEVBQUUsTUFBTSxFQUNsQyxZQUFZLEVBQUUsTUFBTSxFQUlyQztDQUNGO0FBRUQ7Ozs7OztHQU1HO0FBQ0gscUJBQWEsdUJBQXdCLFNBQVEsS0FBSzthQUU5QixJQUFJLEVBQUUsVUFBVTthQUNoQixRQUFRLEVBQUUsUUFBUTthQUNsQiwwQkFBMEIsRUFBRSxNQUFNO2FBQ2xDLG1CQUFtQixFQUFFLE1BQU07YUFDM0Isb0JBQW9CLEVBQUUsTUFBTTthQUM1QixZQUFZLEVBQUUsTUFBTTtJQU50QyxZQUNrQixJQUFJLEVBQUUsVUFBVSxFQUNoQixRQUFRLEVBQUUsUUFBUSxFQUNsQiwwQkFBMEIsRUFBRSxNQUFNLEVBQ2xDLG1CQUFtQixFQUFFLE1BQU0sRUFDM0Isb0JBQW9CLEVBQUUsTUFBTSxFQUM1QixZQUFZLEVBQUUsTUFBTSxFQU9yQztDQUNGIn0=
@@ -1 +1 @@
1
- {"version":3,"file":"errors.d.ts","sourceRoot":"","sources":["../src/errors.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAE9C;;;;GAIG;AACH,qBAAa,sBAAuB,SAAQ,KAAK;aAE7B,IAAI,EAAE,MAAM;aACZ,QAAQ,EAAE,QAAQ;aAClB,YAAY,EAAE,MAAM;IAHtC,YACkB,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,QAAQ,EAClB,YAAY,EAAE,MAAM,EAIrC;CACF;AAED;;;;;;GAMG;AACH,qBAAa,uBAAwB,SAAQ,KAAK;aAE9B,IAAI,EAAE,MAAM;aACZ,QAAQ,EAAE,QAAQ;aAClB,mBAAmB,EAAE,MAAM;aAC3B,oBAAoB,EAAE,MAAM;IAJ9C,YACkB,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,QAAQ,EAClB,mBAAmB,EAAE,MAAM,EAC3B,oBAAoB,EAAE,MAAM,EAO7C;CACF"}
1
+ {"version":3,"file":"errors.d.ts","sourceRoot":"","sources":["../src/errors.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,iCAAiC,CAAC;AAElE,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAE9C;;;;GAIG;AACH,qBAAa,sBAAuB,SAAQ,KAAK;aAE7B,IAAI,EAAE,UAAU;aAChB,QAAQ,EAAE,QAAQ;aAClB,0BAA0B,EAAE,MAAM;aAClC,YAAY,EAAE,MAAM;IAJtC,YACkB,IAAI,EAAE,UAAU,EAChB,QAAQ,EAAE,QAAQ,EAClB,0BAA0B,EAAE,MAAM,EAClC,YAAY,EAAE,MAAM,EAIrC;CACF;AAED;;;;;;GAMG;AACH,qBAAa,uBAAwB,SAAQ,KAAK;aAE9B,IAAI,EAAE,UAAU;aAChB,QAAQ,EAAE,QAAQ;aAClB,0BAA0B,EAAE,MAAM;aAClC,mBAAmB,EAAE,MAAM;aAC3B,oBAAoB,EAAE,MAAM;aAC5B,YAAY,EAAE,MAAM;IANtC,YACkB,IAAI,EAAE,UAAU,EAChB,QAAQ,EAAE,QAAQ,EAClB,0BAA0B,EAAE,MAAM,EAClC,mBAAmB,EAAE,MAAM,EAC3B,oBAAoB,EAAE,MAAM,EAC5B,YAAY,EAAE,MAAM,EAOrC;CACF"}
package/dest/errors.js CHANGED
@@ -7,9 +7,10 @@
7
7
  */ export class DutyAlreadySignedError extends Error {
8
8
  slot;
9
9
  dutyType;
10
+ blockIndexWithinCheckpoint;
10
11
  signedByNode;
11
- constructor(slot, dutyType, signedByNode){
12
- super(`Duty ${dutyType} for slot ${slot} already signed by node ${signedByNode}`), this.slot = slot, this.dutyType = dutyType, this.signedByNode = signedByNode;
12
+ constructor(slot, dutyType, blockIndexWithinCheckpoint, signedByNode){
13
+ super(`Duty ${dutyType} for slot ${slot} already signed by node ${signedByNode}`), this.slot = slot, this.dutyType = dutyType, this.blockIndexWithinCheckpoint = blockIndexWithinCheckpoint, this.signedByNode = signedByNode;
13
14
  this.name = 'DutyAlreadySignedError';
14
15
  }
15
16
  }
@@ -22,10 +23,12 @@
22
23
  */ export class SlashingProtectionError extends Error {
23
24
  slot;
24
25
  dutyType;
26
+ blockIndexWithinCheckpoint;
25
27
  existingMessageHash;
26
28
  attemptedMessageHash;
27
- constructor(slot, dutyType, existingMessageHash, attemptedMessageHash){
28
- super(`Slashing protection: ${dutyType} for slot ${slot} was already signed with different data. ` + `Existing: ${existingMessageHash.slice(0, 10)}..., Attempted: ${attemptedMessageHash.slice(0, 10)}...`), this.slot = slot, this.dutyType = dutyType, this.existingMessageHash = existingMessageHash, this.attemptedMessageHash = attemptedMessageHash;
29
+ signedByNode;
30
+ constructor(slot, dutyType, blockIndexWithinCheckpoint, existingMessageHash, attemptedMessageHash, signedByNode){
31
+ super(`Slashing protection: ${dutyType} for slot ${slot} was already signed with different data. ` + `Existing: ${existingMessageHash.slice(0, 10)}..., Attempted: ${attemptedMessageHash.slice(0, 10)}...`), this.slot = slot, this.dutyType = dutyType, this.blockIndexWithinCheckpoint = blockIndexWithinCheckpoint, this.existingMessageHash = existingMessageHash, this.attemptedMessageHash = attemptedMessageHash, this.signedByNode = signedByNode;
29
32
  this.name = 'SlashingProtectionError';
30
33
  }
31
34
  }
package/dest/factory.d.ts CHANGED
@@ -1,4 +1,4 @@
1
- import type { CreateHASignerConfig } from './config.js';
1
+ import type { ValidatorHASignerConfig } from '@aztec/stdlib/ha-signing';
2
2
  import type { CreateHASignerDeps, SlashingProtectionDatabase } from './types.js';
3
3
  import { ValidatorHASigner } from './validator_ha_signer.js';
4
4
  /**
@@ -16,7 +16,7 @@ import { ValidatorHASigner } from './validator_ha_signer.js';
16
16
  * ```typescript
17
17
  * const { signer, db } = await createHASigner({
18
18
  * databaseUrl: process.env.DATABASE_URL,
19
- * enabled: true,
19
+ * haSigningEnabled: true,
20
20
  * nodeId: 'validator-node-1',
21
21
  * pollingIntervalMs: 100,
22
22
  * signingTimeoutMs: 3000,
@@ -28,23 +28,15 @@ import { ValidatorHASigner } from './validator_ha_signer.js';
28
28
  * await signer.stop(); // On shutdown
29
29
  * ```
30
30
  *
31
- * Example with automatic migrations (simpler for dev/testing):
32
- * ```typescript
33
- * const { signer, db } = await createHASigner({
34
- * databaseUrl: process.env.DATABASE_URL,
35
- * enabled: true,
36
- * nodeId: 'validator-node-1',
37
- * runMigrations: true, // Auto-run migrations on startup
38
- * });
39
- * signer.start();
40
- * ```
31
+ * Note: Migrations must be run separately using `aztec migrate-ha-db up` before
32
+ * creating the signer. The factory will verify the schema is initialized via `db.initialize()`.
41
33
  *
42
34
  * @param config - Configuration for the HA signer
43
35
  * @param deps - Optional dependencies (e.g., for testing)
44
36
  * @returns An object containing the signer and database instances
45
37
  */
46
- export declare function createHASigner(config: CreateHASignerConfig, deps?: CreateHASignerDeps): Promise<{
38
+ export declare function createHASigner(config: ValidatorHASignerConfig, deps?: CreateHASignerDeps): Promise<{
47
39
  signer: ValidatorHASigner;
48
40
  db: SlashingProtectionDatabase;
49
41
  }>;
50
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmFjdG9yeS5kLnRzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2ZhY3RvcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBS0EsT0FBTyxLQUFLLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFFeEQsT0FBTyxLQUFLLEVBQUUsa0JBQWtCLEVBQUUsMEJBQTBCLEVBQUUsTUFBTSxZQUFZLENBQUM7QUFDakYsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFFN0Q7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBeUNHO0FBQ0gsd0JBQXNCLGNBQWMsQ0FDbEMsTUFBTSxFQUFFLG9CQUFvQixFQUM1QixJQUFJLENBQUMsRUFBRSxrQkFBa0IsR0FDeEIsT0FBTyxDQUFDO0lBQ1QsTUFBTSxFQUFFLGlCQUFpQixDQUFDO0lBQzFCLEVBQUUsRUFBRSwwQkFBMEIsQ0FBQztDQUNoQyxDQUFDLENBNEJEIn0=
42
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmFjdG9yeS5kLnRzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2ZhY3RvcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBSUEsT0FBTyxLQUFLLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQU94RSxPQUFPLEtBQUssRUFBRSxrQkFBa0IsRUFBRSwwQkFBMEIsRUFBRSxNQUFNLFlBQVksQ0FBQztBQUNqRixPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUU3RDs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBaUNHO0FBQ0gsd0JBQXNCLGNBQWMsQ0FDbEMsTUFBTSxFQUFFLHVCQUF1QixFQUMvQixJQUFJLENBQUMsRUFBRSxrQkFBa0IsR0FDeEIsT0FBTyxDQUFDO0lBQ1QsTUFBTSxFQUFFLGlCQUFpQixDQUFDO0lBQzFCLEVBQUUsRUFBRSwwQkFBMEIsQ0FBQztDQUNoQyxDQUFDLENBc0NEIn0=
@@ -1 +1 @@
1
- {"version":3,"file":"factory.d.ts","sourceRoot":"","sources":["../src/factory.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,aAAa,CAAC;AAExD,OAAO,KAAK,EAAE,kBAAkB,EAAE,0BAA0B,EAAE,MAAM,YAAY,CAAC;AACjF,OAAO,EAAE,iBAAiB,EAAE,MAAM,0BAA0B,CAAC;AAE7D;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAyCG;AACH,wBAAsB,cAAc,CAClC,MAAM,EAAE,oBAAoB,EAC5B,IAAI,CAAC,EAAE,kBAAkB,GACxB,OAAO,CAAC;IACT,MAAM,EAAE,iBAAiB,CAAC;IAC1B,EAAE,EAAE,0BAA0B,CAAC;CAChC,CAAC,CA4BD"}
1
+ {"version":3,"file":"factory.d.ts","sourceRoot":"","sources":["../src/factory.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,uBAAuB,EAAE,MAAM,0BAA0B,CAAC;AAOxE,OAAO,KAAK,EAAE,kBAAkB,EAAE,0BAA0B,EAAE,MAAM,YAAY,CAAC;AACjF,OAAO,EAAE,iBAAiB,EAAE,MAAM,0BAA0B,CAAC;AAE7D;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiCG;AACH,wBAAsB,cAAc,CAClC,MAAM,EAAE,uBAAuB,EAC/B,IAAI,CAAC,EAAE,kBAAkB,GACxB,OAAO,CAAC;IACT,MAAM,EAAE,iBAAiB,CAAC;IAC1B,EAAE,EAAE,0BAA0B,CAAC;CAChC,CAAC,CAsCD"}
package/dest/factory.js CHANGED
@@ -1,7 +1,10 @@
1
1
  /**
2
2
  * Factory functions for creating validator HA signers
3
- */ import { Pool } from 'pg';
3
+ */ import { DateProvider } from '@aztec/foundation/timer';
4
+ import { getTelemetryClient } from '@aztec/telemetry-client';
5
+ import { Pool } from 'pg';
4
6
  import { PostgresSlashingProtectionDatabase } from './db/postgres.js';
7
+ import { HASignerMetrics } from './metrics.js';
5
8
  import { ValidatorHASigner } from './validator_ha_signer.js';
6
9
  /**
7
10
  * Create a validator HA signer with PostgreSQL backend
@@ -18,7 +21,7 @@ import { ValidatorHASigner } from './validator_ha_signer.js';
18
21
  * ```typescript
19
22
  * const { signer, db } = await createHASigner({
20
23
  * databaseUrl: process.env.DATABASE_URL,
21
- * enabled: true,
24
+ * haSigningEnabled: true,
22
25
  * nodeId: 'validator-node-1',
23
26
  * pollingIntervalMs: 100,
24
27
  * signingTimeoutMs: 3000,
@@ -30,22 +33,19 @@ import { ValidatorHASigner } from './validator_ha_signer.js';
30
33
  * await signer.stop(); // On shutdown
31
34
  * ```
32
35
  *
33
- * Example with automatic migrations (simpler for dev/testing):
34
- * ```typescript
35
- * const { signer, db } = await createHASigner({
36
- * databaseUrl: process.env.DATABASE_URL,
37
- * enabled: true,
38
- * nodeId: 'validator-node-1',
39
- * runMigrations: true, // Auto-run migrations on startup
40
- * });
41
- * signer.start();
42
- * ```
36
+ * Note: Migrations must be run separately using `aztec migrate-ha-db up` before
37
+ * creating the signer. The factory will verify the schema is initialized via `db.initialize()`.
43
38
  *
44
39
  * @param config - Configuration for the HA signer
45
40
  * @param deps - Optional dependencies (e.g., for testing)
46
41
  * @returns An object containing the signer and database instances
47
42
  */ export async function createHASigner(config, deps) {
48
43
  const { databaseUrl, poolMaxCount, poolMinCount, poolIdleTimeoutMs, poolConnectionTimeoutMs, ...signerConfig } = config;
44
+ if (!databaseUrl) {
45
+ throw new Error('databaseUrl is required for createHASigner');
46
+ }
47
+ const telemetryClient = deps?.telemetryClient ?? getTelemetryClient();
48
+ const dateProvider = deps?.dateProvider ?? new DateProvider();
49
49
  // Create connection pool (or use provided pool)
50
50
  let pool;
51
51
  if (!deps?.pool) {
@@ -63,10 +63,15 @@ import { ValidatorHASigner } from './validator_ha_signer.js';
63
63
  const db = new PostgresSlashingProtectionDatabase(pool);
64
64
  // Verify database schema is initialized and version matches
65
65
  await db.initialize();
66
+ // Create metrics
67
+ const metrics = new HASignerMetrics(telemetryClient, signerConfig.nodeId);
66
68
  // Create signer
67
69
  const signer = new ValidatorHASigner(db, {
68
70
  ...signerConfig,
69
71
  databaseUrl
72
+ }, {
73
+ metrics,
74
+ dateProvider
70
75
  });
71
76
  return {
72
77
  signer,
@@ -0,0 +1,51 @@
1
+ import { type TelemetryClient } from '@aztec/telemetry-client';
2
+ export type HACleanupType = 'stuck' | 'old' | 'outdated_rollup';
3
+ /**
4
+ * Metrics for HA signer tracking signing operations, lock acquisition, and cleanup.
5
+ */
6
+ export declare class HASignerMetrics {
7
+ private nodeId;
8
+ private signingDuration;
9
+ private signingSuccessCount;
10
+ private dutyAlreadySignedCount;
11
+ private slashingProtectionCount;
12
+ private signingErrorCount;
13
+ private lockAcquiredCount;
14
+ private cleanupStuckDutiesCount;
15
+ private cleanupOldDutiesCount;
16
+ private cleanupOutdatedRollupDutiesCount;
17
+ constructor(client: TelemetryClient, nodeId: string, name?: string);
18
+ /**
19
+ * Record a successful signing operation.
20
+ * @param dutyType - The type of duty signed
21
+ * @param durationMs - Duration from start of signWithProtection to completion
22
+ */
23
+ recordSigningSuccess(dutyType: string, durationMs: number): void;
24
+ /**
25
+ * Record a DutyAlreadySignedError (expected in HA; another node signed first).
26
+ * @param dutyType - The type of duty
27
+ */
28
+ recordDutyAlreadySigned(dutyType: string): void;
29
+ /**
30
+ * Record a SlashingProtectionError (attempted to sign different data for same duty).
31
+ * @param dutyType - The type of duty
32
+ */
33
+ recordSlashingProtection(dutyType: string): void;
34
+ /**
35
+ * Record a signing function failure (lock will be deleted for retry).
36
+ * @param dutyType - The type of duty
37
+ */
38
+ recordSigningError(dutyType: string): void;
39
+ /**
40
+ * Record lock acquisition.
41
+ * @param acquired - Whether a new lock was acquired (true) or existing record found (false)
42
+ */
43
+ recordLockAcquire(acquired: boolean): void;
44
+ /**
45
+ * Record cleanup metrics.
46
+ * @param type - Type of cleanup
47
+ * @param count - Number of duties cleaned up
48
+ */
49
+ recordCleanup(type: HACleanupType, count: number): void;
50
+ }
51
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0cmljcy5kLnRzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL21ldHJpY3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUlMLEtBQUssZUFBZSxFQUdyQixNQUFNLHlCQUF5QixDQUFDO0FBRWpDLE1BQU0sTUFBTSxhQUFhLEdBQUcsT0FBTyxHQUFHLEtBQUssR0FBRyxpQkFBaUIsQ0FBQztBQUVoRTs7R0FFRztBQUNILHFCQUFhLGVBQWU7SUFrQnhCLE9BQU8sQ0FBQyxNQUFNO0lBaEJoQixPQUFPLENBQUMsZUFBZSxDQUFZO0lBQ25DLE9BQU8sQ0FBQyxtQkFBbUIsQ0FBZ0I7SUFDM0MsT0FBTyxDQUFDLHNCQUFzQixDQUFnQjtJQUM5QyxPQUFPLENBQUMsdUJBQXVCLENBQWdCO0lBQy9DLE9BQU8sQ0FBQyxpQkFBaUIsQ0FBZ0I7SUFHekMsT0FBTyxDQUFDLGlCQUFpQixDQUFnQjtJQUd6QyxPQUFPLENBQUMsdUJBQXVCLENBQWdCO0lBQy9DLE9BQU8sQ0FBQyxxQkFBcUIsQ0FBZ0I7SUFDN0MsT0FBTyxDQUFDLGdDQUFnQyxDQUFnQjtJQUV4RCxZQUNFLE1BQU0sRUFBRSxlQUFlLEVBQ2YsTUFBTSxFQUFFLE1BQU0sRUFDdEIsSUFBSSxTQUFvQixFQXFCekI7SUFFRDs7OztPQUlHO0lBQ0ksb0JBQW9CLENBQUMsUUFBUSxFQUFFLE1BQU0sRUFBRSxVQUFVLEVBQUUsTUFBTSxHQUFHLElBQUksQ0FPdEU7SUFFRDs7O09BR0c7SUFDSSx1QkFBdUIsQ0FBQyxRQUFRLEVBQUUsTUFBTSxHQUFHLElBQUksQ0FNckQ7SUFFRDs7O09BR0c7SUFDSSx3QkFBd0IsQ0FBQyxRQUFRLEVBQUUsTUFBTSxHQUFHLElBQUksQ0FNdEQ7SUFFRDs7O09BR0c7SUFDSSxrQkFBa0IsQ0FBQyxRQUFRLEVBQUUsTUFBTSxHQUFHLElBQUksQ0FNaEQ7SUFFRDs7O09BR0c7SUFDSSxpQkFBaUIsQ0FBQyxRQUFRLEVBQUUsT0FBTyxHQUFHLElBQUksQ0FPaEQ7SUFFRDs7OztPQUlHO0lBQ0ksYUFBYSxDQUFDLElBQUksRUFBRSxhQUFhLEVBQUUsS0FBSyxFQUFFLE1BQU0sR0FBRyxJQUFJLENBWTdEO0NBQ0YifQ==