@aztec/validator-ha-signer 0.0.1-commit.7d4e6cd → 0.0.1-commit.86469d5

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 (46) hide show
  1. package/README.md +42 -37
  2. package/dest/config.d.ts +66 -17
  3. package/dest/config.d.ts.map +1 -1
  4. package/dest/config.js +41 -19
  5. package/dest/db/postgres.d.ts +20 -5
  6. package/dest/db/postgres.d.ts.map +1 -1
  7. package/dest/db/postgres.js +56 -19
  8. package/dest/db/schema.d.ts +11 -7
  9. package/dest/db/schema.d.ts.map +1 -1
  10. package/dest/db/schema.js +35 -15
  11. package/dest/db/types.d.ts +80 -23
  12. package/dest/db/types.d.ts.map +1 -1
  13. package/dest/db/types.js +34 -0
  14. package/dest/errors.d.ts +9 -5
  15. package/dest/errors.d.ts.map +1 -1
  16. package/dest/errors.js +7 -4
  17. package/dest/factory.d.ts +6 -14
  18. package/dest/factory.d.ts.map +1 -1
  19. package/dest/factory.js +6 -11
  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 +9 -3
  24. package/dest/slashing_protection_service.d.ts.map +1 -1
  25. package/dest/slashing_protection_service.js +23 -11
  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 +78 -14
  30. package/dest/types.d.ts.map +1 -1
  31. package/dest/types.js +21 -1
  32. package/dest/validator_ha_signer.d.ts +9 -12
  33. package/dest/validator_ha_signer.d.ts.map +1 -1
  34. package/dest/validator_ha_signer.js +31 -30
  35. package/package.json +9 -8
  36. package/src/config.ts +75 -50
  37. package/src/db/postgres.ts +76 -19
  38. package/src/db/schema.ts +35 -15
  39. package/src/db/types.ts +110 -21
  40. package/src/errors.ts +7 -2
  41. package/src/factory.ts +8 -13
  42. package/src/migrations.ts +17 -1
  43. package/src/slashing_protection_service.ts +55 -13
  44. package/src/test/pglite_pool.ts +256 -0
  45. package/src/types.ts +125 -19
  46. package/src/validator_ha_signer.ts +38 -40
package/src/db/types.ts CHANGED
@@ -1,3 +1,4 @@
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';
3
4
 
@@ -5,9 +6,11 @@ import type { Signature } from '@aztec/foundation/eth-signature';
5
6
  * Row type from PostgreSQL query
6
7
  */
