Fix .dockerignore for build context copies in later stages (#1447)

* Extend .dockerignore integration test with copies in later stages

.dockerignore should continue to apply when copying from the build context in later stages, but it currently doesn't

* Replace excluded global with passed along FileContext struct

This new FileContext struct allows much cleaner handling of context specific file exclusions.
The global excluded file state is no longer needed.

Additionally this also fixes the issue where excluded files aren't being applied for build context copies in later build stages.
This commit is contained in:
Tinjo Schöni 2020-10-08 21:47:14 +02:00 committed by GitHub
parent 0ef3a6b525
commit 5f4e2f1366
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
14 changed files with 150 additions and 126 deletions

View File

@ -243,7 +243,7 @@ func resolveEnvironmentBuildArgs(arguments []string, resolver func(string) strin
// copy Dockerfile to /kaniko/Dockerfile so that if it's specified in the .dockerignore // copy Dockerfile to /kaniko/Dockerfile so that if it's specified in the .dockerignore
// it won't be copied into the image // it won't be copied into the image
func copyDockerfile() error { func copyDockerfile() error {
if _, err := util.CopyFile(opts.DockerfilePath, constants.DockerfilePath, "", util.DoNotChangeUID, util.DoNotChangeGID); err != nil { if _, err := util.CopyFile(opts.DockerfilePath, constants.DockerfilePath, util.FileContext{}, util.DoNotChangeUID, util.DoNotChangeGID); err != nil {
return errors.Wrap(err, "copying dockerfile") return errors.Wrap(err, "copying dockerfile")
} }
opts.DockerfilePath = constants.DockerfilePath opts.DockerfilePath = constants.DockerfilePath

View File

@ -7,5 +7,13 @@ COPY ignore/* /foo
From base as first From base as first
COPY --from=base /foo ignore/bar COPY --from=base /foo ignore/bar
FROM first # Make sure that .dockerignore also applies for later stages
FROM scratch as base2
COPY ignore/* /foo
From base2 as second
COPY --from=base2 /foo ignore/bar
FROM scratch
COPY --from=first ignore/* /fooAnother/ COPY --from=first ignore/* /fooAnother/
COPY --from=second ignore/* /fooAnother2/

View File

@ -32,7 +32,7 @@ import (
type AddCommand struct { type AddCommand struct {
BaseCommand BaseCommand
cmd *instructions.AddCommand cmd *instructions.AddCommand
buildcontext string fileContext util.FileContext
snapshotFiles []string snapshotFiles []string
} }
@ -52,7 +52,7 @@ func (a *AddCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.Bui
return errors.Wrap(err, "getting user group from chown") return errors.Wrap(err, "getting user group from chown")
} }
srcs, dest, err := util.ResolveEnvAndWildcards(a.cmd.SourcesAndDest, a.buildcontext, replacementEnvs) srcs, dest, err := util.ResolveEnvAndWildcards(a.cmd.SourcesAndDest, a.fileContext, replacementEnvs)
if err != nil { if err != nil {
return err return err
} }
@ -64,7 +64,7 @@ func (a *AddCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.Bui
// 1. Download and copy it to the specified dest // 1. Download and copy it to the specified dest
// Else, add to the list of unresolved sources // Else, add to the list of unresolved sources
for _, src := range srcs { for _, src := range srcs {
fullPath := filepath.Join(a.buildcontext, src) fullPath := filepath.Join(a.fileContext.Root, src)
if util.IsSrcRemoteFileURL(src) { if util.IsSrcRemoteFileURL(src) {
urlDest, err := util.URLDestinationFilepath(src, dest, config.WorkingDir, replacementEnvs) urlDest, err := util.URLDestinationFilepath(src, dest, config.WorkingDir, replacementEnvs)
if err != nil { if err != nil {
@ -101,7 +101,7 @@ func (a *AddCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.Bui
SourcesAndDest: append(unresolvedSrcs, dest), SourcesAndDest: append(unresolvedSrcs, dest),
Chown: a.cmd.Chown, Chown: a.cmd.Chown,
}, },
buildcontext: a.buildcontext, fileContext: a.fileContext,
} }
if err := copyCmd.ExecuteCommand(config, buildArgs); err != nil { if err := copyCmd.ExecuteCommand(config, buildArgs); err != nil {
@ -124,7 +124,7 @@ func (a *AddCommand) String() string {
func (a *AddCommand) FilesUsedFromContext(config *v1.Config, buildArgs *dockerfile.BuildArgs) ([]string, error) { func (a *AddCommand) FilesUsedFromContext(config *v1.Config, buildArgs *dockerfile.BuildArgs) ([]string, error) {
replacementEnvs := buildArgs.ReplacementEnvs(config.Env) replacementEnvs := buildArgs.ReplacementEnvs(config.Env)
srcs, _, err := util.ResolveEnvAndWildcards(a.cmd.SourcesAndDest, a.buildcontext, replacementEnvs) srcs, _, err := util.ResolveEnvAndWildcards(a.cmd.SourcesAndDest, a.fileContext, replacementEnvs)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -137,7 +137,7 @@ func (a *AddCommand) FilesUsedFromContext(config *v1.Config, buildArgs *dockerfi
if util.IsFileLocalTarArchive(src) { if util.IsFileLocalTarArchive(src) {
continue continue
} }
fullPath := filepath.Join(a.buildcontext, src) fullPath := filepath.Join(a.fileContext.Root, src)
files = append(files, fullPath) files = append(files, fullPath)
} }

View File

@ -18,6 +18,7 @@ package commands
import ( import (
"github.com/GoogleContainerTools/kaniko/pkg/dockerfile" "github.com/GoogleContainerTools/kaniko/pkg/dockerfile"
"github.com/GoogleContainerTools/kaniko/pkg/util"
v1 "github.com/google/go-containerregistry/pkg/v1" v1 "github.com/google/go-containerregistry/pkg/v1"
"github.com/moby/buildkit/frontend/dockerfile/instructions" "github.com/moby/buildkit/frontend/dockerfile/instructions"
"github.com/pkg/errors" "github.com/pkg/errors"
@ -59,7 +60,7 @@ type DockerCommand interface {
ShouldDetectDeletedFiles() bool ShouldDetectDeletedFiles() bool
} }
func GetCommand(cmd instructions.Command, buildcontext string, useNewRun bool) (DockerCommand, error) { func GetCommand(cmd instructions.Command, fileContext util.FileContext, useNewRun bool) (DockerCommand, error) {
switch c := cmd.(type) { switch c := cmd.(type) {
case *instructions.RunCommand: case *instructions.RunCommand:
if useNewRun { if useNewRun {
@ -67,7 +68,7 @@ func GetCommand(cmd instructions.Command, buildcontext string, useNewRun bool) (
} }
return &RunCommand{cmd: c}, nil return &RunCommand{cmd: c}, nil
case *instructions.CopyCommand: case *instructions.CopyCommand:
return &CopyCommand{cmd: c, buildcontext: buildcontext}, nil return &CopyCommand{cmd: c, fileContext: fileContext}, nil
case *instructions.ExposeCommand: case *instructions.ExposeCommand:
return &ExposeCommand{cmd: c}, nil return &ExposeCommand{cmd: c}, nil
case *instructions.EnvCommand: case *instructions.EnvCommand:
@ -75,7 +76,7 @@ func GetCommand(cmd instructions.Command, buildcontext string, useNewRun bool) (
case *instructions.WorkdirCommand: case *instructions.WorkdirCommand:
return &WorkdirCommand{cmd: c}, nil return &WorkdirCommand{cmd: c}, nil
case *instructions.AddCommand: case *instructions.AddCommand:
return &AddCommand{cmd: c, buildcontext: buildcontext}, nil return &AddCommand{cmd: c, fileContext: fileContext}, nil
case *instructions.CmdCommand: case *instructions.CmdCommand:
return &CmdCommand{cmd: c}, nil return &CmdCommand{cmd: c}, nil
case *instructions.EntrypointCommand: case *instructions.EntrypointCommand:

View File

@ -39,14 +39,14 @@ var (
type CopyCommand struct { type CopyCommand struct {
BaseCommand BaseCommand
cmd *instructions.CopyCommand cmd *instructions.CopyCommand
buildcontext string fileContext util.FileContext
snapshotFiles []string snapshotFiles []string
} }
func (c *CopyCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.BuildArgs) error { func (c *CopyCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.BuildArgs) error {
// Resolve from // Resolve from
if c.cmd.From != "" { if c.cmd.From != "" {
c.buildcontext = filepath.Join(kConfig.KanikoDir, c.cmd.From) c.fileContext = util.FileContext{Root: filepath.Join(kConfig.KanikoDir, c.cmd.From)}
} }
replacementEnvs := buildArgs.ReplacementEnvs(config.Env) replacementEnvs := buildArgs.ReplacementEnvs(config.Env)
@ -55,14 +55,14 @@ func (c *CopyCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.Bu
return errors.Wrap(err, "getting user group from chown") return errors.Wrap(err, "getting user group from chown")
} }
srcs, dest, err := util.ResolveEnvAndWildcards(c.cmd.SourcesAndDest, c.buildcontext, replacementEnvs) srcs, dest, err := util.ResolveEnvAndWildcards(c.cmd.SourcesAndDest, c.fileContext, replacementEnvs)
if err != nil { if err != nil {
return errors.Wrap(err, "resolving src") return errors.Wrap(err, "resolving src")
} }
// For each source, iterate through and copy it over // For each source, iterate through and copy it over
for _, src := range srcs { for _, src := range srcs {
fullPath := filepath.Join(c.buildcontext, src) fullPath := filepath.Join(c.fileContext.Root, src)
fi, err := os.Lstat(fullPath) fi, err := os.Lstat(fullPath)
if err != nil { if err != nil {
@ -89,14 +89,14 @@ func (c *CopyCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.Bu
} }
if fi.IsDir() { if fi.IsDir() {
copiedFiles, err := util.CopyDir(fullPath, destPath, c.buildcontext, uid, gid) copiedFiles, err := util.CopyDir(fullPath, destPath, c.fileContext, uid, gid)
if err != nil { if err != nil {
return errors.Wrap(err, "copying dir") return errors.Wrap(err, "copying dir")
} }
c.snapshotFiles = append(c.snapshotFiles, copiedFiles...) c.snapshotFiles = append(c.snapshotFiles, copiedFiles...)
} else if util.IsSymlink(fi) { } else if util.IsSymlink(fi) {
// If file is a symlink, we want to copy the target file to destPath // If file is a symlink, we want to copy the target file to destPath
exclude, err := util.CopySymlink(fullPath, destPath, c.buildcontext) exclude, err := util.CopySymlink(fullPath, destPath, c.fileContext)
if err != nil { if err != nil {
return errors.Wrap(err, "copying symlink") return errors.Wrap(err, "copying symlink")
} }
@ -106,7 +106,7 @@ func (c *CopyCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.Bu
c.snapshotFiles = append(c.snapshotFiles, destPath) c.snapshotFiles = append(c.snapshotFiles, destPath)
} else { } else {
// ... Else, we want to copy over a file // ... Else, we want to copy over a file
exclude, err := util.CopyFile(fullPath, destPath, c.buildcontext, uid, gid) exclude, err := util.CopyFile(fullPath, destPath, c.fileContext, uid, gid)
if err != nil { if err != nil {
return errors.Wrap(err, "copying file") return errors.Wrap(err, "copying file")
} }
@ -130,7 +130,7 @@ func (c *CopyCommand) String() string {
} }
func (c *CopyCommand) FilesUsedFromContext(config *v1.Config, buildArgs *dockerfile.BuildArgs) ([]string, error) { func (c *CopyCommand) FilesUsedFromContext(config *v1.Config, buildArgs *dockerfile.BuildArgs) ([]string, error) {
return copyCmdFilesUsedFromContext(config, buildArgs, c.cmd, c.buildcontext) return copyCmdFilesUsedFromContext(config, buildArgs, c.cmd, c.fileContext)
} }
func (c *CopyCommand) MetadataOnly() bool { func (c *CopyCommand) MetadataOnly() bool {
@ -186,7 +186,7 @@ func resolveIfSymlink(destPath string) (string, error) {
func copyCmdFilesUsedFromContext( func copyCmdFilesUsedFromContext(
config *v1.Config, buildArgs *dockerfile.BuildArgs, cmd *instructions.CopyCommand, config *v1.Config, buildArgs *dockerfile.BuildArgs, cmd *instructions.CopyCommand,
buildcontext string, fileContext util.FileContext,
) ([]string, error) { ) ([]string, error) {
// We don't use the context if we're performing a copy --from. // We don't use the context if we're performing a copy --from.
if cmd.From != "" { if cmd.From != "" {
@ -196,7 +196,7 @@ func copyCmdFilesUsedFromContext(
replacementEnvs := buildArgs.ReplacementEnvs(config.Env) replacementEnvs := buildArgs.ReplacementEnvs(config.Env)
srcs, _, err := util.ResolveEnvAndWildcards( srcs, _, err := util.ResolveEnvAndWildcards(
cmd.SourcesAndDest, buildcontext, replacementEnvs, cmd.SourcesAndDest, fileContext, replacementEnvs,
) )
if err != nil { if err != nil {
return nil, err return nil, err
@ -204,7 +204,7 @@ func copyCmdFilesUsedFromContext(
files := []string{} files := []string{}
for _, src := range srcs { for _, src := range srcs {
fullPath := filepath.Join(buildcontext, src) fullPath := filepath.Join(fileContext.Root, src)
files = append(files, fullPath) files = append(files, fullPath)
} }

View File

@ -26,6 +26,7 @@ import (
"testing" "testing"
"github.com/GoogleContainerTools/kaniko/pkg/dockerfile" "github.com/GoogleContainerTools/kaniko/pkg/dockerfile"
"github.com/GoogleContainerTools/kaniko/pkg/util"
"github.com/GoogleContainerTools/kaniko/testutil" "github.com/GoogleContainerTools/kaniko/testutil"
v1 "github.com/google/go-containerregistry/pkg/v1" v1 "github.com/google/go-containerregistry/pkg/v1"
"github.com/moby/buildkit/frontend/dockerfile/instructions" "github.com/moby/buildkit/frontend/dockerfile/instructions"
@ -113,6 +114,7 @@ func TestCopyExecuteCmd(t *testing.T) {
Env: []string{}, Env: []string{},
WorkingDir: tempDir, WorkingDir: tempDir,
} }
fileContext := util.FileContext{Root: tempDir}
for _, test := range copyTests { for _, test := range copyTests {
t.Run(test.name, func(t *testing.T) { t.Run(test.name, func(t *testing.T) {
@ -122,7 +124,7 @@ func TestCopyExecuteCmd(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: test.sourcesAndDest, SourcesAndDest: test.sourcesAndDest,
}, },
buildcontext: tempDir, fileContext: fileContext,
} }
buildArgs := copySetUpBuildArgs() buildArgs := copySetUpBuildArgs()
@ -275,7 +277,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{srcDir, "dest"}, SourcesAndDest: []string{srcDir, "dest"},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -307,7 +309,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{filepath.Join(srcDir, "bam.txt"), "dest/"}, SourcesAndDest: []string{filepath.Join(srcDir, "bam.txt"), "dest/"},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -334,7 +336,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{filepath.Join(srcDir, "bam.txt"), "dest"}, SourcesAndDest: []string{filepath.Join(srcDir, "bam.txt"), "dest"},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -363,7 +365,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{filepath.Join(srcDir, "bam.txt"), "dest"}, SourcesAndDest: []string{filepath.Join(srcDir, "bam.txt"), "dest"},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -392,7 +394,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{filepath.Join(srcDir, "sym.link"), "dest/"}, SourcesAndDest: []string{filepath.Join(srcDir, "sym.link"), "dest/"},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -437,7 +439,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{filepath.Join(srcDir, "dead.link"), "dest/"}, SourcesAndDest: []string{filepath.Join(srcDir, "dead.link"), "dest/"},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -478,7 +480,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{"another", "dest"}, SourcesAndDest: []string{"another", "dest"},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -524,7 +526,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{srcDir, "dest"}, SourcesAndDest: []string{srcDir, "dest"},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -568,7 +570,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{"another", "dest"}, SourcesAndDest: []string{"another", "dest"},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -611,7 +613,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{srcDir, linkedDest}, SourcesAndDest: []string{srcDir, linkedDest},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -656,7 +658,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{fmt.Sprintf("%s/bam.txt", srcDir), linkedDest}, SourcesAndDest: []string{fmt.Sprintf("%s/bam.txt", srcDir), linkedDest},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -705,7 +707,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
SourcesAndDest: []string{fmt.Sprintf("%s/bam.txt", srcDir), testDir}, SourcesAndDest: []string{fmt.Sprintf("%s/bam.txt", srcDir), testDir},
Chown: "alice:group", Chown: "alice:group",
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -750,7 +752,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
SourcesAndDest: []string{fmt.Sprintf("%s/bam.txt", srcDir), testDir}, SourcesAndDest: []string{fmt.Sprintf("%s/bam.txt", srcDir), testDir},
Chown: "missing:missing", Chown: "missing:missing",
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{
@ -781,7 +783,7 @@ func TestCopyCommand_ExecuteCommand_Extended(t *testing.T) {
cmd: &instructions.CopyCommand{ cmd: &instructions.CopyCommand{
SourcesAndDest: []string{srcDir, dest}, SourcesAndDest: []string{srcDir, dest},
}, },
buildcontext: testDir, fileContext: util.FileContext{Root: testDir},
} }
cfg := &v1.Config{ cfg := &v1.Config{

View File

@ -73,6 +73,7 @@ type stageBuilder struct {
baseImageDigest string baseImageDigest string
finalCacheKey string finalCacheKey string
opts *config.KanikoOptions opts *config.KanikoOptions
fileContext util.FileContext
cmds []commands.DockerCommand cmds []commands.DockerCommand
args *dockerfile.BuildArgs args *dockerfile.BuildArgs
crossStageDeps map[int][]string crossStageDeps map[int][]string
@ -84,7 +85,7 @@ type stageBuilder struct {
} }
// newStageBuilder returns a new type stageBuilder which contains all the information required to build the stage // newStageBuilder returns a new type stageBuilder which contains all the information required to build the stage
func newStageBuilder(opts *config.KanikoOptions, stage config.KanikoStage, crossStageDeps map[int][]string, dcm map[string]string, sid map[string]string, stageNameToIdx map[string]string) (*stageBuilder, error) { func newStageBuilder(opts *config.KanikoOptions, stage config.KanikoStage, crossStageDeps map[int][]string, dcm map[string]string, sid map[string]string, stageNameToIdx map[string]string, fileContext util.FileContext) (*stageBuilder, error) {
sourceImage, err := image_util.RetrieveSourceImage(stage, opts) sourceImage, err := image_util.RetrieveSourceImage(stage, opts)
if err != nil { if err != nil {
return nil, err return nil, err
@ -117,6 +118,7 @@ func newStageBuilder(opts *config.KanikoOptions, stage config.KanikoStage, cross
snapshotter: snapshotter, snapshotter: snapshotter,
baseImageDigest: digest.String(), baseImageDigest: digest.String(),
opts: opts, opts: opts,
fileContext: fileContext,
crossStageDeps: crossStageDeps, crossStageDeps: crossStageDeps,
digestToCacheKey: dcm, digestToCacheKey: dcm,
stageIdxToDigest: sid, stageIdxToDigest: sid,
@ -127,7 +129,7 @@ func newStageBuilder(opts *config.KanikoOptions, stage config.KanikoStage, cross
} }
for _, cmd := range s.stage.Commands { for _, cmd := range s.stage.Commands {
command, err := commands.GetCommand(cmd, opts.SrcContext, opts.RunV2) command, err := commands.GetCommand(cmd, fileContext, opts.RunV2)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -187,10 +189,8 @@ func (s *stageBuilder) populateCompositeKey(command fmt.Stringer, files []string
compositeKey = s.populateCopyCmdCompositeKey(command, v.From(), compositeKey) compositeKey = s.populateCopyCmdCompositeKey(command, v.From(), compositeKey)
} }
srcCtx := s.opts.SrcContext
for _, f := range files { for _, f := range files {
if err := compositeKey.AddPath(f, srcCtx); err != nil { if err := compositeKey.AddPath(f, s.fileContext); err != nil {
return compositeKey, err return compositeKey, err
} }
} }
@ -572,7 +572,8 @@ func DoBuild(opts *config.KanikoOptions) (v1.Image, error) {
} }
stageNameToIdx := ResolveCrossStageInstructions(kanikoStages) stageNameToIdx := ResolveCrossStageInstructions(kanikoStages)
if err := util.GetExcludedFiles(opts.DockerfilePath, opts.SrcContext); err != nil { fileContext, err := util.NewFileContextFromDockerfile(opts.DockerfilePath, opts.SrcContext)
if err != nil {
return nil, err return nil, err
} }
@ -586,16 +587,14 @@ func DoBuild(opts *config.KanikoOptions) (v1.Image, error) {
} }
logrus.Infof("Built cross stage deps: %v", crossStageDependencies) logrus.Infof("Built cross stage deps: %v", crossStageDependencies)
util.IsFirstStage = true
for index, stage := range kanikoStages { for index, stage := range kanikoStages {
sb, err := newStageBuilder(opts, stage, crossStageDependencies, digestToCacheKey, stageIdxToDigest, stageNameToIdx) sb, err := newStageBuilder(opts, stage, crossStageDependencies, digestToCacheKey, stageIdxToDigest, stageNameToIdx, fileContext)
if err != nil { if err != nil {
return nil, err return nil, err
} }
if err := sb.build(); err != nil { if err := sb.build(); err != nil {
return nil, errors.Wrap(err, "error building stage") return nil, errors.Wrap(err, "error building stage")
} }
util.IsFirstStage = false
reviewConfig(stage, &sb.cf.Config) reviewConfig(stage, &sb.cf.Config)

View File

@ -31,6 +31,7 @@ import (
"github.com/GoogleContainerTools/kaniko/pkg/commands" "github.com/GoogleContainerTools/kaniko/pkg/commands"
"github.com/GoogleContainerTools/kaniko/pkg/config" "github.com/GoogleContainerTools/kaniko/pkg/config"
"github.com/GoogleContainerTools/kaniko/pkg/dockerfile" "github.com/GoogleContainerTools/kaniko/pkg/dockerfile"
"github.com/GoogleContainerTools/kaniko/pkg/util"
"github.com/GoogleContainerTools/kaniko/testutil" "github.com/GoogleContainerTools/kaniko/testutil"
"github.com/google/go-cmp/cmp" "github.com/google/go-cmp/cmp"
v1 "github.com/google/go-containerregistry/pkg/v1" v1 "github.com/google/go-containerregistry/pkg/v1"
@ -679,7 +680,7 @@ func Test_stageBuilder_populateCompositeKey(t *testing.T) {
} }
for _, tc := range testCases { for _, tc := range testCases {
t.Run(tc.description, func(t *testing.T) { t.Run(tc.description, func(t *testing.T) {
sb := &stageBuilder{opts: &config.KanikoOptions{SrcContext: "workspace"}} sb := &stageBuilder{fileContext: util.FileContext{Root: "workspace"}}
ck := CompositeCache{} ck := CompositeCache{}
ck1, err := sb.populateCompositeKey(tc.cmd1.command, []string{}, ck, tc.cmd1.args, tc.cmd1.env) ck1, err := sb.populateCompositeKey(tc.cmd1.command, []string{}, ck, tc.cmd1.args, tc.cmd1.env)
@ -720,7 +721,7 @@ func Test_stageBuilder_build(t *testing.T) {
filePath := filepath.Join(dir, file) filePath := filepath.Join(dir, file)
ch := NewCompositeCache("", "meow") ch := NewCompositeCache("", "meow")
ch.AddPath(filePath, "") ch.AddPath(filePath, util.FileContext{})
hash, err := ch.Hash() hash, err := ch.Hash()
if err != nil { if err != nil {
t.Errorf("couldn't create hash %v", err) t.Errorf("couldn't create hash %v", err)
@ -753,7 +754,7 @@ func Test_stageBuilder_build(t *testing.T) {
filePath := filepath.Join(dir, file) filePath := filepath.Join(dir, file)
ch := NewCompositeCache("", "meow") ch := NewCompositeCache("", "meow")
ch.AddPath(filePath, "") ch.AddPath(filePath, util.FileContext{})
hash, err := ch.Hash() hash, err := ch.Hash()
if err != nil { if err != nil {
t.Errorf("couldn't create hash %v", err) t.Errorf("couldn't create hash %v", err)
@ -856,7 +857,7 @@ COPY %s bar.txt
// hash1 is the read cachekey for the first layer // hash1 is the read cachekey for the first layer
expectedCacheKeys: []string{hash1}, expectedCacheKeys: []string{hash1},
pushedCacheKeys: []string{}, pushedCacheKeys: []string{},
commands: getCommands(dir, cmds), commands: getCommands(util.FileContext{Root: dir}, cmds),
} }
}(), }(),
func() testcase { func() testcase {
@ -872,7 +873,7 @@ COPY %s bar.txt
filePath := filepath.Join(dir, filename) filePath := filepath.Join(dir, filename)
ch := NewCompositeCache("", fmt.Sprintf("COPY %s bar.txt", filename)) ch := NewCompositeCache("", fmt.Sprintf("COPY %s bar.txt", filename))
ch.AddPath(filePath, "") ch.AddPath(filePath, util.FileContext{})
// copy hash // copy hash
_, err = ch.Hash() _, err = ch.Hash()
@ -932,7 +933,7 @@ RUN foobar
image: image, image: image,
expectedCacheKeys: []string{runHash}, expectedCacheKeys: []string{runHash},
pushedCacheKeys: []string{}, pushedCacheKeys: []string{},
commands: getCommands(dir, cmds), commands: getCommands(util.FileContext{Root: dir}, cmds),
} }
}(), }(),
func() testcase { func() testcase {
@ -1139,12 +1140,12 @@ func assertCacheKeys(t *testing.T, expectedCacheKeys, actualCacheKeys []string,
} }
} }
func getCommands(dir string, cmds []instructions.Command) []commands.DockerCommand { func getCommands(fileContext util.FileContext, cmds []instructions.Command) []commands.DockerCommand {
outCommands := make([]commands.DockerCommand, 0) outCommands := make([]commands.DockerCommand, 0)
for _, c := range cmds { for _, c := range cmds {
cmd, err := commands.GetCommand( cmd, err := commands.GetCommand(
c, c,
dir, fileContext,
false, false,
) )
if err != nil { if err != nil {

View File

@ -55,7 +55,7 @@ func (s *CompositeCache) Hash() (string, error) {
return util.SHA256(strings.NewReader(s.Key())) return util.SHA256(strings.NewReader(s.Key()))
} }
func (s *CompositeCache) AddPath(p, context string) error { func (s *CompositeCache) AddPath(p string, context util.FileContext) error {
sha := sha256.New() sha := sha256.New()
fi, err := os.Lstat(p) fi, err := os.Lstat(p)
if err != nil { if err != nil {
@ -70,13 +70,13 @@ func (s *CompositeCache) AddPath(p, context string) error {
// Only add the hash of this directory to the key // Only add the hash of this directory to the key
// if there is any ignored content. // if there is any ignored content.
if !empty || !util.ExcludeFile(p, context) { if !empty || !context.ExcludesFile(p) {
s.keys = append(s.keys, k) s.keys = append(s.keys, k)
} }
return nil return nil
} }
if util.ExcludeFile(p, context) { if context.ExcludesFile(p) {
return nil return nil
} }
fh, err := util.CacheHasher()(p) fh, err := util.CacheHasher()(p)
@ -92,14 +92,14 @@ func (s *CompositeCache) AddPath(p, context string) error {
} }
// HashDir returns a hash of the directory. // HashDir returns a hash of the directory.
func hashDir(p, context string) (bool, string, error) { func hashDir(p string, context util.FileContext) (bool, string, error) {
sha := sha256.New() sha := sha256.New()
empty := true empty := true
if err := filepath.Walk(p, func(path string, fi os.FileInfo, err error) error { if err := filepath.Walk(p, func(path string, fi os.FileInfo, err error) error {
if err != nil { if err != nil {
return err return err
} }
exclude := util.ExcludeFile(path, context) exclude := context.ExcludesFile(path)
if exclude { if exclude {
return nil return nil
} }

View File

@ -80,7 +80,7 @@ func Test_CompositeCache_AddPath_dir(t *testing.T) {
fn := func() string { fn := func() string {
r := NewCompositeCache() r := NewCompositeCache()
if err := r.AddPath(tmpDir, ""); err != nil { if err := r.AddPath(tmpDir, util.FileContext{}); err != nil {
t.Errorf("expected error to be nil but was %v", err) t.Errorf("expected error to be nil but was %v", err)
} }
@ -118,7 +118,7 @@ func Test_CompositeCache_AddPath_file(t *testing.T) {
p := tmpfile.Name() p := tmpfile.Name()
fn := func() string { fn := func() string {
r := NewCompositeCache() r := NewCompositeCache()
if err := r.AddPath(p, ""); err != nil { if err := r.AddPath(p, util.FileContext{}); err != nil {
t.Errorf("expected error to be nil but was %v", err) t.Errorf("expected error to be nil but was %v", err)
} }
@ -158,26 +158,23 @@ func createFilesystemStructure(root string, directories, files []string) error {
return nil return nil
} }
func setIgnoreContext(content string) error { func setIgnoreContext(content string) (util.FileContext, error) {
var fileContext util.FileContext
dockerIgnoreDir, err := ioutil.TempDir("", "") dockerIgnoreDir, err := ioutil.TempDir("", "")
if err != nil { if err != nil {
return err return fileContext, err
} }
defer os.RemoveAll(dockerIgnoreDir) defer os.RemoveAll(dockerIgnoreDir)
err = ioutil.WriteFile(dockerIgnoreDir+".dockerignore", []byte(content), 0644) err = ioutil.WriteFile(dockerIgnoreDir+".dockerignore", []byte(content), 0644)
if err != nil { if err != nil {
return err return fileContext, err
} }
err = util.GetExcludedFiles(dockerIgnoreDir, "") return util.NewFileContextFromDockerfile(dockerIgnoreDir, "")
if err != nil {
return err
}
return nil
} }
func hashDirectory(dirpath string) (string, error) { func hashDirectory(dirpath string, fileContext util.FileContext) (string, error) {
cache1 := NewCompositeCache() cache1 := NewCompositeCache()
err := cache1.AddPath(dirpath, dirpath) err := cache1.AddPath(dirpath, fileContext)
if err != nil { if err != nil {
return "", err return "", err
} }
@ -217,6 +214,8 @@ func Test_CompositeKey_AddPath_Works(t *testing.T) {
}, },
} }
fileContext := util.FileContext{}
for _, test := range tests { for _, test := range tests {
t.Run(test.name, func(t *testing.T) { t.Run(test.name, func(t *testing.T) {
testDir1, err := ioutil.TempDir("", "") testDir1, err := ioutil.TempDir("", "")
@ -239,11 +238,11 @@ func Test_CompositeKey_AddPath_Works(t *testing.T) {
t.Fatalf("Error creating filesytem structure: %s", err) t.Fatalf("Error creating filesytem structure: %s", err)
} }
hash1, err := hashDirectory(testDir1) hash1, err := hashDirectory(testDir1, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }
hash2, err := hashDirectory(testDir2) hash2, err := hashDirectory(testDir2, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }
@ -288,6 +287,8 @@ func Test_CompositeKey_AddPath_WithExtraFile_Works(t *testing.T) {
}, },
} }
fileContext := util.FileContext{}
for _, test := range tests { for _, test := range tests {
t.Run(test.name, func(t *testing.T) { t.Run(test.name, func(t *testing.T) {
testDir1, err := ioutil.TempDir("", "") testDir1, err := ioutil.TempDir("", "")
@ -315,11 +316,11 @@ func Test_CompositeKey_AddPath_WithExtraFile_Works(t *testing.T) {
t.Fatalf("Error creating filesytem structure: %s", err) t.Fatalf("Error creating filesytem structure: %s", err)
} }
hash1, err := hashDirectory(testDir1) hash1, err := hashDirectory(testDir1, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }
hash2, err := hashDirectory(testDir2) hash2, err := hashDirectory(testDir2, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }
@ -364,6 +365,8 @@ func Test_CompositeKey_AddPath_WithExtraDir_Works(t *testing.T) {
}, },
} }
fileContext := util.FileContext{}
for _, test := range tests { for _, test := range tests {
t.Run(test.name, func(t *testing.T) { t.Run(test.name, func(t *testing.T) {
testDir1, err := ioutil.TempDir("", "") testDir1, err := ioutil.TempDir("", "")
@ -391,11 +394,11 @@ func Test_CompositeKey_AddPath_WithExtraDir_Works(t *testing.T) {
t.Fatalf("Error creating filesytem structure: %s", err) t.Fatalf("Error creating filesytem structure: %s", err)
} }
hash1, err := hashDirectory(testDir1) hash1, err := hashDirectory(testDir1, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }
hash2, err := hashDirectory(testDir2) hash2, err := hashDirectory(testDir2, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }
@ -440,7 +443,7 @@ func Test_CompositeKey_AddPath_WithExtraFilIgnored_Works(t *testing.T) {
}, },
} }
err := setIgnoreContext("**/extra") fileContext, err := setIgnoreContext("**/extra")
if err != nil { if err != nil {
t.Fatalf("Error setting exlusion context: %s", err) t.Fatalf("Error setting exlusion context: %s", err)
} }
@ -472,11 +475,11 @@ func Test_CompositeKey_AddPath_WithExtraFilIgnored_Works(t *testing.T) {
t.Fatalf("Error creating filesytem structure: %s", err) t.Fatalf("Error creating filesytem structure: %s", err)
} }
hash1, err := hashDirectory(testDir1) hash1, err := hashDirectory(testDir1, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }
hash2, err := hashDirectory(testDir2) hash2, err := hashDirectory(testDir2, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }
@ -521,7 +524,7 @@ func Test_CompositeKey_AddPath_WithExtraDirIgnored_Works(t *testing.T) {
}, },
} }
err := setIgnoreContext("**/extra") fileContext, err := setIgnoreContext("**/extra")
if err != nil { if err != nil {
t.Fatalf("Error setting exlusion context: %s", err) t.Fatalf("Error setting exlusion context: %s", err)
} }
@ -553,11 +556,11 @@ func Test_CompositeKey_AddPath_WithExtraDirIgnored_Works(t *testing.T) {
t.Fatalf("Error creating filesytem structure: %s", err) t.Fatalf("Error creating filesytem structure: %s", err)
} }
hash1, err := hashDirectory(testDir1) hash1, err := hashDirectory(testDir1, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }
hash2, err := hashDirectory(testDir2) hash2, err := hashDirectory(testDir2, fileContext)
if err != nil { if err != nil {
t.Fatalf("Failed to calculate hash: %s", err) t.Fatalf("Failed to calculate hash: %s", err)
} }

