-
Notifications
You must be signed in to change notification settings - Fork 20
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[WIP] Add developing chaincode tutorial
See #127 Signed-off-by: James Taylor <[email protected]>
- Loading branch information
Showing
4 changed files
with
177 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,150 @@ | ||
# Developing and debuging chaincode | ||
|
||
Publishing a chaincode Docker image and using the image digest to deploy the chaincode works well with the [Fabric chaincode lifecycle](https://hyperledger-fabric.readthedocs.io/en/latest/chaincode_lifecycle.html), however it is not as convenient while developing and debugging chaincode. | ||
|
||
This tutorial describes how to deploy and debug the same chaincode using the chaincode-as-a-service builder and the [fabric-samples](https://github.com/hyperledger/fabric-samples/) nano test network. | ||
|
||
TODO: see also... | ||
https://github.com/hyperledger/fabric-samples/blob/main/test-network/CHAINCODE_AS_A_SERVICE_TUTORIAL.md | ||
|
||
First create a directory to download all the required files and run the demo. | ||
|
||
```shell | ||
mkdir hlf-debug-demo | ||
cd hlf-debug-demo | ||
``` | ||
|
||
Now follow the steps below to deploy your first smart contract using the k8s builder! | ||
|
||
## Setup the nano test network | ||
|
||
Download the sample nano test network (fabric-samples isn't tagged so we'll use a known good commit). | ||
|
||
```shell | ||
export FABRIC_SAMPLES_COMMIT=0db64487e5e89a81d68e6871af3f0907c67e7d75 | ||
curl -sSL "https://github.com/hyperledger/fabric-samples/archive/${FABRIC_SAMPLES_COMMIT}.tar.gz" | tar -xzf - --strip-components=1 fabric-samples-${FABRIC_SAMPLES_COMMIT}/test-network-nano-bash | ||
``` | ||
|
||
Install the Fabric binaries | ||
|
||
TODO: ... if they're not already in your path! | ||
|
||
```shell | ||
curl -sSLO https://raw.githubusercontent.com/hyperledger/fabric/main/scripts/install-fabric.sh && chmod +x install-fabric.sh | ||
./install-fabric.sh binary | ||
``` | ||
|
||
Update chaincode-as-a-service builder configuration | ||
|
||
The sample `core.yaml` file needs to be updated with the correct chaincode-as-a-service builder location since it assumes that the peer is running in a docker container. Either edit the `path` for the `ccaas_builder` in `externalBuilders`, or use the following [yq](https://github.com/mikefarah/yq) command. | ||
|
||
```shell | ||
yq -i '( .chaincode.externalBuilders[] | select(.name == "ccaas_builder") | .path ) = strenv(PWD) + "/builders/ccaas"' config/core.yaml | ||
``` | ||
|
||
Start the nano test network | ||
|
||
```shell | ||
cd test-network-nano-bash | ||
./network.sh start | ||
``` | ||
|
||
## Deploy a chaincode-as-a-service chaincode package | ||
|
||
With chaincode-as-a-service you still have to go through the chaincode lifecycle once to tell Fabric where the chaincode is running but after that you can stop, restart, update, and debug the chaincode all without needing to repeat the chaincode lifecycle steps as you would do normally. | ||
This is unlikely to be acceptable in a production environment but it is ideal in a development environment. | ||
|
||
Like the k8s builder, the chaincode-as-a-service chaincode package does not contain any source code. | ||
Instead, it contains the details a Fabric peer needs to connect to the chaincode server. | ||
These details are contained in a `connection.json` and do not include anything specific to the chaincode you want to debug. | ||
You can use exactly the same chaincode-as-a-service chaincode package to debug any chaincode, in any language. | ||
|
||
Start by creating an `connection.json` file. | ||
|
||
```shell | ||
cat << CONNECTIONJSON_EOF > connection.json | ||
{ | ||
"address": "127.0.0.1:9999", | ||
"dial_timeout": "10s", | ||
"tls_required": false | ||
} | ||
CONNECTIONJSON_EOF | ||
``` | ||
|
||
Create a `metadata.json` file for the chaincode package. The chaincode-as-a-service builder provided with Fabric will detect the type of `ccaas`. | ||
|
||
```shell | ||
cat << METADATAJSON_EOF > metadata.json | ||
{ | ||
"type": "ccaas", | ||
"label": "dev-contract" | ||
} | ||
METADATAJSON_EOF | ||
``` | ||
|
||
Create the chaincode package archive. | ||
|
||
```shell | ||
tar -czf code.tar.gz connection.json | ||
tar -czf dev-contract.tgz metadata.json code.tar.gz | ||
``` | ||
|
||
|
||
|
||
```shell | ||
cd test-network-nano-bash | ||
. ./peer1admin.sh | ||
peer channel list | ||
``` | ||
|
||
```shell | ||
peer lifecycle chaincode install ../dev-contract.tgz | ||
``` | ||
|
||
Set the CHAINCODE_ID environment variable for use in subsequent commands: | ||
|
||
```shell | ||
export CHAINCODE_ID=$(peer lifecycle chaincode calculatepackageid ../dev-contract.tgz) && echo $CHAINCODE_ID | ||
``` | ||
|
||
```shell | ||
peer lifecycle chaincode approveformyorg -o 127.0.0.1:6050 --channelID mychannel --name dev-contract --version 1 --package-id $CHAINCODE_ID --sequence 1 --tls --cafile ${PWD}/crypto-config/ordererOrganizations/example.com/orderers/orderer.example.com/tls/ca.crt | ||
``` | ||
|
||
```shell | ||
peer lifecycle chaincode commit -o 127.0.0.1:6050 --channelID mychannel --name dev-contract --version 1 --sequence 1 --tls --cafile "${PWD}"/crypto-config/ordererOrganizations/example.com/orderers/orderer.example.com/tls/ca.crt | ||
``` | ||
|
||
## Debugging the chaincode | ||
|
||
Normally Fabric peers start chaincode but with chaincode-as-a-service you are responsible for starting the chaincode, which is why it is so useful for debugging. | ||
|
||
Importantly chaincode can be started in chaincode-as-a-service mode without any code changes in recent Fabric versions, making it easy to move from a development environment, to using the k8s builder in a production environment. | ||
|
||
Go and Java chaincode will start in chaincode-as-a-service mode if `CHAINCODE_SERVER_ADDRESS` and `CORE_CHAINCODE_ID_NAME` environment variables are configured. The `CHAINCODE_SERVER_ADDRESS` must match the `address` field from the `connection.json` file in the chaincode package. The `CORE_CHAINCODE_ID_NAME` can be found using the `peer lifecycle chaincode calculatepackageid` command. | ||
|
||
Node.js chaincode can be started in chaincode-as-a-service mode using `--chaincode-address` and `--chaincode-id` command line arguments. The sample Node.js contract includes a `debug` script in `package.json` which uses the same `CHAINCODE_SERVER_ADDRESS` and `CORE_CHAINCODE_ID_NAME` environment variables for these command line arguments as Go and Java chaincode. | ||
|
||
|
||
https://code.visualstudio.com/docs/nodejs/nodejs-debugging | ||
|
||
Check everything is working. | ||
|
||
```shell | ||
peer chaincode query -C mychannel -n dev-contract -c '{"Args":["org.hyperledger.fabric:GetMetadata"]}' | ||
``` | ||
|
||
Remember to set a breakpoint at the start of the transaction function you want to debug. | ||
|
||
Remember the client/fabric transaction timeout, whilst you have the chaincode stopped in the debugger, the timeout is still 'ticking' | ||
|
||
|
||
```shell | ||
peer chaincode invoke -o 127.0.0.1:6050 -C mychannel -n dev-contract -c '{"Args":["PutValue","asset1","green"]}' --waitForEvent --tls --cafile "${PWD}"/crypto-config/ordererOrganizations/example.com/orderers/orderer.example.com/tls/ca.crt | ||
``` | ||
|
||
```shell | ||
peer chaincode query -C mychannel -n dev-contract -c '{"Args":["GetValue","asset1"]}' | ||
``` | ||
|
||
TODO: check that Java works! |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
# CHAINCODE_SERVER_ADDRESS must be set to the host and port where the peer can | ||
# connect to the chaincode server | ||
CHAINCODE_SERVER_ADDRESS=localhost | ||
|
||
# CHAINCODE_ID must be set to the Package ID that is assigned to the chaincode | ||
# on install. The `peer lifecycle chaincode queryinstalled` command can be | ||
# used to get the ID after install if required | ||
CHAINCODE_ID=test | ||
|
||
# Optional parameters that will be used for TLS connection between peer node | ||
# and the chaincode. | ||
# TLS is disabled by default, uncomment the following line to enable TLS connection | ||
# CHAINCODE_TLS_DISABLED=false | ||
|
||
# Following variables will be ignored if TLS is not enabled. | ||
# They need to be in PEM format | ||
# CHAINCODE_TLS_KEY=/path/to/private/key/file | ||
# CHAINCODE_TLS_CERT=/path/to/public/cert/file | ||
|
||
# The following variable will be used by the chaincode server to verify the | ||
# connection from the peer node. | ||
# Note that when this is set a single chaincode server cannot be shared | ||
# across organizations unless their root CA is same. | ||
# CHAINCODE_CLIENT_CA_CERT=/path/to/peer/organization/root/ca/cert/file |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters