Skip to content

Commit

Permalink
feat: add WebsocketProvider options to quorum LP
Browse files Browse the repository at this point in the history
- add WebsocketProvider options to quorum LP
- add test for calling getBlock with verifier

Signed-off-by: Tomasz Awramski <[email protected]>
  • Loading branch information
rwat17 committed Aug 22, 2023
1 parent d8d538d commit bb748f4
Show file tree
Hide file tree
Showing 7 changed files with 98 additions and 8 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,7 @@
"sanitize-html": "2.7.0",
"typescript-optional": "2.0.1",
"web3": "1.5.2",
"web3-core-helpers": "1.10.0",
"web3-eth-contract": "1.5.2",
"web3-utils": "1.5.2"
},
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,10 @@ import Web3JsQuorum, {
ISendRawTransaction,
IPrivateTransactionReceipt,
} from "web3js-quorum";
import {
WebsocketProviderOptions,
HttpProviderOptions,
} from "web3-core-helpers";
import { AbiItem } from "web3-utils";
import { Contract } from "web3-eth-contract";
import { ContractSendMethod } from "web3-eth-contract";
Expand Down Expand Up @@ -84,6 +88,8 @@ export interface IPluginLedgerConnectorQuorumOptions
prometheusExporter?: PrometheusExporter;
pluginRegistry: PluginRegistry;
privateUrl?: string;
wsProviderOptions?: WebsocketProviderOptions;
httpProviderOptions?: HttpProviderOptions;
}

export class PluginLedgerConnectorQuorum
Expand Down Expand Up @@ -113,9 +119,20 @@ export class PluginLedgerConnectorQuorum

private getWeb3Provider() {
if (!this.options.rpcApiWsHost) {
return new Web3.providers.HttpProvider(this.options.rpcApiHttpHost);
}
return new Web3.providers.WebsocketProvider(this.options.rpcApiWsHost);
return this.options.httpProviderOptions
? new Web3.providers.HttpProvider(
this.options.rpcApiHttpHost,
this.options.httpProviderOptions,
)
: new Web3.providers.HttpProvider(this.options.rpcApiHttpHost);
}

return this.options.wsProviderOptions
? new Web3.providers.WebsocketProvider(
this.options.rpcApiWsHost,
this.options.wsProviderOptions,
)
: new Web3.providers.WebsocketProvider(this.options.rpcApiWsHost);
}

