@hyperledger/cactus-plugin-bungee-hermes v2.0.0
@hyperledger/cactus-plugin-bungee-hermes
The package provides Hyperledger Cacti a way to create blockchain snapshots and views for different distributed ledgers currently supported by Cacti. The implementation follows the paper BUNGEE (Blockchain UNifier view GEnErator).
The plugin will allow, as suggested in the paper, "new applications built on top of dependable blockchain interoperability, such as stakeholder-centric snapshots for audits, cross-chain analysis, blockchain migration, and combined on-chain-off-chain analytics".
Summary
Getting Started
Clone the git repository on your local machine. Follow these instructions that will get you a copy of the project up and running on your local machine for development and testing purposes.
Prerequisites
In the root of the project to install the dependencies execute the command:
npm run configureKnow how to use the following plugins of the project:
- cactus-plugin-ledger-connector-fabric
- cactus-plugin-ledger-connector-besu
- cactus-plugin-ledger-connector-ethereum
Architecture
The plugin interacts with a cactus ledger connector, using strategies with custom logic for each different network.
Note that, so far, only strategies for Fabric, Besu and Ethereum networks were implemented. Smart-contracts for Fabric and EVM based chains must implement the interface provided in the files ITraceableContract.ts and ITraceableContract.sol, in the test directory
The plugin stands behind a cacti-ledger-connector, which is used to fetch information from the ledger to create the snapshot.
.......................................
.            Cacti Node               .
.                                     .
.                                     .
. ..........            ............. .          .........
. .        .            .   Cacti   . .          .       .
. . BUNGEE . ---------- . Connector . -----------. _DLT  . 
. .        .            ............. .          .       .
. ..........                          .          .........  
.                                     .
.                                     .
.                                     .
.......................................The plugin can then serve multiple purposes, and thus serve also other plugins or apps in more complex deployment architectures (where we link bungee to other components).
BUNGEE - Hermes API
This plugin uses OpenAPI to generate the API paths. Endpoints exposed:
- CreateViewV1
- GetPublicKey
- GetAvailableStrategies
- VerifyMerkleRoot
- MergeViewsV1
- ProcessViewV1
Running the tests
- besu-test-basic.test.ts: A test using strategy-besu and a besu connector, testing creating views for different timeframes and states.
- ethereum-test-basic.test.ts: A test using strategy-ethereum and a ethereum connector, testing creating views for different timeframes and states.
- fabric-test-basic.test.ts: A test using strategy-fabric and a fabric connector, testing creating views for different timeframes and states.
- besu-test-pruning.test.ts: A test using strategy-besu and a besu connector, testing creating views for specific timeframes.
- fabric-test-pruning.test.ts: A test using strategy-fabric and a fabric connector, testing creating views for specific timeframes.
- bungee-api-test.test.ts: A more complex test, using multiple strategies, connectors, and calls to bungee-hermes API. Tests new functionalities like view proof validation.
- bungee-merge-views.test.ts: A test using besu-strategy and connector, to demonstrate and test the mergeView functionality of bungee-hermes.
- bungee-process-views.test.ts: A test using besu-strategy and connector, to demonstrate and test processView functionality of the plugin.
Tests developed using JEST testing framework.
Usage
Lets imagine we want bungee-hermes to create views for a besu ledger. Let us consider a besu ledger connector, exposing its API on URL: http://localhost:4000
Then we instantiate the plugin as follows:
const pluginBungeeHermesOptions = {
    keyPair: Secp256k1Keys.generateKeyPairsBuffer(),
    instanceId: uuidv4(),
    logLevel,
};
const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions);We add the desired strategies:
bungee.addStrategy('BESU_STRATEGY', new StrategyBesu("INFO"));Save network details for our target ledger:
const besuNetworkDetails: BesuNetworkDetails = {
    signingCredential: besuSigningCredential,
    contractName: besuContractName,
    connectorApiPath: 'http://localhost:4000',
    keychainId: besuKeychainPlugin.getKeychainId(),
    contractAddress: besuContractAddress,
    participant: accountAddress,
  };And we can request views, after exposing the api:
