package dockertest /* Copyright 2014 The Camlistore Authors 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. */ import ( "bytes" "encoding/json" "errors" "fmt" "log" "math/rand" "os/exec" "regexp" "strings" "time" // Import postgres driver _ "github.com/lib/pq" "github.com/pborman/uuid" ) /// runLongTest checks all the conditions for running a docker container // based on image. func runLongTest(image string) error { DockerMachineAvailable = false if haveDockerMachine() && UseDockerMachine != "" { DockerMachineAvailable = true if !startDockerMachine() { log.Printf(`Starting docker machine "%s" failed. This could be because the image is already running or because the image does not exist. Tests will fail if the image does not exist.`, DockerMachineName) } } else if !haveDocker() { return errors.New("Neither 'docker' nor 'docker-machine' available on this system.") } if ok, err := HaveImage(image); !ok || err != nil { if err != nil { return fmt.Errorf("Error checking for docker image %s: %v", image, err) } log.Printf("Pulling docker image %s ...", image) if err := Pull(image); err != nil { return fmt.Errorf("Error pulling %s: %v", image, err) } } return nil } func runDockerCommand(command string, args ...string) *exec.Cmd { if DockerMachineAvailable { command = "/usr/local/bin/" + strings.Join(append([]string{command}, args...), " ") cmd := exec.Command("docker-machine", "ssh", DockerMachineName, command) return cmd } return exec.Command(command, args...) } // haveDockerMachine returns whether the "docker" command was found. func haveDockerMachine() bool { _, err := exec.LookPath("docker-machine") return err == nil } // startDockerMachine starts the docker machine and returns false if the command failed to execute func startDockerMachine() bool { _, err := exec.Command("docker-machine", "start", DockerMachineName).Output() return err == nil } // haveDocker returns whether the "docker" command was found. func haveDocker() bool { _, err := exec.LookPath("docker") return err == nil } type dockerImage struct { repo string tag string } type dockerImageList []dockerImage func (l dockerImageList) contains(repo string, tag string) bool { if tag == "" { tag = "latest" } for _, image := range l { if image.repo == repo && image.tag == tag { return true } } return false } func parseDockerImagesOutput(data []byte) (images dockerImageList) { lines := strings.Split(string(data), "\n") if len(lines) < 2 { return } // skip first line with columns names images = make(dockerImageList, 0, len(lines)-1) for _, line := range lines[1:] { cols := strings.Fields(line) if len(cols) < 2 { continue } image := dockerImage{ repo: cols[0], tag: cols[1], } images = append(images, image) } return } func parseImageName(name string) (repo string, tag string) { if fields := strings.SplitN(name, ":", 2); len(fields) == 2 { repo, tag = fields[0], fields[1] } else { repo = name } return } // HaveImage reports if docker have image 'name'. func HaveImage(name string) (bool, error) { out, err := runDockerCommand("docker", "images", "--no-trunc").Output() if err != nil { return false, err } repo, tag := parseImageName(name) images := parseDockerImagesOutput(out) return images.contains(repo, tag), nil } func run(args ...string) (containerID string, err error) { var stdout, stderr bytes.Buffer validID := regexp.MustCompile(`^([a-zA-Z0-9]+)$`) cmd := runDockerCommand("docker", append([]string{"run"}, args...)...) cmd.Stdout, cmd.Stderr = &stdout, &stderr if err = cmd.Run(); err != nil { err = fmt.Errorf("Error running docker\nStdOut: %s\nStdErr: %s\nError: %v\n\n", stdout.String(), stderr.String(), err) return } containerID = strings.TrimSpace(string(stdout.String())) if !validID.MatchString(containerID) { return "", fmt.Errorf("Error running docker: %s", containerID) } if containerID == "" { return "", errors.New("Unexpected empty output from `docker run`") } return containerID, nil } // KillContainer runs docker kill on a container. func KillContainer(container string) error { if container != "" { return runDockerCommand("docker", "kill", container).Run() } return nil } // StartContainer runs 'docker start' on a container. func StartContainer(container string) error { if container != "" { return runDockerCommand("docker", "start", container).Run() } return nil } // StopContainer runs 'docker stop' on a container. func StopContainer(container string) error { if container != "" { return runDockerCommand("docker", "stop", container).Run() } return nil } // Pull retrieves the docker image with 'docker pull'. func Pull(image string) error { out, err := runDockerCommand("docker", "pull", image).CombinedOutput() if err != nil { err = fmt.Errorf("%v: %s", err, out) } return err } // IP returns the IP address of the container. func IP(containerID string) (string, error) { out, err := runDockerCommand("docker", "inspect", containerID).Output() if err != nil { return "", err } type networkSettings struct { IPAddress string } type container struct { NetworkSettings networkSettings } var c []container if err := json.NewDecoder(bytes.NewReader(out)).Decode(&c); err != nil { return "", err } if len(c) == 0 { return "", errors.New("no output from docker inspect") } if ip := c[0].NetworkSettings.IPAddress; ip != "" { return ip, nil } return "", errors.New("could not find an IP. Not running?") } // SetupMultiportContainer sets up a container, using the start function to run the given image. // It also looks up the IP address of the container, and tests this address with the given // ports and timeout. It returns the container ID and its IP address, or makes the test // fail on error. func SetupMultiportContainer(image string, ports []int, timeout time.Duration, start func() (string, error)) (c ContainerID, ip string, err error) { err = runLongTest(image) if err != nil { return "", "", err } containerID, err := start() if err != nil { return "", "", err } c = ContainerID(containerID) ip, err = c.lookup(ports, timeout) if err != nil { c.KillRemove() return "", "", err } return c, ip, nil } // SetupContainer sets up a container, using the start function to run the given image. // It also looks up the IP address of the container, and tests this address with the given // port and timeout. It returns the container ID and its IP address, or makes the test // fail on error. func SetupContainer(image string, port int, timeout time.Duration, start func() (string, error)) (c ContainerID, ip string, err error) { return SetupMultiportContainer(image, []int{port}, timeout, start) } // RandomPort returns a random non-priviledged port. func RandomPort() int { min := 1025 max := 65534 return min + rand.Intn(max-min) } // GenerateContainerID generated a random container id. func GenerateContainerID() string { return ContainerPrefix + uuid.New() } func init() { rand.Seed(time.Now().UTC().UnixNano()) }