View File

@ -85,7 +85,7 @@ func ResolveEnvironmentReplacement(value string, envs []string, isFilepath bool)
return fp, nil return fp, nil
} }
func ResolveEnvAndWildcards(sd instructions.SourcesAndDest, buildcontext string, envs []string) ([]string, string, error) { func ResolveEnvAndWildcards(sd instructions.SourcesAndDest, fileContext FileContext, envs []string) ([]string, string, error) {
// First, resolve any environment replacement // First, resolve any environment replacement
resolvedEnvs, err := ResolveEnvironmentReplacementList(sd, envs, true) resolvedEnvs, err := ResolveEnvironmentReplacementList(sd, envs, true)
if err != nil { if err != nil {
@ -96,11 +96,11 @@ func ResolveEnvAndWildcards(sd instructions.SourcesAndDest, buildcontext string,
} }
dest := resolvedEnvs[len(resolvedEnvs)-1] dest := resolvedEnvs[len(resolvedEnvs)-1]
// Resolve wildcards and get a list of resolved sources // Resolve wildcards and get a list of resolved sources
srcs, err := ResolveSources(resolvedEnvs[0:len(resolvedEnvs)-1], buildcontext) srcs, err := ResolveSources(resolvedEnvs[0:len(resolvedEnvs)-1], fileContext.Root)
if err != nil { if err != nil {
return nil, "", errors.Wrap(err, "failed to resolve sources") return nil, "", errors.Wrap(err, "failed to resolve sources")
} }
err = IsSrcsValid(sd, srcs, buildcontext) err = IsSrcsValid(sd, srcs, fileContext)
return srcs, dest, err return srcs, dest, err
} }
@ -220,14 +220,14 @@ func URLDestinationFilepath(rawurl, dest, cwd string, envs []string) (string, er
return destPath, nil return destPath, nil
} }
func IsSrcsValid(srcsAndDest instructions.SourcesAndDest, resolvedSources []string, root string) error { func IsSrcsValid(srcsAndDest instructions.SourcesAndDest, resolvedSources []string, fileContext FileContext) error {
srcs := srcsAndDest[:len(srcsAndDest)-1] srcs := srcsAndDest[:len(srcsAndDest)-1]
dest := srcsAndDest[len(srcsAndDest)-1] dest := srcsAndDest[len(srcsAndDest)-1]
if !ContainsWildcards(srcs) { if !ContainsWildcards(srcs) {
totalSrcs := 0 totalSrcs := 0
for _, src := range srcs { for _, src := range srcs {
if ExcludeFile(src, root) { if fileContext.ExcludesFile(src) {
continue continue
} }
totalSrcs++ totalSrcs++
@ -242,7 +242,7 @@ func IsSrcsValid(srcsAndDest instructions.SourcesAndDest, resolvedSources []stri
if IsSrcRemoteFileURL(resolvedSources[0]) { if IsSrcRemoteFileURL(resolvedSources[0]) {
return nil return nil
} }
path := filepath.Join(root, resolvedSources[0]) path := filepath.Join(fileContext.Root, resolvedSources[0])
fi, err := os.Lstat(path) fi, err := os.Lstat(path)
if err != nil { if err != nil {
return errors.Wrap(err, fmt.Sprintf("failed to get fileinfo for %v", path)) return errors.Wrap(err, fmt.Sprintf("failed to get fileinfo for %v", path))
@ -259,12 +259,12 @@ func IsSrcsValid(srcsAndDest instructions.SourcesAndDest, resolvedSources []stri
continue continue
} }
src = filepath.Clean(src) src = filepath.Clean(src)
files, err := RelativeFiles(src, root) files, err := RelativeFiles(src, fileContext.Root)
if err != nil { if err != nil {
return errors.Wrap(err, "failed to get relative files") return errors.Wrap(err, "failed to get relative files")
} }
for _, file := range files { for _, file := range files {
if ExcludeFile(file, root) { if fileContext.ExcludesFile(file) {
continue continue
} }
totalFiles++ totalFiles++

View File

@ -478,10 +478,11 @@ var isSrcValidTests = []struct {
func Test_IsSrcsValid(t *testing.T) { func Test_IsSrcsValid(t *testing.T) {
for _, test := range isSrcValidTests { for _, test := range isSrcValidTests {
t.Run(test.name, func(t *testing.T) { t.Run(test.name, func(t *testing.T) {
if err := GetExcludedFiles("", buildContextPath); err != nil { fileContext, err := NewFileContextFromDockerfile("", buildContextPath)
t.Fatalf("error getting excluded files: %v", err) if err != nil {
t.Fatalf("error creating file context: %v", err)
} }
err := IsSrcsValid(test.srcsAndDest, test.resolvedSources, buildContextPath) err = IsSrcsValid(test.srcsAndDest, test.resolvedSources, fileContext)
testutil.CheckError(t, test.shouldErr, err) testutil.CheckError(t, test.shouldErr, err)
}) })
} }

View File

@ -74,8 +74,10 @@ var ignorelist = initialIgnoreList
var volumes = []string{} var volumes = []string{}
var excluded []string type FileContext struct {
var IsFirstStage = true Root string
ExcludedFiles []string
}
type ExtractFunction func(string, *tar.Header, io.Reader) error type ExtractFunction func(string, *tar.Header, io.Reader) error
@ -581,7 +583,7 @@ func DetermineTargetFileOwnership(fi os.FileInfo, uid, gid int64) (int64, int64)
// CopyDir copies the file or directory at src to dest // CopyDir copies the file or directory at src to dest
// It returns a list of files it copied over // It returns a list of files it copied over
func CopyDir(src, dest, buildcontext string, uid, gid int64) ([]string, error) { func CopyDir(src, dest string, context FileContext, uid, gid int64) ([]string, error) {
files, err := RelativeFiles("", src) files, err := RelativeFiles("", src)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "copying dir") return nil, errors.Wrap(err, "copying dir")
@ -593,7 +595,7 @@ func CopyDir(src, dest, buildcontext string, uid, gid int64) ([]string, error) {
if err != nil { if err != nil {
return nil, errors.Wrap(err, "copying dir") return nil, errors.Wrap(err, "copying dir")
} }
if ExcludeFile(fullPath, buildcontext) { if context.ExcludesFile(fullPath) {
logrus.Debugf("%s found in .dockerignore, ignoring", src) logrus.Debugf("%s found in .dockerignore, ignoring", src)
continue continue
} }
@ -608,12 +610,12 @@ func CopyDir(src, dest, buildcontext string, uid, gid int64) ([]string, error) {
} }
} else if IsSymlink(fi) { } else if IsSymlink(fi) {
// If file is a symlink, we want to create the same relative symlink // If file is a symlink, we want to create the same relative symlink
if _, err := CopySymlink(fullPath, destPath, buildcontext); err != nil { if _, err := CopySymlink(fullPath, destPath, context); err != nil {
return nil, err return nil, err
} }
} else { } else {
// ... Else, we want to copy over a file // ... Else, we want to copy over a file
if _, err := CopyFile(fullPath, destPath, buildcontext, uid, gid); err != nil { if _, err := CopyFile(fullPath, destPath, context, uid, gid); err != nil {
return nil, err return nil, err
} }
} }
@ -623,8 +625,8 @@ func CopyDir(src, dest, buildcontext string, uid, gid int64) ([]string, error) {
} }
// CopySymlink copies the symlink at src to dest. // CopySymlink copies the symlink at src to dest.
func CopySymlink(src, dest, buildcontext string) (bool, error) { func CopySymlink(src, dest string, context FileContext) (bool, error) {
if ExcludeFile(src, buildcontext) { if context.ExcludesFile(src) {
logrus.Debugf("%s found in .dockerignore, ignoring", src) logrus.Debugf("%s found in .dockerignore, ignoring", src)
return true, nil return true, nil
} }
@ -644,8 +646,8 @@ func CopySymlink(src, dest, buildcontext string) (bool, error) {
} }
// CopyFile copies the file at src to dest // CopyFile copies the file at src to dest
func CopyFile(src, dest, buildcontext string, uid, gid int64) (bool, error) { func CopyFile(src, dest string, context FileContext, uid, gid int64) (bool, error) {
if ExcludeFile(src, buildcontext) { if context.ExcludesFile(src) {
logrus.Debugf("%s found in .dockerignore, ignoring", src) logrus.Debugf("%s found in .dockerignore, ignoring", src)
return true, nil return true, nil
} }
@ -669,40 +671,46 @@ func CopyFile(src, dest, buildcontext string, uid, gid int64) (bool, error) {
return false, CreateFile(dest, srcFile, fi.Mode(), uint32(uid), uint32(gid)) return false, CreateFile(dest, srcFile, fi.Mode(), uint32(uid), uint32(gid))
} }
// GetExcludedFiles gets a list of files to exclude from the .dockerignore func NewFileContextFromDockerfile(dockerfilePath, buildcontext string) (FileContext, error) {
func GetExcludedFiles(dockerfilepath string, buildcontext string) error { fileContext := FileContext{Root: buildcontext}
path := dockerfilepath + ".dockerignore" excludedFiles, err := getExcludedFiles(dockerfilePath, buildcontext)
if err != nil {
return fileContext, err
}
fileContext.ExcludedFiles = excludedFiles
return fileContext, nil
}
// getExcludedFiles returns a list of files to exclude from the .dockerignore
func getExcludedFiles(dockerfilePath, buildcontext string) ([]string, error) {
path := dockerfilePath + ".dockerignore"
if !FilepathExists(path) { if !FilepathExists(path) {
path = filepath.Join(buildcontext, ".dockerignore") path = filepath.Join(buildcontext, ".dockerignore")
} }
if !FilepathExists(path) { if !FilepathExists(path) {
return nil return nil, nil
} }
logrus.Infof("Using dockerignore file: %v", path) logrus.Infof("Using dockerignore file: %v", path)
contents, err := ioutil.ReadFile(path) contents, err := ioutil.ReadFile(path)
if err != nil { if err != nil {
return errors.Wrap(err, "parsing .dockerignore") return nil, errors.Wrap(err, "parsing .dockerignore")
} }
reader := bytes.NewBuffer(contents) reader := bytes.NewBuffer(contents)
excluded, err = dockerignore.ReadAll(reader) return dockerignore.ReadAll(reader)
return err
} }
// ExcludeFile returns true if the .dockerignore specified this file should be ignored // ExcludesFile returns true if the file context specified this file should be ignored.
func ExcludeFile(path, buildcontext string) bool { // Usually this is specified via .dockerignore
// Apply dockerfile excludes for first stage only func (c FileContext) ExcludesFile(path string) bool {
if !IsFirstStage { if HasFilepathPrefix(path, c.Root, false) {
return false
}
if HasFilepathPrefix(path, buildcontext, false) {
var err error var err error
path, err = filepath.Rel(buildcontext, path) path, err = filepath.Rel(c.Root, path)
if err != nil { if err != nil {
logrus.Errorf("unable to get relative path, including %s in build: %v", path, err) logrus.Errorf("unable to get relative path, including %s in build: %v", path, err)
return false return false
} }
} }
match, err := fileutils.Matches(path, excluded) match, err := fileutils.Matches(path, c.ExcludedFiles)
if err != nil { if err != nil {
logrus.Errorf("error matching, including %s in build: %v", path, err) logrus.Errorf("error matching, including %s in build: %v", path, err)
return false return false

View File

@ -880,7 +880,7 @@ func TestCopySymlink(t *testing.T) {
if err := os.Symlink(tc.linkTarget, link); err != nil { if err := os.Symlink(tc.linkTarget, link); err != nil {
t.Fatal(err) t.Fatal(err)
} }
if _, err := CopySymlink(link, dest, ""); err != nil { if _, err := CopySymlink(link, dest, FileContext{}); err != nil {
t.Fatal(err) t.Fatal(err)
} }
if _, err := os.Lstat(dest); err != nil { if _, err := os.Lstat(dest); err != nil {
@ -966,19 +966,20 @@ func Test_correctDockerignoreFileIsUsed(t *testing.T) {
}, },
} }
for _, tt := range tests { for _, tt := range tests {
if err := GetExcludedFiles(tt.args.dockerfilepath, tt.args.buildcontext); err != nil { fileContext, err := NewFileContextFromDockerfile(tt.args.dockerfilepath, tt.args.buildcontext)
if err != nil {
t.Fatal(err) t.Fatal(err)
} }
for _, excl := range tt.args.excluded { for _, excl := range tt.args.excluded {
t.Run(tt.name+" to exclude "+excl, func(t *testing.T) { t.Run(tt.name+" to exclude "+excl, func(t *testing.T) {
if !ExcludeFile(excl, tt.args.buildcontext) { if !fileContext.ExcludesFile(excl) {
t.Errorf("'%v' not excluded", excl) t.Errorf("'%v' not excluded", excl)
} }
}) })
} }
for _, incl := range tt.args.included { for _, incl := range tt.args.included {
t.Run(tt.name+" to include "+incl, func(t *testing.T) { t.Run(tt.name+" to include "+incl, func(t *testing.T) {
if ExcludeFile(incl, tt.args.buildcontext) { if fileContext.ExcludesFile(incl) {
t.Errorf("'%v' not included", incl) t.Errorf("'%v' not included", incl)
} }
}) })
@ -1004,7 +1005,7 @@ func Test_CopyFile_skips_self(t *testing.T) {
t.Fatal(err) t.Fatal(err)
} }
ignored, err := CopyFile(tempFile, tempFile, "", DoNotChangeUID, DoNotChangeGID) ignored, err := CopyFile(tempFile, tempFile, FileContext{}, DoNotChangeUID, DoNotChangeGID)
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }