yaci

1. High Level Fetchers

There are two high level fetchers, BlockSync and BlockRangeSync. Using these fetchers, you can receive blocks and other related events through a listener, BlockChainDataListener.

BlockChainDataListener interface has following methods

void onByronBlock(ByronMainBlock byronBlock)
void onByronEbBlock(ByronEbBlock byronEbBlock)
void onBlock(Block block)
void onRollback(Point point)
void onTransactions(Era era, BlockHeader blockHeader, List<Transaction> transactions)
void onBatchDone()

BlockSync

Sync from tip or a specified point

Use BlockSync to connect to a remote/local Cardano node and receive blocks from tip or a specific point in real-time through a listener. You can receive various related events through BlockChainDataListener.

Sync blocks from tip

BlockSync blockSync = new BlockSync(ip, port, protocolMagic, Constants.WELL_KNOWN_PREPROD_POINT);
blockSync.startSyncFromTip(new BlockChainDataListener() {
    
            @Override
            public void onBlock(Block block) {
                System.out.println(block.getHeader().getHeaderBody().getBlockNumber());
            }

            @Override
            public void onTransactions(Era era, BlockHeader blockHeader, List<Transaction> transactions) {
                System.out.println("# of transactions >> " + transactions.size());
            }
});

Sync blocks from a specific point

BlockSync blockSync = new BlockSync(ip, port, protocolMagic, Constants.WELL_KNOWN_PREPROD_POINT);
blockSync.startSync(
                new Point(13107195, "ad2ceec67a07069d6e9295ed2144015860602c29f42505dc6ea2f55b9fc0dd93"),
                new BlockChainDataListener() {
                    
            @Override
            public void onBlock(Block block) {
                System.out.println(block.getHeader().getHeaderBody().getBlockNumber());
            }

            @Override
            public void onTransactions(Era era, BlockHeader blockHeader, List<Transaction> transactions) {
                System.out.println("# of transactions >> " + transactions.size());
            }
});

BlockRangeSync

Fetch blocks from point-1 to point-2

Use BlockRangeSync to connect to a remote/local Cardano node and fetch blocks from point-1 to point-2 and listen to other related events through BlockChainDataListener

BlockRangeSync blockRangeSync = new BlockRangeSync(ip, port, protocolMagic);
blockRangeSync.start(new BlockChainDataListener() {
    
     @Override
     public void onBlock(Block block) {
          System.out.println(block.getHeader().getHeaderBody().getBlockNumber());
      }

      @Override
      public void onTransactions(Era era, BlockHeader blockHeader, List<Transaction> transactions) {
          System.out.println("# of transactions >> " + transactions.size());
      }
});

Point from = new Point(13107194, "5b6194ab222088fdd0f3bad7e7343808ec10e52db107160e412bc204f58cf020");
Point to = new Point(13107220, "b26af1198cb891107fafead2388c8e8019f0157d101fda7896ba7358b58c7b83");

blockRangeSync.fetch(from, to);

2. Low level fetchers

This section explains other low level fetchers which can be used to get all agent level events. These low level fetchers are also used in high level fetchers mentioned in the previous section.

a. Follow the Chain

Follow a remote Cardano node using N2NChainSyncFetcher

Using N2NChainSyncFetcher, you can connect to a remote/local Cardano node through Cardano server port. Each fetcher api provides different callback apis to receive the result.

Consumer Function to receive Block

In the below example, we are passing a Consumer function to the start method of N2NChainSyncFetcher. The Consumer function is applied whenever there is a new block.

N2NChainSyncFetcher chainSyncFetcher = new N2NChainSyncFetcher(Constants.MAINNET_IOHK_RELAY_ADDR, Constants.MAINNET_IOHK_RELAY_PORT,
        Constants.WELL_KNOWN_MAINNET_POINT, Constants.MAINNET_PROTOCOL_MAGIC);

