2014-02-25 16:17:48 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2014-05-13 18:49:12 +00:00
|
|
|
"os"
|
2014-02-25 16:17:48 +00:00
|
|
|
"os/exec"
|
|
|
|
"path/filepath"
|
2014-05-13 18:49:12 +00:00
|
|
|
"strings"
|
2014-02-25 16:17:48 +00:00
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
2014-05-20 21:29:19 +00:00
|
|
|
func TestBuildCacheADD(t *testing.T) {
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestBuildCacheADD", "1")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testcacheadd1", ".")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
exitCode, err := runCommand(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v", err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
buildDirectory = filepath.Join(workingDirectory, "build_tests", "TestBuildCacheADD", "2")
|
|
|
|
buildCmd = exec.Command(dockerBinary, "build", "-t", "testcacheadd2", ".")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v %v", out, err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
if strings.Contains(out, "Using cache") {
|
|
|
|
t.Fatal("2nd build used cache on ADD, it shouldn't")
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteImages("testcacheadd1")
|
|
|
|
deleteImages("testcacheadd2")
|
|
|
|
|
|
|
|
logDone("build - build two images with ADD")
|
|
|
|
}
|
|
|
|
|
2014-02-25 16:17:48 +00:00
|
|
|
func TestBuildSixtySteps(t *testing.T) {
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestBuildSixtySteps")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "foobuildsixtysteps", ".")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v %v", out, err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
2014-04-04 00:22:32 +00:00
|
|
|
deleteImages("foobuildsixtysteps")
|
2014-02-25 16:17:48 +00:00
|
|
|
|
|
|
|
logDone("build - build an image with sixty build steps")
|
|
|
|
}
|
|
|
|
|
2014-04-09 14:21:22 +00:00
|
|
|
func TestAddSingleFileToRoot(t *testing.T) {
|
2014-05-21 18:29:11 +00:00
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestAdd", "SingleFileToRoot")
|
|
|
|
f, err := os.OpenFile(filepath.Join(buildDirectory, "test_file"), os.O_CREATE, 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
f.Close()
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testaddimg", ".")
|
2014-04-09 14:21:22 +00:00
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v %v", out, err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteImages("testaddimg")
|
|
|
|
|
|
|
|
logDone("build - add single file to root")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAddSingleFileToExistDir(t *testing.T) {
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestAdd")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testaddimg", "SingleFileToExistDir")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v %v", out, err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteImages("testaddimg")
|
|
|
|
|
|
|
|
logDone("build - add single file to existing dir")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAddSingleFileToNonExistDir(t *testing.T) {
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestAdd")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testaddimg", "SingleFileToNonExistDir")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v %v", out, err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteImages("testaddimg")
|
|
|
|
|
|
|
|
logDone("build - add single file to non-existing dir")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAddDirContentToRoot(t *testing.T) {
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestAdd")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testaddimg", "DirContentToRoot")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v %v", out, err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteImages("testaddimg")
|
|
|
|
|
|
|
|
logDone("build - add directory contents to root")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAddDirContentToExistDir(t *testing.T) {
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestAdd")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testaddimg", "DirContentToExistDir")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v %v", out, err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteImages("testaddimg")
|
|
|
|
|
|
|
|
logDone("build - add directory contents to existing dir")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAddWholeDirToRoot(t *testing.T) {
|
2014-05-21 18:29:11 +00:00
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestAdd", "WholeDirToRoot")
|
|
|
|
test_dir := filepath.Join(buildDirectory, "test_dir")
|
|
|
|
if err := os.MkdirAll(test_dir, 0755); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
f, err := os.OpenFile(filepath.Join(test_dir, "test_file"), os.O_CREATE, 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
f.Close()
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testaddimg", ".")
|
2014-04-09 14:21:22 +00:00
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v %v", out, err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteImages("testaddimg")
|
|
|
|
|
|
|
|
logDone("build - add whole directory to root")
|
|
|
|
}
|
|
|
|
|
2014-05-20 20:39:46 +00:00
|
|
|
func TestAddEtcToRoot(t *testing.T) {
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestAdd")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testaddimg", "EtcToRoot")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
errorOut(err, t, fmt.Sprintf("build failed to complete: %v %v", out, err))
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteImages("testaddimg")
|
|
|
|
logDone("build - add etc directory to root")
|
|
|
|
}
|
|
|
|
|
2014-05-13 18:49:12 +00:00
|
|
|
// Issue #5270 - ensure we throw a better error than "unexpected EOF"
|
|
|
|
// when we can't access files in the context.
|
|
|
|
func TestBuildWithInaccessibleFilesInContext(t *testing.T) {
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestBuildWithInaccessibleFilesInContext")
|
|
|
|
|
|
|
|
{
|
|
|
|
// This is used to ensure we detect inaccessible files early during build in the cli client
|
|
|
|
pathToInaccessibleFileBuildDirectory := filepath.Join(buildDirectory, "inaccessiblefile")
|
|
|
|
pathToFileWithoutReadAccess := filepath.Join(pathToInaccessibleFileBuildDirectory, "fileWithoutReadAccess")
|
|
|
|
|
2014-05-19 20:55:28 +00:00
|
|
|
err := os.Chown(pathToFileWithoutReadAccess, 0, 0)
|
2014-05-13 18:49:12 +00:00
|
|
|
errorOut(err, t, fmt.Sprintf("failed to chown file to root: %s", err))
|
|
|
|
err = os.Chmod(pathToFileWithoutReadAccess, 0700)
|
|
|
|
errorOut(err, t, fmt.Sprintf("failed to chmod file to 700: %s", err))
|
|
|
|
|
|
|
|
buildCommandStatement := fmt.Sprintf("%s build -t inaccessiblefiles .", dockerBinary)
|
|
|
|
buildCmd := exec.Command("su", "unprivilegeduser", "-c", buildCommandStatement)
|
|
|
|
buildCmd.Dir = pathToInaccessibleFileBuildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
if err == nil || exitCode == 0 {
|
|
|
|
t.Fatalf("build should have failed: %s %s", err, out)
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if we've detected the failure before we started building
|
|
|
|
if !strings.Contains(out, "no permission to read from ") {
|
2014-05-21 18:29:11 +00:00
|
|
|
t.Fatalf("output should've contained the string: no permission to read from but contained: %s", out)
|
2014-05-13 18:49:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.Contains(out, "Error checking context is accessible") {
|
|
|
|
t.Fatalf("output should've contained the string: Error checking context is accessible")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
{
|
|
|
|
// This is used to ensure we detect inaccessible directories early during build in the cli client
|
|
|
|
pathToInaccessibleDirectoryBuildDirectory := filepath.Join(buildDirectory, "inaccessibledirectory")
|
|
|
|
pathToDirectoryWithoutReadAccess := filepath.Join(pathToInaccessibleDirectoryBuildDirectory, "directoryWeCantStat")
|
|
|
|
pathToFileInDirectoryWithoutReadAccess := filepath.Join(pathToDirectoryWithoutReadAccess, "bar")
|
|
|
|
|
2014-05-19 20:55:28 +00:00
|
|
|
err := os.Chown(pathToDirectoryWithoutReadAccess, 0, 0)
|
2014-05-13 18:49:12 +00:00
|
|
|
errorOut(err, t, fmt.Sprintf("failed to chown directory to root: %s", err))
|
|
|
|
err = os.Chmod(pathToDirectoryWithoutReadAccess, 0444)
|
|
|
|
errorOut(err, t, fmt.Sprintf("failed to chmod directory to 755: %s", err))
|
|
|
|
err = os.Chmod(pathToFileInDirectoryWithoutReadAccess, 0700)
|
|
|
|
errorOut(err, t, fmt.Sprintf("failed to chmod file to 444: %s", err))
|
|
|
|
|
|
|
|
buildCommandStatement := fmt.Sprintf("%s build -t inaccessiblefiles .", dockerBinary)
|
|
|
|
buildCmd := exec.Command("su", "unprivilegeduser", "-c", buildCommandStatement)
|
|
|
|
buildCmd.Dir = pathToInaccessibleDirectoryBuildDirectory
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
if err == nil || exitCode == 0 {
|
|
|
|
t.Fatalf("build should have failed: %s %s", err, out)
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if we've detected the failure before we started building
|
|
|
|
if !strings.Contains(out, "can't stat") {
|
|
|
|
t.Fatalf("output should've contained the string: can't access %s", out)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.Contains(out, "Error checking context is accessible") {
|
|
|
|
t.Fatalf("output should've contained the string: Error checking context is accessible")
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
{
|
|
|
|
// This is used to ensure we don't follow links when checking if everything in the context is accessible
|
|
|
|
// This test doesn't require that we run commands as an unprivileged user
|
|
|
|
pathToDirectoryWhichContainsLinks := filepath.Join(buildDirectory, "linksdirectory")
|
|
|
|
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testlinksok", ".")
|
|
|
|
buildCmd.Dir = pathToDirectoryWhichContainsLinks
|
|
|
|
out, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatalf("build should have worked: %s %s", err, out)
|
|
|
|
}
|
|
|
|
|
|
|
|
deleteImages("testlinksok")
|
|
|
|
|
|
|
|
}
|
|
|
|
deleteImages("inaccessiblefiles")
|
|
|
|
logDone("build - ADD from context with inaccessible files must fail")
|
|
|
|
logDone("build - ADD from context with accessible links must work")
|
|
|
|
}
|
|
|
|
|
2014-05-19 16:46:25 +00:00
|
|
|
func TestBuildForceRm(t *testing.T) {
|
|
|
|
containerCountBefore, err := getContainerCount()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to get the container count: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestBuildForceRm")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "--force-rm", ".")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
_, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
|
|
|
|
if err == nil || exitCode == 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
containerCountAfter, err := getContainerCount()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to get the container count: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if containerCountBefore != containerCountAfter {
|
|
|
|
t.Fatalf("--force-rm shouldn't have left containers behind")
|
|
|
|
}
|
|
|
|
|
|
|
|
logDone("build - ensure --force-rm doesn't leave containers behind")
|
|
|
|
}
|
|
|
|
|
2014-05-19 18:21:25 +00:00
|
|
|
func TestBuildRm(t *testing.T) {
|
|
|
|
{
|
|
|
|
containerCountBefore, err := getContainerCount()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to get the container count: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestBuildRm")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "--rm", "-t", "testbuildrm", ".")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
_, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
containerCountAfter, err := getContainerCount()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to get the container count: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if containerCountBefore != containerCountAfter {
|
|
|
|
t.Fatalf("-rm shouldn't have left containers behind")
|
|
|
|
}
|
|
|
|
deleteImages("testbuildrm")
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
containerCountBefore, err := getContainerCount()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to get the container count: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestBuildRm")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "-t", "testbuildrm", ".")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
_, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
containerCountAfter, err := getContainerCount()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to get the container count: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if containerCountBefore != containerCountAfter {
|
|
|
|
t.Fatalf("--rm shouldn't have left containers behind")
|
|
|
|
}
|
|
|
|
deleteImages("testbuildrm")
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
containerCountBefore, err := getContainerCount()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to get the container count: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
buildDirectory := filepath.Join(workingDirectory, "build_tests", "TestBuildRm")
|
|
|
|
buildCmd := exec.Command(dockerBinary, "build", "--rm=false", "-t", "testbuildrm", ".")
|
|
|
|
buildCmd.Dir = buildDirectory
|
|
|
|
_, exitCode, err := runCommandWithOutput(buildCmd)
|
|
|
|
|
|
|
|
if err != nil || exitCode != 0 {
|
|
|
|
t.Fatal("failed to build the image")
|
|
|
|
}
|
|
|
|
|
|
|
|
containerCountAfter, err := getContainerCount()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to get the container count: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if containerCountBefore == containerCountAfter {
|
|
|
|
t.Fatalf("--rm=false should have left containers behind")
|
|
|
|
}
|
|
|
|
deleteAllContainers()
|
|
|
|
deleteImages("testbuildrm")
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
logDone("build - ensure --rm doesn't leave containers behind and that --rm=true is the default")
|
|
|
|
logDone("build - ensure --rm=false overrides the default")
|
|
|
|
}
|
|
|
|
|
2014-02-25 16:17:48 +00:00
|
|
|
// TODO: TestCaching
|
|
|
|
|
|
|
|
// TODO: TestADDCacheInvalidation
|