mirror of
https://github.com/nspcc-dev/neo-go.git
synced 2025-01-03 09:22:49 +00:00
4b7b2ef701
Signed-off-by: Anna Shaleva <shaleva.ann@nspcc.ru>
104 lines
No EOL
5.3 KiB
Markdown
104 lines
No EOL
5.3 KiB
Markdown
# NeoFS block storage
|
|
|
|
Using NeoFS to store chain's blocks and snapshots was proposed in
|
|
[#3463](https://github.com/neo-project/neo/issues/3463). NeoGo contains several
|
|
extensions utilizing NeoFS block storage aimed to improve node synchronization
|
|
efficiency and reduce node storage size.
|
|
|
|
## Components and functionality
|
|
|
|
### Block storage schema
|
|
|
|
A single NeoFS container is used to store blocks and index files. Each container
|
|
has network magic attribute (`Magic:56753`). Each block is stored in a binary
|
|
form as a separate object with a unique OID and a set of attributes:
|
|
- block object identifier with block index value (`Block:1`)
|
|
- primary node index (`Primary:0`)
|
|
- block hash in the LE form (`Hash:5412a781caf278c0736556c0e544c7cfdbb6e3c62ae221ef53646be89364566b`)
|
|
- previous block hash in the LE form (`PrevHash:3654a054d82a8178c7dfacecc2c57282e23468a42ee407f14506368afe22d929`)
|
|
- millisecond-precision block creation timestamp (`BlockTime:1627894840919`)
|
|
- second-precision block uploading timestamp (`Timestamp:1627894840`)
|
|
|
|
Each index file is an object containing a constant-sized batch of raw block object
|
|
IDs in binary form ordered by block index. Each index file is marked with the
|
|
following attributes:
|
|
- index file identifier with consecutive file index value (`Index:0`)
|
|
- the number of OIDs included into index file (`IndexSize:128000`)
|
|
- second-precision index file uploading timestamp (`Timestamp:1627894840`)
|
|
|
|
### NeoFS BlockFetcher
|
|
|
|
NeoFS BlockFetcher service is designed as an alternative to P2P synchronisation
|
|
protocol. It allows to download blocks from a trusted container in the NeoFS network
|
|
and persist them to database using standard verification flow. NeoFS BlockFetcher
|
|
service primarily used during the node's bootstrap, providing a fast alternative to
|
|
P2P blocks synchronisation.
|
|
|
|
NeoFS BlockFetcher service has two modes of operation:
|
|
- Index File Search: Search for index files, which contain batches of block object
|
|
IDs and fetch blocks from NeoFS by retrieved OIDs.
|
|
- Direct Block Search: Search and fetch blocks directly from NeoFS container via
|
|
built-in NeoFS object search mechanism.
|
|
|
|
Operation mode of BlockFetcher can be configured via `SkipIndexFilesSearch`
|
|
parameter.
|
|
|
|
#### Operation flow
|
|
|
|
1. **OID Fetching**:
|
|
Depending on the mode, the service either:
|
|
- Searches for index files by index file attribute and reads block OIDs from index
|
|
file object-by-object.
|
|
- Searches blocks one by one directly by block attribute.
|
|
|
|
Once the OIDs are retrieved, they are immediately redirected to the
|
|
block downloading routines for further processing. The channel that
|
|
is used to redirect block OIDs to downloading routines is buffered
|
|
to provide smooth OIDs delivery without delays. The size of this channel
|
|
can be configured via `OIDBatchSize` parameter and equals to `2*OIDBatchSize`.
|
|
2. **Parallel Block Downloading**:
|
|
The number of downloading routines can be configured via
|
|
`DownloaderWorkersCount` parameter. It's up to the user to find the
|
|
balance between the downloading speed and blocks persist speed for every
|
|
node that uses NeoFS BlockFetcher. Downloaded blocks are placed to the
|
|
block queue directly.
|
|
3. **Block Insertion**:
|
|
Downloaded blocks are inserted into the blockchain using the same logic
|
|
as in the P2P synchronisation protocol. The block queue is used to order
|
|
downloaded blocks before they are inserted into the blockchain. The
|
|
size of the queue can be configured via the `BQueueSize` parameter
|
|
and should be larger than the `OIDBatchSize` parameter to avoid blocking
|
|
the downloading routines.
|
|
|
|
Once all blocks available in the NeoFS container are processed, the service
|
|
shuts down automatically.
|
|
|
|
### NeoFS Upload Command
|
|
The `upload-bin` command is designed to fetch blocks from the RPC node and upload
|
|
them to the NeoFS container. It also creates and uploads index files. Below is an
|
|
example usage of the command:
|
|
|
|
```shell
|
|
./bin/neo-go util upload-bin --cid 9iVfUg8aDHKjPC4LhQXEkVUM4HDkR7UCXYLs8NQwYfSG --wallet-config ./wallet-config.yml --block-attribute Block --index-attribute Index --rpc-endpoint https://rpc.t5.n3.nspcc.ru:20331 -fsr st1.t5.fs.neo.org:8080 -fsr st2.t5.fs.neo.org:8080 -fsr st3.t5.fs.neo.org:8080
|
|
```
|
|
|
|
Run `./bin/neo-go util upload-bin --help` to see the full list of supported options.
|
|
|
|
This command works as follows:
|
|
1. Fetches the current block height from the RPC node.
|
|
2. Searches for the index files stored in NeoFS.
|
|
3. Searches for the stored blocks from the latest incomplete index file.
|
|
4. Fetches missing blocks from the RPC node and uploads them to the NeoFS container.
|
|
5. After uploading the blocks, it creates index file based on the uploaded block OIDs.
|
|
6. Uploads the created index file to the NeoFS container.
|
|
7. Repeats steps 4-6 until the current block height is reached.
|
|
|
|
If the command is interrupted, it can be resumed. It starts the uploading process
|
|
from the last uploaded index file.
|
|
|
|
For a given block sequence, only one type of index file is supported. If new index
|
|
files are needed (different `index-file-size` or `index-attribute`), `upload-bin`
|
|
will upload the entire block sequence starting from genesis since no migration is
|
|
supported yet by this command. Please, add a comment to the
|
|
[#3744](https://github.com/nspcc-dev/neo-go/issues/3744) issue if you need this
|
|
functionality. |