const expressApp = express();
  expressApp.use(bodyParser.json({ limit: "250mb" }));
  bungeeServer = http.createServer(expressApp);
  const listenOptions: IListenOptions = {
    hostname: "127.0.0.1",
    port: 3000,
    server: bungeeServer,
  };
const addressInfo = (await Servers.listen(listenOptions)) as AddressInfo;
const { address, port } = addressInfo;
await bungee.getOrCreateWebServices();
await bungee.registerWebServices(expressApp);
const bungeePath = `http://${address}:${port}`;
const config = new Configuration({ basePath: bungeePath });
const bungeeApi = new BungeeApi(config);
const viewBesu = await bungeeApi.createViewV1({
    strategyId: 'BESU_STRATEGY',
    networkDetails: besuNetworkDetails,
    tI: undefined,          //default to 0 (UNIX timestamp seconds)
    tF: undefined,          //default to Number.MAX_SAFE_INTEGER.toString() (UNIX timestamp seconds)
    stateIds: undefined,    //default to capture all assets
    viewID: undefined,      //plugin generates the id, if not given
  } as CreateViewRequest);Note that each strategy can be used to query different ledgers (ledgers of the same type, but on different locations), and BUNGEE also supports adding multiple strategies to each bungee-hermes-plugin instance. Each strategy implements the logic to query information from each different ledger (i.e. capture set of asset states), while bungee-hermes plugin handles the snapshot and view creation.
We can merge views to create an IntegratedView. Integrated views are the result of merging multiple views, with or without using a privacy policy. IntegratedViews have a map of ExtendedStates (stateId -> ExtendedState). And ExtendedState is the data gathered for a specific stateId, from different views map(viewId -> State).
Example:
const mergeViewsNoPolicyReq = await bungeeApi.mergeViewsV1({
    //for previously obtained views
    serializedViews: [JSON.stringify(view.view), JSON.stringify(view2.view)],
    //no policy. When policy is requested, take note that order of arguments matter
    mergePolicy: undefined,
    policyArguments: [],
  });
expect(mergeViewsNoPolicyReq.status).toBe(200);
expect(mergeViewsNoPolicyReq.data.integratedView).toBeTruthy();
expect(mergeViewsNoPolicyReq.data.signature).toBeTruthy();'View' object contains a 'viewProof'. 'viewProof' is composed by two merkle trees, one for 'stateProofs' and another for 'transactionProofs'. The analogous exists for IntegratedView objects, a merkle tree is created for all transactions and states in the IntegratedView. One can check if the content of a view or integratedView has no inconsistencies, by querying the VerifyMerkleRoot endpoint with the appropriate input:
  //using a previously created View object
  const stateProofs = view?.getSnapshot()
    .getStateBins()
    .map((x) => JSON.stringify(x.getStateProof()));
  const transactionProofs: string[] = [];
  view?.getAllTransactions()
    .forEach((t) => transactionProofs.push(JSON.stringify(t.getProof()))),
  );
  const verifyStateRoot = await bungeeApi.verifyMerkleRoot({
    input: stateProofs?.reverse(), //check integrity, order should not matter
    root: proof?.statesMerkleRoot,
  });
  expect(verifyStateRoot.data.result).toBeTrue();
  const verifyTransactionsRoot = await bungeeApi.verifyMerkleRoot({
    input: transactionProofs?.reverse(), //check integrity, order should not matter
    root: proof?.transactionsMerkleRoot,
  });
  expect(verifyTransactionsRoot.data.result).toBeTrue();Lastly, we can also process views according to a privacy-policy as follows:
import { hideState } from "../../../main/typescript/view-creation/privacy-policies";
bungee.addViewPrivacyPolicy("policy1", hideState);
const processed = await bungeeApi.processViewV1({
    serializedView: JSON.stringify({
      view: JSON.stringify(view.view as View),
      signature: view.signature,
    }),
    policyId: "hideState",
    policyArguments: [BESU_ASSET_ID],
  });This example uses the hideState policy, which takes as arguments a stateId to remove from the new view. When we apply a policy, the old view metadata is stored in the new View for traceability. This process can be applied multiple times
Contributing
We welcome contributions to Hyperledger Cactus in many forms, and there’s always plenty to do!
Please review CONTRIBUTING.md to get started.
License
This distribution is published under the Apache License Version 2.0 found in the LICENSE file.