From 841ac13a966450647488ccf591e035f66af520d9 Mon Sep 17 00:00:00 2001 From: Carlos Amaro Date: Tue, 4 Jun 2024 12:41:28 +0100 Subject: [PATCH] feat(bungee-hermes): ability to use connectors without instanciating APIs Signed-off-by: Carlos Amaro --- .../cactus-plugin-bungee-hermes/package.json | 5 + .../strategy/obtain-ledger-strategy.ts | 4 +- .../main/typescript/strategy/strategy-besu.ts | 255 +++++++++---- .../typescript/strategy/strategy-ethereum.ts | 181 +++++---- .../typescript/strategy/strategy-fabric.ts | 202 +++++++--- .../integration/besu-test-basic.test.ts | 236 ++++++------ .../integration/besu-test-pruning.test.ts | 183 +++++---- .../integration/bungee-api-test.test.ts | 65 ++-- .../integration/bungee-merge-views.test.ts | 346 ++++++++++-------- .../integration/bungee-process-views.test.ts | 165 +++++---- .../integration/ethereum-test-basic.test.ts | 297 ++++++++------- .../integration/fabric-test-basic.test.ts | 243 ++++++------ .../integration/fabric-test-pruning.test.ts | 201 +++++----- 13 files changed, 1439 insertions(+), 944 deletions(-) diff --git a/packages/cactus-plugin-bungee-hermes/package.json b/packages/cactus-plugin-bungee-hermes/package.json index 22469ee4467..54de038e1fd 100644 --- a/packages/cactus-plugin-bungee-hermes/package.json +++ b/packages/cactus-plugin-bungee-hermes/package.json @@ -35,6 +35,11 @@ "name": "André Augusto", "email": "andre.augusto@tecnico.ulisboa.pt", "url": "https://github.com/AndreAugusto11" + }, + { + "name": "Carlos Amaro", + "email": "carlosrscamaro@tecnico.ulisboa.pt", + "url": "https://github.com/LordKubaya" } ], "main": "dist/lib/main/typescript/index.js", diff --git a/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/obtain-ledger-strategy.ts b/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/obtain-ledger-strategy.ts index eeb1b68aa12..d4f412116c0 100644 --- a/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/obtain-ledger-strategy.ts +++ b/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/obtain-ledger-strategy.ts @@ -1,8 +1,10 @@ import { Logger } from "@hyperledger/cactus-common"; import { State } from "../view-creation/state"; +import { IPluginLedgerConnector } from "@hyperledger/cactus-core-api/src/main/typescript/plugin/ledger-connector/i-plugin-ledger-connector"; export interface NetworkDetails { - connectorApiPath: string; + connectorApiPath?: string; + connector?: IPluginLedgerConnector; participant: string; } export interface ObtainLedgerStrategy { diff --git a/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-besu.ts b/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-besu.ts index cf26e702bdd..894e5ba39a7 100644 --- a/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-besu.ts +++ b/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-besu.ts @@ -10,10 +10,12 @@ import { EthContractInvocationType, EvmBlock, EvmLog, + EvmTransaction, GetBlockV1Request, GetPastLogsV1Request, GetTransactionV1Request, InvokeContractV1Request, + PluginLedgerConnectorBesu, Web3SigningCredential, } from "@hyperledger/cactus-plugin-ledger-connector-besu"; import { State } from "../view-creation/state"; @@ -29,6 +31,7 @@ export interface BesuNetworkDetails extends NetworkDetails { contractName: string; contractAddress: string; } + export class StrategyBesu implements ObtainLedgerStrategy { public static readonly CLASS_NAME = "StrategyBesu"; @@ -48,20 +51,32 @@ export class StrategyBesu implements ObtainLedgerStrategy { this.log.debug(`Generating ledger snapshot`); Checks.truthy(networkDetails, `${fnTag} networkDetails`); - const config = new Configuration({ - basePath: networkDetails.connectorApiPath, - }); - const besuApi = new BesuApi(config); + let besuApi: BesuApi | undefined; + let connector: PluginLedgerConnectorBesu | undefined; + + if (networkDetails.connector) { + connector = networkDetails.connector as PluginLedgerConnectorBesu; + } else if (networkDetails.connectorApiPath) { + const config = new Configuration({ + basePath: networkDetails.connectorApiPath, + }); + besuApi = new BesuApi(config); + } else { + throw new Error( + `${StrategyBesu.CLASS_NAME}#generateLedgerStates: networkDetails must have either connector or connectorApiPath`, + ); + } const ledgerStates = new Map(); const assetsKey = stateIds.length == 0 - ? await this.getAllAssetsKey(networkDetails, besuApi) + ? await this.getAllAssetsKey(networkDetails, connector, besuApi) : stateIds; this.log.debug("Current assets detected to capture: " + assetsKey); for (const assetKey of assetsKey) { const { transactions, values, blocks } = await this.getAllInfoByKey( assetKey, networkDetails, + connector, besuApi, ); @@ -92,7 +107,8 @@ export class StrategyBesu implements ObtainLedgerStrategy { async getAllAssetsKey( networkDetails: BesuNetworkDetails, - api: BesuApi, + connector: PluginLedgerConnectorBesu | undefined, + api: BesuApi | undefined, ): Promise { const parameters = { contractName: networkDetails.contractName, @@ -103,29 +119,19 @@ export class StrategyBesu implements ObtainLedgerStrategy { signingCredential: networkDetails.signingCredential, gas: 1000000, }; - const response = await api.invokeContractV1( + const response = await this.invokeContract( parameters as InvokeContractV1Request, + connector, + api, ); - - if (response.status >= 200 && response.status < 300) { - if (response.data.callOutput) { - return response.data.callOutput as string[]; - } else { - throw new Error( - `${StrategyBesu.CLASS_NAME}#getAllAssetsKey: contract ${networkDetails.contractName} method getAllAssetsIDs output is falsy`, - ); - } - } - throw new Error( - `${StrategyBesu.CLASS_NAME}#getAllAssetsKey: BesuAPI error with status ${response.status}: ` + - response.data, - ); + return response; } async getAllInfoByKey( key: string, networkDetails: BesuNetworkDetails, - api: BesuApi, + connector: PluginLedgerConnectorBesu | undefined, + api: BesuApi | undefined, ): Promise<{ transactions: Transaction[]; values: string[]; @@ -137,29 +143,144 @@ export class StrategyBesu implements ObtainLedgerStrategy { address: networkDetails.contractAddress, topics: [[null], [Web3.utils.keccak256(key)]], //filter logs by asset key }; - const response = await api.getPastLogsV1(req as GetPastLogsV1Request); - if (response.status < 200 || response.status >= 300) { - throw new Error( - `${StrategyBesu.CLASS_NAME}#getAllInfoByKey: BesuAPI getPastLogsV1 error with status ${response.status}: ` + - response.data, - ); - } - if (!response.data.logs) { - throw new Error( - `${StrategyBesu.CLASS_NAME}#getAllInfoByKey: BesuAPI getPastLogsV1 API call successfull but output data is falsy`, - ); - } - const decoded = response.data.logs as EvmLog[]; + const decoded = await this.getPastLogs(req, connector, api); const transactions: Transaction[] = []; const blocks: Map = new Map(); const values: string[] = []; this.log.debug("Getting transaction logs for asset: " + key); for (const log of decoded) { - const txTx = await api.getTransactionV1({ - transactionHash: log.transactionHash, - } as GetTransactionV1Request); + const txTx = await this.getTransaction( + { + transactionHash: log.transactionHash, + } as GetTransactionV1Request, + connector, + api, + ); + + const txBlock = await this.getBlock( + { + blockHashOrBlockNumber: log.blockHash, + } as GetBlockV1Request, + connector, + api, + ); + + this.log.debug( + "Transaction: " + + log.transactionHash + + "\nData: " + + JSON.stringify(log.data) + + "\n =========== \n", + ); + const proof = new Proof({ + creator: txTx.from as string, //no sig for besu + }); + const transaction: Transaction = new Transaction( + log.transactionHash, + txBlock.timestamp, + new TransactionProof(proof, log.transactionHash), + ); + transaction.setStateId(key); + transaction.setTarget(networkDetails.contractAddress as string); + transaction.setPayload(txTx.input ? txTx.input : ""); //FIXME: payload = transaction input ? + transactions.push(transaction); + values.push(JSON.stringify(log.data)); + + blocks.set(transaction.getId(), txBlock); + } + + return { transactions: transactions, values: values, blocks: blocks }; + } + + async invokeContract( + parameters: InvokeContractV1Request, + connector: PluginLedgerConnectorBesu | undefined, + api: BesuApi | undefined, + ): Promise { + if (connector) { + const response = await connector.invokeContract(parameters); + if (response.callOutput) { + return response.callOutput as string[]; + } else { + throw new Error( + `${StrategyBesu.CLASS_NAME}#invokeContract: contract ${parameters.contractName} method ${parameters.methodName} output is falsy`, + ); + } + } else if (api) { + const response = await api.invokeContractV1(parameters); + if (response.status >= 200 && response.status < 300) { + if (response.data.callOutput) { + return response.data.callOutput as string[]; + } else { + throw new Error( + `${StrategyBesu.CLASS_NAME}#invokeContract: contract ${parameters.contractName} method ${parameters.methodName} output is falsy`, + ); + } + } + throw new Error( + `${StrategyBesu.CLASS_NAME}#invokeContract: BesuAPI error with status ${response.status}: ` + + response.data, + ); + } + throw new Error( + `${StrategyBesu.CLASS_NAME}#invokeContract: BesuAPI or Connector were not defined`, + ); + } + + async getPastLogs( + req: GetPastLogsV1Request, + connector: PluginLedgerConnectorBesu | undefined, + api: BesuApi | undefined, + ): Promise { + if (connector) { + const response = await connector.getPastLogs(req); + if (response.logs) { + return response.logs; + } else { + throw new Error( + `${StrategyBesu.CLASS_NAME}#getPastLogs: BesuAPI getPastLogs output is falsy`, + ); + } + } + if (api) { + const response = await api.getPastLogsV1(req); + if (response.status < 200 || response.status >= 300) { + throw new Error( + `${StrategyBesu.CLASS_NAME}#getAllInfoByKey: BesuAPI getPastLogsV1 error with status ${response.status}: ` + + response.data, + ); + } + if (!response.data.logs) { + throw new Error( + `${StrategyBesu.CLASS_NAME}#getAllInfoByKey: BesuAPI getPastLogsV1 API call successfull but output data is falsy`, + ); + } + return response.data.logs as EvmLog[]; + } + throw new Error( + `${StrategyBesu.CLASS_NAME}#getTransaction: BesuAPI or Connector were not defined`, + ); + } + + async getTransaction( + req: GetTransactionV1Request, + connector: PluginLedgerConnectorBesu | undefined, + api: BesuApi | undefined, + ): Promise { + if (connector) { + const response = await connector.getTransaction(req); + if (response.transaction) { + return response.transaction; + } else { + throw new Error( + `${StrategyBesu.CLASS_NAME}#getTransaction: BesuAPI getTransaction output is falsy`, + ); + } + } + if (api) { + const txTx = await api.getTransactionV1(req); if (txTx.status < 200 || txTx.status >= 300) { throw new Error( @@ -172,11 +293,30 @@ export class StrategyBesu implements ObtainLedgerStrategy { `${StrategyBesu.CLASS_NAME}#getAllInfoByKey: BesuAPI getTransactionV1 call successfull but output data is falsy`, ); } + return txTx.data.transaction; + } + throw new Error( + `${StrategyBesu.CLASS_NAME}#getTransaction: BesuAPI or Connector were not defined`, + ); + } - const txBlock = await api.getBlockV1({ - blockHashOrBlockNumber: log.blockHash, - } as GetBlockV1Request); - + async getBlock( + req: GetBlockV1Request, + connector: PluginLedgerConnectorBesu | undefined, + api: BesuApi | undefined, + ): Promise { + if (connector) { + const response = await connector.getBlock(req); + if (response.block) { + return response.block; + } else { + throw new Error( + `${StrategyBesu.CLASS_NAME}#getBlock: BesuAPI getBlock output is falsy`, + ); + } + } + if (api) { + const txBlock = await api.getBlockV1(req); if (txBlock.status < 200 || txBlock.status >= 300) { throw new Error( `${StrategyBesu.CLASS_NAME}#getAllInfoByKey: BesuAPI getBlockV1 error with status ${txBlock.status}: ` + @@ -188,33 +328,10 @@ export class StrategyBesu implements ObtainLedgerStrategy { `${StrategyBesu.CLASS_NAME}#getAllInfoByKey: BesuAPI getBlockV1 call successfull but output data is falsy`, ); } - - this.log.debug( - "Transaction: " + - log.transactionHash + - "\nData: " + - JSON.stringify(log.data) + - "\n =========== \n", - ); - const proof = new Proof({ - creator: txTx.data.transaction.from as string, //no sig for besu - }); - const transaction: Transaction = new Transaction( - log.transactionHash, - txBlock.data.block.timestamp, - new TransactionProof(proof, log.transactionHash), - ); - transaction.setStateId(key); - transaction.setTarget(networkDetails.contractAddress as string); - transaction.setPayload( - txTx.data.transaction.input ? txTx.data.transaction.input : "", - ); //FIXME: payload = transaction input ? - transactions.push(transaction); - values.push(JSON.stringify(log.data)); - - blocks.set(transaction.getId(), txBlock.data.block); + return txBlock.data.block; } - - return { transactions: transactions, values: values, blocks: blocks }; + throw new Error( + `${StrategyBesu.CLASS_NAME}#getTransaction: BesuAPI or Connector were not defined`, + ); } } diff --git a/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-ethereum.ts b/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-ethereum.ts index c77824b7274..29683646bd1 100644 --- a/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-ethereum.ts +++ b/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-ethereum.ts @@ -10,6 +10,8 @@ import { EthContractInvocationType, InvokeContractV1Request, InvokeRawWeb3EthMethodV1Request, + PluginLedgerConnectorEthereum, + InvokeRawWeb3EthMethodV1Response, } from "@hyperledger/cactus-plugin-ledger-connector-ethereum"; import { NetworkDetails, ObtainLedgerStrategy } from "./obtain-ledger-strategy"; import { Configuration } from "@hyperledger/cactus-core-api"; @@ -77,21 +79,33 @@ export class StrategyEthereum implements ObtainLedgerStrategy { this.log.debug(`Generating ledger snapshot`); Checks.truthy(networkDetails, `${fnTag} networkDetails`); - const config = new Configuration({ - basePath: networkDetails.connectorApiPath, - }); - const ethereumApi = new EthereumApi(config); + let ethereumApi: EthereumApi | undefined; + let connector: PluginLedgerConnectorEthereum | undefined; + + if (networkDetails.connector) { + connector = networkDetails.connector as PluginLedgerConnectorEthereum; + } else if (networkDetails.connectorApiPath) { + const config = new Configuration({ + basePath: networkDetails.connectorApiPath, + }); + ethereumApi = new EthereumApi(config); + } else { + throw new Error( + `${StrategyEthereum.CLASS_NAME}#generateLedgerStates: networkDetails must have either connector or connectorApiPath`, + ); + } const ledgerStates = new Map(); const assetsKey = stateIds.length == 0 - ? await this.getAllAssetsKey(networkDetails, ethereumApi) + ? await this.getAllAssetsKey(networkDetails, connector, ethereumApi) : stateIds; this.log.debug("Current assets detected to capture: " + assetsKey); for (const assetKey of assetsKey) { const { transactions, values, blocks } = await this.getAllInfoByKey( assetKey, networkDetails, + connector, ethereumApi, ); @@ -122,7 +136,8 @@ export class StrategyEthereum implements ObtainLedgerStrategy { async getAllAssetsKey( networkDetails: EthereumNetworkDetails, - api: EthereumApi, + connector: PluginLedgerConnectorEthereum | undefined, + api: EthereumApi | undefined, ): Promise { const parameters = { contract: { @@ -134,28 +149,19 @@ export class StrategyEthereum implements ObtainLedgerStrategy { params: [], signingCredential: networkDetails.signingCredential, }; - const response = await api.invokeContractV1( + const response = await this.invokeContract( parameters as InvokeContractV1Request, + connector, + api, ); - if (response.status >= 200 && response.status < 300) { - if (response.data.callOutput) { - return response.data.callOutput as string[]; - } else { - throw new Error( - `${StrategyEthereum.CLASS_NAME}#getAllAssetsKey: contract ${networkDetails.contractName} method getAllAssetsIDs output is falsy`, - ); - } - } - throw new Error( - `${StrategyEthereum.CLASS_NAME}#getAllAssetsKey: EthereumAPI error with status ${response.status}: ` + - response.data, - ); + return response; } async getAllInfoByKey( key: string, networkDetails: EthereumNetworkDetails, - api: EthereumApi, + connector: PluginLedgerConnectorEthereum | undefined, + api: EthereumApi | undefined, ): Promise<{ transactions: Transaction[]; values: string[]; @@ -171,19 +177,8 @@ export class StrategyEthereum implements ObtainLedgerStrategy { methodName: "getPastLogs", params: [filter], }; - const response = await api.invokeWeb3EthMethodV1(getLogsReq); - if (response.status < 200 || response.status >= 300) { - throw new Error( - `${StrategyEthereum.CLASS_NAME}#getAllInfoByKey: BesuAPI getPastLogsV1 error with status ${response.status}: ` + - response.data, - ); - } - if (!response.data.data) { - throw new Error( - `${StrategyEthereum.CLASS_NAME}#getAllInfoByKey: BesuAPI getPastLogsV1 API call successfull but output data is falsy`, - ); - } - const decoded = response.data.data as EvmLog[]; + const response = await this.invokeWeb3EthMethod(getLogsReq, connector, api); + const decoded = response.data as EvmLog[]; const transactions: Transaction[] = []; const blocks: Map = new Map(); const values: string[] = []; @@ -194,37 +189,21 @@ export class StrategyEthereum implements ObtainLedgerStrategy { methodName: "getTransaction", params: [log.transactionHash], }; - const txTx = await api.invokeWeb3EthMethodV1(getTransactionReq); - - if (txTx.status < 200 || txTx.status >= 300) { - throw new Error( - `${StrategyEthereum.CLASS_NAME}#getAllInfoByKey: EthereumAPI invokeWeb3EthMethodV1 getTransaction error with status ${txTx.status}: ` + - txTx.data, - ); - } - if (!txTx.data.data) { - throw new Error( - `${StrategyEthereum.CLASS_NAME}#getAllInfoByKey: EthereumAPI invokeWeb3EthMethodV1 getTransaction successfull but output data is falsy`, - ); - } + const txTx = await this.invokeWeb3EthMethod( + getTransactionReq, + connector, + api, + ); const getBlockReq: InvokeRawWeb3EthMethodV1Request = { methodName: "getBlock", params: [log.blockHash], }; - const txBlock = await api.invokeWeb3EthMethodV1(getBlockReq); - - if (txBlock.status < 200 || txBlock.status >= 300) { - throw new Error( - `${StrategyEthereum.CLASS_NAME}#getAllInfoByKey: EthereumAPI invokeWeb3EthMethodV1 getBlock error with status ${txBlock.status}: ` + - txBlock.data, - ); - } - if (!txBlock.data.data) { - throw new Error( - `${StrategyEthereum.CLASS_NAME}#getAllInfoByKey: EthereumAPI invokeWeb3EthMethodV1 getBlock successfull but output data is falsy`, - ); - } + const txBlock = await this.invokeWeb3EthMethod( + getBlockReq, + connector, + api, + ); this.log.debug( "Transaction: " + @@ -234,22 +213,96 @@ export class StrategyEthereum implements ObtainLedgerStrategy { "\n =========== \n", ); const proof = new Proof({ - creator: txTx.data.data.from as string, //no sig for ethereum + creator: txTx.data.from as string, //no sig for ethereum }); const transaction: Transaction = new Transaction( log.transactionHash, - txBlock.data.data.timestamp, + txBlock.data.timestamp, new TransactionProof(proof, log.transactionHash), ); transaction.setStateId(key); transaction.setTarget(networkDetails.contractAddress as string); - transaction.setPayload(txTx.data.data.input ? txTx.data.data.input : ""); //FIXME: payload = transaction input ? + transaction.setPayload(txTx.data.input ? txTx.data.input : ""); //FIXME: payload = transaction input ? transactions.push(transaction); values.push(JSON.stringify(log.data)); - blocks.set(transaction.getId(), txBlock.data.data); + blocks.set(transaction.getId(), txBlock.data); } return { transactions: transactions, values: values, blocks: blocks }; } + + async invokeContract( + parameters: InvokeContractV1Request, + connector: PluginLedgerConnectorEthereum | undefined, + api: EthereumApi | undefined, + ): Promise { + if (connector) { + const response = await connector.invokeContract(parameters); + if (response.callOutput) { + return response.callOutput as string[]; + } else { + throw new Error( + `${StrategyEthereum.CLASS_NAME}#getAllAssetsKey: contract ${parameters.contract} method invokeContract output is falsy`, + ); + } + } else if (api) { + const response = await api.invokeContractV1( + parameters as InvokeContractV1Request, + ); + if (response.status >= 200 && response.status < 300) { + if (response.data.callOutput) { + return response.data.callOutput as string[]; + } else { + throw new Error( + `${StrategyEthereum.CLASS_NAME}#getAllAssetsKey: contract ${parameters.contract} method invokeContract output is falsy`, + ); + } + } + throw new Error( + `${StrategyEthereum.CLASS_NAME}#getAllAssetsKey: EthereumAPI error with status ${response.status}: ` + + response.data, + ); + } + throw new Error( + `${StrategyEthereum.CLASS_NAME}#invokeContract: EthereumAPI or Connector were not defined`, + ); + } + + async invokeWeb3EthMethod( + parameters: InvokeRawWeb3EthMethodV1Request, + connector: PluginLedgerConnectorEthereum | undefined, + api: EthereumApi | undefined, + ): Promise { + if (connector) { + const response = { + data: await connector.invokeRawWeb3EthMethod(parameters), + } as InvokeRawWeb3EthMethodV1Response; + if (response) { + return response; + } else { + throw new Error( + `${StrategyEthereum.CLASS_NAME}#invokeWeb3EthMethod: output is falsy`, + ); + } + } else if (api) { + const response = await api.invokeWeb3EthMethodV1(parameters); + if (response.status >= 200 && response.status < 300) { + if (response.data.data) { + return response.data; + } else { + throw new Error( + `${StrategyEthereum.CLASS_NAME}#invokeWeb3EthMethod: output is falsy`, + ); + } + } + throw new Error( + `${StrategyEthereum.CLASS_NAME}#invokeWeb3EthMethod: EthereumAPI error with status ${response.status}: ` + + response.data, + ); + } + throw new Error( + `${StrategyEthereum.CLASS_NAME}#invokeWeb3EthMethod: EthereumAPI was not defined`, + ); + } } diff --git a/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-fabric.ts b/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-fabric.ts index 106958db7ba..9e692918309 100644 --- a/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-fabric.ts +++ b/packages/cactus-plugin-bungee-hermes/src/main/typescript/strategy/strategy-fabric.ts @@ -4,6 +4,7 @@ import { Configuration, FabricContractInvocationType, RunTransactionRequest, + PluginLedgerConnectorFabric, } from "@hyperledger/cactus-plugin-ledger-connector-fabric"; import { NetworkDetails, ObtainLedgerStrategy } from "./obtain-ledger-strategy"; import { @@ -43,14 +44,27 @@ export class StrategyFabric implements ObtainLedgerStrategy { this.log.debug(`Generating ledger snapshot`); Checks.truthy(networkDetails, `${fnTag} networkDetails`); - const config = new Configuration({ - basePath: networkDetails.connectorApiPath, - }); - const fabricApi = new FabricApi(config); + let fabricApi: FabricApi | undefined; + let connector: PluginLedgerConnectorFabric | undefined; + + if (networkDetails.connector) { + connector = networkDetails.connector as PluginLedgerConnectorFabric; + } else if (networkDetails.connectorApiPath) { + const config = new Configuration({ + basePath: networkDetails.connectorApiPath, + }); + fabricApi = new FabricApi(config); + } else { + throw new Error( + `${StrategyFabric.CLASS_NAME}#generateLedgerStates: networkDetails must have either connector or connectorApiPath`, + ); + } const assetsKey = stateIds.length == 0 - ? (await this.getAllAssetsKey(fabricApi, networkDetails)).split(",") + ? ( + await this.getAllAssetsKey(networkDetails, connector, fabricApi) + ).split(",") : stateIds; const ledgerStates = new Map(); //For each key in ledgerAssetsKey @@ -58,15 +72,21 @@ export class StrategyFabric implements ObtainLedgerStrategy { const assetValues: string[] = []; const txWithTimeS: Transaction[] = []; - const txs = await this.getAllTxByKey(assetKey, fabricApi, networkDetails); + const txs = await this.getAllTxByKey( + networkDetails, + assetKey, + connector, + fabricApi, + ); //For each tx get receipt let last_receipt; for (const tx of txs) { const receipt = JSON.parse( await this.fabricGetTxReceiptByTxIDV1( + networkDetails, tx.getId(), + connector, fabricApi, - networkDetails, ), ); tx.getProof().setCreator( @@ -95,9 +115,10 @@ export class StrategyFabric implements ObtainLedgerStrategy { last_receipt = receipt; } const block = await this.fabricGetBlockByTxID( + networkDetails, txs[txs.length - 1].getId(), + connector, fabricApi, - networkDetails, ); const state = new State(assetKey, assetValues, txWithTimeS); ledgerStates.set(assetKey, state); @@ -122,38 +143,58 @@ export class StrategyFabric implements ObtainLedgerStrategy { } async fabricGetTxReceiptByTxIDV1( - transactionId: string, - api: FabricApi, networkDetails: FabricNetworkDetails, + transactionId: string, + connector: PluginLedgerConnectorFabric | undefined, + api: FabricApi | undefined, ): Promise { - const receiptLockRes = await api.getTransactionReceiptByTxIDV1({ + const parameters = { signingCredential: networkDetails.signingCredential, channelName: networkDetails.channelName, contractName: "qscc", invocationType: FabricContractInvocationType.Call, methodName: "GetBlockByTxID", params: [networkDetails.channelName, transactionId], - } as RunTransactionRequest); + } as RunTransactionRequest; - if (receiptLockRes.status >= 200 && receiptLockRes.status < 300) { - if (receiptLockRes.data) { - return JSON.stringify(receiptLockRes.data); + if (connector) { + const receiptLockRes = + await connector.getTransactionReceiptByTxID(parameters); + if (receiptLockRes) { + return JSON.stringify(receiptLockRes); } else { throw new Error( `${StrategyFabric.CLASS_NAME}#fabricGetTxReceiptByTxIDV1: contract qscc method GetBlockByTxID invocation output is falsy`, ); } + } else if (api) { + const receiptLockRes = + await api.getTransactionReceiptByTxIDV1(parameters); + if (receiptLockRes.status >= 200 && receiptLockRes.status < 300) { + if (receiptLockRes.data) { + return JSON.stringify(receiptLockRes.data); + } else { + throw new Error( + `${StrategyFabric.CLASS_NAME}#fabricGetTxReceiptByTxIDV1: contract qscc method GetBlockByTxID invocation output is falsy`, + ); + } + } + throw new Error( + `${StrategyFabric.CLASS_NAME}#fabricGetTxReceiptByTxIDV1: FabricAPI error with status 500: ` + + receiptLockRes.data, + ); + } else { + throw new Error( + `${StrategyFabric.CLASS_NAME}#fabricGetTxReceiptByTxIDV1: FabricAPI or Connector were not defined`, + ); } - throw new Error( - `${StrategyFabric.CLASS_NAME}#fabricGetTxReceiptByTxIDV1: FabricAPI error with status 500: ` + - receiptLockRes.data, - ); } async fabricGetBlockByTxID( - txId: string, - api: FabricApi, networkDetails: FabricNetworkDetails, + txId: string, + connector: PluginLedgerConnectorFabric | undefined, + api: FabricApi | undefined, ): Promise<{ hash: string; signers: string[] }> { const gatewayOptions = { identity: networkDetails.signingCredential.keychainRef, @@ -173,23 +214,39 @@ export class StrategyFabric implements ObtainLedgerStrategy { skipDecode: false, }; - const getBlockResponse = await api.getBlockV1(getBlockReq); + let data; - if (getBlockResponse.status < 200 || getBlockResponse.status >= 300) { - throw new Error( - `${StrategyFabric.CLASS_NAME}#fabricGetTxReceiptByTxIDV1: FabricAPI getBlockV1 error with status ${getBlockResponse.status}: ` + - getBlockResponse.data, - ); - } - if (!getBlockResponse.data) { + if (connector) { + const getBlockResponse = await connector.getBlock(getBlockReq); + if (getBlockResponse) { + data = getBlockResponse; + } else { + throw new Error( + `${StrategyFabric.CLASS_NAME}#fabricGetBlockByTxID: getBlockV1 API call output data is falsy`, + ); + } + } else if (api) { + const getBlockResponse = await api.getBlockV1(getBlockReq); + + if (getBlockResponse.status < 200 || getBlockResponse.status >= 300) { + throw new Error( + `${StrategyFabric.CLASS_NAME}#fabricGetTxReceiptByTxIDV1: FabricAPI getBlockV1 error with status ${getBlockResponse.status}: ` + + getBlockResponse.data, + ); + } + if (!getBlockResponse.data) { + throw new Error( + `${StrategyFabric.CLASS_NAME}#fabricGetBlockByTxID: getBlockV1 API call output data is falsy`, + ); + } + data = getBlockResponse.data; + } else { throw new Error( - `${StrategyFabric.CLASS_NAME}#fabricGetBlockByTxID: getBlockV1 API call output data is falsy`, + `${StrategyFabric.CLASS_NAME}#fabricGetBlockByTxID: FabricAPI or Connector were not defined`, ); } - const block = JSON.parse( - JSON.stringify(getBlockResponse?.data), - ).decodedBlock; + const block = JSON.parse(JSON.stringify(data)).decodedBlock; const blockSig = block.metadata.metadata[0].signatures; const sigs = []; @@ -212,60 +269,95 @@ export class StrategyFabric implements ObtainLedgerStrategy { } async getAllAssetsKey( - api: FabricApi, networkDetails: FabricNetworkDetails, + connector: PluginLedgerConnectorFabric | undefined, + api: FabricApi | undefined, ): Promise { - const response = await api.runTransactionV1({ + const parameters = { signingCredential: networkDetails.signingCredential, channelName: networkDetails.channelName, contractName: networkDetails.contractName, methodName: "GetAllAssetsKey", invocationType: FabricContractInvocationType.Call, params: [], - } as RunTransactionRequest); - - if (response.status >= 200 && response.status < 300) { - if (response.data.functionOutput) { - return response.data.functionOutput; + } as RunTransactionRequest; + if (connector) { + const response = await connector.transact(parameters); + if (response.functionOutput) { + return response.functionOutput; } else { throw new Error( `${StrategyFabric.CLASS_NAME}#getAllAssetsKey: contract ${networkDetails.contractName} method GetAllAssetsKey invocation output is falsy`, ); } + } else if (api) { + const response = await api.runTransactionV1(parameters); + + if (response.status >= 200 && response.status < 300) { + if (response.data.functionOutput) { + return response.data.functionOutput; + } else { + throw new Error( + `${StrategyFabric.CLASS_NAME}#getAllAssetsKey: contract ${networkDetails.contractName} method GetAllAssetsKey invocation output is falsy`, + ); + } + } + throw new Error( + `${StrategyFabric.CLASS_NAME}#getAllAssetsKey: FabricAPI error with status 500: ` + + response.data, + ); + } else { + throw new Error( + `${StrategyFabric.CLASS_NAME}#fabricGetBlockByTxID: FabricAPI or Connector were not defined`, + ); } - throw new Error( - `${StrategyFabric.CLASS_NAME}#getAllAssetsKey: FabricAPI error with status 500: ` + - response.data, - ); } async getAllTxByKey( - key: string, - api: FabricApi, networkDetails: FabricNetworkDetails, + key: string, + connector: PluginLedgerConnectorFabric | undefined, + api: FabricApi | undefined, ): Promise { - const response = await api.runTransactionV1({ + const parameters = { signingCredential: networkDetails.signingCredential, channelName: networkDetails.channelName, contractName: networkDetails.contractName, methodName: "GetAllTxByKey", invocationType: FabricContractInvocationType.Call, params: [key], - } as RunTransactionRequest); + } as RunTransactionRequest; - if (response.status >= 200 && response.status < 300) { - if (response.data.functionOutput) { - return this.txsStringToTxs(response.data.functionOutput); + if (connector) { + const response = await connector.transact(parameters); + if (response.functionOutput) { + return this.txsStringToTxs(response.functionOutput); } else { throw new Error( `${StrategyFabric.CLASS_NAME}#getAllTxByKey: contract ${networkDetails.contractName} method GetAllTxByKey invocation output is falsy`, ); } + } else if (api) { + const response = await api.runTransactionV1(parameters); + + if (response.status >= 200 && response.status < 300) { + if (response.data.functionOutput) { + return this.txsStringToTxs(response.data.functionOutput); + } else { + throw new Error( + `${StrategyFabric.CLASS_NAME}#getAllTxByKey: contract ${networkDetails.contractName} method GetAllTxByKey invocation output is falsy`, + ); + } + } + throw new Error( + `${StrategyFabric.CLASS_NAME}#getAllTxByKey: FabricAPI error with status 500: ` + + response.data, + ); + } else { + throw new Error( + `${StrategyFabric.CLASS_NAME}#getAllTxByKey: FabricAPI or Connector were not defined`, + ); } - throw new Error( - `${StrategyFabric.CLASS_NAME}#getAllTxByKey: FabricAPI error with status 500: ` + - response.data, - ); } // Receive transactions in string format and parses to Transaction [] diff --git a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/besu-test-basic.test.ts b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/besu-test-basic.test.ts index b9967b9c7eb..744df37daab 100644 --- a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/besu-test-basic.test.ts +++ b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/besu-test-basic.test.ts @@ -73,7 +73,9 @@ let bungeeContractAddress: string; let keychainPlugin: PluginKeychainMemory; -beforeAll(async () => { +let networkDetailsList: BesuNetworkDetails[]; + +beforeEach(async () => { pruneDockerAllIfGithubAction({ logLevel }) .then(() => { log.info("Pruning throw OK"); @@ -240,113 +242,141 @@ beforeAll(async () => { logLevel, }; } + networkDetailsList = [ + { + signingCredential: bungeeSigningCredential, + contractName, + connectorApiPath: besuPath, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: firstHighNetWorthAccount, + } as BesuNetworkDetails, + { + signingCredential: bungeeSigningCredential, + contractName, + connector: connector, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: firstHighNetWorthAccount, + } as BesuNetworkDetails, + ]; }); -test("test creation of views for different timeframes and states", async () => { - const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); - const strategy = "BESU"; - bungee.addStrategy(strategy, new StrategyBesu("INFO")); - const networkDetails: BesuNetworkDetails = { - signingCredential: bungeeSigningCredential, - contractName, - connectorApiPath: besuPath, - keychainId: bungeeKeychainId, - contractAddress: bungeeContractAddress, - participant: firstHighNetWorthAccount, - }; - - const snapshot = await bungee.generateSnapshot([], strategy, networkDetails); - const view = bungee.generateView( - snapshot, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - //expect to return a view - expect(view.view).toBeTruthy(); - expect(view.signature).toBeTruthy(); - - //expect the view to have capture the new asset BESU_ASSET_ID, and attributes to match - expect(snapshot.getStateBins().length).toEqual(1); - expect(snapshot.getStateBins()[0].getId()).toEqual(BESU_ASSET_ID); - expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); - - const view1 = bungee.generateView(snapshot, "0", "9999", undefined); - - //expects nothing to limit time of 9999 - expect(view1.view).toBeUndefined(); - expect(view1.signature).toBeUndefined(); - - //changing BESU_ASSET_ID value - const lockAsset = await connector.invokeContract({ - contractName, - keychainId: keychainPlugin.getKeychainId(), - invocationType: EthContractInvocationType.Send, - methodName: "lockAsset", - params: [BESU_ASSET_ID], - signingCredential: { - ethAccount: firstHighNetWorthAccount, - secret: besuKeyPair.privateKey, - type: Web3SigningCredentialType.PrivateKeyHex, - }, - gas: 1000000, - }); - expect(lockAsset).not.toBeUndefined(); - expect(lockAsset.success).toBeTrue(); - - //creating new asset - const new_asset_id = uuidv4(); - const depNew = await connector.invokeContract({ - contractName, - keychainId: keychainPlugin.getKeychainId(), - invocationType: EthContractInvocationType.Send, - methodName: "createAsset", - params: [new_asset_id, 10], - signingCredential: { - ethAccount: firstHighNetWorthAccount, - secret: besuKeyPair.privateKey, - type: Web3SigningCredentialType.PrivateKeyHex, - }, - gas: 1000000, - }); - expect(depNew).not.toBeUndefined(); - expect(depNew.success).toBeTrue(); - - const snapshot1 = await bungee.generateSnapshot([], strategy, networkDetails); - const view2 = bungee.generateView( - snapshot1, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - //expect to return a view - expect(view2.view).toBeTruthy(); - expect(view2.signature).toBeTruthy(); - - const stateBins = snapshot1.getStateBins(); - expect(stateBins.length).toEqual(2); //expect to have captured state for both assets - - const bins = [stateBins[0].getId(), stateBins[1].getId()]; - - //checks if values match: - // - new value of BESU_ASSET_ID state in new snapshot different than value from old snapshot) - // - successfully captured transaction that created the new asset - if (bins[0] === BESU_ASSET_ID) { - expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(2); - expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( - snapshot.getStateBins()[0].getValue(), +test.each([{ apiPath: true }, { apiPath: false }])( + //test for both BesuApiPath and BesuConnector + "test creation of views for different timeframes and states using", + async ({ apiPath }) => { + let networkDetails: BesuNetworkDetails; + if (apiPath) { + networkDetails = networkDetailsList[0]; + } else { + networkDetails = networkDetailsList[1]; + } + const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); + const strategy = "BESU"; + bungee.addStrategy(strategy, new StrategyBesu("INFO")); + + const snapshot = await bungee.generateSnapshot( + [], + strategy, + networkDetails, ); - expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(1); - } else { - expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); - expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(2); - expect(snapshot1.getStateBins()[1].getValue()).not.toEqual( - snapshot.getStateBins()[0].getValue(), + const view = bungee.generateView( + snapshot, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, ); - } -}); + //expect to return a view + expect(view.view).toBeTruthy(); + expect(view.signature).toBeTruthy(); + + //expect the view to have capture the new asset BESU_ASSET_ID, and attributes to match + expect(snapshot.getStateBins().length).toEqual(1); + expect(snapshot.getStateBins()[0].getId()).toEqual(BESU_ASSET_ID); + expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); + + const view1 = bungee.generateView(snapshot, "0", "9999", undefined); + + //expects nothing to limit time of 9999 + expect(view1.view).toBeUndefined(); + expect(view1.signature).toBeUndefined(); + + //changing BESU_ASSET_ID value + const lockAsset = await connector?.invokeContract({ + contractName, + keychainId: keychainPlugin.getKeychainId(), + invocationType: EthContractInvocationType.Send, + methodName: "lockAsset", + params: [BESU_ASSET_ID], + signingCredential: { + ethAccount: firstHighNetWorthAccount, + secret: besuKeyPair.privateKey, + type: Web3SigningCredentialType.PrivateKeyHex, + }, + gas: 1000000, + }); + expect(lockAsset).not.toBeUndefined(); + expect(lockAsset.success).toBeTrue(); + + //creating new asset + const new_asset_id = uuidv4(); + const depNew = await connector?.invokeContract({ + contractName, + keychainId: keychainPlugin.getKeychainId(), + invocationType: EthContractInvocationType.Send, + methodName: "createAsset", + params: [new_asset_id, 10], + signingCredential: { + ethAccount: firstHighNetWorthAccount, + secret: besuKeyPair.privateKey, + type: Web3SigningCredentialType.PrivateKeyHex, + }, + gas: 1000000, + }); + expect(depNew).not.toBeUndefined(); + expect(depNew.success).toBeTrue(); -afterAll(async () => { + const snapshot1 = await bungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + const view2 = bungee.generateView( + snapshot1, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + //expect to return a view + expect(view2.view).toBeTruthy(); + expect(view2.signature).toBeTruthy(); + + const stateBins = snapshot1.getStateBins(); + expect(stateBins.length).toEqual(2); //expect to have captured state for both assets + + const bins = [stateBins[0].getId(), stateBins[1].getId()]; + + //checks if values match: + // - new value of BESU_ASSET_ID state in new snapshot different than value from old snapshot) + // - successfully captured transaction that created the new asset + if (bins[0] === BESU_ASSET_ID) { + expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(2); + expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( + snapshot.getStateBins()[0].getValue(), + ); + expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(1); + } else { + expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); + expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(2); + expect(snapshot1.getStateBins()[1].getValue()).not.toEqual( + snapshot.getStateBins()[0].getValue(), + ); + } + }, +); + +afterEach(async () => { await Servers.shutdown(besuServer); await besuLedger.stop(); await besuLedger.destroy(); diff --git a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/besu-test-pruning.test.ts b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/besu-test-pruning.test.ts index 4bd8df5822a..000194d07b4 100644 --- a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/besu-test-pruning.test.ts +++ b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/besu-test-pruning.test.ts @@ -72,7 +72,9 @@ let bungeeContractAddress: string; let keychainPlugin: PluginKeychainMemory; -beforeAll(async () => { +let networkDetailsList: BesuNetworkDetails[]; + +beforeEach(async () => { pruneDockerAllIfGithubAction({ logLevel }) .then(() => { log.info("Pruning throw OK"); @@ -239,84 +241,113 @@ beforeAll(async () => { logLevel, }; } + networkDetailsList = [ + { + signingCredential: bungeeSigningCredential, + contractName, + connectorApiPath: besuPath, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: firstHighNetWorthAccount, + } as BesuNetworkDetails, + { + signingCredential: bungeeSigningCredential, + contractName, + connector: connector, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: firstHighNetWorthAccount, + } as BesuNetworkDetails, + ]; }); -test("test creation of views for specific timeframes", async () => { - const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); - const strategy = "BESU"; - bungee.addStrategy(strategy, new StrategyBesu("INFO")); - const netwokDetails: BesuNetworkDetails = { - signingCredential: bungeeSigningCredential, - contractName, - connectorApiPath: besuPath, - keychainId: bungeeKeychainId, - contractAddress: bungeeContractAddress, - participant: firstHighNetWorthAccount, - }; - - const snapshot = await bungee.generateSnapshot([], strategy, netwokDetails); - const view = bungee.generateView( - snapshot, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - - //expect to return a view - expect(view.view).toBeTruthy(); - expect(view.signature).toBeTruthy(); - - //expect the view to have capture the new asset BESU_ASSET_ID, and attributes to match - expect(snapshot.getStateBins().length).toEqual(1); - expect(snapshot.getStateBins()[0].getId()).toEqual(BESU_ASSET_ID); - expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); - - //changing BESU_ASSET_ID value - const lockAsset = await connector.invokeContract({ - contractName, - keychainId: keychainPlugin.getKeychainId(), - invocationType: EthContractInvocationType.Send, - methodName: "lockAsset", - params: [BESU_ASSET_ID], - signingCredential: { - ethAccount: firstHighNetWorthAccount, - secret: besuKeyPair.privateKey, - type: Web3SigningCredentialType.PrivateKeyHex, - }, - gas: 1000000, - }); - expect(lockAsset).not.toBeUndefined(); - expect(lockAsset.success).toBeTrue(); - - const snapshot1 = await bungee.generateSnapshot([], strategy, netwokDetails); - - //tI is the time of the first transaction + 1 - const tI = ( - parseInt(snapshot.getStateBins()[0].getTransactions()[0].getTimeStamp()) + 1 - ).toString(); - - const view1 = bungee.generateView( - snapshot1, - tI, - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - - //expect to return a view - expect(view1.view).toBeTruthy(); - expect(view1.signature).toBeTruthy(); - - expect(snapshot1.getStateBins().length).toEqual(1); - expect(snapshot1.getStateBins()[0].getId()).toEqual(BESU_ASSET_ID); - //expect the view to not include first transaction (made before tI) - expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); - //expect old and new snapshot state values to differ - expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( - snapshot.getStateBins()[0].getValue(), - ); -}); +test.each([{ apiPath: true }, { apiPath: false }])( + //test for both BesuApiPath and BesuConnector + "test creation of views for specific timeframes", + async ({ apiPath }) => { + let networkDetails: BesuNetworkDetails; + if (apiPath) { + networkDetails = networkDetailsList[0]; + } else { + networkDetails = networkDetailsList[1]; + } + const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); + const strategy = "BESU"; + bungee.addStrategy(strategy, new StrategyBesu("INFO")); + + const snapshot = await bungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + const view = bungee.generateView( + snapshot, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + + //expect to return a view + expect(view.view).toBeTruthy(); + expect(view.signature).toBeTruthy(); + + //expect the view to have capture the new asset BESU_ASSET_ID, and attributes to match + expect(snapshot.getStateBins().length).toEqual(1); + expect(snapshot.getStateBins()[0].getId()).toEqual(BESU_ASSET_ID); + expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); + + //changing BESU_ASSET_ID value + const lockAsset = await connector.invokeContract({ + contractName, + keychainId: keychainPlugin.getKeychainId(), + invocationType: EthContractInvocationType.Send, + methodName: "lockAsset", + params: [BESU_ASSET_ID], + signingCredential: { + ethAccount: firstHighNetWorthAccount, + secret: besuKeyPair.privateKey, + type: Web3SigningCredentialType.PrivateKeyHex, + }, + gas: 1000000, + }); + expect(lockAsset).not.toBeUndefined(); + expect(lockAsset.success).toBeTrue(); + + const snapshot1 = await bungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + + //tI is the time of the first transaction + 1 + const tI = ( + parseInt(snapshot.getStateBins()[0].getTransactions()[0].getTimeStamp()) + + 1 + ).toString(); + + const view1 = bungee.generateView( + snapshot1, + tI, + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + + //expect to return a view + expect(view1.view).toBeTruthy(); + expect(view1.signature).toBeTruthy(); + + expect(snapshot1.getStateBins().length).toEqual(1); + expect(snapshot1.getStateBins()[0].getId()).toEqual(BESU_ASSET_ID); + //expect the view to not include first transaction (made before tI) + expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); + //expect old and new snapshot state values to differ + expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( + snapshot.getStateBins()[0].getValue(), + ); + }, +); -afterAll(async () => { +afterEach(async () => { await Servers.shutdown(besuServer); await besuLedger.stop(); await besuLedger.destroy(); diff --git a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-api-test.test.ts b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-api-test.test.ts index 355da3bbe89..078f0c61030 100644 --- a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-api-test.test.ts +++ b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-api-test.test.ts @@ -56,14 +56,6 @@ import { } from "@hyperledger/cactus-plugin-ledger-connector-fabric"; import path from "path"; import { DiscoveryOptions } from "fabric-network"; -import { - FabricNetworkDetails, - StrategyFabric, -} from "../../../main/typescript/strategy/strategy-fabric"; -import { - BesuNetworkDetails, - StrategyBesu, -} from "../../../main/typescript/strategy/strategy-besu"; import { PluginLedgerConnectorEthereum, DefaultApi as EthereumApi, @@ -72,10 +64,33 @@ import { GethTestLedger, WHALE_ACCOUNT_ADDRESS, } from "@hyperledger/cactus-test-geth-ledger"; -import { - EthereumNetworkDetails, - StrategyEthereum, -} from "../../../main/typescript/strategy/strategy-ethereum"; +import { StrategyEthereum } from "../../../main/typescript/strategy/strategy-ethereum"; +import { StrategyFabric } from "../../../main/typescript/strategy/strategy-fabric"; +import { StrategyBesu } from "../../../main/typescript/strategy/strategy-besu"; + +interface BesuNetworkDetails { + connectorApiPath: string; + participant: string; + signingCredential: Web3SigningCredential; + keychainId: string; + contractName: string; + contractAddress: string; +} +interface FabricNetworkDetails { + connectorApiPath: string; + participant: string; + signingCredential: FabricSigningCredential; + contractName: string; + channelName: string; +} +interface EthereumNetworkDetails { + connectorApiPath: string; + participant: string; + signingCredential: Web3SigningCredential; + keychainId: string; + contractName: string; + contractAddress: string; +} const logLevel: LogLevelDesc = "INFO"; @@ -101,10 +116,10 @@ const log = LoggerProvider.getOrCreate({ label: "BUNGEE - Hermes", }); -let ethereumSigningCredential; +let ethereumPath: string; +let ethereumSigningCredential: Web3SigningCredential; let ethereumKeychainId: string; let ethereumContractAddress: string; -let ethereumNetworkDetails: EthereumNetworkDetails; let ethereumServer: Server; let ethereumLedger: GethTestLedger; @@ -175,6 +190,15 @@ test("tests bungee api using different strategies", async () => { participant: "Org1MSP", }; + const ethereumNetworkDetails: EthereumNetworkDetails = { + signingCredential: ethereumSigningCredential, + contractName: LockAssetContractJson.contractName, + connectorApiPath: ethereumPath, + keychainId: ethereumKeychainId, + contractAddress: ethereumContractAddress, + participant: WHALE_ACCOUNT_ADDRESS, + }; + const expressApp = express(); expressApp.use(bodyParser.json({ limit: "250mb" })); bungeeServer = http.createServer(expressApp); @@ -755,8 +779,8 @@ async function setupEthereumTestLedger(): Promise { }; const addressInfo = (await Servers.listen(listenOptions)) as AddressInfo; const { address, port } = addressInfo; - const apiHost = `http://${address}:${port}`; - const apiConfig = new Configuration({ basePath: apiHost }); + ethereumPath = `http://${address}:${port}`; + const apiConfig = new Configuration({ basePath: ethereumPath }); const apiClient = new EthereumApi(apiConfig); const rpcApiHttpHost = await ledger.getRpcApiHttpHost(); const web3 = new Web3(rpcApiHttpHost); @@ -872,15 +896,6 @@ async function setupEthereumTestLedger(): Promise { ethereumContractAddress = deployOut.data.transactionReceipt .contractAddress as string; - ethereumNetworkDetails = { - signingCredential: ethereumSigningCredential, - contractName: LockAssetContractJson.contractName, - connectorApiPath: apiHost, - keychainId: ethereumKeychainId, - contractAddress: ethereumContractAddress, - participant: WHALE_ACCOUNT_ADDRESS, - }; - return "Ethereum Network setup successful"; } diff --git a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-merge-views.test.ts b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-merge-views.test.ts index bc0519c7392..d1421a825d7 100644 --- a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-merge-views.test.ts +++ b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-merge-views.test.ts @@ -76,7 +76,9 @@ let bungeeServer: Server; let keychainPlugin: PluginKeychainMemory; -beforeAll(async () => { +let networkDetailsList: BesuNetworkDetails[]; + +beforeEach(async () => { pruneDockerAllIfGithubAction({ logLevel }) .then(() => { log.info("Pruning throw OK"); @@ -243,166 +245,196 @@ beforeAll(async () => { logLevel, }; } + networkDetailsList = [ + { + signingCredential: bungeeSigningCredential, + contractName, + connectorApiPath: besuPath, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: firstHighNetWorthAccount, + } as BesuNetworkDetails, + { + signingCredential: bungeeSigningCredential, + contractName, + connector: connector, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: firstHighNetWorthAccount, + } as BesuNetworkDetails, + ]; }); -test("test merging views, and integrated view proofs", async () => { - const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); - const strategy = "BESU"; - bungee.addStrategy(strategy, new StrategyBesu("INFO")); - const networkDetails: BesuNetworkDetails = { - signingCredential: bungeeSigningCredential, - contractName, - connectorApiPath: besuPath, - keychainId: bungeeKeychainId, - contractAddress: bungeeContractAddress, - participant: firstHighNetWorthAccount, - }; - - const snapshot = await bungee.generateSnapshot([], strategy, networkDetails); - const view = bungee.generateView( - snapshot, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - //expect to return a view - expect(view.view).toBeTruthy(); - expect(view.signature).toBeTruthy(); - - //changing BESU_ASSET_ID value - const lockAsset = await connector.invokeContract({ - contractName, - keychainId: keychainPlugin.getKeychainId(), - invocationType: EthContractInvocationType.Send, - methodName: "lockAsset", - params: [BESU_ASSET_ID], - signingCredential: { - ethAccount: firstHighNetWorthAccount, - secret: besuKeyPair.privateKey, - type: Web3SigningCredentialType.PrivateKeyHex, - }, - gas: 1000000, - }); - expect(lockAsset).not.toBeUndefined(); - expect(lockAsset.success).toBeTrue(); - - //creating new asset - const new_asset_id = uuidv4(); - const depNew = await connector.invokeContract({ - contractName, - keychainId: keychainPlugin.getKeychainId(), - invocationType: EthContractInvocationType.Send, - methodName: "createAsset", - params: [new_asset_id, 10], - signingCredential: { - ethAccount: firstHighNetWorthAccount, - secret: besuKeyPair.privateKey, - type: Web3SigningCredentialType.PrivateKeyHex, - }, - gas: 1000000, - }); - expect(depNew).not.toBeUndefined(); - expect(depNew.success).toBeTrue(); - - const snapshot1 = await bungee.generateSnapshot([], strategy, networkDetails); - const view2 = bungee.generateView( - snapshot1, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - //expect to return a view - expect(view2.view).toBeTruthy(); - expect(view2.signature).toBeTruthy(); - - 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 mergeViewsNoPolicyReq = await bungeeApi.mergeViewsV1({ - serializedViews: [ - JSON.stringify({ - view: JSON.stringify(view.view as View), - signature: view.signature, - }), - // eslint-disable-next-line prettier/prettier +test.each([{ apiPath: true }, { apiPath: false }])( + //test for both BesuApiPath and BesuConnector + "test merging views, and integrated view proofs", + async ({ apiPath }) => { + let networkDetails: BesuNetworkDetails; + if (apiPath) { + networkDetails = networkDetailsList[0]; + } else { + networkDetails = networkDetailsList[1]; + } + const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); + const strategy = "BESU"; + bungee.addStrategy(strategy, new StrategyBesu("INFO")); + + const snapshot = await bungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + const view = bungee.generateView( + snapshot, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + //expect to return a view + expect(view.view).toBeTruthy(); + expect(view.signature).toBeTruthy(); + + //changing BESU_ASSET_ID value + const lockAsset = await connector.invokeContract({ + contractName, + keychainId: keychainPlugin.getKeychainId(), + invocationType: EthContractInvocationType.Send, + methodName: "lockAsset", + params: [BESU_ASSET_ID], + signingCredential: { + ethAccount: firstHighNetWorthAccount, + secret: besuKeyPair.privateKey, + type: Web3SigningCredentialType.PrivateKeyHex, + }, + gas: 1000000, + }); + expect(lockAsset).not.toBeUndefined(); + expect(lockAsset.success).toBeTrue(); + + //creating new asset + const new_asset_id = uuidv4(); + const depNew = await connector.invokeContract({ + contractName, + keychainId: keychainPlugin.getKeychainId(), + invocationType: EthContractInvocationType.Send, + methodName: "createAsset", + params: [new_asset_id, 10], + signingCredential: { + ethAccount: firstHighNetWorthAccount, + secret: besuKeyPair.privateKey, + type: Web3SigningCredentialType.PrivateKeyHex, + }, + gas: 1000000, + }); + expect(depNew).not.toBeUndefined(); + expect(depNew.success).toBeTrue(); + + const snapshot1 = await bungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + const view2 = bungee.generateView( + snapshot1, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + //expect to return a view + expect(view2.view).toBeTruthy(); + expect(view2.signature).toBeTruthy(); + + 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 mergeViewsNoPolicyReq = await bungeeApi.mergeViewsV1({ + serializedViews: [ + JSON.stringify({ + view: JSON.stringify(view.view as View), + signature: view.signature, + }), + // eslint-disable-next-line prettier/prettier JSON.stringify({ view: JSON.stringify(view2.view as View), signature: view2.signature }), - ], - mergePolicy: MergePolicyOpts.NONE, - }); - expect(mergeViewsNoPolicyReq.status).toBe(200); - - expect(mergeViewsNoPolicyReq.data.integratedView).toBeTruthy(); - expect(mergeViewsNoPolicyReq.data.signature).toBeTruthy(); - - const mergeViewsNoPolicy = bungee.mergeViews( - [view.view as View, view2.view as View], - [view.signature as string, view2.signature as string], - MergePolicyOpts.NONE, - [], - ); - //1 transaction captured in first view, and 3 in the second - expect(mergeViewsNoPolicy.integratedView.getAllTransactions().length).toBe(4); - //1 state captured in first view, and 2 in the second - expect(mergeViewsNoPolicy.integratedView.getAllStates().length).toBe(3); - - const transactionReceipts: string[] = []; - - mergeViewsNoPolicy.integratedView.getAllTransactions().forEach((t) => { - transactionReceipts.push(JSON.stringify(t.getProof())); - }); - expect( - ( - await bungeeApi.verifyMerkleRoot({ - input: transactionReceipts, - root: mergeViewsNoPolicy.integratedView.getIntegratedViewProof() - .transactionsMerkleRoot, - }) - ).data.result, - ).toBeTrue(); - - const mergeViewsWithPolicy = bungee.mergeViews( - [view.view as View, view2.view as View], - [view.signature as string, view2.signature as string], - MergePolicyOpts.PruneState, - [BESU_ASSET_ID], //should remove all states related to this asset - ); - - //0 transactions captured in first view, and 1 in the second (because of policy) - // eslint-disable-next-line prettier/prettier - expect(mergeViewsWithPolicy.integratedView.getAllTransactions().length).toBe(1); - //0 state captured in first view, and 1 in the second (because of policy) - expect(mergeViewsWithPolicy.integratedView.getAllStates().length).toBe(1); - - const mergeViewsWithPolicy2 = bungee.mergeViews( - [view.view as View, view2.view as View], - [view.signature as string, view2.signature as string], - MergePolicyOpts.PruneStateFromView, - [BESU_ASSET_ID, view2.view?.getKey() as string], //should remove all states related to this asset - ); - - //1 transactions captured in first view, and 1 in the second (because of policy) - // eslint-disable-next-line prettier/prettier - expect(mergeViewsWithPolicy2.integratedView.getAllTransactions().length).toBe(2); - //1 state captured in first view, and only 1 in the second (because of policy) - expect(mergeViewsWithPolicy2.integratedView.getAllStates().length).toBe(2); -}); + ], + mergePolicy: MergePolicyOpts.NONE, + }); + expect(mergeViewsNoPolicyReq.status).toBe(200); + + expect(mergeViewsNoPolicyReq.data.integratedView).toBeTruthy(); + expect(mergeViewsNoPolicyReq.data.signature).toBeTruthy(); + + const mergeViewsNoPolicy = bungee.mergeViews( + [view.view as View, view2.view as View], + [view.signature as string, view2.signature as string], + MergePolicyOpts.NONE, + [], + ); + //1 transaction captured in first view, and 3 in the second + expect(mergeViewsNoPolicy.integratedView.getAllTransactions().length).toBe( + 4, + ); + //1 state captured in first view, and 2 in the second + expect(mergeViewsNoPolicy.integratedView.getAllStates().length).toBe(3); + + const transactionReceipts: string[] = []; + + mergeViewsNoPolicy.integratedView.getAllTransactions().forEach((t) => { + transactionReceipts.push(JSON.stringify(t.getProof())); + }); + expect( + ( + await bungeeApi.verifyMerkleRoot({ + input: transactionReceipts, + root: mergeViewsNoPolicy.integratedView.getIntegratedViewProof() + .transactionsMerkleRoot, + }) + ).data.result, + ).toBeTrue(); + + const mergeViewsWithPolicy = bungee.mergeViews( + [view.view as View, view2.view as View], + [view.signature as string, view2.signature as string], + MergePolicyOpts.PruneState, + [BESU_ASSET_ID], //should remove all states related to this asset + ); + + //0 transactions captured in first view, and 1 in the second (because of policy) + // eslint-disable-next-line prettier/prettier + expect(mergeViewsWithPolicy.integratedView.getAllTransactions().length).toBe(1); + //0 state captured in first view, and 1 in the second (because of policy) + expect(mergeViewsWithPolicy.integratedView.getAllStates().length).toBe(1); + + const mergeViewsWithPolicy2 = bungee.mergeViews( + [view.view as View, view2.view as View], + [view.signature as string, view2.signature as string], + MergePolicyOpts.PruneStateFromView, + [BESU_ASSET_ID, view2.view?.getKey() as string], //should remove all states related to this asset + ); + + //1 transactions captured in first view, and 1 in the second (because of policy) + // eslint-disable-next-line prettier/prettier + expect(mergeViewsWithPolicy2.integratedView.getAllTransactions().length).toBe(2); + //1 state captured in first view, and only 1 in the second (because of policy) + expect(mergeViewsWithPolicy2.integratedView.getAllStates().length).toBe(2); + }, +); -afterAll(async () => { +afterEach(async () => { await Servers.shutdown(besuServer); await Servers.shutdown(bungeeServer); await besuLedger.stop(); diff --git a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-process-views.test.ts b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-process-views.test.ts index 2daeba45180..b935dc28d58 100644 --- a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-process-views.test.ts +++ b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/bungee-process-views.test.ts @@ -78,7 +78,9 @@ let bungeeServer: Server; let keychainPlugin: PluginKeychainMemory; -beforeAll(async () => { +let networkDetailsList: BesuNetworkDetails[]; + +beforeEach(async () => { pruneDockerAllIfGithubAction({ logLevel }) .then(() => { log.info("Pruning throw OK"); @@ -280,77 +282,102 @@ beforeAll(async () => { logLevel, }; } + networkDetailsList = [ + { + signingCredential: bungeeSigningCredential, + contractName, + connectorApiPath: besuPath, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: firstHighNetWorthAccount, + } as BesuNetworkDetails, + { + signingCredential: bungeeSigningCredential, + contractName, + connector: connector, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: firstHighNetWorthAccount, + } as BesuNetworkDetails, + ]; }); -test("test merging views, and integrated view proofs", async () => { - const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); - const strategy = "BESU"; - bungee.addStrategy(strategy, new StrategyBesu("INFO")); - const networkDetails: BesuNetworkDetails = { - signingCredential: bungeeSigningCredential, - contractName, - connectorApiPath: besuPath, - keychainId: bungeeKeychainId, - contractAddress: bungeeContractAddress, - participant: firstHighNetWorthAccount, - }; - - const snapshot = await bungee.generateSnapshot([], strategy, networkDetails); - const view = bungee.generateView( - snapshot, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - //expect to return a view - expect(view.view).toBeTruthy(); - expect(view.signature).toBeTruthy(); - - 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 processed = await bungeeApi.processViewV1({ - serializedView: JSON.stringify({ - view: JSON.stringify(view.view as View), - signature: view.signature, - }), - policyId: PrivacyPolicyOpts.PruneState, - policyArguments: [BESU_ASSET_ID], - }); - - expect(processed.status).toBe(200); - expect(processed.data.view).toBeTruthy(); - expect(processed.data.signature).toBeTruthy(); - - const processedView = deserializeView(JSON.stringify(processed.data)); - - //check view deserializer - expect(JSON.stringify(processedView)).toEqual(processed.data.view); - - expect(processedView.getPolicy()).toBeTruthy(); - expect(processedView.getOldVersionsMetadata().length).toBe(1); - expect(processedView.getOldVersionsMetadata()[0].signature).toBe( - view.signature, - ); - expect(processedView.getAllTransactions().length).toBe(1); -}); +test.each([{ apiPath: true }, { apiPath: false }])( + //test for both BesuApiPath and BesuConnector + "test merging views, and integrated view proofs", + async ({ apiPath }) => { + let networkDetails: BesuNetworkDetails; + if (apiPath) { + networkDetails = networkDetailsList[0]; + } else { + networkDetails = networkDetailsList[1]; + } + + const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); + const strategy = "BESU"; + bungee.addStrategy(strategy, new StrategyBesu("INFO")); + + const snapshot = await bungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + const view = bungee.generateView( + snapshot, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + //expect to return a view + expect(view.view).toBeTruthy(); + expect(view.signature).toBeTruthy(); + + 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 processed = await bungeeApi.processViewV1({ + serializedView: JSON.stringify({ + view: JSON.stringify(view.view as View), + signature: view.signature, + }), + policyId: PrivacyPolicyOpts.PruneState, + policyArguments: [BESU_ASSET_ID], + }); + + expect(processed.status).toBe(200); + expect(processed.data.view).toBeTruthy(); + expect(processed.data.signature).toBeTruthy(); + + const processedView = deserializeView(JSON.stringify(processed.data)); + + //check view deserializer + expect(JSON.stringify(processedView)).toEqual(processed.data.view); + + expect(processedView.getPolicy()).toBeTruthy(); + expect(processedView.getOldVersionsMetadata().length).toBe(1); + expect(processedView.getOldVersionsMetadata()[0].signature).toBe( + view.signature, + ); + expect(processedView.getAllTransactions().length).toBe(1); + }, +); -afterAll(async () => { +afterEach(async () => { await Servers.shutdown(besuServer); await Servers.shutdown(bungeeServer); await besuLedger.stop(); diff --git a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/ethereum-test-basic.test.ts b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/ethereum-test-basic.test.ts index f4a6ec2987e..f95afbbe6d4 100644 --- a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/ethereum-test-basic.test.ts +++ b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/ethereum-test-basic.test.ts @@ -14,9 +14,9 @@ const testLogLevel: LogLevelDesc = "info"; import "jest-extended"; import express from "express"; import bodyParser from "body-parser"; -import http from "http"; import { v4 as uuidV4 } from "uuid"; import { AddressInfo } from "net"; +import http, { Server } from "http"; import { Server as SocketIoServer } from "socket.io"; import Web3 from "web3"; @@ -88,23 +88,16 @@ describe("Ethereum contract deploy and invoke using keychain", () => { let bungeeContractAddress: string; let pluginBungeeHermesOptions: IPluginBungeeHermesOptions; const ETH_ASSET_NAME = uuidV4(); - const expressApp = express(); - expressApp.use(bodyParser.json({ limit: "250mb" })); - const server = http.createServer(expressApp); - // set to address Type Error returned by Response.json() - // "Can't serialize BigInt" - expressApp.set("json replacer", stringifyBigIntReplacer); - - const wsApi = new SocketIoServer(server, { - path: Constants.SocketIoConnectionPathV1, - }); + let server: Server; ////////////////////////////////// // Setup ////////////////////////////////// - beforeAll(async () => { + let networkDetailsList: EthereumNetworkDetails[]; + + beforeEach(async () => { pruneDockerAllIfGithubAction({ logLevel: testLogLevel }) .then(() => { log.info("Pruning throw OK"); @@ -120,6 +113,15 @@ describe("Ethereum contract deploy and invoke using keychain", () => { }); await ledger.start(); + const expressApp = express(); + + expressApp.use(bodyParser.json({ limit: "250mb" })); + // set to address Type Error returned by Response.json() + // "Can't serialize BigInt" + expressApp.set("json replacer", stringifyBigIntReplacer); + + server = http.createServer(expressApp); + const listenOptions: IListenOptions = { hostname: "127.0.0.1", port: 5000, @@ -158,6 +160,10 @@ describe("Ethereum contract deploy and invoke using keychain", () => { // Instantiate connector with the keychain plugin that already has the // private key we want to use for one of our tests await connector.getOrCreateWebServices(); + const wsApi = new SocketIoServer(server, { + path: Constants.SocketIoConnectionPathV1, + }); + await connector.registerWebServices(expressApp, wsApi); const initTransferValue = web3.utils.toWei("5000", "ether"); @@ -232,130 +238,169 @@ describe("Ethereum contract deploy and invoke using keychain", () => { instanceId: uuidV4(), logLevel: testLogLevel, }; + networkDetailsList = [ + { + signingCredential: bungeeSigningCredential, + contractName: LockAssetContractJson.contractName, + connectorApiPath: apiHost, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: WHALE_ACCOUNT_ADDRESS, + } as EthereumNetworkDetails, + { + signingCredential: bungeeSigningCredential, + contractName: LockAssetContractJson.contractName, + connector: connector, + keychainId: bungeeKeychainId, + contractAddress: bungeeContractAddress, + participant: WHALE_ACCOUNT_ADDRESS, + } as EthereumNetworkDetails, + ]; }); - afterAll(async () => { + afterEach(async () => { + await Servers.shutdown(server); await ledger.stop(); await ledger.destroy(); - await Servers.shutdown(server); - const pruning = pruneDockerAllIfGithubAction({ logLevel: testLogLevel }); - await expect(pruning).resolves.toBeTruthy(); + await pruneDockerAllIfGithubAction({ logLevel: testLogLevel }) + .then(() => { + log.info("Pruning throw OK"); + }) + .catch(async () => { + await Containers.logDiagnostics({ logLevel: testLogLevel }); + fail("Pruning didn't throw OK"); + }); }); - test("test creation of views for different timeframes and states", async () => { - const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); - const strategy = "ETH"; - bungee.addStrategy(strategy, new StrategyEthereum("INFO")); - const networkDetails: EthereumNetworkDetails = { - signingCredential: bungeeSigningCredential, - contractName: LockAssetContractJson.contractName, - connectorApiPath: apiHost, - keychainId: bungeeKeychainId, - contractAddress: bungeeContractAddress, - participant: WHALE_ACCOUNT_ADDRESS, - }; - - const snapshot = await bungee.generateSnapshot( - [], - strategy, - networkDetails, - ); - const view = bungee.generateView( - snapshot, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - - //expect to return a view - expect(view.view).toBeTruthy(); - expect(view.signature).toBeTruthy(); - - //expect the view to have capture the new asset ETH_ASSET_NAME, and attributes to match - expect(snapshot.getStateBins().length).toEqual(1); - expect(snapshot.getStateBins()[0].getId()).toEqual(ETH_ASSET_NAME); - expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); - - const view1 = bungee.generateView(snapshot, "0", "9999", undefined); - - //expects nothing to limit time of 9999 - expect(view1.view).toBeUndefined(); - expect(view1.signature).toBeUndefined(); - - //changing ETH_ASSET_NAME value - const lockAsset = await apiClient.invokeContractV1({ - contract: { - contractName: LockAssetContractJson.contractName, - keychainId: keychainPlugin.getKeychainId(), - }, - invocationType: EthContractInvocationType.Send, - methodName: "lockAsset", - params: [ETH_ASSET_NAME], - web3SigningCredential: { - ethAccount: WHALE_ACCOUNT_ADDRESS, - secret: "", - type: Web3SigningCredentialType.GethKeychainPassword, - }, - }); - expect(lockAsset).not.toBeUndefined(); - expect(lockAsset.status).toBe(200); + test.each([{ apiPath: true }, { apiPath: false }])( + //test for both EthereumApiPath and EthereumConnector + "test creation of views for different timeframes and states", + async ({ apiPath }) => { + if (!apiPath) { + // set to address Type Error returned by Response.json() when using the connector by it self + // "Can't serialize BigInt" + const originalStringify = JSON.stringify; + const mock = jest.spyOn(JSON, "stringify"); + mock.mockImplementation((value: any) => { + try { + return originalStringify(value); + } catch (error) { + return value.toString(); + } + }); + } + + let networkDetails: EthereumNetworkDetails; + if (apiPath) { + networkDetails = networkDetailsList[0]; + } else { + networkDetails = networkDetailsList[1]; + } + const bungee = new PluginBungeeHermes(pluginBungeeHermesOptions); + const strategy = "ETH"; + bungee.addStrategy(strategy, new StrategyEthereum("INFO")); + const snapshot = await bungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + const view = bungee.generateView( + snapshot, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); - //changing ETH_ASSET_NAME value - const new_asset_id = uuidV4(); - const depNew = await apiClient.invokeContractV1({ - contract: { - contractName: LockAssetContractJson.contractName, - keychainId: keychainPlugin.getKeychainId(), - }, - invocationType: EthContractInvocationType.Send, - methodName: "createAsset", - params: [new_asset_id, 10], - web3SigningCredential: { - ethAccount: WHALE_ACCOUNT_ADDRESS, - secret: "", - type: Web3SigningCredentialType.GethKeychainPassword, - }, - }); - expect(depNew).not.toBeUndefined(); - expect(depNew.status).toBe(200); + //expect to return a view + expect(view.view).toBeTruthy(); + expect(view.signature).toBeTruthy(); + + //expect the view to have capture the new asset ETH_ASSET_NAME, and attributes to match + expect(snapshot.getStateBins().length).toEqual(1); + expect(snapshot.getStateBins()[0].getId()).toEqual(ETH_ASSET_NAME); + expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); + + const view1 = bungee.generateView(snapshot, "0", "9999", undefined); + + //expects nothing to limit time of 9999 + expect(view1.view).toBeUndefined(); + expect(view1.signature).toBeUndefined(); + + //changing ETH_ASSET_NAME value + const lockAsset = await apiClient.invokeContractV1({ + contract: { + contractName: LockAssetContractJson.contractName, + keychainId: keychainPlugin.getKeychainId(), + }, + invocationType: EthContractInvocationType.Send, + methodName: "lockAsset", + params: [ETH_ASSET_NAME], + web3SigningCredential: { + ethAccount: WHALE_ACCOUNT_ADDRESS, + secret: "", + type: Web3SigningCredentialType.GethKeychainPassword, + }, + }); + expect(lockAsset).not.toBeUndefined(); + expect(lockAsset.status).toBe(200); + + //changing ETH_ASSET_NAME value + const new_asset_id = uuidV4(); + const depNew = await apiClient.invokeContractV1({ + contract: { + contractName: LockAssetContractJson.contractName, + keychainId: keychainPlugin.getKeychainId(), + }, + invocationType: EthContractInvocationType.Send, + methodName: "createAsset", + params: [new_asset_id, 10], + web3SigningCredential: { + ethAccount: WHALE_ACCOUNT_ADDRESS, + secret: "", + type: Web3SigningCredentialType.GethKeychainPassword, + }, + }); + expect(depNew).not.toBeUndefined(); + expect(depNew.status).toBe(200); - const snapshot1 = await bungee.generateSnapshot( - [], - strategy, - networkDetails, - ); - const view2 = bungee.generateView( - snapshot1, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - //expect to return a view - expect(view2.view).toBeTruthy(); - expect(view2.signature).toBeTruthy(); - - const stateBins = snapshot1.getStateBins(); - expect(stateBins.length).toEqual(2); //expect to have captured state for both assets - - const bins = [stateBins[0].getId(), stateBins[1].getId()]; - - //checks if values match: - // - new value of ETH_ASSET_NAME state in new snapshot different than value from old snapshot) - // - successfully captured transaction that created the new asset - if (bins[0] === ETH_ASSET_NAME) { - expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(2); - expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( - snapshot.getStateBins()[0].getValue(), + const snapshot1 = await bungee.generateSnapshot( + [], + strategy, + networkDetails, ); - expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(1); - } else { - expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); - expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(2); - expect(snapshot1.getStateBins()[1].getValue()).not.toEqual( - snapshot.getStateBins()[0].getValue(), + const view2 = bungee.generateView( + snapshot1, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, ); - } - }); + //expect to return a view + expect(view2.view).toBeTruthy(); + expect(view2.signature).toBeTruthy(); + + const stateBins = snapshot1.getStateBins(); + expect(stateBins.length).toEqual(2); //expect to have captured state for both assets + + const bins = [stateBins[0].getId(), stateBins[1].getId()]; + + //checks if values match: + // - new value of ETH_ASSET_NAME state in new snapshot different than value from old snapshot) + // - successfully captured transaction that created the new asset + if (bins[0] === ETH_ASSET_NAME) { + expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(2); + expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( + snapshot.getStateBins()[0].getValue(), + ); + expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(1); + } else { + expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); + expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(2); + expect(snapshot1.getStateBins()[1].getValue()).not.toEqual( + snapshot.getStateBins()[0].getValue(), + ); + } + }, + ); }); function stringifyBigIntReplacer(key: string, value: bigint): string { diff --git a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/fabric-test-basic.test.ts b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/fabric-test-basic.test.ts index 08c49cbf6a8..eb446222d88 100644 --- a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/fabric-test-basic.test.ts +++ b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/fabric-test-basic.test.ts @@ -68,14 +68,14 @@ let pluginBungeeFabricOptions: IPluginBungeeHermesOptions; let pluginBungee: PluginBungeeHermes; const FABRIC_ASSET_ID = uuidv4(); -let networkDetails: FabricNetworkDetails; +let networkDetailsList: FabricNetworkDetails[]; const log = LoggerProvider.getOrCreate({ level: logLevel, label: "BUNGEE - Hermes", }); -beforeAll(async () => { +beforeEach(async () => { pruneDockerAllIfGithubAction({ logLevel }) .then(() => { log.info("Pruning throw OK"); @@ -333,120 +333,143 @@ beforeAll(async () => { instanceId: uuidv4(), }; - networkDetails = { - connectorApiPath: fabricPath, - signingCredential: fabricSigningCredential, - channelName: fabricChannelName, - contractName: fabricContractName, - participant: "Org1MSP", - }; + networkDetailsList = [ + { + connectorApiPath: fabricPath, + signingCredential: fabricSigningCredential, + channelName: fabricChannelName, + contractName: fabricContractName, + participant: "Org1MSP", + }, + { + connector: fabricConnector, + signingCredential: fabricSigningCredential, + channelName: fabricChannelName, + contractName: fabricContractName, + participant: "Org1MSP", + }, + ]; pluginBungee = new PluginBungeeHermes(pluginBungeeFabricOptions); } }); -test("test creation of views for different timeframes and states", async () => { - const strategy = "FABRIC"; - pluginBungee.addStrategy(strategy, new StrategyFabric("INFO")); - - const snapshot = await pluginBungee.generateSnapshot( - [], - strategy, - networkDetails, - ); - const view = pluginBungee.generateView( - snapshot, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - - //expect to return a view - expect(view.view).toBeTruthy(); - expect(view.signature).toBeTruthy(); - - //expect the view to have capture the new asset Fabric_ASSET_ID, and attributes to match - expect(snapshot.getStateBins().length).toEqual(1); - expect(snapshot.getStateBins()[0].getId()).toEqual(FABRIC_ASSET_ID); - expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); - - //fabric transaction proofs include endorsements - expect( - snapshot.getStateBins()[0].getTransactions()[0].getProof().getEndorsements() - ?.length, - ).toEqual(2); - - //no valid states for this time frame - const view1 = pluginBungee.generateView(snapshot, "0", "9999", undefined); - expect(view1.view).toBeUndefined(); - expect(view1.signature).toBeUndefined(); - - //creating new asset - const new_asset_id = uuidv4(); - const createResponse = await apiClient.runTransactionV1({ - contractName: fabricContractName, - channelName: fabricChannelName, - params: [new_asset_id, "10"], - methodName: "CreateAsset", - invocationType: FabricContractInvocationType.Send, - signingCredential: fabricSigningCredential, - }); - expect(createResponse).not.toBeUndefined(); - expect(createResponse.status).toBeGreaterThan(199); - expect(createResponse.status).toBeLessThan(300); - - //changing FABRIC_ASSET_ID value - const modifyResponse = await apiClient.runTransactionV1({ - contractName: fabricContractName, - channelName: fabricChannelName, - params: [FABRIC_ASSET_ID, "18"], - methodName: "UpdateAsset", - invocationType: FabricContractInvocationType.Send, - signingCredential: fabricSigningCredential, - }); - expect(modifyResponse).not.toBeUndefined(); - expect(modifyResponse.status).toBeGreaterThan(199); - expect(modifyResponse.status).toBeLessThan(300); - - const snapshot1 = await pluginBungee.generateSnapshot( - [], - strategy, - networkDetails, - ); - const view2 = pluginBungee.generateView( - snapshot1, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - - //expect to return a view - expect(view2.view).toBeTruthy(); - expect(view2.signature).toBeTruthy(); - - //expect to have captured state for both assets - const stateBins = snapshot1.getStateBins(); - expect(stateBins.length).toEqual(2); - const bins = [stateBins[0].getId(), stateBins[1].getId()]; - - expect(bins.includes(FABRIC_ASSET_ID)).toBeTrue(); - expect(bins.includes(new_asset_id)).toBeTrue(); - - //checks if values match: - // - new value of FABRIC_ASSET_ID state in new snapshot equals to new value) - // - successfully captured transaction that created the new asset - if (bins[0] === FABRIC_ASSET_ID) { - expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(2); - expect(snapshot1.getStateBins()[0].getValue()).toEqual("18"); - expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(1); - } else { - expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); - expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(2); - expect(snapshot1.getStateBins()[1].getValue()).toEqual("18"); - } -}); +test.each([{ apiPath: true }, { apiPath: false }])( + //test for both FabricApiPath and FabricConnector + "test creation of views for different timeframes and states", + async ({ apiPath }) => { + let networkDetails: FabricNetworkDetails; + if (apiPath) { + networkDetails = networkDetailsList[0]; + } else { + networkDetails = networkDetailsList[1]; + } + + const strategy = "FABRIC"; + pluginBungee.addStrategy(strategy, new StrategyFabric("INFO")); + + const snapshot = await pluginBungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + const view = pluginBungee.generateView( + snapshot, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + + //expect to return a view + expect(view.view).toBeTruthy(); + expect(view.signature).toBeTruthy(); + + //expect the view to have capture the new asset Fabric_ASSET_ID, and attributes to match + expect(snapshot.getStateBins().length).toEqual(1); + expect(snapshot.getStateBins()[0].getId()).toEqual(FABRIC_ASSET_ID); + expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); + + //fabric transaction proofs include endorsements + expect( + snapshot + .getStateBins()[0] + .getTransactions()[0] + .getProof() + .getEndorsements()?.length, + ).toEqual(2); + + //no valid states for this time frame + const view1 = pluginBungee.generateView(snapshot, "0", "9999", undefined); + expect(view1.view).toBeUndefined(); + expect(view1.signature).toBeUndefined(); + + //creating new asset + const new_asset_id = uuidv4(); + const createResponse = await apiClient.runTransactionV1({ + contractName: fabricContractName, + channelName: fabricChannelName, + params: [new_asset_id, "10"], + methodName: "CreateAsset", + invocationType: FabricContractInvocationType.Send, + signingCredential: fabricSigningCredential, + }); + expect(createResponse).not.toBeUndefined(); + expect(createResponse.status).toBeGreaterThan(199); + expect(createResponse.status).toBeLessThan(300); + + //changing FABRIC_ASSET_ID value + const modifyResponse = await apiClient.runTransactionV1({ + contractName: fabricContractName, + channelName: fabricChannelName, + params: [FABRIC_ASSET_ID, "18"], + methodName: "UpdateAsset", + invocationType: FabricContractInvocationType.Send, + signingCredential: fabricSigningCredential, + }); + expect(modifyResponse).not.toBeUndefined(); + expect(modifyResponse.status).toBeGreaterThan(199); + expect(modifyResponse.status).toBeLessThan(300); + + const snapshot1 = await pluginBungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + const view2 = pluginBungee.generateView( + snapshot1, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + + //expect to return a view + expect(view2.view).toBeTruthy(); + expect(view2.signature).toBeTruthy(); + + //expect to have captured state for both assets + const stateBins = snapshot1.getStateBins(); + expect(stateBins.length).toEqual(2); + const bins = [stateBins[0].getId(), stateBins[1].getId()]; + + expect(bins.includes(FABRIC_ASSET_ID)).toBeTrue(); + expect(bins.includes(new_asset_id)).toBeTrue(); + + //checks if values match: + // - new value of FABRIC_ASSET_ID state in new snapshot equals to new value) + // - successfully captured transaction that created the new asset + if (bins[0] === FABRIC_ASSET_ID) { + expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(2); + expect(snapshot1.getStateBins()[0].getValue()).toEqual("18"); + expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(1); + } else { + expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); + expect(snapshot1.getStateBins()[1].getTransactions().length).toEqual(2); + expect(snapshot1.getStateBins()[1].getValue()).toEqual("18"); + } + }, +); -afterAll(async () => { +afterEach(async () => { await fabricLedger.stop(); await fabricLedger.destroy(); await Servers.shutdown(fabricServer); diff --git a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/fabric-test-pruning.test.ts b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/fabric-test-pruning.test.ts index 0a810c00fff..63ae4768345 100644 --- a/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/fabric-test-pruning.test.ts +++ b/packages/cactus-plugin-bungee-hermes/src/test/typescript/integration/fabric-test-pruning.test.ts @@ -68,14 +68,14 @@ let pluginBungeeFabricOptions: IPluginBungeeHermesOptions; let pluginBungee: PluginBungeeHermes; const FABRIC_ASSET_ID = uuidv4(); -let networkDetails: FabricNetworkDetails; +let networkDetailsList: FabricNetworkDetails[]; const log = LoggerProvider.getOrCreate({ level: logLevel, label: "BUNGEE - Hermes", }); -beforeAll(async () => { +beforeEach(async () => { pruneDockerAllIfGithubAction({ logLevel }) .then(() => { log.info("Pruning throw OK"); @@ -333,99 +333,122 @@ beforeAll(async () => { instanceId: uuidv4(), }; - networkDetails = { - connectorApiPath: fabricPath, - signingCredential: fabricSigningCredential, - channelName: fabricChannelName, - contractName: fabricContractName, - participant: "Org1MSP", - }; + networkDetailsList = [ + { + connectorApiPath: fabricPath, + signingCredential: fabricSigningCredential, + channelName: fabricChannelName, + contractName: fabricContractName, + participant: "Org1MSP", + }, + { + connector: fabricConnector, + signingCredential: fabricSigningCredential, + channelName: fabricChannelName, + contractName: fabricContractName, + participant: "Org1MSP", + }, + ]; pluginBungee = new PluginBungeeHermes(pluginBungeeFabricOptions); } }); -test("test creation of views for specific timeframes", async () => { - const strategy = "FABRIC"; - pluginBungee.addStrategy(strategy, new StrategyFabric("INFO")); - - const snapshot = await pluginBungee.generateSnapshot( - [], - strategy, - networkDetails, - ); - const view = pluginBungee.generateView( - snapshot, - "0", - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - //expect to return a view - expect(view.view).toBeTruthy(); - expect(view.signature).toBeTruthy(); - - //expect the view to have capture the new asset FABRIC_ASSET_ID, and attributes to match - expect(snapshot.getStateBins().length).toEqual(1); - expect(snapshot.getStateBins()[0].getId()).toEqual(FABRIC_ASSET_ID); - expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); - //fabric transaction proofs include endorsements - expect( - snapshot.getStateBins()[0].getTransactions()[0].getProof().getEndorsements() - ?.length, - ).toEqual(2); - - //changing FABRIC_ASSET_ID value - const modifyResponse = await apiClient.runTransactionV1({ - contractName: fabricContractName, - channelName: fabricChannelName, - params: [FABRIC_ASSET_ID, "18"], - methodName: "UpdateAsset", - invocationType: FabricContractInvocationType.Send, - signingCredential: fabricSigningCredential, - }); - expect(modifyResponse).not.toBeUndefined(); - expect(modifyResponse.status).toBeGreaterThan(199); - expect(modifyResponse.status).toBeLessThan(300); - - const snapshot1 = await pluginBungee.generateSnapshot( - [], - strategy, - networkDetails, - ); - - //tI is the time of the first transaction +1 - const tI = ( - BigInt(snapshot.getStateBins()[0].getTransactions()[0].getTimeStamp()) + - BigInt(1) - ).toString(); - - expect(snapshot1.getStateBins().length).toEqual(1); - expect(snapshot1.getStateBins()[0].getId()).toEqual(FABRIC_ASSET_ID); - expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(2); - expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( - snapshot.getStateBins()[0].getValue(), - ); - const view1 = pluginBungee.generateView( - snapshot1, - tI, - Number.MAX_SAFE_INTEGER.toString(), - undefined, - ); - //expect to return a view - expect(view1.view).toBeTruthy(); - expect(view1.signature).toBeTruthy(); - - expect(snapshot1.getStateBins().length).toEqual(1); - expect(snapshot1.getStateBins()[0].getId()).toEqual(FABRIC_ASSET_ID); - //expect the view to not include first transaction (made before tI) - expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); - //expect old and new snapshot state values to differ - expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( - snapshot.getStateBins()[0].getValue(), - ); -}); +test.each([{ apiPath: true }, { apiPath: false }])( + //test for both FabricApiPath and FabricConnector + "test creation of views for specific timeframes", + async ({ apiPath }) => { + let networkDetails: FabricNetworkDetails; + if (apiPath) { + networkDetails = networkDetailsList[0]; + } else { + networkDetails = networkDetailsList[1]; + } + + const strategy = "FABRIC"; + pluginBungee.addStrategy(strategy, new StrategyFabric("INFO")); + + const snapshot = await pluginBungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + const view = pluginBungee.generateView( + snapshot, + "0", + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + //expect to return a view + expect(view.view).toBeTruthy(); + expect(view.signature).toBeTruthy(); + + //expect the view to have capture the new asset FABRIC_ASSET_ID, and attributes to match + expect(snapshot.getStateBins().length).toEqual(1); + expect(snapshot.getStateBins()[0].getId()).toEqual(FABRIC_ASSET_ID); + expect(snapshot.getStateBins()[0].getTransactions().length).toEqual(1); + //fabric transaction proofs include endorsements + expect( + snapshot + .getStateBins()[0] + .getTransactions()[0] + .getProof() + .getEndorsements()?.length, + ).toEqual(2); + + //changing FABRIC_ASSET_ID value + const modifyResponse = await apiClient.runTransactionV1({ + contractName: fabricContractName, + channelName: fabricChannelName, + params: [FABRIC_ASSET_ID, "18"], + methodName: "UpdateAsset", + invocationType: FabricContractInvocationType.Send, + signingCredential: fabricSigningCredential, + }); + expect(modifyResponse).not.toBeUndefined(); + expect(modifyResponse.status).toBeGreaterThan(199); + expect(modifyResponse.status).toBeLessThan(300); + + const snapshot1 = await pluginBungee.generateSnapshot( + [], + strategy, + networkDetails, + ); + + //tI is the time of the first transaction +1 + const tI = ( + BigInt(snapshot.getStateBins()[0].getTransactions()[0].getTimeStamp()) + + BigInt(1) + ).toString(); + + expect(snapshot1.getStateBins().length).toEqual(1); + expect(snapshot1.getStateBins()[0].getId()).toEqual(FABRIC_ASSET_ID); + expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(2); + expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( + snapshot.getStateBins()[0].getValue(), + ); + const view1 = pluginBungee.generateView( + snapshot1, + tI, + Number.MAX_SAFE_INTEGER.toString(), + undefined, + ); + //expect to return a view + expect(view1.view).toBeTruthy(); + expect(view1.signature).toBeTruthy(); + + expect(snapshot1.getStateBins().length).toEqual(1); + expect(snapshot1.getStateBins()[0].getId()).toEqual(FABRIC_ASSET_ID); + //expect the view to not include first transaction (made before tI) + expect(snapshot1.getStateBins()[0].getTransactions().length).toEqual(1); + //expect old and new snapshot state values to differ + expect(snapshot1.getStateBins()[0].getValue()).not.toEqual( + snapshot.getStateBins()[0].getValue(), + ); + }, +); -afterAll(async () => { +afterEach(async () => { await fabricLedger.stop(); await fabricLedger.destroy(); await Servers.shutdown(fabricServer);