171 lines
4.6 KiB
Go
171 lines
4.6 KiB
Go
/*
|
|
Copyright IBM Corp. All Rights Reserved.
|
|
|
|
SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
package chaincode
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/hex"
|
|
"fmt"
|
|
"reflect"
|
|
"strings"
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
cb "github.com/hyperledger/fabric-protos-go/common"
|
|
pb "github.com/hyperledger/fabric-protos-go/peer"
|
|
"github.com/hyperledger/fabric/bccsp"
|
|
"github.com/hyperledger/fabric/protoutil"
|
|
"github.com/pkg/errors"
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
var (
|
|
chaincodeListCmd *cobra.Command
|
|
getInstalledChaincodes bool
|
|
getInstantiatedChaincodes bool
|
|
)
|
|
|
|
// listCmd returns the cobra command for listing
|
|
// the installed or instantiated chaincodes
|
|
func listCmd(cf *ChaincodeCmdFactory, cryptoProvider bccsp.BCCSP) *cobra.Command {
|
|
chaincodeListCmd = &cobra.Command{
|
|
Use: "list",
|
|
Short: "Get the instantiated chaincodes on a channel or installed chaincodes on a peer.",
|
|
Long: "Get the instantiated chaincodes in the channel if specify channel, or get installed chaincodes on the peer",
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
return getChaincodes(cmd, cf, cryptoProvider)
|
|
},
|
|
}
|
|
|
|
flagList := []string{
|
|
"channelID",
|
|
"installed",
|
|
"instantiated",
|
|
"peerAddresses",
|
|
"tlsRootCertFiles",
|
|
"connectionProfile",
|
|
}
|
|
attachFlags(chaincodeListCmd, flagList)
|
|
|
|
return chaincodeListCmd
|
|
}
|
|
|
|
func getChaincodes(cmd *cobra.Command, cf *ChaincodeCmdFactory, cryptoProvider bccsp.BCCSP) error {
|
|
if getInstantiatedChaincodes && channelID == "" {
|
|
return errors.New("The required parameter 'channelID' is empty. Rerun the command with -C flag")
|
|
}
|
|
// Parsing of the command line is done so silence cmd usage
|
|
cmd.SilenceUsage = true
|
|
|
|
var err error
|
|
if cf == nil {
|
|
cf, err = InitCmdFactory(cmd.Name(), true, false, cryptoProvider)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
creator, err := cf.Signer.Serialize()
|
|
if err != nil {
|
|
return fmt.Errorf("error serializing identity: %s", err)
|
|
}
|
|
|
|
var proposal *pb.Proposal
|
|
|
|
if getInstalledChaincodes == getInstantiatedChaincodes {
|
|
return errors.New("must explicitly specify \"--installed\" or \"--instantiated\"")
|
|
}
|
|
|
|
if getInstalledChaincodes {
|
|
proposal, _, err = protoutil.CreateGetInstalledChaincodesProposal(creator)
|
|
}
|
|
|
|
if getInstantiatedChaincodes {
|
|
proposal, _, err = protoutil.CreateGetChaincodesProposal(channelID, creator)
|
|
}
|
|
|
|
if err != nil {
|
|
return errors.WithMessage(err, "error creating proposal")
|
|
}
|
|
|
|
var signedProposal *pb.SignedProposal
|
|
signedProposal, err = protoutil.GetSignedProposal(proposal, cf.Signer)
|
|
if err != nil {
|
|
return errors.WithMessage(err, "error creating signed proposal")
|
|
}
|
|
|
|
// list is currently only supported for one peer
|
|
proposalResponse, err := cf.EndorserClients[0].ProcessProposal(context.Background(), signedProposal)
|
|
if err != nil {
|
|
return errors.WithMessage(err, "error endorsing proposal")
|
|
}
|
|
|
|
if proposalResponse.Response == nil {
|
|
return errors.Errorf("proposal response had nil response")
|
|
}
|
|
|
|
if proposalResponse.Response.Status != int32(cb.Status_SUCCESS) {
|
|
return errors.Errorf("bad response: %d - %s", proposalResponse.Response.Status, proposalResponse.Response.Message)
|
|
}
|
|
|
|
return printResponse(getInstalledChaincodes, getInstantiatedChaincodes, proposalResponse)
|
|
}
|
|
|
|
// printResponse prints the information included in the response
|
|
// from the server. If getInstalledChaincodes is set to false, the
|
|
// proposal response will be interpreted as containing instantiated
|
|
// chaincode information.
|
|
func printResponse(getInstalledChaincodes, getInstantiatedChaincodes bool, proposalResponse *pb.ProposalResponse) error {
|
|
cqr := &pb.ChaincodeQueryResponse{}
|
|
err := proto.Unmarshal(proposalResponse.Response.Payload, cqr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if getInstalledChaincodes {
|
|
fmt.Println("Get installed chaincodes on peer:")
|
|
} else {
|
|
fmt.Printf("Get instantiated chaincodes on channel %s:\n", channelID)
|
|
}
|
|
|
|
for _, chaincode := range cqr.Chaincodes {
|
|
fmt.Printf("%v\n", ccInfo{chaincode}.String())
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type ccInfo struct {
|
|
*pb.ChaincodeInfo
|
|
}
|
|
|
|
func (cci ccInfo) String() string {
|
|
b := bytes.Buffer{}
|
|
md := reflect.ValueOf(*cci.ChaincodeInfo)
|
|
md2 := reflect.Indirect(reflect.ValueOf(*cci.ChaincodeInfo)).Type()
|
|
for i := 0; i < md.NumField(); i++ {
|
|
f := md.Field(i)
|
|
val := f.String()
|
|
if isBytes(f) {
|
|
val = hex.EncodeToString(f.Bytes())
|
|
}
|
|
if len(val) == 0 {
|
|
continue
|
|
}
|
|
// Skip the proto-internal generated fields
|
|
if strings.HasPrefix(md2.Field(i).Name, "XXX") {
|
|
continue
|
|
}
|
|
b.WriteString(fmt.Sprintf("%s: %s, ", md2.Field(i).Name, val))
|
|
}
|
|
return b.String()[:len(b.String())-2]
|
|
}
|
|
|
|
func isBytes(v reflect.Value) bool {
|
|
return v.Kind() == reflect.Slice && v.Type().Elem().Kind() == reflect.Uint8
|
|
}
|