As mentioned in some sections above, this application uses a custom module that will dynamically create and mount middleware functions that connect to a Fabric gateway. It is used for any routes in your application that need to connect to the Fabric network. In a nutshell, this module allows you to specify in a config file which routes in your application need to connect to which channels, chaincodes, and contracts in your Fabric network. The module then parses that file, creates a gateway instance and connects to it, and dynamically creates middleware functions that get and store the gateway and the specified channel, chaincodes, and its respective contract instances for use in the route handler function. The middleware functions are then mounted to the routes specified in the config file.
Let's go through the config file in detail to fully understand what happens under the covers and to fully understand how to configure our file.
The module is found at server/middlewares/fabric-routes.ts and searches for the config file at server/config/fabric-connections.json. There is already an example file there. Below is an example to follow along with as you read through the rest of this section.
{
"serviceDiscovery": {
"enabled": true,
"asLocalhost": false
},
"fabricConnections": {
"conn1": [
{
"channel": "defaultchannel",
"chaincodes": {
"pingcc": []
}
}
],
"conn2": [
{
"channel": "channel2",
"chaincodes": {
"pingcc": [
"contract1",
"contract2"
],
"chaincode2": [
"contract1"
]
}
},
{
"channel": "channel3",
"chaincodes": {
"pingcc": [
"contract1",
"contract2"
],
"chaincode3": [
"contract1"
]
}
}
]
},
"routes": [
{
"path": "/ping",
"fabricConnection": "conn1",
"modulePath": "routes/ping"
},
{
"path": "/securedPing",
"fabricConnection": "conn1",
"modulePath": "routes/securedPing",
"protected" : {
"enabled": true,
"allowedClients": [
"<clientIds>"
]
}
}
]
}
The fabric-connections JSON file has three main keys: serviceDiscovery
, fabricConnections
and routes
. The fabricConnections
object defines the different Fabric connections you will need in your application. For example, in the example you see two connections specified: conn1
and conn2
. conn1
is the simplest connection you can define; you connect to 1 channel and 1 chaincode with no contracts specified.
It's important to note here that if you do not specify contracts in the array, fabric-routes will pass just the chaincode name to the Contract API. This will always be true for Go chaincode as multiple contacts are not yet supported in v1.4.
conn2
shows something a little more complex; you connect to 2 different channels and 2 different chaincodes (with various contracts) on each channel. Each connection defines what you will need in order to do your invokes and queries in a specific route handler. serviceDiscovery
has configuration for enabling Service Discovery, which is the default in IBM Blockchain Platform 2.0. For Service Discovery, you must add anchor peers to your channel. Without Service Discovery enabled, you will need to add the channels
field in your connection profile to explicitly give the peer endpoints for endorsement. For example, something like:
"channels": {
"channel1": {
"orderers": ["<orderer endpoint>"],
"peers": {
"<peer1 endpoint>": {},
"<peer2 endpoint": {}
}
}
},
In the third key, routes
, you add all the routes in your app that need to connect to the Fabric network. For each route, you specify the route path, the fabric connection the handler for that route will need and the path to the router module where you have defined this route and its handler function. The router module path should be relative to the root of your server app, aka from within the server directory. For this example, we have one route defined that uses the conn1
connection and whose router module is defined at routes/ping.ts (read the Development section above if you need further clarification on the router). Each route can have only a single fabric connection, but a fabric connection can be mapped to multiple routes. Inside of routes
you have the ability to enable authentication with IBM App ID by providing the protected
object with the allowedClients
array of client id's that are allowed to access the endpoint. See the securedPing
route example in the snippet above.
Following the example file, when it comes time to implement our ping route handler, we already have a ready-to-use fabric-network Gateway instance where we can invoke and query the chaincode(s) we need. To invoke or query the chaincode(s) defined in the fabric connection for this route, you simply need to do the following call: await res.locals.<channelName>.<chaincode>.<contractName>.submitTransaction()
or await res.locals.<channelName>.<chaincode>.<contractName>.evaluateTransaction()
.
you can skip the <contractName> if you are using Go chaincode or only have one contract.
Everything is configured and defined, but how does it all come together and how is it all registered to our Express app? In order to register these routes and mount their dynamically created fabric connection middleware functions, you have to add the following in the routes/index.ts file:
import FabricRoutes from '../middlewares/fabric-routes';
// Hyperledger Fabric routes
// add specified routes and create their middleware functions to connect to the fabric network
const fabricRoutes = new FabricRoutes(router);
await fabricRoutes.setup();
This will connect the Gateway instance and wrap all of our routes in our config file with middleware that stores the gateway and needed channels and chaincodes for use in each particular route handler. This is what happens under the covers of the fabricRoutes.setup()
call:
- A middleware function for each connection in the config file is created
- Gateway instance is connected
- Config file is parsed
- For each fabric connection in config file, a middleware function is created that:
- Stores the gateway instance at
res.locals.gateway
- Gets a Network instance for each channel specified in the connection. Stores the network instance at
res.local.<channelName>
- Gets the Contract instance for each chaincode specified in the channel. Stores the contract instance at
res.local.<channelName>.<chaincodeName>.<contractName>
- Stores the gateway instance at
- For each route, the middleware to connect to Fabric is mounted and then the router module for that path is registered. The final flow of the route is: Fabric gateway middleware -> other middleware specified in the router module -> route handler.
To deep dive into this module, take a look at the source code and its comments: server/middlewares/fabric-routes.ts
NOTE: The gateway instance is created and connected only once, then stored in memory for reuse across every request. Currently, this application supports the use of only one Blockchain identity.