代码拉取完成,页面将自动刷新
/*
Copyright IBM Corp. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0
*/
package externalbuilder
import (
"encoding/json"
"io/ioutil"
"os"
"os/exec"
"path/filepath"
"syscall"
"time"
"github.com/hyperledger/fabric/core/comm"
"github.com/hyperledger/fabric/core/container/ccintf"
"github.com/pkg/errors"
)
const (
DialTimeout = 3 * time.Second
CCServerReleaseDir = "chaincode/server"
)
type Instance struct {
PackageID string
BldDir string
ReleaseDir string
Builder *Builder
Session *Session
TermTimeout time.Duration
}
// Duration used for the DialTimeout property
type Duration struct {
time.Duration
}
func (d Duration) MarshalJSON() ([]byte, error) {
return json.Marshal(d.Seconds())
}
func (d *Duration) UnmarshalJSON(b []byte) error {
var v interface{}
if err := json.Unmarshal(b, &v); err != nil {
return err
}
switch value := v.(type) {
case float64:
d.Duration = time.Duration(value)
return nil
case string:
var err error
if d.Duration, err = time.ParseDuration(value); err != nil {
return err
}
return nil
default:
return errors.New("invalid duration")
}
}
// ChaincodeServerUserData holds "connection.json" information
type ChaincodeServerUserData struct {
Address string `json:"address"`
DialTimeout Duration `json:"dial_timeout"`
TLSRequired bool `json:"tls_required"`
ClientAuthRequired bool `json:"client_auth_required"`
ClientKey string `json:"client_key"` // PEM encoded client key
ClientCert string `json:"client_cert"` // PEM encoded client certificate
RootCert string `json:"root_cert"` // PEM encoded peer chaincode certificate
}
func (c *ChaincodeServerUserData) ChaincodeServerInfo(cryptoDir string) (*ccintf.ChaincodeServerInfo, error) {
if c.Address == "" {
return nil, errors.New("chaincode address not provided")
}
connInfo := &ccintf.ChaincodeServerInfo{Address: c.Address}
if c.DialTimeout == (Duration{}) {
connInfo.ClientConfig.Timeout = DialTimeout
} else {
connInfo.ClientConfig.Timeout = c.DialTimeout.Duration
}
// we can expose this if necessary
connInfo.ClientConfig.KaOpts = comm.DefaultKeepaliveOptions
if !c.TLSRequired {
return connInfo, nil
}
if c.ClientAuthRequired && c.ClientKey == "" {
return nil, errors.New("chaincode tls key not provided")
}
if c.ClientAuthRequired && c.ClientCert == "" {
return nil, errors.New("chaincode tls cert not provided")
}
if c.RootCert == "" {
return nil, errors.New("chaincode tls root cert not provided")
}
connInfo.ClientConfig.SecOpts.UseTLS = true
if c.ClientAuthRequired {
connInfo.ClientConfig.SecOpts.RequireClientCert = true
connInfo.ClientConfig.SecOpts.Certificate = []byte(c.ClientCert)
connInfo.ClientConfig.SecOpts.Key = []byte(c.ClientKey)
}
connInfo.ClientConfig.SecOpts.ServerRootCAs = [][]byte{[]byte(c.RootCert)}
return connInfo, nil
}
func (i *Instance) ChaincodeServerReleaseDir() string {
return filepath.Join(i.ReleaseDir, CCServerReleaseDir)
}
func (i *Instance) ChaincodeServerInfo() (*ccintf.ChaincodeServerInfo, error) {
ccinfoPath := filepath.Join(i.ChaincodeServerReleaseDir(), "connection.json")
_, err := os.Stat(ccinfoPath)
if os.IsNotExist(err) {
return nil, nil
}
if err != nil {
return nil, errors.WithMessage(err, "connection information not provided")
}
b, err := ioutil.ReadFile(ccinfoPath)
if err != nil {
return nil, errors.WithMessagef(err, "could not read '%s' for chaincode info", ccinfoPath)
}
ccdata := &ChaincodeServerUserData{}
err = json.Unmarshal(b, &ccdata)
if err != nil {
return nil, errors.WithMessagef(err, "malformed chaincode info at '%s'", ccinfoPath)
}
return ccdata.ChaincodeServerInfo(i.ChaincodeServerReleaseDir())
}
func (i *Instance) Start(peerConnection *ccintf.PeerConnection) error {
sess, err := i.Builder.Run(i.PackageID, i.BldDir, peerConnection)
if err != nil {
return errors.WithMessage(err, "could not execute run")
}
i.Session = sess
return nil
}
// Stop signals the process to terminate with SIGTERM. If the process doesn't
// terminate within TermTimeout, the process is killed with SIGKILL.
func (i *Instance) Stop() error {
if i.Session == nil {
return errors.Errorf("instance has not been started")
}
done := make(chan struct{})
go func() { i.Wait(); close(done) }()
i.Session.Signal(syscall.SIGTERM)
select {
case <-time.After(i.TermTimeout):
i.Session.Signal(syscall.SIGKILL)
case <-done:
return nil
}
select {
case <-time.After(5 * time.Second):
return errors.Errorf("failed to stop instance '%s'", i.PackageID)
case <-done:
return nil
}
}
func (i *Instance) Wait() (int, error) {
if i.Session == nil {
return -1, errors.Errorf("instance was not successfully started")
}
err := i.Session.Wait()
err = errors.Wrapf(err, "builder '%s' run failed", i.Builder.Name)
if exitErr, ok := errors.Cause(err).(*exec.ExitError); ok {
return exitErr.ExitCode(), err
}
return 0, err
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。