Nethereum Quorum is the Nethereum extension to interact with Quorum, the permissioned implementation of Ethereum supporting data privacy created by JP Morgan.
The cross-platform library supports .Net Core, Mono, Linux, iOS, Android, Raspberry PI, Xbox and of course Windows.
Issues, Requests and help
We should be able to answer there any simple queries, general comments or requests, everyone is welcome. In a similar feel free to raise any issue or pull request.
Quick installation
Here is a list of all the nuget packages. Nethereum.Portable combines all the Nethereum packages including Quorum and Nethereum.Quorum is the single Quorum library.
| Package | Nuget |
|---|---|
| Nethereum.Portable | |
| Nethereum.Quorum |
Usage
Public interaction with Quorum is the same as interacting with Geth, if you require more information on this please refer to the general Netherum documentation in Read the docs.
For private interaction or to access the Quorum RPC methods, a specialised Web3, Web3Quorum is required.
PrivateFor will be set as following:
var web3Node1 = new Web3Quorum(urlNode1); var privateFor = new List<string>(new[] { "ROAZBWtSacxXQrOe3FGAqJDyJjFePR5ce4TSIzmJ0Bc=" }); web3Node1.SetPrivateRequestParameters(privateFor);
afterwards all the transactions will use the PrivateFor parameter.
var contract = web3Node1.Eth.GetContract(abi, address); var functionSet = contract.GetFunction("set"); var txnHash = await transactionService.SendRequestAsync(() => functionSet.SendTransactionAsync(account, 4));
For a full example of usage using the 7nodes sample of Quorum check the Unit Test
Quorum RPC
The Quorum RPC methods can be accessed using web3.Quorum to retrieve the CanonicalHash, BlockMaker, Voter, MakeBlock.. Here are some examples from the Unit tests using the 7nodes sample of Quorum
[Fact] public async void ShouldReturnCanonicalHashForBlockNumber1() { var web3 = new Web3Quorum(DefaultSettings.GetDefaultUrl()); var canonicalHash = await web3.Quorum.CanonicalHash.SendRequestAsync(1); Assert.Equal("0x8bb911238205c6d5e9841335c9c5aff3dfae4c0f6b0df28100737c2660a15f8d", canonicalHash); } [Fact] public async void ShouldReturnTrueWhenCallingIsBlockMakerForCoinbase() { var web3 = new Web3Quorum(DefaultSettings.GetDefaultUrl()); var isBlockMaker = await web3.Quorum.IsBlockMaker.SendRequestAsync(await web3.Eth.CoinBase.SendRequestAsync()); Assert.Equal(true, isBlockMaker); } [Fact] public async void ShouldReturnTrueWhenCallingIsVoterForCoinbase() { var web3 = new Web3Quorum(DefaultSettings.GetDefaultUrl()); var isVoter = await web3.Quorum.IsVoter.SendRequestAsync(await web3.Eth.CoinBase.SendRequestAsync()); Assert.Equal(true, isVoter); } [Fact] public async void ShouldBeAbleToMakeBlocksBypassingBlockStrategy() { //Node2 is a Block Maker //This might fail.. depending on the canonicalhash.. (quorum) var web3 = new Web3Quorum(DefaultSettings.QuorumIPAddress + ":22001"); var hash = await web3.Quorum.MakeBlock.SendRequestAsync(); Assert.NotNull(hash); } [Fact] public async void ShouldBeAbleToVote() { //Node 5 can vote var web3 = new Web3Quorum(DefaultSettings.QuorumIPAddress + ":22004"); var block = await web3.Eth.Blocks.GetBlockWithTransactionsByNumber.SendRequestAsync(BlockParameter.CreateLatest()); var hash = await web3.Quorum.Vote.SendRequestAsync(block.BlockHash); Assert.NotNull(hash); } [Fact] public async void ShouldPauseResumeBlockMaker() { //Node 2 is a BlockMaker var web3 = new Web3Quorum(DefaultSettings.QuorumIPAddress + ":22001"); await web3.Quorum.PauseBlockMaker.SendRequestAsync(); await web3.Quorum.ResumeBlockMaker.SendRequestAsync(); } [Fact] public async void ShouldGetNodeInfoOfBlockMaker() { var web3 = new Web3Quorum(DefaultSettings.QuorumIPAddress + ":22001"); var nodeInfo = await web3.Quorum.NodeInfo.SendRequestAsync(); Assert.Equal("active", nodeInfo.BlockMakeStratregy.Status); }