7
8
  export interface DutyRow {
9
+ rollup_address: string;
8
10
  validator_address: string;
9
11
  slot: string;
10
12
  block_number: string;
13
+ block_index_within_checkpoint: number;
11
14
  duty_type: DutyType;
12
15
  status: DutyStatus;
13
16
  message_hash: string;
@@ -31,8 +34,13 @@ export interface InsertOrGetRow extends DutyRow {
31
34
  */
32
35
  export enum DutyType {
33
36
  BLOCK_PROPOSAL = 'BLOCK_PROPOSAL',
37
+ CHECKPOINT_PROPOSAL = 'CHECKPOINT_PROPOSAL',
34
38
  ATTESTATION = 'ATTESTATION',
35
39
  ATTESTATIONS_AND_SIGNERS = 'ATTESTATIONS_AND_SIGNERS',
40
+ GOVERNANCE_VOTE = 'GOVERNANCE_VOTE',
41
+ SLASHING_VOTE = 'SLASHING_VOTE',
42
+ AUTH_REQUEST = 'AUTH_REQUEST',
43
+ TXS = 'TXS',
36
44
  }
37
45
 
38
46
  /**
@@ -47,12 +55,16 @@ export enum DutyStatus {
47
55
  * Record of a validator duty in the database
48
56
  */
49
57
  export interface ValidatorDutyRecord {
58
+ /** Ethereum address of the rollup contract */
59
+ rollupAddress: EthAddress;
50
60
  /** Ethereum address of the validator */
51
61
  validatorAddress: EthAddress;
52
62
  /** Slot number for this duty */
53
- slot: bigint;
63
+ slot: SlotNumber;
54
64
  /** Block number for this duty */
55
- blockNumber: bigint;
65
+ blockNumber: BlockNumber;
66
+ /** Block index within checkpoint (0, 1, 2... for block proposals, -1 for other duty types) */
67
+ blockIndexWithinCheckpoint: number;
56
68
  /** Type of duty being performed */
57
69
  dutyType: DutyType;
58
70
  /** Current status of the duty */
@@ -74,44 +86,121 @@ export interface ValidatorDutyRecord {
74
86
  }
75
87
 
76
88
  /**
77
- * Minimal info needed to identify a unique duty
89
+ * Duty identifier for block proposals.
90
+ * blockIndexWithinCheckpoint is REQUIRED and must be >= 0.
78
91
  */
79
- export interface DutyIdentifier {
92
+ export interface BlockProposalDutyIdentifier {
93
+ rollupAddress: EthAddress;
80
94
  validatorAddress: EthAddress;
81
- slot: bigint;
82
- dutyType: DutyType;
95
+ slot: SlotNumber;
96
+ /** Block index within checkpoint (0, 1, 2...). Required for block proposals. */
97
+ blockIndexWithinCheckpoint: IndexWithinCheckpoint;
98
+ dutyType: DutyType.BLOCK_PROPOSAL;
83
99
  }
84
100
 
85
101
  /**
86
- * Parameters for checking and recording a new duty
102
+ * Duty identifier for non-block-proposal duties.
103
+ * blockIndexWithinCheckpoint is not applicable (internally stored as -1).
87
104
  */
88
- export interface CheckAndRecordParams {
105
+ export interface OtherDutyIdentifier {
106
+ rollupAddress: EthAddress;
89
107
  validatorAddress: EthAddress;
90
- slot: bigint;
91
- blockNumber: bigint;
92
- dutyType: DutyType;
108
+ slot: SlotNumber;
109
+ dutyType:
110
+ | DutyType.CHECKPOINT_PROPOSAL
111
+ | DutyType.ATTESTATION
112
+ | DutyType.ATTESTATIONS_AND_SIGNERS
113
+ | DutyType.GOVERNANCE_VOTE
114
+ | DutyType.SLASHING_VOTE
115
+ | DutyType.AUTH_REQUEST
116
+ | DutyType.TXS;
117
+ }
118
+
119
+ /**
120
+ * Minimal info needed to identify a unique duty.
121
+ * Uses discriminated union to enforce type safety:
122
+ * - BLOCK_PROPOSAL duties MUST have blockIndexWithinCheckpoint >= 0
123
+ * - Other duty types do NOT have blockIndexWithinCheckpoint (internally -1)
124
+ */
125
+ export type DutyIdentifier = BlockProposalDutyIdentifier | OtherDutyIdentifier;
126
+
127
+ /**
128
+ * Validates and normalizes the block index for a duty.
129
+ * - BLOCK_PROPOSAL: validates blockIndexWithinCheckpoint is provided and >= 0
130
+ * - Other duty types: always returns -1
131
+ *
132
+ * @throws Error if BLOCK_PROPOSAL is missing blockIndexWithinCheckpoint or has invalid value
133
+ */
134
+ export function normalizeBlockIndex(dutyType: DutyType, blockIndexWithinCheckpoint: number | undefined): number {
135
+ if (dutyType === DutyType.BLOCK_PROPOSAL) {
136
+ if (blockIndexWithinCheckpoint === undefined) {
137
+ throw new Error('BLOCK_PROPOSAL duties require blockIndexWithinCheckpoint to be specified');
138
+ }
139
+ if (blockIndexWithinCheckpoint < 0) {
140
+ throw new Error(
141
+ `BLOCK_PROPOSAL duties require blockIndexWithinCheckpoint >= 0, got ${blockIndexWithinCheckpoint}`,
142
+ );
143
+ }
144
+ return blockIndexWithinCheckpoint;
145
+ }
146
+ // For all other duty types, always use -1
147
+ return -1;
148
+ }
149
+
150
+ /**
151
+ * Gets the block index from a DutyIdentifier.
152
+ * - BLOCK_PROPOSAL: returns the blockIndexWithinCheckpoint
153
+ * - Other duty types: returns -1
154
+ */
155
+ export function getBlockIndexFromDutyIdentifier(duty: DutyIdentifier): number {
156
+ if (duty.dutyType === DutyType.BLOCK_PROPOSAL) {
157
+ return duty.blockIndexWithinCheckpoint;
158
+ }
159
+ return -1;
160
+ }
161
+
162
+ /**
163
+ * Additional parameters for checking and recording a new duty
164
+ */
165
+ interface CheckAndRecordExtra {
166
+ /** Block number for this duty */
167
+ blockNumber: BlockNumber | CheckpointNumber;
168
+ /** The signing root (hash) for this duty */
93
169
  messageHash: string;
170
+ /** Identifier for the node that acquired the lock */
94
171
  nodeId: string;
95
172
  }
96
173
 
97
174
  /**
98
- * Parameters for recording a successful signing
175
+ * Parameters for checking and recording a new duty.
176
+ * Uses intersection with DutyIdentifier to preserve the discriminated union.
99
177
  */
100
- export interface RecordSuccessParams {
101
- validatorAddress: EthAddress;
102
- slot: bigint;
103
- dutyType: DutyType;
178
+ export type CheckAndRecordParams = DutyIdentifier & CheckAndRecordExtra;
179
+
180
+ /**
181
+ * Additional parameters for recording a successful signing
182
+ */
183
+ interface RecordSuccessExtra {
104
184
  signature: Signature;
105
185
  nodeId: string;
106
186
  lockToken: string;
107
187
  }
108
188
 
109
189
  /**
110
- * Parameters for deleting a duty
190
+ * Parameters for recording a successful signing.
191
+ * Uses intersection with DutyIdentifier to preserve the discriminated union.
111
192
  */
112
- export interface DeleteDutyParams {
113
- validatorAddress: EthAddress;
114
- slot: bigint;
115
- dutyType: DutyType;
193
+ export type RecordSuccessParams = DutyIdentifier & RecordSuccessExtra;
194
+
195
+ /**
196
+ * Additional parameters for deleting a duty
197
+ */
198
+ interface DeleteDutyExtra {
116
199
  lockToken: string;
117
200
  }
201
+
202
+ /**
203
+ * Parameters for deleting a duty.
204
+ * Uses intersection with DutyIdentifier to preserve the discriminated union.
205
+ */
206
+ export type DeleteDutyParams = DutyIdentifier & DeleteDutyExtra;
package/src/errors.ts CHANGED
@@ -1,6 +1,8 @@
1
1
  /**
2
2
  * Custom errors for the validator HA signer
3
3
  */
4
+ import type { SlotNumber } from '@aztec/foundation/branded-types';
5
+
4
6
  import type { DutyType } from './db/types.js';
5
7
 
6
8
  /**
@@ -10,8 +12,9 @@ import type { DutyType } from './db/types.js';
10
12
  */
11
13
  export class DutyAlreadySignedError extends Error {
12
14
  constructor(
13
- public readonly slot: bigint,
15
+ public readonly slot: SlotNumber,
14
16
  public readonly dutyType: DutyType,
17
+ public readonly blockIndexWithinCheckpoint: number,
15
18
  public readonly signedByNode: string,
16
19
  ) {
17
20
  super(`Duty ${dutyType} for slot ${slot} already signed by node ${signedByNode}`);
@@ -28,10 +31,12 @@ export class DutyAlreadySignedError extends Error {
28
31
  */
29
32
  export class SlashingProtectionError extends Error {
30
33
  constructor(
31
- public readonly slot: bigint,
34
+ public readonly slot: SlotNumber,
32
35
  public readonly dutyType: DutyType,
36
+ public readonly blockIndexWithinCheckpoint: number,
33
37
  public readonly existingMessageHash: string,
34
38
  public readonly attemptedMessageHash: string,
39
+ public readonly signedByNode: string,
35
40
  ) {
36
41
  super(
37
42
  `Slashing protection: ${dutyType} for slot ${slot} was already signed with different data. ` +
package/src/factory.ts CHANGED
@@ -3,7 +3,7 @@
3
3
  */
4
4
  import { Pool } from 'pg';
5
5
 
6
- import type { CreateHASignerConfig } from './config.js';
6
+ import type { ValidatorHASignerConfig } from './config.js';
7
7
  import { PostgresSlashingProtectionDatabase } from './db/postgres.js';
8
8
  import type { CreateHASignerDeps, SlashingProtectionDatabase } from './types.js';
9
9
  import { ValidatorHASigner } from './validator_ha_signer.js';
@@ -23,7 +23,7 @@ import { ValidatorHASigner } from './validator_ha_signer.js';
23
23
  * ```typescript
24
24
  * const { signer, db } = await createHASigner({
25
25
  * databaseUrl: process.env.DATABASE_URL,
26
- * enabled: true,
26
+ * haSigningEnabled: true,
27
27
  * nodeId: 'validator-node-1',
28
28
  * pollingIntervalMs: 100,
29
29
  * signingTimeoutMs: 3000,
@@ -35,23 +35,15 @@ import { ValidatorHASigner } from './validator_ha_signer.js';
35
35
  * await signer.stop(); // On shutdown
36
36
  * ```
37
37
  *
38
- * Example with automatic migrations (simpler for dev/testing):
39
- * ```typescript
40
- * const { signer, db } = await createHASigner({
41
- * databaseUrl: process.env.DATABASE_URL,
42
- * enabled: true,
43
- * nodeId: 'validator-node-1',
44
- * runMigrations: true, // Auto-run migrations on startup
45
- * });
46
- * signer.start();
47
- * ```
38
+ * Note: Migrations must be run separately using `aztec migrate-ha-db up` before
39
+ * creating the signer. The factory will verify the schema is initialized via `db.initialize()`.
48
40
  *
49
41
  * @param config - Configuration for the HA signer
50
42
  * @param deps - Optional dependencies (e.g., for testing)
51
43
  * @returns An object containing the signer and database instances
52
44
  */
53
45
  export async function createHASigner(
54
- config: CreateHASignerConfig,
46
+ config: ValidatorHASignerConfig,
55
47
  deps?: CreateHASignerDeps,
56
48
  ): Promise<{
57
49
  signer: ValidatorHASigner;
@@ -60,6 +52,9 @@ export async function createHASigner(
60
52
  const { databaseUrl, poolMaxCount, poolMinCount, poolIdleTimeoutMs, poolConnectionTimeoutMs, ...signerConfig } =
61
53
  config;
62
54
 
55
+ if (!databaseUrl) {
56
+ throw new Error('databaseUrl is required for createHASigner');
57
+ }
63
58
  // Create connection pool (or use provided pool)
64
59
  let pool: Pool;
65
60
  if (!deps?.pool) {
package/src/migrations.ts CHANGED
@@ -3,6 +3,7 @@
3
3
  */
4
4
  import { createLogger } from '@aztec/foundation/log';
5
5
 
6
+ import { readdirSync } from 'fs';
6
7
  import { runner } from 'node-pg-migrate';
7
8
  import { dirname, join } from 'path';
8
9
  import { fileURLToPath } from 'url';
@@ -30,17 +31,32 @@ export async function runMigrations(databaseUrl: string, options: RunMigrationsO
30
31
 
31
32
  const log = createLogger('validator-ha-signer:migrations');
32
33
 
34
+ const migrationsDir = join(__dirname, 'db', 'migrations');
35
+
33
36
  try {
34
37
  log.info(`Running migrations ${direction}...`);
35
38
 
39
+ // Filter out .d.ts and .d.ts.map files - node-pg-migrate only needs .js files
40
+ const migrationFiles = readdirSync(migrationsDir);
41
+ const jsMigrationFiles = migrationFiles.filter(
42
+ file => file.endsWith('.js') && !file.endsWith('.d.ts') && !file.endsWith('.d.ts.map'),
43
+ );
44
+
45
+ if (jsMigrationFiles.length === 0) {
46
+ log.info('No migration files found');
47
+ return [];
48
+ }
49
+
36
50
  const appliedMigrations = await runner({
37
51
  databaseUrl,
38
- dir: join(__dirname, 'db', 'migrations'),
52
+ dir: migrationsDir,
39
53
  direction,
40
54
  migrationsTable: 'pgmigrations',
41
55
  count: direction === 'down' ? 1 : Infinity,
42
56
  verbose,
43
57
  log: msg => (verbose ? log.info(msg) : log.debug(msg)),
58
+ // Ignore TypeScript declaration files - node-pg-migrate will try to import them otherwise
59
+ ignorePattern: '.*\\.d\\.(ts|js)$|.*\\.d\\.ts\\.map$',
44
60
  });
45
61
 
46
62
  if (appliedMigrations.length === 0) {
@@ -8,9 +8,15 @@ import { type Logger, createLogger } from '@aztec/foundation/log';
8
8
  import { RunningPromise } from '@aztec/foundation/promise';
9
9
  import { sleep } from '@aztec/foundation/sleep';
10
10
 
11
- import { type CheckAndRecordParams, type DeleteDutyParams, DutyStatus, type RecordSuccessParams } from './db/types.js';
11
+ import {
12
+ type CheckAndRecordParams,
13
+ type DeleteDutyParams,
14
+ DutyStatus,
15
+ type RecordSuccessParams,
16
+ getBlockIndexFromDutyIdentifier,
17
+ } from './db/types.js';
12
18
  import { DutyAlreadySignedError, SlashingProtectionError } from './errors.js';
13
- import type { SlashingProtectionConfig, SlashingProtectionDatabase } from './types.js';
19
+ import type { SlashingProtectionDatabase, ValidatorHASignerConfig } from './types.js';
14
20
 
15
21
  /**
16
22
  * Slashing Protection Service
@@ -31,21 +37,24 @@ export class SlashingProtectionService {
31
37
  private readonly log: Logger;
32
38
  private readonly pollingIntervalMs: number;
33
39
  private readonly signingTimeoutMs: number;
40
+ private readonly maxStuckDutiesAgeMs: number;
34
41
 
35
42
  private cleanupRunningPromise: RunningPromise;
36
43
 
37
44
  constructor(
38
45
  private readonly db: SlashingProtectionDatabase,
39
- private readonly config: SlashingProtectionConfig,
46
+ private readonly config: ValidatorHASignerConfig,
40
47
  ) {
41
48
  this.log = createLogger('slashing-protection');
42
49
  this.pollingIntervalMs = config.pollingIntervalMs;
43
50
  this.signingTimeoutMs = config.signingTimeoutMs;
51
+ // Default to 144s (2x 72s Aztec slot duration) if not explicitly configured
52
+ this.maxStuckDutiesAgeMs = config.maxStuckDutiesAgeMs ?? 144_000;
44
53
 
45
54
  this.cleanupRunningPromise = new RunningPromise(
46
55
  this.cleanupStuckDuties.bind(this),
47
56
  this.log,
48
- this.config.maxStuckDutiesAgeMs,
57
+ this.maxStuckDutiesAgeMs,
49
58
  );
50
59
  }
51
60
 
@@ -98,9 +107,16 @@ export class SlashingProtectionService {
98
107
  existingNodeId: record.nodeId,
99
108
  attemptingNodeId: nodeId,
100
109
  });
101
- throw new SlashingProtectionError(slot, dutyType, record.messageHash, messageHash);
110
+ throw new SlashingProtectionError(
111
+ slot,
112
+ dutyType,
113
+ record.blockIndexWithinCheckpoint,
114
+ record.messageHash,
115
+ messageHash,
116
+ record.nodeId,
117
+ );
102
118
  }
103
- throw new DutyAlreadySignedError(slot, dutyType, record.nodeId);
119
+ throw new DutyAlreadySignedError(slot, dutyType, record.blockIndexWithinCheckpoint, record.nodeId);
104
120
  } else if (record.status === DutyStatus.SIGNING) {
105
121
  // Another node is currently signing - check for timeout
106
122
  if (Date.now() - startTime > this.signingTimeoutMs) {
@@ -109,7 +125,7 @@ export class SlashingProtectionService {
109
125
  timeoutMs: this.signingTimeoutMs,
110
126
  signingNodeId: record.nodeId,
111
127
  });
112
- throw new DutyAlreadySignedError(slot, dutyType, 'unknown (timeout)');
128
+ throw new DutyAlreadySignedError(slot, dutyType, record.blockIndexWithinCheckpoint, 'unknown (timeout)');
113
129
  }
114
130
 
115
131
  // Wait and poll
@@ -133,9 +149,18 @@ export class SlashingProtectionService {
133
149
  * @returns true if the update succeeded, false if token didn't match
134
150
  */
135
151
  async recordSuccess(params: RecordSuccessParams): Promise<boolean> {
136
- const { validatorAddress, slot, dutyType, signature, nodeId, lockToken } = params;
152
+ const { rollupAddress, validatorAddress, slot, dutyType, signature, nodeId, lockToken } = params;
153
+ const blockIndexWithinCheckpoint = getBlockIndexFromDutyIdentifier(params);
137
154
 
138
- const success = await this.db.updateDutySigned(validatorAddress, slot, dutyType, signature.toString(), lockToken);
155
+ const success = await this.db.updateDutySigned(
156
+ rollupAddress,
157
+ validatorAddress,
158
+ slot,
159
+ dutyType,
160
+ signature.toString(),
161
+ lockToken,
162
+ blockIndexWithinCheckpoint,
163
+ );
139
164
 
140
165
  if (success) {
141
166
  this.log.info(`Recorded successful signing for duty ${dutyType} at slot ${slot}`, {
@@ -160,9 +185,17 @@ export class SlashingProtectionService {
160
185
  * @returns true if the delete succeeded, false if token didn't match
161
186
  */
162
187
  async deleteDuty(params: DeleteDutyParams): Promise<boolean> {
163
- const { validatorAddress, slot, dutyType, lockToken } = params;
188
+ const { rollupAddress, validatorAddress, slot, dutyType, lockToken } = params;
189
+ const blockIndexWithinCheckpoint = getBlockIndexFromDutyIdentifier(params);
164
190
 
165
- const success = await this.db.deleteDuty(validatorAddress, slot, dutyType, lockToken);
191
+ const success = await this.db.deleteDuty(
192
+ rollupAddress,
193
+ validatorAddress,
194
+ slot,
195
+ dutyType,
196
+ lockToken,
197
+ blockIndexWithinCheckpoint,
198
+ );
166
199
 
167
200
  if (success) {
168
201
  this.log.info(`Deleted duty ${dutyType} at slot ${slot} to allow retry`, {
@@ -201,15 +234,24 @@ export class SlashingProtectionService {
201
234
  this.log.info('Slashing protection service stopped', { nodeId: this.config.nodeId });
202
235
  }
203
236
 
237
+ /**
238
+ * Close the database connection.
239
+ * Should be called after stop() during graceful shutdown.
240
+ */
241
+ async close() {
242
+ await this.db.close();
243
+ this.log.info('Slashing protection database connection closed');
244
+ }
245
+
204
246
  /**
205
247
  * Cleanup own stuck duties
206
248
  */
207
249
  private async cleanupStuckDuties() {
208
- const numDuties = await this.db.cleanupOwnStuckDuties(this.config.nodeId, this.config.maxStuckDutiesAgeMs);
250
+ const numDuties = await this.db.cleanupOwnStuckDuties(this.config.nodeId, this.maxStuckDutiesAgeMs);
209
251
  if (numDuties > 0) {
210
252
  this.log.info(`Cleaned up ${numDuties} stuck duties`, {
211
253
  nodeId: this.config.nodeId,
212
- maxStuckDutiesAgeMs: this.config.maxStuckDutiesAgeMs,
254
+ maxStuckDutiesAgeMs: this.maxStuckDutiesAgeMs,
213
255
  });
214
256
  }
215
257
  }