2018-02-02 17:23:12 -05:00
|
|
|
package swarm
|
|
|
|
|
|
|
|
import (
|
2017-06-15 15:06:08 -04:00
|
|
|
"context"
|
2018-03-26 23:36:55 -04:00
|
|
|
"runtime"
|
2018-02-02 17:23:12 -05:00
|
|
|
"testing"
|
2018-03-26 23:36:55 -04:00
|
|
|
"time"
|
2018-02-02 17:23:12 -05:00
|
|
|
|
2017-06-15 15:06:08 -04:00
|
|
|
"github.com/docker/docker/api/types"
|
|
|
|
"github.com/docker/docker/api/types/filters"
|
2018-02-02 17:23:12 -05:00
|
|
|
swarmtypes "github.com/docker/docker/api/types/swarm"
|
2018-04-11 06:10:17 -04:00
|
|
|
"github.com/docker/docker/internal/test/daemon"
|
2018-02-02 17:23:12 -05:00
|
|
|
"github.com/docker/docker/internal/test/environment"
|
2018-06-11 09:32:11 -04:00
|
|
|
"gotest.tools/assert"
|
|
|
|
"gotest.tools/poll"
|
|
|
|
"gotest.tools/skip"
|
2018-02-02 17:23:12 -05:00
|
|
|
)
|
|
|
|
|
2018-03-26 23:36:55 -04:00
|
|
|
// ServicePoll tweaks the pollSettings for `service`
|
|
|
|
func ServicePoll(config *poll.Settings) {
|
|
|
|
// Override the default pollSettings for `service` resource here ...
|
2018-04-18 10:18:53 -04:00
|
|
|
config.Timeout = 30 * time.Second
|
|
|
|
config.Delay = 100 * time.Millisecond
|
2018-03-26 23:36:55 -04:00
|
|
|
if runtime.GOARCH == "arm64" || runtime.GOARCH == "arm" {
|
2018-05-17 03:42:42 -04:00
|
|
|
config.Timeout = 90 * time.Second
|
2018-03-26 23:36:55 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NetworkPoll tweaks the pollSettings for `network`
|
|
|
|
func NetworkPoll(config *poll.Settings) {
|
|
|
|
// Override the default pollSettings for `network` resource here ...
|
|
|
|
config.Timeout = 30 * time.Second
|
|
|
|
config.Delay = 100 * time.Millisecond
|
|
|
|
|
|
|
|
if runtime.GOARCH == "arm64" || runtime.GOARCH == "arm" {
|
|
|
|
config.Timeout = 50 * time.Second
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ContainerPoll tweaks the pollSettings for `container`
|
|
|
|
func ContainerPoll(config *poll.Settings) {
|
|
|
|
// Override the default pollSettings for `container` resource here ...
|
|
|
|
|
|
|
|
if runtime.GOARCH == "arm64" || runtime.GOARCH == "arm" {
|
|
|
|
config.Timeout = 30 * time.Second
|
|
|
|
config.Delay = 100 * time.Millisecond
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-02 17:36:59 -05:00
|
|
|
// NewSwarm creates a swarm daemon for testing
|
2018-04-11 06:10:17 -04:00
|
|
|
func NewSwarm(t *testing.T, testEnv *environment.Execution, ops ...func(*daemon.Daemon)) *daemon.Daemon {
|
2018-04-18 11:16:55 -04:00
|
|
|
t.Helper()
|
2018-04-25 05:03:43 -04:00
|
|
|
skip.If(t, testEnv.IsRemoteDaemon)
|
2018-04-11 06:10:17 -04:00
|
|
|
if testEnv.DaemonInfo.ExperimentalBuild {
|
|
|
|
ops = append(ops, daemon.WithExperimental)
|
2018-02-02 17:23:12 -05:00
|
|
|
}
|
2018-04-11 06:10:17 -04:00
|
|
|
d := daemon.New(t, ops...)
|
2018-04-13 11:02:56 -04:00
|
|
|
d.StartAndSwarmInit(t)
|
2018-02-02 17:23:12 -05:00
|
|
|
return d
|
|
|
|
}
|
2017-06-15 15:06:08 -04:00
|
|
|
|
|
|
|
// ServiceSpecOpt is used with `CreateService` to pass in service spec modifiers
|
|
|
|
type ServiceSpecOpt func(*swarmtypes.ServiceSpec)
|
|
|
|
|
|
|
|
// CreateService creates a service on the passed in swarm daemon.
|
2018-04-11 06:10:17 -04:00
|
|
|
func CreateService(t *testing.T, d *daemon.Daemon, opts ...ServiceSpecOpt) string {
|
2018-04-18 11:16:55 -04:00
|
|
|
t.Helper()
|
2017-06-15 15:06:08 -04:00
|
|
|
spec := defaultServiceSpec()
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&spec)
|
|
|
|
}
|
|
|
|
|
2018-04-13 11:02:56 -04:00
|
|
|
client := d.NewClientT(t)
|
|
|
|
defer client.Close()
|
2017-06-15 15:06:08 -04:00
|
|
|
|
|
|
|
resp, err := client.ServiceCreate(context.Background(), spec, types.ServiceCreateOptions{})
|
2018-03-13 15:28:34 -04:00
|
|
|
assert.NilError(t, err, "error creating service")
|
2017-06-15 15:06:08 -04:00
|
|
|
return resp.ID
|
|
|
|
}
|
|
|
|
|
|
|
|
func defaultServiceSpec() swarmtypes.ServiceSpec {
|
|
|
|
var spec swarmtypes.ServiceSpec
|
|
|
|
ServiceWithImage("busybox:latest")(&spec)
|
|
|
|
ServiceWithCommand([]string{"/bin/top"})(&spec)
|
|
|
|
ServiceWithReplicas(1)(&spec)
|
|
|
|
return spec
|
|
|
|
}
|
|
|
|
|
2018-06-01 06:47:38 -04:00
|
|
|
// ServiceWithInit sets whether the service should use init or not
|
|
|
|
func ServiceWithInit(b *bool) func(*swarmtypes.ServiceSpec) {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
ensureContainerSpec(spec)
|
|
|
|
spec.TaskTemplate.ContainerSpec.Init = b
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-15 15:06:08 -04:00
|
|
|
// ServiceWithImage sets the image to use for the service
|
|
|
|
func ServiceWithImage(image string) func(*swarmtypes.ServiceSpec) {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
ensureContainerSpec(spec)
|
|
|
|
spec.TaskTemplate.ContainerSpec.Image = image
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ServiceWithCommand sets the command to use for the service
|
|
|
|
func ServiceWithCommand(cmd []string) ServiceSpecOpt {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
ensureContainerSpec(spec)
|
|
|
|
spec.TaskTemplate.ContainerSpec.Command = cmd
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ServiceWithConfig adds the config reference to the service
|
|
|
|
func ServiceWithConfig(configRef *swarmtypes.ConfigReference) ServiceSpecOpt {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
ensureContainerSpec(spec)
|
|
|
|
spec.TaskTemplate.ContainerSpec.Configs = append(spec.TaskTemplate.ContainerSpec.Configs, configRef)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ServiceWithSecret adds the secret reference to the service
|
|
|
|
func ServiceWithSecret(secretRef *swarmtypes.SecretReference) ServiceSpecOpt {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
ensureContainerSpec(spec)
|
|
|
|
spec.TaskTemplate.ContainerSpec.Secrets = append(spec.TaskTemplate.ContainerSpec.Secrets, secretRef)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ServiceWithReplicas sets the replicas for the service
|
|
|
|
func ServiceWithReplicas(n uint64) ServiceSpecOpt {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
spec.Mode = swarmtypes.ServiceMode{
|
|
|
|
Replicated: &swarmtypes.ReplicatedService{
|
|
|
|
Replicas: &n,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ServiceWithName sets the name of the service
|
|
|
|
func ServiceWithName(name string) ServiceSpecOpt {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
spec.Annotations.Name = name
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-05 17:26:29 -04:00
|
|
|
// ServiceWithNetwork sets the network of the service
|
|
|
|
func ServiceWithNetwork(network string) ServiceSpecOpt {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
spec.TaskTemplate.Networks = append(spec.TaskTemplate.Networks,
|
|
|
|
swarmtypes.NetworkAttachmentConfig{Target: network})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ServiceWithEndpoint sets the Endpoint of the service
|
|
|
|
func ServiceWithEndpoint(endpoint *swarmtypes.EndpointSpec) ServiceSpecOpt {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
spec.EndpointSpec = endpoint
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-22 16:24:14 -04:00
|
|
|
// ServiceWithSysctls sets the Sysctls option of the service's ContainerSpec.
|
|
|
|
func ServiceWithSysctls(sysctls map[string]string) ServiceSpecOpt {
|
|
|
|
return func(spec *swarmtypes.ServiceSpec) {
|
|
|
|
ensureContainerSpec(spec)
|
|
|
|
spec.TaskTemplate.ContainerSpec.Sysctls = sysctls
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-15 15:06:08 -04:00
|
|
|
// GetRunningTasks gets the list of running tasks for a service
|
2018-04-11 06:10:17 -04:00
|
|
|
func GetRunningTasks(t *testing.T, d *daemon.Daemon, serviceID string) []swarmtypes.Task {
|
2018-04-18 11:16:55 -04:00
|
|
|
t.Helper()
|
2018-04-13 11:02:56 -04:00
|
|
|
client := d.NewClientT(t)
|
|
|
|
defer client.Close()
|
2017-06-15 15:06:08 -04:00
|
|
|
|
|
|
|
filterArgs := filters.NewArgs()
|
|
|
|
filterArgs.Add("desired-state", "running")
|
|
|
|
filterArgs.Add("service", serviceID)
|
|
|
|
|
|
|
|
options := types.TaskListOptions{
|
|
|
|
Filters: filterArgs,
|
|
|
|
}
|
|
|
|
tasks, err := client.TaskList(context.Background(), options)
|
2018-03-13 15:28:34 -04:00
|
|
|
assert.NilError(t, err)
|
2017-06-15 15:06:08 -04:00
|
|
|
return tasks
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExecTask runs the passed in exec config on the given task
|
2018-04-11 06:10:17 -04:00
|
|
|
func ExecTask(t *testing.T, d *daemon.Daemon, task swarmtypes.Task, config types.ExecConfig) types.HijackedResponse {
|
2018-04-18 11:16:55 -04:00
|
|
|
t.Helper()
|
2018-04-13 11:02:56 -04:00
|
|
|
client := d.NewClientT(t)
|
|
|
|
defer client.Close()
|
2017-06-15 15:06:08 -04:00
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
resp, err := client.ContainerExecCreate(ctx, task.Status.ContainerStatus.ContainerID, config)
|
2018-03-13 15:28:34 -04:00
|
|
|
assert.NilError(t, err, "error creating exec")
|
2017-06-15 15:06:08 -04:00
|
|
|
|
|
|
|
startCheck := types.ExecStartCheck{}
|
|
|
|
attach, err := client.ContainerExecAttach(ctx, resp.ID, startCheck)
|
2018-03-13 15:28:34 -04:00
|
|
|
assert.NilError(t, err, "error attaching to exec")
|
2017-06-15 15:06:08 -04:00
|
|
|
return attach
|
|
|
|
}
|
|
|
|
|
|
|
|
func ensureContainerSpec(spec *swarmtypes.ServiceSpec) {
|
|
|
|
if spec.TaskTemplate.ContainerSpec == nil {
|
|
|
|
spec.TaskTemplate.ContainerSpec = &swarmtypes.ContainerSpec{}
|
|
|
|
}
|
|
|
|
}
|