9.1 KiB
Demo
-
Clone the stack-orchestrator repo.
-
Create a
config.sh
file.cd stack-orchestrator/helper-scripts ./create-config.sh
-
Setup the required repositories.
./setup-repositories.sh -p ssh
-
Checkout v4 release in go-ethereum repo. The path for go-ethereum is specified by
vulcanize_go_ethereum
variable inconfig.sh
file created in stack-orchestrator repo.# In go-ethereum repo. git checkout v1.10.19-statediff-4.0.2-alpha
-
To run the stack-orchestrator, the docker-compose version used is:
docker-compose version # docker-compose version 1.29.2, build 5becea4c
-
Run the stack-orchestrator
cd stack-orchestrator/helper-scripts
./wrapper.sh -f true \ -m true \ -s v4 \ -l latest \ -v remove \ -p ../config.sh
-
Run the IPFS (go-ipfs version 0.12.2) daemon:
ipfs daemon # API server listening on /ip4/127.0.0.1/tcp/5001
The IPFS API address can be seen in the output.
-
In the config file update the
server.ipfsApiAddr
config with the IPFS API address. -
Create a postgres12 database for the watcher:
sudo su - postgres # If database already exists # dropdb erc721-watcher createdb erc721-watcher
-
Create database for the job queue and enable the
pgcrypto
extension on them (https://github.com/timgit/pg-boss/blob/master/docs/usage.md#intro):# If database already exists # dropdb erc721-watcher-job-queue createdb erc721-watcher-job-queue
postgres@tesla:~$ psql -U postgres -h localhost erc721-watcher-job-queue Password for user postgres: psql (12.7 (Ubuntu 12.7-1.pgdg18.04+1)) SSL connection (protocol: TLSv1.3, cipher: TLS_AES_256_GCM_SHA384, bits: 256, compression: off) Type "help" for help. erc721-watcher-job-queue=# CREATE EXTENSION pgcrypto; CREATE EXTENSION erc721-watcher-job-queue=# exit
-
In the config file update the
database
connection settings. -
In
graph-watcher
repo, follow the instructions in Setup for installing and building packages.# After setup yarn && yarn build
-
Run the watcher:
yarn server
-
Run the job-runner:
yarn job-runner
-
Deploy an ERC721 token:
yarn nft:deploy # NFT deployed to: NFT_ADDRESS
Export the address of the deployed token to a shell variable for later use:
export NFT_ADDRESS="<NFT_ADDRESS>"
-
Run the following GQL mutation in generated watcher GraphQL endpoint http://127.0.0.1:3006/graphql
mutation { watchContract( address: "NFT_ADDRESS" kind: "ERC721" checkpoint: true ) }
-
Get the signer account address and export to a shell variable:
yarn account
export SIGNER_ADDRESS="<SIGNER_ADDRESS>"
-
Connect MetaMask to
http://localhost:8545
(with chain ID99
) -
Add a second account to Metamask and export the account address to a shell variable for later use:
export RECIPIENT_ADDRESS="<RECIPIENT_ADDRESS>"
-
To get the current block hash at any time, run:
yarn block:latest
-
Run the following GQL query (
eth_call
) in generated watcher GraphQL endpoint http://127.0.0.1:3006/graphqlquery { name( blockHash: "LATEST_BLOCK_HASH" contractAddress: "NFT_ADDRESS" ) { value proof { data } } symbol( blockHash: "LATEST_BLOCK_HASH" contractAddress: "NFT_ADDRESS" ) { value proof { data } } balanceOf( blockHash: "LATEST_BLOCK_HASH" contractAddress: "NFT_ADDRESS" owner: "SIGNER_ADDRESS" ) { value proof { data } } }
-
Run the following GQL query (
storage
) in generated watcher GraphQL endpoint http://127.0.0.1:3006/graphqlquery { _name( blockHash: "LATEST_BLOCK_HASH" contractAddress: "NFT_ADDRESS" ) { value proof { data } } _symbol( blockHash: "LATEST_BLOCK_HASH" contractAddress: "NFT_ADDRESS" ) { value proof { data } } _balances( blockHash: "LATEST_BLOCK_HASH" contractAddress: "NFT_ADDRESS" key0: "SIGNER_ADDRESS" ) { value proof { data } } }
-
Run the following GQL subscription in generated watcher GraphQL endpoint:
subscription { onEvent { event { __typename ... on TransferEvent { from to tokenId }, ... on ApprovalEvent { owner approved tokenId } }, block { number hash } } }
-
Mint token
yarn nft:mint --nft $NFT_ADDRESS --to $SIGNER_ADDRESS --token-id 1
-
A Transfer event to SIGNER_ADDRESS shall be visible in the subscription at endpoint.
-
An auto-generated
diff_staged
IPLDBlock should be added with parent CID pointing to the initial checkpoint IPLDBlock. -
Custom property
transferCount
should be 1 initially.
-
-
Run the
getState
query at the endpoint to get the latest IPLDBlock for NFT_ADDRESS:query { getState ( blockHash: "EVENT_BLOCK_HASH" contractAddress: "NFT_ADDRESS" # kind: "checkpoint" # kind: "diff" kind: "diff_staged" ) { cid block { cid hash number timestamp parentHash } contractAddress data } }
-
diff
IPLDBlocks get created corresponding to thediff_staged
blocks when their respective eth_blocks reach the pruned region. -
data
contains the default state and also the custom state propertytransferCount
that is indexed in hooks.ts file.
-
-
Get the latest blockHash and run the following query for
transferCount
entity:query { transferCount( block: { hash: "LATEST_BLOCK_HASH" } id: "NFT_ADDRESS" ) { id count } }
Note: Contract address is assigned to the Entity ID.
-
With the latest blockHash, run the following query for
balanceOf
andownerOf
(eth_call
):query { fromBalanceOf: balanceOf( blockHash: "LATEST_BLOCK_HASH" contractAddress: "NFT_ADDRESS" owner: "SIGNER_ADDRESS" ) { value proof { data } } toBalanceOf: balanceOf( blockHash: "LATEST_BLOCK_HASH" contractAddress: "NFT_ADDRESS" owner: "RECIPIENT_ADDRESS" ) { value proof { data } } ownerOf( blockHash: "LATEST_BLOCK_HASH" contractAddress: "NFT_ADDRESS" tokenId: 1 ) { value proof { data } } }
-
Transfer token
yarn nft:transfer --nft $NFT_ADDRESS --from $SIGNER_ADDRESS --to $RECIPIENT_ADDRESS --token-id 1
-
An Approval event for SIGNER_ADDRESS shall be visible in the subscription at endpoint.
-
A Transfer event to $RECIPIENT_ADDRESS shall be visible in the subscription at endpoint.
-
An auto-generated
diff_staged
IPLDBlock should be added with parent CID pointing to the previous IPLDBlock. -
Custom property
transferCount
should be incremented after transfer. This can be checked in thegetState
query and in IPFS webUI mentioned in the later steps.
-
-
Get the latest blockHash and replace the blockHash in the above
eth_call
query. The result should be different and the token should be transferred to the recipient. -
Run the
getState
query again at the endpoint with the event blockHash. -
Run the
transferCount
entity query again with the latest blockHash. The updated count should be returned. -
After the
diff
block has been created (can check if event block number pruned in yarn server log), create a checkpoint using CLI inpackages/erc721-watcher
:yarn checkpoint --address $NFT_ADDRESS
-
Run the
getState
query again with the output blockHash and kindcheckpoint
at the endpoint. -
The latest checkpoint should have the aggregate of state diffs since the last checkpoint.
-
The IPLDBlock entries can be seen in pg-admin in table ipld_block.
-
-
All the diff and checkpoint IPLDBlocks should pushed to IPFS.
-
Open IPFS WebUI http://127.0.0.1:5001/webui and search for IPLDBlocks using their CIDs.
-
The state should have auto indexed data and also custom property
transferCount
according to code in hooks filehandleEvent
method.
Reset / Clean up
-
To close down services in stack-orchestrator, hit
ctrl + c
in the terminal where it was run. -
To stop and remove stack-orchestrator services running in background run:
cd stack-orchestrator docker-compose -f ./docker/latest/docker-compose-db-sharding.yml down -v --remove-orphans