This Beta MirrorNode implementation exposes Hedera Hashgraph transactions, transaction records, account balances, and events, generated by the Hedera mainnet (or testnet, if so configured) via a REST API.
Hedera mirror nodes receive the information from the mainnet nodes and they provide value-added services such as providing audit support, access to historical data, transaction analytics, visibility services, security threat modeling, data monetization services, etc. Mirror nodes can also run additional business logic to support applications built using Hedera mainnet.
While mirror nodes receive information from the mainnet nodes, they do not contribute to consensus on the mainnet, and their votes are not counted. Only the votes from the mainnet nodes are counted for determining consensus. The trust of Hedera mainnet is derived based on the the consensus reached by the mainnet nodes. That trust is transferred to the mirror nodes using cryptographic signatures on a chain of records (account balances, events, transactions, etc).
Eventually, the mirror nodes can run the same code as the Hedera mainnet nodes so that they can see the transactions in real time. To make the initial deployments easier, the beta mirror node strives to take away the burden of running a full Hedera node through creation of periodic files that contain processed information (such as account balances or transaction records), and have the full trust of the Hedera mainnet nodes. The beta mirror node software reduces the processing burden by receiving pre-constructed files from the mainnet, validating those, populating a database and providing REST APIs.
- Lower compute, bandwidth requirement
- It allows users to only save what they care about, and discard what they don’t (lower storage requirement)
- Easy searchable database so the users can add value quickly
- Easy to consume REST APIs to make integrations faster
The Beta mirror node works as follows:
-
When a transaction reaches consensus, Hedera nodes add the transaction and its associated record to a record file.
-
The file is closed on a regular cadence and a new file is created for the next batch of transactions and records. The interval is currently set to 5 seconds but may vary between networks.
-
Once the file is closed, nodes generate a signature file which contains the signature generated by the node for the record file.
-
Record files also contain the hash of the previous record file, thus creating an unbreakable validation chain.
-
The signature and record files are then uploaded from the nodes to Amazon S3 and Google File Storage.
-
This mirror node software downloads signature files from either S3 or Google File Storage.
-
The signature files are validated to ensure more than 2/3 of the nodes in the address book (stored in a
0.0.102
file) have the same signature. -
For each valid signature file, the corresponding record file is then downloaded from the cloud.
-
Record files can then be processed and transactions and records processed for long term storage.
-
Event files are handled in exactly the same manner.
-
In addition, nodes regularly generate a balance file which contains the list of Hedera accounts and their corresponding balance which is also uploaded to S3 and Google File Storage.
-
The files are also signed by the nodes.
-
This mirror node software can download the balance files, validate 2/3rd of nodes have signed and then process the balance files for long term storage.
- JDK 11
- Docker
- Docker-compose
git clone [email protected]:hashgraph/hedera-mirror-node.git
cd hedera-mirror-node
cp config/config.json.sample config/config.json
cp config/dotenv.sample config/.env
nano config/config.json # Insert AWS S3 credentials and bucket name
nano config/.env # Adjust usernames and passwords
./mvnw clean install
docker-compose up
Containers use persisted volumes as follows:
./MirrorNodePostgresData
on your local machine maps to/var/lib/postgresql/data
in the containers. This contains the files for the PostgreSQL database. Note: If you database container fails to initialise properly and the database fails to run, you will have to delete this folder prior to attempting a restart otherwise the database initialisation scripts will not be run../config
on your local machine maps to/usr/etc/mirror-node
in the containers. This contains the configuration files for loading and parsing files../MirrorNodeData
on your local machine maps to/var/lib/mirror-node
in the container. This contains files downloaded from S3 or GCP.
These are necessary not only for the database data to be persisted, but also so that the parsing containers can access file obtained via the downloading containers
Note: Shutting down the database container via docker-compose down
may result in a corrupted database that may not restart or may take longer than usual to restart.
In order to avoid this, shell into the container and issue the following command:
Use docker ps
to get the name of the database container, it should be something like mirror-node-postgres
.
Use the command docker exec -it docker_mirror-node-postgres_1 /bin/sh
to get a shell in the container.
su - postgres -c "PGDATA=$PGDATA /usr/local/bin/pg_ctl -w stop"
You may now power down the docker image itself.
This mirror node beta requires Java version 11.
If you are planning on using the docker compose images, you'll need Docker
installed.
Without Docker
, you will need to install PostgreSQL
versions 10 or 11.
Run ./mvnw package -DskipTests
from the MirrorNode
directory.
This will compile a runnable mirror node jar file in the target
directory.
Pay close attention to the contents of these configuration files, they directly affect how the mirror node operates.
-
config.json.sample
- rename this file toconfig.json
and edit so that the configuration parameters that are appropriate to your environment are setup. See section below on configuration file specifics. -
The file prefixed with '0.0.102' is the contents of a file hosted on Hedera with file ID
0.0.102
. This file contains the address book from the Hedera network which lists nodes and their public keys for signature verification purposes. Ensure the appropriate one for your network is identified in theconfig.json
file (addressBookFile entry) otherwise signature verification will fail.
The 0.0.102
file contains the address book, that is the list of nodes, their account number and public key(s). This file is different on every network so it is imperative to ensure you have the correct one for each network, else the signature verification process will fail. There are some sample address book files for various environments in the config
folder that can be used. Alternatively, one can be manually downloaded from the network:
Set the following environment variables or add them to a .env
file.
NODE_ADDRESS=127.0.0.1:50211
NODE_ID=0.0.x
OPERATOR_ID=0.0.x
OPERATOR_KEY=your account's private key
NODE_ADDRESS
is the IP address/url + port of the node you wish to request the file from.
NODE_ID
is the account number of the node (0.0.x).
OPERATOR_ID
is your own account number on the network (0.0.x).
OPERATOR_KEY
is your private key for the above account.
Run the following command to update the address book at the location specified in config.json
.
java -cp target/mirror-node-*.jar -Dloader.main=com.hedera.addressBook.NetworkAddressBook org.springframework.boot.loader.PropertiesLauncher
If no errors are output, the file specified by the addressBookFile
parameter of the config.json
file will now contain the network's address book.
Once setup, the file will be automatically updated as the mirror node software parses fileUpdate transactions that pertain to this file.
Note: Changes to this file while downloading or processing is taking place may be overwritten by the software. Make sure all processes are stopped before making changes.
Parameter name | Default value | Description |
---|---|---|
cloud-provider | "S3" |
Either S3 or GCP depending on where you want to download files from |
clientRegion | "us-east-2" |
The region which you want to download from |
bucketName | "hedera-export" |
The name of the bucket containing the files to download |
accessKey | "" |
Your S3 or GCP access key |
secretKey | "" |
Your S3 or GCP secret key |
downloadToDir | "/MirrorNodeData" |
The location where downloaded files will reside |
addressBookFile | "./config/0.0.102" |
The location of the address book file file |
accountBalancesS3Location | "accountBalances/balance" |
The location of the account balances files in the cloud bucket |
recordFilesS3Location | "recordstreams/record" |
The location of the record files in the cloud bucket |
persistClaims | false |
Determines whether claim data is persisted to the database or not |
persistFiles | "ALL" |
Determines whether file data is persisted to the database or not, can be set to ALL , NONE or SYSTEM . SYSTEM means only files with a file number lower than 1000 will be persisted |
persistContracts | true |
Determines whether contract data is persisted to the database or not |
persistCryptoTransferAmounts | true |
Determines whether crypto transfer amount data is persisted to the database or not |
The following environment variables may be used instead of values in the config.json
file for additional security.
Environment variables if set will take precedence over values in the config.json
file.
Environment variables may be set through the command line export varname=value
or via a .env
file located in the folder where the java classes are executed from
Note: this requires additional information to be stored in the config.json
, .env
or environment variables as follows:
json parameter name | corresponding environment variable |
---|---|
accessKey | HEDERA_S3_ACCESS_KEY |
secretKey | HEDERA_S3_SECRET_KEY |
Sample ./.env
file.
HEDERA_S3_ACCESS_KEY=accessKey
HEDERA_S3_SECRET_KEY=secretKey
You can skip this step if you're using Docker containers.
Ensure you have a postgreSQL server running (versions 10 and 11 have been tested) with the mirror node software.
Flyway (https://flywaydb.org/getstarted/) is used to manage the database schema.
All database scripts reside in src/main/resources/postgres
.
init.sql
should be used to initialise the database and owner.
- Edit the file with usernames, passwords, etc... you wish to use.
- Run the script as a DB admin user:
psql postgres -f src/main/resources/db/scripts/init.sql
Then flyway should be used to build the initial set of tables, and apply any changes. Those are determined by files names Vx.x__
.
Note: The Vx.x
scripts use variables which you should set prior to running the scripts.
Example: \set db_name='mydatabasename'
Make sure the config/config.json
or .env
file have values that match the above.
Check the output of the script carefully to ensure no errors occurred.
You can skip this section if you're running docker containers. To run the Java downloader and parser:
java -jar target/mirror-node-*.jar
The error below may appear on the console when running the .jar
file, this is normal and nothing to be concerned about.
WARNING: An illegal reflective access operation has occurred
WARNING: Illegal reflective access by com.google.protobuf.UnsafeUtil (file:/home/greg/mirrornode/lib/protobuf-java-3.5.1.jar) to field java.nio.Buffer.address
WARNING: Please consider reporting this to the maintainers of com.google.protobuf.UnsafeUtil
WARNING: Use --illegal-access=warn to enable warnings of further illegal reflective access operations
WARNING: All illegal access operations will be denied in a future release
Docker compose scripts are provided and run all the mirror node components:
- PostgreSQL database
- Java backend
- REST API
A REST API to the database is available under rest-api
.
To start it, cd rest-api
then npm install
.
Create a .env
file as per below and run with npm start
.
Unit tests: You can also unittest using jest by using npm test
.
Acceptance tests: To test against a live instance, you can use npm run acceptancetest
. You need to specify the instance ip:port as TARGET
environment variable for this.
example .env
file (note: there is a env.sample file in this directory):
DB_USER=mirror_api
DB_PASS=mirror_api_pass
DB_NAME=mirror_node
# This is the port the REST API will listen onto
PORT=5551
# server hosting the database
DB_HOST=localhost
# Shard number
SHARD_NUM=0
# Whether to include the host:port in the next links returned from the rest api
INCLUDE_PATH_IN_NEXT_LINKS=0
PORT
is the port number the REST API will listen onto.
The mirror node saves its current state to the database in a table called t_application_status
.
While the values in this table are updated in real time, any changes you wish to make here to promote to the application require that you first stop the application, make the changes in the database, then restart the application
Status name | Description |
---|---|
Event hash mismatch bypass until after | If a hash mismatch occurs before this event file name, processing into the database will stop. Leave blank to catch all mismatches, set to X to bypass all or set to the filename before which hash mismatches are ok and understood |
Record hash mismatch bypass until after | If a hash mismatch occurs before this record name, processing into the database will stop. Leave blank to catch all mismatches, set to X to bypass all or set to the filename before which hash mismatches are ok and understood |
Last processed record hash | The hash of the last record file processed into the database |
Last processed event hash | The hash of the last event file processed into the database |
Last valid downloaded record file name | The name of the last record file to have passed signature verification |
Last valid downloaded record file hash | The hash of the last record file to have passed signature verification |
Last valid downloaded event file name | The name of the last event file to have passed signature verification |
Last valid downloaded event file hash | The hash of the last event file to have passed signature verification |
Last valid downloaded balance file name | The name of the last balance file to have passed signature verification |
Refer to CONTRIBUTING.md
Apache License 2.0, see LICENSE.