Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

docs: Design document for Block Verification Process #384

Merged
merged 16 commits into from
Jan 9, 2025
Merged
Changes from 14 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
247 changes: 247 additions & 0 deletions server/docs/design/block-verification.md
jsync-swirlds marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,247 @@
# Block Verification Design

## Table of Contents

1. [Purpose](#purpose)
1. [Goals](#goals)
1. [Terms](#terms)
1. [Entities](#entities)
1. [Design](#design)
1. [Interfaces](#interfaces)
1. [Enums](#enums)
1. [Metrics](#metrics)
1. [Exceptions](#exceptions)

## Purpose

The purpose of the Block Verification feature is to ensure that blocks received
from consensus nodes are valid and have not been tampered with. This is achieved
by re-calculating the block hash and verifying it against the signature provided
by the consensus node.

## Goals

1. The block-node must re-create the block hash from the block items and verify
that it matches the hash implied by the signature.
1. If verification fails, the block should be considered invalid, and
appropriate error-handling procedures must be triggered.

## Terms

<dl>
<dt>Consensus Node (CN)</dt><dd>A node that produces and provides blocks.</dd>
<dt>Block Items</dt><dd>The block data pieces (header, events, transactions,
transaction result, state changes, proof) that make up a block.</dd>
<dt>Block Hash</dt><dd>A cryptographic hash representing the block’s integrity.</dd>
<dt>Signature</dt><dd>The cryptographic signature of the block hash created by
Network private key (aka: LedgerId).</dd>
<dt>Public Key</dt><dd>The public key of the LedgerId that signed the block.</dd>
jsync-swirlds marked this conversation as resolved.
Show resolved Hide resolved
</dl>

## Entities
ata-nas marked this conversation as resolved.
Show resolved Hide resolved

- #### VerificationHandler
ata-nas marked this conversation as resolved.
Show resolved Hide resolved
- Receives the stream of block items from the (partially) parsed and
unverified block items ring buffer.
jsync-swirlds marked this conversation as resolved.
Show resolved Hide resolved
- When it detects a block_header, it creates a BlockHashingSession using the
BlockHashingSessionFactory, providing it with the initial block items (and
internally, the session will handle asynchronous hashing).
- Adds subsequent block items to the session, including the block_proof.
- Does not block waiting for verification; the hash computation and
verification continue asynchronously.
- ### BlockHashingSessionFactory
- Creates new BlockHashingSession instances, provides them with a
ExecutorService.
- ### BlockHashingSession
- Holds all necessary block data while it is waiting to hash it.
- Accepts block items incrementally. (continues to compute them
asynchronously)
- Once the block_proof is provided, finalizes the hash computation
asynchronously.
- After computing the final hash, calls SignatureVerifier for verification.
- ### SignatureVerifier
- Verifies the block signature is valid (using the ledger ID) and signed the
same hash that was computed by the `BlockHashingSession`.
- Report results to BlockStatusManager.
- ### BlockStatusManager
- Receives verification results from SignatureVerifier.
- Updates block status and triggers any necessary recovery or follow-up
processes depending on the outcome.

## Design

1. The `VerificationHandler` receives the block items from the unverified ring
buffer.
1. When the block_header is detected, the `VerificationHandler` creates a
`BlockHashingSession` using the `BlockHashingSessionFactory`.
1. The `BlockHashingSession` accepts subsequent block items incrementally.
1. Once the block_proof is received, the `BlockHashingSession` calls
`completeHashing()` to finalize the hash computation.
1. Upon completion of computing the final block hash, the `BlockHashingSession`
calls the `SignatureVerifier` to verify the signature.
1. The `SignatureVerifier` compares the computed hash to the hash signed by the
Block Proof signature.
1. If the verification fails, the `SignatureVerifier` calls the
`BlockStatusManager` to update the block status as SIGNATURE_INVALID.
1. If the verification succeeds, the `SignatureVerifier` calls the
`BlockStatusManager` to update the block status as VERIFIED.
1. The `BlockStatusManager` initiates any necessary recovery or follow-up
processes depending on the verification result.

Sequence Diagram:

```mermaid
sequenceDiagram
participant U as UnverifiedRingBuffer
participant V as VerificationHandler
participant F as BlockHashingSessionFactory
participant S as BlockHashingSession
participant U as UnverifiedRingBuffer
participant V as VerificationHandler
participant F as BlockHashingSessionFactory
participant S as BlockHashingSession
participant SV as SignatureVerifier
participant BSM as BlockStatusManager

U->>V: (1) onBlockItemsReceived(blockItems)

alt (2) Detects block_header

V->>F: createSession(initialBlockItems, executorService, signatureVerifier)
Note over S: New instance of BlockHashingSession created
F-->>V: returns BlockHashingSession (S)
V-->>U: Returns without blocking
jsync-swirlds marked this conversation as resolved.
Show resolved Hide resolved
S->>S: *Starts hash computation asynchronously

else (3) Append more Block Items
loop
V->>S: addBlockItems(items)
V-->>U: return without blocking
S->>S: *Continues hash computation asynchronously
end

else (4) Append BlockItems with block_proof

V->>S: addBlockItems(items with block_proof)
V-->>U: return without blocking
S->>S: async completeHashing()
S->>SV: (5) verifySignature(signature, computedHash, blockNumber)

Note over SV,BSM: (6) Compare computed hash and signature
alt (7) Verification Fails
SV->>BSM: updateBlockStatus(blockNumber, ERROR, SIGNATURE_INVALID
else (8) Verification Succeeds
SV->>BSM: updateBlockStatus(blockNumber, VERIFIED, NONE)
end
Note over BSM: (9) Follow-up to downstream services
end

```

## Interfaces
ata-nas marked this conversation as resolved.
Show resolved Hide resolved

### VerificationHandler

```java
public interface VerificationHandler {
void onBlockItemsReceived(List<BlockItem> blockItems);
}
```

### BlockHashingSessionFactory

```java
public interface BlockHashingSessionFactory {
BlockHashingSession createSession(List<BlockItem> initialBlockItems,
ExecutorService executorService,
SignatureVerifier signatureVerifier);
}
```

### BlockHashingSession

```java
public interface BlockHashingSession {
void addBlockItem(BlockItem item);

CompletableFuture<Void> completeHashing();
}
```

### SignatureVerifier

```java
public interface SignatureVerifier {
void verifySignature(byte[] signature,
byte[] computedHash,
long blockNumber);
}
```

### BlockStatusManager

```java
public interface BlockStatusManager {
void updateBlockStatus(long blockNumber,
BlockVerificationStatus status,
VerificationError error);
}
```

## Enums

```java
public enum BlockVerificationStatus {
VERIFIED,
ERROR
}

public enum VerificationError {
NONE,
SIGNATURE_INVALID,
SYSTEM_ERROR
}
```

## Configuration
ata-nas marked this conversation as resolved.
Show resolved Hide resolved

As with other services already implemented in the Block Node system, the current
Block Verification Service should be able to be configurable via a Config Object
`VerificationConfig` and all the configurations needed shall be defined in
there.

Specifics on the properties names and possible values will be defined at the
Configuration document, but should include a way to enable/disable the module,
select Hashing algorithm and the amount of concurrent workers for the
`ServiceExecutor` pool.

## Metrics

<dl>
<dt>blocks_received</dt><dd>Counter of the number of blocks received for
verification.</dd>
<dt>blocks_verified</dt><dd>Counter of the number of blocks verified.</dd>
<dt>blocks_unverified</dt><dd>Counter of the number of blocks that have not been
verified, is the difference between blocks_received and blocks_verified.</dd>
<dt>blocks_signature_invalid</dt><dd>Counter of the number of blocks with
invalid signatures.</dd>
<dt>blocks_system_error</dt><dd>Counter of the number of blocks with system
errors.</dd>
<dt>block_verification_time</dt><dd>Histogram of the time taken to verify a
block, gives the node operator an idea of the time taken to verify a block.</dd>
</dl>

## Exceptions
jsync-swirlds marked this conversation as resolved.
Show resolved Hide resolved

- **SYSTEM_ERROR:** Issues with node configuration or bugs. The node logs
details, updates metrics, and might attempt recovery or halt.
- **SIGNATURE_INVALID:** If verification fails, SIGNATURE_INVALID is used. The
block is marked invalid, and the BlockStatusManager triggers error-handling
routines (requesting re-sends, removing corrupted data, notifying subscribers,
etc.).

### Signature invalid

If the computed hash does not match the hash signed by the network, the block is
considered unverified. It is marked as such and publishers are requested to
resend the block.
Loading