代码拉取完成,页面将自动刷新
/*
Copyright IBM Corp. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0
*/
package runner
import (
"context"
"fmt"
"io"
"net"
"os"
"sync"
"time"
docker "github.com/fsouza/go-dockerclient"
"github.com/pkg/errors"
"github.com/tedsuo/ifrit"
)
const ZooKeeperDefaultImage = "hyperledger/fabric-zookeeper:latest"
type ZooKeeper struct {
Client *docker.Client
Image string
HostIP string
HostPort []int
ContainerPorts []docker.Port
Name string
StartTimeout time.Duration
NetworkName string
ClientPort docker.Port
LeaderPort docker.Port
PeerPort docker.Port
ZooMyID int
ZooServers string
ErrorStream io.Writer
OutputStream io.Writer
containerID string
hostAddress string
containerAddress string
address string
mutex sync.Mutex
stopped bool
}
func (z *ZooKeeper) Run(sigCh <-chan os.Signal, ready chan<- struct{}) error {
if z.Image == "" {
z.Image = ZooKeeperDefaultImage
}
if z.Name == "" {
z.Name = DefaultNamer()
}
if z.HostIP == "" {
z.HostIP = "127.0.0.1"
}
if z.ContainerPorts == nil {
if z.ClientPort == docker.Port("") {
z.ClientPort = docker.Port("2181/tcp")
}
if z.LeaderPort == docker.Port("") {
z.LeaderPort = docker.Port("3888/tcp")
}
if z.PeerPort == docker.Port("") {
z.PeerPort = docker.Port("2888/tcp")
}
z.ContainerPorts = []docker.Port{
z.ClientPort,
z.LeaderPort,
z.PeerPort,
}
}
if z.StartTimeout == 0 {
z.StartTimeout = DefaultStartTimeout
}
if z.ZooMyID == 0 {
z.ZooMyID = 1
}
if z.Client == nil {
client, err := docker.NewClientFromEnv()
if err != nil {
return err
}
z.Client = client
}
containerOptions := docker.CreateContainerOptions{
Name: z.Name,
HostConfig: &docker.HostConfig{
AutoRemove: true,
},
Config: &docker.Config{
Image: z.Image,
Env: []string{
fmt.Sprintf("ZOO_MY_ID=%d", z.ZooMyID),
fmt.Sprintf("ZOO_SERVERS=%s", z.ZooServers),
},
},
}
if z.NetworkName != "" {
nw, err := z.Client.NetworkInfo(z.NetworkName)
if err != nil {
return err
}
containerOptions.NetworkingConfig = &docker.NetworkingConfig{
EndpointsConfig: map[string]*docker.EndpointConfig{
z.NetworkName: &docker.EndpointConfig{
NetworkID: nw.ID,
},
},
}
}
container, err := z.Client.CreateContainer(containerOptions)
if err != nil {
return err
}
z.containerID = container.ID
err = z.Client.StartContainer(container.ID, nil)
if err != nil {
return err
}
defer z.Stop()
container, err = z.Client.InspectContainer(container.ID)
if err != nil {
return err
}
z.containerAddress = net.JoinHostPort(
container.NetworkSettings.IPAddress,
z.ContainerPorts[0].Port(),
)
streamCtx, streamCancel := context.WithCancel(context.Background())
defer streamCancel()
go z.streamLogs(streamCtx)
containerExit := z.wait()
ctx, cancel := context.WithTimeout(context.Background(), z.StartTimeout)
defer cancel()
select {
case <-ctx.Done():
return errors.Wrapf(ctx.Err(), "zookeeper in container %s did not start", z.containerID)
case <-containerExit:
return errors.New("container exited before ready")
default:
z.address = z.containerAddress
}
close(ready)
for {
select {
case err := <-containerExit:
return err
case <-sigCh:
if err := z.Stop(); err != nil {
return err
}
}
}
}
func (z *ZooKeeper) wait() <-chan error {
exitCh := make(chan error)
go func() {
exitCode, err := z.Client.WaitContainer(z.containerID)
if err == nil {
err = fmt.Errorf("zookeeper: process exited with %d", exitCode)
}
exitCh <- err
}()
return exitCh
}
func (z *ZooKeeper) streamLogs(ctx context.Context) error {
if z.ErrorStream == nil && z.OutputStream == nil {
return nil
}
logOptions := docker.LogsOptions{
Context: ctx,
Container: z.ContainerID(),
ErrorStream: z.ErrorStream,
OutputStream: z.OutputStream,
Stderr: z.ErrorStream != nil,
Stdout: z.OutputStream != nil,
Follow: true,
}
return z.Client.Logs(logOptions)
}
func (z *ZooKeeper) ContainerID() string {
return z.containerID
}
func (z *ZooKeeper) ContainerAddress() string {
return z.containerAddress
}
func (z *ZooKeeper) Start() error {
p := ifrit.Invoke(z)
select {
case <-p.Ready():
return nil
case err := <-p.Wait():
return err
}
}
func (z *ZooKeeper) Stop() error {
z.mutex.Lock()
if z.stopped {
z.mutex.Unlock()
return errors.Errorf("container %s already stopped", z.Name)
}
z.stopped = true
z.mutex.Unlock()
err := z.Client.StopContainer(z.containerID, 0)
if err != nil {
return err
}
_, err = z.Client.PruneVolumes(docker.PruneVolumesOptions{})
return err
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。