constructor(public readonly options: IPluginLedgerConnectorQuorumOptions) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,9 @@ describe("invokeRawWeb3EthMethod Tests", () => {
await quorumTestLedger.start();

const rpcApiHttpHost = await quorumTestLedger.getRpcApiHttpHost();
log.debug("rpcApiHttpHost:", rpcApiHttpHost);
const getRpcApiWsHost = await quorumTestLedger.getRpcApiWsHost();
log.warn("rpcApiHttpHost:", rpcApiHttpHost);
log.warn("getRpcApiWsHost:", getRpcApiWsHost);

log.info("Create PluginLedgerConnectorQuorum...");
connector = new PluginLedgerConnectorQuorum({
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ import "jest-extended";
import lodash from "lodash";
import { v4 as uuidv4 } from "uuid";
import Web3 from "web3";
import { WebsocketProviderOptions } from "web3-core-helpers";
import { AbiItem } from "web3-utils";
import { PluginRegistry } from "@hyperledger/cactus-core";
import {
Expand Down Expand Up @@ -112,9 +113,19 @@ describe("Verifier integration with quorum connector tests", () => {
plugins.push(keychainPlugin);

log.info("Create PluginLedgerConnectorQuorum...");

const wsProviderOptions: WebsocketProviderOptions = {
clientConfig: {
// Useful if requests are large
maxReceivedFrameSize: 100000000,
maxReceivedMessageSize: 100000000,
},
};

connector = new PluginLedgerConnectorQuorum({
rpcApiHttpHost: connectionProfile.quorum.member1.url,
rpcApiWsHost: connectionProfile.quorum.member1.wsUrl,
wsProviderOptions: wsProviderOptions,
logLevel: sutLogLevel,
instanceId: uuidv4(),
pluginRegistry: new PluginRegistry({ plugins: [keychainPlugin] }),
Expand Down Expand Up @@ -270,6 +281,26 @@ describe("Verifier integration with quorum connector tests", () => {
};
});

test("Using verifier for calling getBlock method works", async () => {
const correctContract = {};
const correctMethod = {
type: "web3Eth",
command: "getBlock",
};

const correctArgs: any = { args: ["latest"] };

const resultCorrect = await verifier.sendSyncRequest(
correctContract,
correctMethod,
correctArgs,
);

log.warn("verifier", verifier);
log.warn("###Results", resultCorrect);
// expect(resultCorrect.status).toEqual(200);
});

test("Invalid web3EthContract calls are rejected by QuorumApiClient", async () => {
// Define correct input parameters
const correctContract: Record<string, unknown> = lodash.clone(
Expand Down Expand Up @@ -500,6 +531,7 @@ describe("Verifier integration with quorum connector tests", () => {
methodCall,
argsCall,
);
log.error(resultsCall);
expect(resultsCall.status).toEqual(200);
expect(resultsCall.data).toEqual(newName);
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ export interface IQuorumTestLedgerConstructorOptions {
containerImageVersion?: string;
containerImageName?: string;
rpcApiHttpPort?: number;
rpcApiWsPort?: number;
logLevel?: LogLevelDesc;
emitContainerLogs?: boolean;
readonly envVars?: string[];
Expand All @@ -32,6 +33,7 @@ export const QUORUM_TEST_LEDGER_DEFAULT_OPTIONS = Object.freeze({
containerImageVersion: "2021-01-08-7a055c3",
containerImageName: "ghcr.io/hyperledger/cactus-quorum-all-in-one",
rpcApiHttpPort: 8545,
rpcApiWsPort: 8546,
});

export const QUORUM_TEST_LEDGER_OPTIONS_JOI_SCHEMA: Joi.Schema = Joi.object().keys(
Expand All @@ -51,6 +53,7 @@ export class QuorumTestLedger implements ITestLedger {
public readonly containerImageVersion: string;
public readonly containerImageName: string;
public readonly rpcApiHttpPort: number;
public readonly rpcApiWsPort: number;
public readonly emitContainerLogs: boolean;

private readonly log: Logger;
Expand All @@ -73,6 +76,8 @@ export class QuorumTestLedger implements ITestLedger {
this.rpcApiHttpPort =
options.rpcApiHttpPort ||
QUORUM_TEST_LEDGER_DEFAULT_OPTIONS.rpcApiHttpPort;
this.rpcApiWsPort =
options.rpcApiWsPort || QUORUM_TEST_LEDGER_DEFAULT_OPTIONS.rpcApiWsPort;

this.envVars = options.envVars || [];

Expand Down Expand Up @@ -102,9 +107,14 @@ export class QuorumTestLedger implements ITestLedger {

public async getRpcApiHttpHost(): Promise<string> {
const ipAddress = "127.0.0.1";
const hostPort = await this.getRpcApiPublicPort();
const hostPort = await this.getRpcHttpApiPublicPort();
return `http://${ipAddress}:${hostPort}`;
}
public async getRpcApiWsHost(): Promise<string> {
const ipAddress = "127.0.0.1";
const hostPort = await this.getRpcWsApiPublicPort();
return `ws://${ipAddress}:${hostPort}`;
}

public async getFileContents(filePath: string): Promise<string> {
const response: NodeJS.ReadableStream = await this.getContainer().getArchive(
Expand Down Expand Up @@ -221,7 +231,7 @@ export class QuorumTestLedger implements ITestLedger {
Env: this.envVars,
ExposedPorts: {
[`${this.rpcApiHttpPort}/tcp`]: {}, // quorum RPC - HTTP
"8546/tcp": {}, // quorum RPC - WebSocket
[`${this.rpcApiWsPort}/tcp`]: {}, // quorum RPC - WebSocket
"8888/tcp": {}, // orion Client Port - HTTP
"8080/tcp": {}, // orion Node Port - HTTP
"9001/tcp": {}, // supervisord - HTTP
Expand Down Expand Up @@ -339,7 +349,7 @@ export class QuorumTestLedger implements ITestLedger {
}
}

public async getRpcApiPublicPort(): Promise<number> {
public async getRpcHttpApiPublicPort(): Promise<number> {
const fnTag = "QuorumTestLedger#getRpcApiPublicPort()";
const aContainerInfo = await this.getContainerInfo();
const { rpcApiHttpPort: thePort } = this;
Expand All @@ -362,6 +372,29 @@ export class QuorumTestLedger implements ITestLedger {
}
}

public async getRpcWsApiPublicPort(): Promise<number> {
const fnTag = "QuorumTestLedger#getRpcWsApiPublicPort()";
const aContainerInfo = await this.getContainerInfo();
const { rpcApiWsPort: thePort } = this;
const { Ports: ports } = aContainerInfo;

if (ports.length < 1) {
throw new Error(`${fnTag} no ports exposed or mapped at all`);
}
const mapping = ports.find((x) => x.PrivatePort === thePort);
if (mapping) {
if (!mapping.PublicPort) {
throw new Error(`${fnTag} port ${thePort} mapped but not public`);
} else if (mapping.IP !== "0.0.0.0") {
throw new Error(`${fnTag} port ${thePort} mapped to localhost`);
} else {
return mapping.PublicPort;
}
} else {
throw new Error(`${fnTag} no mapping found for ${thePort}`);
}
}

public async getContainerIpAddress(): Promise<string> {
const fnTag = "QuorumTestLedger#getContainerIpAddress()";
const aContainerInfo = await this.getContainerInfo();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ tap.test("starts/stops/destroys a docker container", async (assert: any) => {
assert.ok(ipAddress);
assert.ok(ipAddress.length);

const hostPort: number = await ledger.getRpcApiPublicPort();
const hostPort: number = await ledger.getRpcHttpApiPublicPort();
assert.ok(hostPort, "getRpcApiPublicPort() returns truthy OK");
assert.ok(isFinite(hostPort), "getRpcApiPublicPort() returns finite OK");

Expand Down
5 changes: 5 additions & 0 deletions tools/docker/quorum-all-in-one/start-quorum.sh
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,11 @@ geth \
--rpcvhosts "*" \
--rpcaddr 0.0.0.0 \
--rpcport 8545 \
--ws \
--wsaddr "0.0.0.0" \
--wsport "8546" \
--wsapi admin,db,eth,debug,miner,net,shh,txpool,personal,web3,quorum,${QUORUM_CONSENSUS:-raft} \
--wsorigins "*" \
--rpcapi admin,db,eth,debug,miner,net,shh,txpool,personal,web3,quorum,${QUORUM_CONSENSUS:-raft} \
--port 21000 \
--allow-insecure-unlock \
Expand Down

0 comments on commit bb748f4

Please sign in to comment.