#Hyperledger Client SDK for Node.js
The Hyperledger Client SDK (HLC) provides a powerful and easy to use API to interact with a Hyperledger blockchain.
The sections in this document are as follows:
-
The Getting Started section is intended to help you quickly get a feel for HLC, how to use it, and some of it's common capabilities. This is demonstrated by example.
-
The Getting Set Up section shows you how to setup up your environment and how to run the unit tests. Looking at the implementation of the unit tests will also help you learn more about the APIs by example, including asset management and confidentiality.
-
The Going Deeper section discusses HLC's pluggability or extensibility design. It also describes the main object hierarchy to help you get started in navigating the reference documentation. The top-level class is
Chain
.WARNING: To view the reference documentation correctly, you first need to build the SDK and confirm functionality by running the unit tests. Subsequently open the following URLs directly in your browser. Be sure to replace YOUR-FABRIC-DIR with the path to your fabric directory.
file:///YOUR-FABRIC-DIR/sdk/node/doc/modules/_hlc_.html
file:///YOUR-FABRIC-DIR/sdk/node/doc/classes/_hlc_.chain.html
-
The Future Work section describes some upcoming work to be done.
This purpose of this section is to help you quickly get a feel for HLC and how you may use it. It is not intended to demonstrate all of it's power, but to demonstrate common use cases by example.
First, there is some basic terminology you should understand. In order to transact on a hyperledger blockchain, you must first have an identity which has been both registered and enrolled.
Think of registration as issuing a user invitation to join a blockchain. It consists of adding a new user name (also called an enrollment ID) to the membership service configuration. This can be done programatically with the Member.register
method, or by adding the enrollment ID directly to the membersrvc.yaml configuration file.
Think of enrollment as accepting a user invitation to join a blockchain. This is always done by the entity that will transact on the blockchain. This can be done programatically via the Member.enroll
method.
The best way to quickly learn HLC is by example.
The following example demonstrates a typical web app. The web app authenticates a user and then transacts on a blockchain on behalf of that user.
/**
* This example shows how to do the following in a web app.
* 1) At initialization time, enroll the web app with the block chain.
* The identity must have already been registered.
* 2) At run time, after a user has authenticated with the web app:
* a) register and enroll an identity for the user;
* b) use this identity to deploy, query, and invoke a chaincode.
*/
// To include the package from your hyperledger fabric directory:
// var hlc = require("myFabricDir/sdk/node");
// To include the package from npm:
// var hlc = require('hlc');
var hlc = require('hlc');
// Create a client chain.
// The name can be anything as it is only used internally.
var chain = hlc.newChain("targetChain");
// Configure the KeyValStore which is used to store sensitive keys
// as so it is important to secure this storage.
// The FileKeyValStore is a simple file-based KeyValStore, but you
// can easily implement your own to store whereever you want.
chain.setKeyValStore( hlc.newFileKeyValStore('/tmp/keyValStore') );
// Set the URL for member services
chain.setMemberServicesUrl("grpc://localhost:50051");
// Add a peer's URL
chain.addPeer("grpc://localhost:30303");
// Enroll "WebAppAdmin" which is already registered because it is
// listed in fabric/membersrvc/membersrvc.yaml with it's one time password.
// If "WebAppAdmin" has already been registered, this will still succeed
// because it stores the state in the KeyValStore
// (i.e. in '/tmp/keyValStore' in this sample).
chain.enroll("WebAppAdmin", "DJY27pEnl16d", function(err, webAppAdmin) {
if (err) return console.log("ERROR: failed to register %s: %s",err);
// Successfully enrolled WebAppAdmin during initialization.
// Set this user as the chain's registrar which is authorized to register other users.
chain.setRegistrar(webAppAdmin);
// Now begin listening for web app requests
listenForUserRequests();
});
// Main web app function to listen for and handle requests
function listenForUserRequests() {
for (;;) {
// WebApp-specific logic goes here to await the next request.
// ...
// Assume that we received a request from an authenticated user
// 'userName', and determined that we need to invoke the chaincode
// with 'chaincodeID' and function named 'fcn' with arguments 'args'.
handleUserRequest(userName,chaincodeID,fcn,args);
}
}
// Handle a user request
function handleUserRequest(userName, chaincodeID, fcn, args) {
// Register and enroll this user.
// If this user has already been registered and/or enrolled, this will
// still succeed because the state is kept in the KeyValStore
// (i.e. in '/tmp/keyValStore' in this sample).
var registrationRequest = {
enrollmentID: userName,
// Customize account & affiliation
account: "bank_a",
affiliation: "00001"
};
chain.registerAndEnroll( registrationRequest, function(err, user) {
if (err) return console.log("ERROR: %s",err);
// Issue an invoke request
var invokeRequest = {
// Name (hash) required for invoke
chaincodeID: chaincodeID,
// Function to trigger
fcn: fcn,
// Parameters for the invoke function
args: args
};
// Invoke the request from the user object.
var tx = user.invoke(invokeRequest);
// Listen for the 'submitted' event
tx.on('submitted', function(results) {
console.log("submitted invoke: %j",results);
});
// Listen for the 'complete' event.
tx.on('complete', function(results) {
console.log("completed invoke: %j",results;
});
// Listen for the 'error' event.
tx.on('error', function(err) {
console.log("error on invoke: %j",err);
});
});
}
To install hlc
from npm simply execute the following command.
npm install -g hlc
To have the chaincode deployment succeed in network mode, you must properly set up the chaincode project outside of your Hyperledger Fabric source tree. These instructions will demonstrate how to properly set up the directory structure to deploy chaincode_example02 in network mode.
The chaincode project must be placed inside the src
directory in your local $GOPATH
. For example, the chaincode_example02
project may be placed inside $GOPATH/src/
as shown below.
$GOPATH/src/github.com/chaincode_example02/
The chaincode project directory must contain project related code and also a vendor
folder which contains the entire Hyperledger Fabric source tree. Currently, this is still a dependency to have the chaincode project deploy successfully. However, the entire fabric directory is not packaged into the deploy transaction when the payload is generated. The deployment process selects a specific set of files when creating the transaction payload. Correct project directory structure is shown below.
ls -la $GOPATH/src/github.com/chaincode_example02/
.
..
chaincode_example02.go
vendor
ls -la $GOPATH/src/github.com/chaincode_example02/vendor/github.com/hyperledger/
.
..
fabric
Once you have placed your chaincode project inside the src
directory in your local $GOPATH
together with the vendor
directory containing the Hyperledger Fabric, you need to verify that the chaincode builds in this directory. To do so execute go build
. This step verifies that all of the chaincode dependencies are present.
cd $GOPATH/src/github.com/chaincode_example02/
go build
Once the chaincode is built, you need to verify that chain-tests.js unit test file points to the appropriate chaincode project path. The default directory is set to github.com/chaincode_example02/
as shown below.
// Path to the local directory containing the chaincode project under $GOPATH
var testChaincodePath = "github.com/chaincode_example02/";
Set the DEPLOY_MODE
environment variable to net
and run the chain-tests as follows:
cd $GOPATH/src/github.com/hyperledger/fabric/sdk/node
export DEPLOY_MODE='net'
node test/unit/chain-tests.js | node_modules/.bin/tap-spec
If you wish to activate TLS connection with the member services the following actions are needed:
- Modify
$GOPATH/src/github.com/hyperledger/fabric/membersrvc/membersrvc.yaml
as follows:
server:
tls:
cert:
file: "/var/hyperledger/production/.membersrvc/tlsca.cert"
key:
file: "/var/hyperledger/production/.membersrvc/tlsca.priv"
This is needed to instruct the member services on which tls cert and key to use.
- Modify
$GOPATH/src/github.com/hyperledger/fabric/peer/core.yaml
as follows:
peer:
pki:
tls:
enabled: true
rootcert:
file: "/var/hyperledger/production/.membersrvc/tlsca.cert"
This is needed to allow the peer to connect to the member services using TLS, otherwise the connection will fail.
-
Bootstrap your member services and the peer. This is needed in order to have the file tlsca.cert generated by the member services.
-
Copy
/var/hyperledger/production/.membersrvc/tlsca.cert
to$GOPATH/src/github.com/hyperledger/fabric/sdk/node
.
Note: If you cleanup the folder /var/hyperledger/production
then don't forget to copy again the tlsca.cert file as described above.
HLC includes a set of unit tests implemented with the tape framework. The unit test script builds and runs both the membership service server and the peer node for you, therefore you do not have to start those manually.
To run the unit tests, execute the following commands.
cd $GOPATH/src/github.com/hyperledger/fabric
make node-sdk-unit-tests
The following are brief descriptions of each of the unit tests that are being run.
The registrar.js test case exercises registering users with member services. It also tests registering a designated registrar user which can then register additional users.
The chain-tests.js test case exercises the chaincode_example02.go chaincode when it has been deployed in both development mode and network mode.
The asset-mgmt.js test case exercises the asset_management.go chaincode when it has been deployed in both development mode and network mode.
The asset-mgmt-with-roles.js test case exercises the asset_management_with_roles.go chaincode when it has been deployed in both development mode and network mode.
If you see errors stating that the client has already been registered/enrolled, keep in mind that you can perform the enrollment process only once, as the enrollmentSecret is a one-time-use password. You will see these errors if you have performed a user registration/enrollment and subsequently deleted the crypto tokens stored on the client side. The next time you try to enroll, errors similar to the ones below will be seen.
Error: identity or token do not match
Error: user is already registered
To address this, remove any stored crypto material from the CA server by following the instructions here. You will also need to remove any of the crypto tokens stored on the client side by deleting the KeyValStore directory. That directory is configurable and is set to /tmp/keyValStore
within the unit tests.
HLC was designed to support two pluggable components:
-
Pluggable key value store which is used to retrieve and store keys associated with a member. The key value store is used to store sensitive private keys, so care must be taken to properly protect access.
-
Pluggable member service which is used to register and enroll members. Member services enables hyperledger to be a permissioned blockchain, providing security services such as anonymity, unlinkability of transactions, and confidentiality
HLC is written primarily in typescript and is object-oriented. The source can be found in the fabric/sdk/node/src
directory.
To go deeper, you can view the reference documentation in your browser by opening the reference documentation and clicking on "hlc" on the right-hand side under "Globals". This will work after you have built the SDK per the instruction here.
The following is a high-level description of the HLC objects (classes and interfaces) to help guide you through the object hierarchy.
-
The main top-level class is Chain. It is the client's representation of a chain. HLC allows you to interact with multiple chains and to share a single KeyValStore and MemberServices object with multiple Chain objects as needed. For each chain, you add one or more Peer objects which represents the endpoint(s) to which HLC connects to transact on the chain.
-
The KeyValStore is a very simple interface which HLC uses to store and retrieve all persistent data. This data includes private keys, so it is very important to keep this storage secure. The default implementation is a simple file-based version found in the FileKeyValStore class.
-
The MemberServices interface is implemented by the MemberServicesImpl class and provides security and identity related features such as privacy, unlinkability, and confidentiality. This implementation issues ECerts (enrollment certificates) and TCerts (transaction certificates). ECerts are for enrollment identity and TCerts are for transactions.
-
The Member class most often represents an end user who transacts on the chain, but it may also represent other types of members such as peers. From the Member class, you can register and enroll members or users. This interacts with the MemberServices object. You can also deploy, query, and invoke chaincode directly, which interacts with the Peer. The implementation for deploy, query and invoke simply creates a temporary TransactionContext object and delegates the work to it.
-
The TransactionContext class implements the bulk of the deploy, invoke, and query logic. It interacts with MemberServices to get a TCert to perform these operations. Note that there is a one-to-one relationship between TCert and TransactionContext; in other words, a single TransactionContext will always use the same TCert. If you want to issue multiple transactions with the same TCert, then you can get a TransactionContext object from a Member object directly and issue multiple deploy, invoke, or query operations on it. Note however that if you do this, these transactions are linkable, which means someone could tell that they came from the same user, though not know which user. For this reason, you will typically just call deploy, invoke, and query on the User or Member object.
The following is a list of known remaining work to be done.
-
The reference documentation needs to be made simpler to follow as there are currently some internal classes and interfaces that are being exposed.
-
We are investigating how to make deployment of a chaincode simpler by not requiring you to set up a specific directory structure with dependencies.
-
Implement events appropriately, both custom and non-custom. The 'complete' event for
deploy
andinvoke
is currently implemented by simply waiting a set number of seconds (5 for invoke, 20 for deploy). It needs to receive a complete event from the server with the result of the transaction and make this available to the caller. This has not yet been implemented. -
Support SHA2. HLC currently supports SHA3.