mirror of
https://github.com/docker/compose.git
synced 2025-07-23 13:45:00 +02:00
commit
3ad1e699a2
2
.gitattributes
vendored
Normal file
2
.gitattributes
vendored
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
core.autocrlf false
|
||||||
|
*.golden text eol=lf
|
33
.github/workflows/ci.yml
vendored
33
.github/workflows/ci.yml
vendored
@ -21,7 +21,7 @@ jobs:
|
|||||||
- name: Run golangci-lint
|
- name: Run golangci-lint
|
||||||
run: |
|
run: |
|
||||||
curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b . v1.27.0
|
curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/master/install.sh | sh -s -- -b . v1.27.0
|
||||||
./golangci-lint run
|
./golangci-lint run --timeout 10m0s
|
||||||
|
|
||||||
build:
|
build:
|
||||||
name: Build
|
name: Build
|
||||||
@ -62,3 +62,34 @@ jobs:
|
|||||||
|
|
||||||
- name: E2E Test
|
- name: E2E Test
|
||||||
run: make e2e-local
|
run: make e2e-local
|
||||||
|
|
||||||
|
windows-build:
|
||||||
|
name: Windows Build
|
||||||
|
runs-on: windows-latest
|
||||||
|
env:
|
||||||
|
GO111MODULE: "on"
|
||||||
|
steps:
|
||||||
|
- name: Set up Go 1.14
|
||||||
|
uses: actions/setup-go@v1
|
||||||
|
with:
|
||||||
|
go-version: 1.14
|
||||||
|
id: go
|
||||||
|
|
||||||
|
- name: Checkout code into the Go module directory
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- uses: actions/cache@v1
|
||||||
|
with:
|
||||||
|
path: ~/go/pkg/mod
|
||||||
|
key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-go-
|
||||||
|
|
||||||
|
- name: Test
|
||||||
|
run: make -f builder.Makefile test
|
||||||
|
|
||||||
|
- name: Build
|
||||||
|
run: make -f builder.Makefile cli
|
||||||
|
|
||||||
|
- name: E2E Test
|
||||||
|
run: make e2e-win-ci
|
||||||
|
5
Makefile
5
Makefile
@ -37,7 +37,10 @@ cli: ## Compile the cli
|
|||||||
--output ./bin
|
--output ./bin
|
||||||
|
|
||||||
e2e-local: ## Run End to end local tests
|
e2e-local: ## Run End to end local tests
|
||||||
go test -v ./tests/e2e ./moby/e2e
|
go test -v ./tests/e2e ./tests/skip-win-ci-e2e ./moby/e2e
|
||||||
|
|
||||||
|
e2e-win-ci: ## Run End to end local tests on windows CI, no docker for linux containers available ATM
|
||||||
|
go test -v ./tests/e2e
|
||||||
|
|
||||||
e2e-aci: ## Run End to end ACI tests (requires azure login)
|
e2e-aci: ## Run End to end ACI tests (requires azure login)
|
||||||
go test -v ./tests/aci-e2e
|
go test -v ./tests/aci-e2e
|
||||||
|
@ -3,7 +3,6 @@ package convert
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
"path/filepath"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
@ -113,7 +112,8 @@ func (v *volumeInput) parse(name string, s string) error {
|
|||||||
v.name = name
|
v.name = name
|
||||||
v.target = volumeURL.Path
|
v.target = volumeURL.Path
|
||||||
if v.target == "" {
|
if v.target == "" {
|
||||||
v.target = filepath.Join("/run/volumes/", v.share)
|
// Do not use filepath.Join, on Windows it will replace / by \
|
||||||
|
v.target = "/run/volumes/" + v.share
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -28,13 +28,8 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -65,7 +60,7 @@ func (s *E2eSuite) TestContextDefault() {
|
|||||||
output := s.NewDockerCommand("context", "show").ExecOrDie()
|
output := s.NewDockerCommand("context", "show").ExecOrDie()
|
||||||
Expect(output).To(ContainSubstring("default"))
|
Expect(output).To(ContainSubstring("default"))
|
||||||
output = s.NewCommand("docker", "context", "ls").ExecOrDie()
|
output = s.NewCommand("docker", "context", "ls").ExecOrDie()
|
||||||
golden.Assert(s.T(), output, "ls-out-default.golden")
|
golden.Assert(s.T(), output, GoldenFile("ls-out-default"))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -106,7 +101,7 @@ func (s *E2eSuite) TestSetupError() {
|
|||||||
It("should display an error if cannot shell out to docker-classic", func() {
|
It("should display an error if cannot shell out to docker-classic", func() {
|
||||||
err := os.Setenv("PATH", s.BinDir)
|
err := os.Setenv("PATH", s.BinDir)
|
||||||
Expect(err).To(BeNil())
|
Expect(err).To(BeNil())
|
||||||
err = os.Remove(filepath.Join(s.BinDir, "docker-classic"))
|
err = os.Remove(filepath.Join(s.BinDir, DockerClassicExecutable()))
|
||||||
Expect(err).To(BeNil())
|
Expect(err).To(BeNil())
|
||||||
output, err := s.NewDockerCommand("ps").Exec()
|
output, err := s.NewDockerCommand("ps").Exec()
|
||||||
Expect(output).To(ContainSubstring("docker-classic"))
|
Expect(output).To(ContainSubstring("docker-classic"))
|
||||||
@ -115,37 +110,6 @@ func (s *E2eSuite) TestSetupError() {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *E2eSuite) TestKillChildOnCancel() {
|
|
||||||
It("should kill docker-classic if parent command is cancelled", func() {
|
|
||||||
out := s.ListProcessesCommand().ExecOrDie()
|
|
||||||
Expect(out).NotTo(ContainSubstring("docker-classic"))
|
|
||||||
|
|
||||||
dir := s.ConfigDir
|
|
||||||
Expect(ioutil.WriteFile(filepath.Join(dir, "Dockerfile"), []byte(`FROM alpine:3.10
|
|
||||||
RUN sleep 100`), 0644)).To(Succeed())
|
|
||||||
shutdown := make(chan time.Time)
|
|
||||||
errs := make(chan error)
|
|
||||||
ctx := s.NewDockerCommand("build", "--no-cache", "-t", "test-sleep-image", ".").WithinDirectory(dir).WithTimeout(shutdown)
|
|
||||||
go func() {
|
|
||||||
_, err := ctx.Exec()
|
|
||||||
errs <- err
|
|
||||||
}()
|
|
||||||
err := WaitFor(time.Second, 10*time.Second, errs, func() bool {
|
|
||||||
out := s.ListProcessesCommand().ExecOrDie()
|
|
||||||
return strings.Contains(out, "docker-classic")
|
|
||||||
})
|
|
||||||
Expect(err).NotTo(HaveOccurred())
|
|
||||||
log.Println("Killing docker process")
|
|
||||||
|
|
||||||
close(shutdown)
|
|
||||||
err = WaitFor(time.Second, 12*time.Second, nil, func() bool {
|
|
||||||
out := s.ListProcessesCommand().ExecOrDie()
|
|
||||||
return !strings.Contains(out, "docker-classic")
|
|
||||||
})
|
|
||||||
Expect(err).NotTo(HaveOccurred())
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *E2eSuite) TestLegacy() {
|
func (s *E2eSuite) TestLegacy() {
|
||||||
It("should list all legacy commands", func() {
|
It("should list all legacy commands", func() {
|
||||||
output := s.NewDockerCommand("--help").ExecOrDie()
|
output := s.NewDockerCommand("--help").ExecOrDie()
|
||||||
@ -159,7 +123,7 @@ func (s *E2eSuite) TestLegacy() {
|
|||||||
|
|
||||||
It("should run local container in less than 10 secs", func() {
|
It("should run local container in less than 10 secs", func() {
|
||||||
s.NewDockerCommand("pull", "hello-world").ExecOrDie()
|
s.NewDockerCommand("pull", "hello-world").ExecOrDie()
|
||||||
output := s.NewDockerCommand("run", "--rm", "hello-world").WithTimeout(time.NewTimer(10 * time.Second).C).ExecOrDie()
|
output := s.NewDockerCommand("run", "--rm", "hello-world").WithTimeout(time.NewTimer(20 * time.Second).C).ExecOrDie()
|
||||||
Expect(output).To(ContainSubstring("Hello from Docker!"))
|
Expect(output).To(ContainSubstring("Hello from Docker!"))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -187,7 +151,7 @@ func (s *E2eSuite) TestMockBackend() {
|
|||||||
currentContext := s.NewDockerCommand("context", "use", "test-example").ExecOrDie()
|
currentContext := s.NewDockerCommand("context", "use", "test-example").ExecOrDie()
|
||||||
Expect(currentContext).To(ContainSubstring("test-example"))
|
Expect(currentContext).To(ContainSubstring("test-example"))
|
||||||
output := s.NewDockerCommand("context", "ls").ExecOrDie()
|
output := s.NewDockerCommand("context", "ls").ExecOrDie()
|
||||||
golden.Assert(s.T(), output, "ls-out-test-example.golden")
|
golden.Assert(s.T(), output, GoldenFile("ls-out-test-example"))
|
||||||
output = s.NewDockerCommand("context", "show").ExecOrDie()
|
output = s.NewDockerCommand("context", "show").ExecOrDie()
|
||||||
Expect(output).To(ContainSubstring("test-example"))
|
Expect(output).To(ContainSubstring("test-example"))
|
||||||
})
|
})
|
||||||
@ -222,40 +186,6 @@ func (s *E2eSuite) TestMockBackend() {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *E2eSuite) TestAPIServer() {
|
|
||||||
_, err := exec.LookPath("yarn")
|
|
||||||
if err != nil || os.Getenv("SKIP_NODE") != "" {
|
|
||||||
s.T().Skip("skipping, yarn not installed")
|
|
||||||
}
|
|
||||||
It("can run 'serve' command", func() {
|
|
||||||
cName := "test-example"
|
|
||||||
s.NewDockerCommand("context", "create", cName, "example").ExecOrDie()
|
|
||||||
|
|
||||||
sPath := fmt.Sprintf("unix:///%s/docker.sock", s.ConfigDir)
|
|
||||||
server, err := serveAPI(s.ConfigDir, sPath)
|
|
||||||
Expect(err).To(BeNil())
|
|
||||||
defer killProcess(server)
|
|
||||||
|
|
||||||
s.NewCommand("yarn", "install").WithinDirectory("../node-client").ExecOrDie()
|
|
||||||
output := s.NewCommand("yarn", "run", "start", cName, sPath).WithinDirectory("../node-client").ExecOrDie()
|
|
||||||
Expect(output).To(ContainSubstring("nginx"))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestE2e(t *testing.T) {
|
func TestE2e(t *testing.T) {
|
||||||
suite.Run(t, new(E2eSuite))
|
suite.Run(t, new(E2eSuite))
|
||||||
}
|
}
|
||||||
|
|
||||||
func killProcess(process *os.Process) {
|
|
||||||
err := process.Kill()
|
|
||||||
Expect(err).To(BeNil())
|
|
||||||
}
|
|
||||||
|
|
||||||
func serveAPI(configDir string, address string) (*os.Process, error) {
|
|
||||||
cmd := exec.Command("../../bin/docker", "--config", configDir, "serve", "--address", address)
|
|
||||||
err := cmd.Start()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return cmd.Process, nil
|
|
||||||
}
|
|
||||||
|
2
tests/e2e/testdata/ls-out-default-windows.golden
vendored
Normal file
2
tests/e2e/testdata/ls-out-default-windows.golden
vendored
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
NAME TYPE DESCRIPTION DOCKER ENPOINT KUBERNETES ENDPOINT ORCHESTRATOR
|
||||||
|
default * docker Current DOCKER_HOST based configuration npipe:////./pipe/docker_engine swarm
|
3
tests/e2e/testdata/ls-out-test-example-windows.golden
vendored
Normal file
3
tests/e2e/testdata/ls-out-test-example-windows.golden
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
NAME TYPE DESCRIPTION DOCKER ENPOINT KUBERNETES ENDPOINT ORCHESTRATOR
|
||||||
|
default docker Current DOCKER_HOST based configuration npipe:////./pipe/docker_engine swarm
|
||||||
|
test-example * example
|
@ -28,6 +28,7 @@
|
|||||||
package framework
|
package framework
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/robpike/filter"
|
"github.com/robpike/filter"
|
||||||
@ -48,6 +49,19 @@ func Columns(line string) []string {
|
|||||||
return filter.Choose(strings.Split(line, " "), nonEmptyString).([]string)
|
return filter.Choose(strings.Split(line, " "), nonEmptyString).([]string)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GoldenFile golden file specific to platform
|
||||||
|
func GoldenFile(name string) string {
|
||||||
|
if IsWindows() {
|
||||||
|
return name + "-windows.golden"
|
||||||
|
}
|
||||||
|
return name + ".golden"
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsWindows windows or other GOOS
|
||||||
|
func IsWindows() bool {
|
||||||
|
return runtime.GOOS == "windows"
|
||||||
|
}
|
||||||
|
|
||||||
// It runs func
|
// It runs func
|
||||||
func It(description string, test func()) {
|
func It(description string, test func()) {
|
||||||
test()
|
test()
|
||||||
|
@ -33,7 +33,6 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"runtime"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/onsi/gomega"
|
"github.com/onsi/gomega"
|
||||||
@ -56,13 +55,14 @@ func (s *Suite) SetupSuite() {
|
|||||||
log.Error(message)
|
log.Error(message)
|
||||||
cp := filepath.Join(s.ConfigDir, "config.json")
|
cp := filepath.Join(s.ConfigDir, "config.json")
|
||||||
d, _ := ioutil.ReadFile(cp)
|
d, _ := ioutil.ReadFile(cp)
|
||||||
|
fmt.Printf("Bin dir:%s\n", s.BinDir)
|
||||||
fmt.Printf("Contents of %s:\n%s\n\nContents of config dir:\n", cp, string(d))
|
fmt.Printf("Contents of %s:\n%s\n\nContents of config dir:\n", cp, string(d))
|
||||||
for _, p := range dirContents(s.ConfigDir) {
|
for _, p := range dirContents(s.ConfigDir) {
|
||||||
fmt.Println(p)
|
fmt.Println(p)
|
||||||
}
|
}
|
||||||
s.T().Fail()
|
s.T().Fail()
|
||||||
})
|
})
|
||||||
s.linkClassicDocker()
|
s.copyExecutablesInBinDir()
|
||||||
}
|
}
|
||||||
|
|
||||||
// TearDownSuite is run after all tests
|
// TearDownSuite is run after all tests
|
||||||
@ -79,22 +79,42 @@ func dirContents(dir string) []string {
|
|||||||
return res
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Suite) linkClassicDocker() {
|
func (s *Suite) copyExecutablesInBinDir() {
|
||||||
p, err := exec.LookPath("docker-classic")
|
p, err := exec.LookPath(DockerClassicExecutable())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
p, err = exec.LookPath("docker")
|
p, err = exec.LookPath(dockerExecutable())
|
||||||
}
|
}
|
||||||
gomega.Expect(err).To(gomega.BeNil())
|
gomega.Expect(err).To(gomega.BeNil())
|
||||||
err = os.Symlink(p, filepath.Join(s.BinDir, "docker-classic"))
|
err = copyFile(p, filepath.Join(s.BinDir, DockerClassicExecutable()))
|
||||||
gomega.Expect(err).To(gomega.BeNil())
|
gomega.Expect(err).To(gomega.BeNil())
|
||||||
dockerPath, err := filepath.Abs("../../bin/docker")
|
dockerPath, err := filepath.Abs("../../bin/" + dockerExecutable())
|
||||||
gomega.Expect(err).To(gomega.BeNil())
|
gomega.Expect(err).To(gomega.BeNil())
|
||||||
err = os.Symlink(dockerPath, filepath.Join(s.BinDir, "docker"))
|
err = copyFile(dockerPath, filepath.Join(s.BinDir, dockerExecutable()))
|
||||||
gomega.Expect(err).To(gomega.BeNil())
|
gomega.Expect(err).To(gomega.BeNil())
|
||||||
err = os.Setenv("PATH", fmt.Sprintf("%s:%s", s.BinDir, os.Getenv("PATH")))
|
err = os.Setenv("PATH", concatenatePath(s.BinDir))
|
||||||
gomega.Expect(err).To(gomega.BeNil())
|
gomega.Expect(err).To(gomega.BeNil())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func concatenatePath(path string) string {
|
||||||
|
if IsWindows() {
|
||||||
|
return fmt.Sprintf("%s;%s", path, os.Getenv("PATH"))
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s:%s", path, os.Getenv("PATH"))
|
||||||
|
}
|
||||||
|
|
||||||
|
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
|
||||||
|
}
|
||||||
|
|
||||||
// BeforeTest is run before each test
|
// BeforeTest is run before each test
|
||||||
func (s *Suite) BeforeTest(suite, test string) {
|
func (s *Suite) BeforeTest(suite, test string) {
|
||||||
d, _ := ioutil.TempDir("", "")
|
d, _ := ioutil.TempDir("", "")
|
||||||
@ -109,7 +129,7 @@ func (s *Suite) AfterTest(suite, test string) {
|
|||||||
|
|
||||||
// ListProcessesCommand creates a command to list processes, "tasklist" on windows, "ps" otherwise.
|
// ListProcessesCommand creates a command to list processes, "tasklist" on windows, "ps" otherwise.
|
||||||
func (s *Suite) ListProcessesCommand() *CmdContext {
|
func (s *Suite) ListProcessesCommand() *CmdContext {
|
||||||
if runtime.GOOS == "windows" {
|
if IsWindows() {
|
||||||
return s.NewCommand("tasklist")
|
return s.NewCommand("tasklist")
|
||||||
}
|
}
|
||||||
return s.NewCommand("ps")
|
return s.NewCommand("ps")
|
||||||
@ -125,12 +145,20 @@ func (s *Suite) NewCommand(command string, args ...string) *CmdContext {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func dockerExecutable() string {
|
func dockerExecutable() string {
|
||||||
if runtime.GOOS == "windows" {
|
if IsWindows() {
|
||||||
return "docker.exe"
|
return "docker.exe"
|
||||||
}
|
}
|
||||||
return "docker"
|
return "docker"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DockerClassicExecutable binary name based on platform
|
||||||
|
func DockerClassicExecutable() string {
|
||||||
|
if IsWindows() {
|
||||||
|
return "docker-classic.exe"
|
||||||
|
}
|
||||||
|
return "docker-classic"
|
||||||
|
}
|
||||||
|
|
||||||
// NewDockerCommand creates a docker builder.
|
// NewDockerCommand creates a docker builder.
|
||||||
func (s *Suite) NewDockerCommand(args ...string) *CmdContext {
|
func (s *Suite) NewDockerCommand(args ...string) *CmdContext {
|
||||||
return s.NewCommand(dockerExecutable(), args...)
|
return s.NewCommand(dockerExecutable(), args...)
|
||||||
|
127
tests/skip-win-ci-e2e/skip_win_ci_test.go
Normal file
127
tests/skip-win-ci-e2e/skip_win_ci_test.go
Normal file
@ -0,0 +1,127 @@
|
|||||||
|
/*
|
||||||
|
Copyright (c) 2020 Docker Inc.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person
|
||||||
|
obtaining a copy of this software and associated documentation
|
||||||
|
files (the "Software"), to deal in the Software without
|
||||||
|
restriction, including without limitation the rights to use, copy,
|
||||||
|
modify, merge, publish, distribute, sublicense, and/or sell copies
|
||||||
|
of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED,
|
||||||
|
INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||||||
|
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
||||||
|
HOLDERS BE LIABLE FOR ANY CLAIM,
|
||||||
|
DAMAGES OR OTHER LIABILITY,
|
||||||
|
WHETHER IN AN ACTION OF CONTRACT,
|
||||||
|
TORT OR OTHERWISE,
|
||||||
|
ARISING FROM, OUT OF OR IN CONNECTION WITH
|
||||||
|
THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
. "github.com/onsi/gomega"
|
||||||
|
"github.com/stretchr/testify/suite"
|
||||||
|
|
||||||
|
. "github.com/docker/api/tests/framework"
|
||||||
|
)
|
||||||
|
|
||||||
|
type NonWinCIE2eSuite struct {
|
||||||
|
Suite
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NonWinCIE2eSuite) TestKillChildOnCancel() {
|
||||||
|
It("should kill docker-classic if parent command is cancelled", func() {
|
||||||
|
out := s.ListProcessesCommand().ExecOrDie()
|
||||||
|
Expect(out).NotTo(ContainSubstring("docker-classic"))
|
||||||
|
|
||||||
|
dir := s.ConfigDir
|
||||||
|
Expect(ioutil.WriteFile(filepath.Join(dir, "Dockerfile"), []byte(`FROM alpine:3.10
|
||||||
|
RUN sleep 100`), 0644)).To(Succeed())
|
||||||
|
shutdown := make(chan time.Time)
|
||||||
|
errs := make(chan error)
|
||||||
|
ctx := s.NewDockerCommand("build", "--no-cache", "-t", "test-sleep-image", ".").WithinDirectory(dir).WithTimeout(shutdown)
|
||||||
|
go func() {
|
||||||
|
_, err := ctx.Exec()
|
||||||
|
errs <- err
|
||||||
|
}()
|
||||||
|
err := WaitFor(time.Second, 10*time.Second, errs, func() bool {
|
||||||
|
out := s.ListProcessesCommand().ExecOrDie()
|
||||||
|
return strings.Contains(out, "docker-classic")
|
||||||
|
})
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
log.Println("Killing docker process")
|
||||||
|
|
||||||
|
close(shutdown)
|
||||||
|
err = WaitFor(time.Second, 12*time.Second, nil, func() bool {
|
||||||
|
out := s.ListProcessesCommand().ExecOrDie()
|
||||||
|
return !strings.Contains(out, "docker-classic")
|
||||||
|
})
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NonWinCIE2eSuite) TestAPIServer() {
|
||||||
|
_, err := exec.LookPath("yarn")
|
||||||
|
if err != nil || os.Getenv("SKIP_NODE") != "" {
|
||||||
|
s.T().Skip("skipping, yarn not installed")
|
||||||
|
}
|
||||||
|
It("can run 'serve' command", func() {
|
||||||
|
cName := "test-example"
|
||||||
|
s.NewDockerCommand("context", "create", cName, "example").ExecOrDie()
|
||||||
|
|
||||||
|
//sPath := fmt.Sprintf("unix:///%s/docker.sock", s.ConfigDir)
|
||||||
|
sPath, cliAddress := s.getGrpcServerAndCLientAddress()
|
||||||
|
server, err := serveAPI(s.ConfigDir, sPath)
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
defer killProcess(server)
|
||||||
|
|
||||||
|
s.NewCommand("yarn", "install").WithinDirectory("../node-client").ExecOrDie()
|
||||||
|
output := s.NewCommand("yarn", "run", "start", cName, cliAddress).WithinDirectory("../node-client").ExecOrDie()
|
||||||
|
Expect(output).To(ContainSubstring("nginx"))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NonWinCIE2eSuite) getGrpcServerAndCLientAddress() (string, string) {
|
||||||
|
if IsWindows() {
|
||||||
|
return "npipe:////./pipe/clibackend", "unix:////./pipe/clibackend"
|
||||||
|
}
|
||||||
|
socketName := fmt.Sprintf("unix:///%s/docker.sock", s.ConfigDir)
|
||||||
|
return socketName, socketName
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestE2e(t *testing.T) {
|
||||||
|
suite.Run(t, new(NonWinCIE2eSuite))
|
||||||
|
}
|
||||||
|
|
||||||
|
func killProcess(process *os.Process) {
|
||||||
|
err := process.Kill()
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
}
|
||||||
|
|
||||||
|
func serveAPI(configDir string, address string) (*os.Process, error) {
|
||||||
|
cmd := exec.Command("../../bin/docker", "--config", configDir, "serve", "--address", address)
|
||||||
|
err := cmd.Start()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return cmd.Process, nil
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user