chainSyncFetcher.start(block -> {
        System.out.println("Received Block >> " + block.getHeader().getHeaderBody().getBlockNumber());
         System.out.println("Total # of Txns >> " + block.getTransactionBodies().size());
});

Listeners to recieve blocks and other events

You can also attach a BlockFetchListener to the fetcher to receive blocks and to listen to other events (e.g; noBlockFound) published by the fetcher.

To receive other low level events like “rollForward”, “rollBackward” (to handle specific scenarios during rollback), you need to add a ChainSyncAgentListener.

chainSyncFetcher.addBlockFetchListener(new BlockfetchAgentListener() {
    @Override
    public void blockFound(Block block) {
       System.out.println("Received Block >> " + block.getHeader().getHeaderBody().getBlockNumber());
       System.out.println("Total # of Txns >> " + block.getTransactionBodies().size());
    }
});

chainSyncFetcher.start();

Add ChainSyncAgentLister to listen to other low level events specific to ChainSyncAgent

chainSyncFetcher.addChainSyncListener(new ChainSyncAgentListener() {
     @Override
     public void rollforward(Tip tip, BlockHeader blockHeader) {
         System.out.println("RollForward >> " + tip);
     }
     
     @Override
     public void rollbackward(Tip tip, Point toPoint) {
          System.out.println("Rollbakcward >> " + toPoint);
     }
});

Note: All listeners should be added before start method call

Follow a local Cardano Node

If you want to use n2c protocol with a local Cardano node, you can use N2CChainSyncFetcher.

b. Fetch blocks from point-1 to point-2 using BlockFetcher

To fetch block bodies from point-1 to point-2, you need to use BlockFetcher.

Note: Currently full parsing is supported only for Shelly and post-Shelley era blocks. For Byron era blocks, only header is parsed.

Fetch and receive blocks in a consumer function

The following example fetches blocks from last block of Byron era to a babbage era block. The blocks are received in the Consumer function passed to the start() method.

Note: Consumer function only supports Shelley or post-Shelley era blocks

//Last Byron Block
Point from = new Point(4492799, "f8084c61b6a238acec985b59310b6ecec49c0ab8352249afd7268da5cff2a457");
//Some point in Babbage era
Point to = new Point(72719195, "a75cd55e48be3d341a7f8404094ee3cb9ce1f9fc923d23f675846d65546ef1dd");

BlockFetcher blockFetcher = new BlockFetcher("192.168.0.228", 6000, NetworkType.MAINNET.getProtocolMagic());
blockFetcher.start(block -> {
     System.out.println("Block >>> {} -- {} {}" + block.getHeader().getHeaderBody().getBlockNumber() + "  " + block.getHeader().getHeaderBody().getSlot() + "  " + block.getEra());
});

blockFetcher.fetch(from, to);

Fetch and receive blocks in Listener

You can also receive blocks using BlockfetchAgentListener

//Last Byron Block
Point from = new Point(4492799, "f8084c61b6a238acec985b59310b6ecec49c0ab8352249afd7268da5cff2a457");
//Some point in Babbage era
Point to = new Point(72719195, "a75cd55e48be3d341a7f8404094ee3cb9ce1f9fc923d23f675846d65546ef1dd");

BlockFetcher blockFetcher = new BlockFetcher(ip, port, NetworkType.MAINNET.getProtocolMagic());
blockFetcher.addBlockFetchListener(new BlockfetchAgentListener() {
     @Override
     public void byronBlockFound(ByronBlock block) {
          System.out.println("Byron block > " + block.getHeader().getConsensusData().getSlotId().getSlot());
     }

     @Override
     public void blockFound(Block block) {
          System.out.println("Block >>> {} -- {} {}" + block.getHeader().getHeaderBody().getBlockNumber() + "  " + block.getHeader().getHeaderBody().getSlot() + "  " + block.getEra());
     }
});

blockFetcher.start();
blockFetcher.fetch(from, to);

Shutdown

shutdown() can be called at the end to release the connection.

blockFetcher.shutdown();