2020-08-06 15:40:36 +02:00
|
|
|
/*
|
2020-09-22 12:13:00 +02:00
|
|
|
Copyright 2020 Docker Compose CLI authors
|
2020-08-06 15:40:36 +02:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package framework
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
2020-08-21 18:39:25 +02:00
|
|
|
"fmt"
|
2020-08-06 15:40:36 +02:00
|
|
|
"io/ioutil"
|
2020-09-07 13:43:16 +02:00
|
|
|
"net/http"
|
2020-08-06 15:40:36 +02:00
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"path/filepath"
|
|
|
|
"runtime"
|
2020-08-21 18:39:25 +02:00
|
|
|
"strings"
|
2020-08-06 15:40:36 +02:00
|
|
|
"testing"
|
2020-09-07 13:43:16 +02:00
|
|
|
"time"
|
2020-08-06 15:40:36 +02:00
|
|
|
|
|
|
|
"gotest.tools/v3/assert"
|
|
|
|
is "gotest.tools/v3/assert/cmp"
|
|
|
|
"gotest.tools/v3/icmd"
|
|
|
|
|
2020-09-07 13:22:08 +02:00
|
|
|
"github.com/docker/compose-cli/api/containers"
|
2020-08-06 15:40:36 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// DockerExecutableName is the OS dependent Docker CLI binary name
|
|
|
|
DockerExecutableName = "docker"
|
|
|
|
existingExectuableName = "com.docker.cli"
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
DockerExecutableName = DockerExecutableName + ".exe"
|
|
|
|
existingExectuableName = existingExectuableName + ".exe"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// E2eCLI is used to wrap the CLI for end to end testing
|
|
|
|
type E2eCLI struct {
|
|
|
|
BinDir string
|
|
|
|
ConfigDir string
|
2020-08-21 18:52:07 +02:00
|
|
|
test *testing.T
|
2020-08-06 15:40:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewParallelE2eCLI returns a configured TestE2eCLI with t.Parallel() set
|
|
|
|
func NewParallelE2eCLI(t *testing.T, binDir string) *E2eCLI {
|
|
|
|
t.Parallel()
|
|
|
|
return newE2eCLI(t, binDir)
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewE2eCLI returns a configured TestE2eCLI
|
|
|
|
func NewE2eCLI(t *testing.T, binDir string) *E2eCLI {
|
|
|
|
return newE2eCLI(t, binDir)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newE2eCLI(t *testing.T, binDir string) *E2eCLI {
|
|
|
|
d, err := ioutil.TempDir("", "")
|
|
|
|
assert.Check(t, is.Nil(err))
|
|
|
|
|
|
|
|
t.Cleanup(func() {
|
|
|
|
if t.Failed() {
|
|
|
|
conf, _ := ioutil.ReadFile(filepath.Join(d, "config.json"))
|
|
|
|
t.Errorf("Config: %s\n", string(conf))
|
|
|
|
t.Error("Contents of config dir:")
|
|
|
|
for _, p := range dirContents(d) {
|
|
|
|
t.Errorf(p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ = os.RemoveAll(d)
|
|
|
|
})
|
|
|
|
|
2020-08-21 17:34:02 +02:00
|
|
|
return &E2eCLI{binDir, d, t}
|
2020-08-06 15:40:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func dirContents(dir string) []string {
|
|
|
|
res := []string{}
|
|
|
|
_ = filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
|
|
|
res = append(res, filepath.Join(dir, path))
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetupExistingCLI copies the existing CLI in a temporary directory so that the
|
|
|
|
// new CLI can be configured to use it
|
|
|
|
func SetupExistingCLI() (string, func(), error) {
|
|
|
|
p, err := exec.LookPath(existingExectuableName)
|
|
|
|
if err != nil {
|
|
|
|
p, err = exec.LookPath(DockerExecutableName)
|
|
|
|
if err != nil {
|
|
|
|
return "", nil, errors.New("existing CLI not found in PATH")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
d, err := ioutil.TempDir("", "")
|
|
|
|
if err != nil {
|
|
|
|
return "", nil, err
|
|
|
|
}
|
|
|
|
if err := CopyFile(p, filepath.Join(d, existingExectuableName)); err != nil {
|
|
|
|
return "", nil, err
|
|
|
|
}
|
|
|
|
bin, err := filepath.Abs("../../bin/" + DockerExecutableName)
|
|
|
|
if err != nil {
|
|
|
|
return "", nil, err
|
|
|
|
}
|
|
|
|
if err := CopyFile(bin, filepath.Join(d, DockerExecutableName)); err != nil {
|
|
|
|
return "", nil, err
|
|
|
|
}
|
|
|
|
cleanup := func() {
|
|
|
|
_ = os.RemoveAll(d)
|
|
|
|
}
|
|
|
|
return d, cleanup, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CopyFile copies a file from a path to a path setting permissions to 0777
|
|
|
|
func CopyFile(sourceFile string, destinationFile string) error {
|
|
|
|
input, err := ioutil.ReadFile(sourceFile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ioutil.WriteFile(destinationFile, input, 0777)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewCmd creates a cmd object configured with the test environment set
|
|
|
|
func (c *E2eCLI) NewCmd(command string, args ...string) icmd.Cmd {
|
|
|
|
env := append(os.Environ(),
|
|
|
|
"DOCKER_CONFIG="+c.ConfigDir,
|
|
|
|
"KUBECONFIG=invalid",
|
2020-09-17 13:58:20 +02:00
|
|
|
"PATH="+c.PathEnvVar(),
|
2020-08-06 15:40:36 +02:00
|
|
|
)
|
|
|
|
return icmd.Cmd{
|
|
|
|
Command: append([]string{command}, args...),
|
|
|
|
Env: env,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewDockerCmd creates a docker cmd without running it
|
|
|
|
func (c *E2eCLI) NewDockerCmd(args ...string) icmd.Cmd {
|
|
|
|
return c.NewCmd(filepath.Join(c.BinDir, DockerExecutableName), args...)
|
|
|
|
}
|
|
|
|
|
2020-08-24 10:23:14 +02:00
|
|
|
// RunDockerOrExitError runs a docker command and returns a result
|
|
|
|
func (c *E2eCLI) RunDockerOrExitError(args ...string) *icmd.Result {
|
2020-08-21 18:52:07 +02:00
|
|
|
fmt.Printf(" [%s] docker %s\n", c.test.Name(), strings.Join(args, " "))
|
2020-08-06 15:40:36 +02:00
|
|
|
return icmd.RunCmd(c.NewDockerCmd(args...))
|
|
|
|
}
|
|
|
|
|
2020-08-24 10:23:14 +02:00
|
|
|
// RunDockerCmd runs a docker command, expects no error and returns a result
|
|
|
|
func (c *E2eCLI) RunDockerCmd(args ...string) *icmd.Result {
|
|
|
|
res := c.RunDockerOrExitError(args...)
|
2020-08-21 17:34:02 +02:00
|
|
|
res.Assert(c.test, icmd.Success)
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
2020-09-17 13:58:20 +02:00
|
|
|
// PathEnvVar returns path (os sensitive) for running test
|
|
|
|
func (c *E2eCLI) PathEnvVar() string {
|
|
|
|
path := c.BinDir + ":" + os.Getenv("PATH")
|
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
path = c.BinDir + ";" + os.Getenv("PATH")
|
|
|
|
}
|
|
|
|
return path
|
|
|
|
}
|
|
|
|
|
2020-08-06 15:40:36 +02:00
|
|
|
// GoldenFile golden file specific to platform
|
|
|
|
func GoldenFile(name string) string {
|
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
return name + "-windows.golden"
|
|
|
|
}
|
|
|
|
return name + ".golden"
|
|
|
|
}
|
|
|
|
|
|
|
|
// ParseContainerInspect parses the output of a `docker inspect` command for a
|
|
|
|
// container
|
|
|
|
func ParseContainerInspect(stdout string) (*containers.Container, error) {
|
|
|
|
var res containers.Container
|
|
|
|
rdr := bytes.NewReader([]byte(stdout))
|
|
|
|
if err := json.NewDecoder(rdr).Decode(&res); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &res, nil
|
|
|
|
}
|
2020-09-07 13:43:16 +02:00
|
|
|
|
|
|
|
// HTTPGetWithRetry performs an HTTP GET on an `endpoint`.
|
|
|
|
// In the case of an error it retries the same request after a 5 second sleep,
|
|
|
|
// returning the error if count of `tries` is reached
|
|
|
|
func HTTPGetWithRetry(endpoint string, tries int) (*http.Response, error) {
|
|
|
|
var (
|
|
|
|
r *http.Response
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
for t := 0; t < tries; t++ {
|
|
|
|
r, err = http.Get(endpoint)
|
|
|
|
if err == nil || t == tries-1 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
time.Sleep(5 * time.Second)
|
|
|
|
}
|
|
|
|
return r, err
|
|
|
|
}
|