kaniko/pkg/commands/workdir.go

196 lines
5.3 KiB
Go

/*
Copyright 2018 Google LLC
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package commands
import (
"fmt"
"os"
"path/filepath"
kConfig "github.com/GoogleContainerTools/kaniko/pkg/config"
"github.com/GoogleContainerTools/kaniko/pkg/dockerfile"
"github.com/pkg/errors"
"github.com/GoogleContainerTools/kaniko/pkg/util"
v1 "github.com/google/go-containerregistry/pkg/v1"
"github.com/moby/buildkit/frontend/dockerfile/instructions"
"github.com/sirupsen/logrus"
)
type WorkdirCommand struct {
BaseCommand
cmd *instructions.WorkdirCommand
snapshotFiles []string
shdCache bool
}
func ToAbsPath(path string, workdir string) string {
if filepath.IsAbs(path) {
return path
} else {
if workdir != "" {
return filepath.Join(workdir, path)
} else {
return filepath.Join("/", path)
}
}
}
// For testing
var mkdirAllWithPermissions = util.MkdirAllWithPermissions
func (w *WorkdirCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.BuildArgs) error {
logrus.Info("Cmd: workdir")
workdirPath := w.cmd.Path
replacementEnvs := buildArgs.ReplacementEnvs(config.Env)
resolvedWorkingDir, err := util.ResolveEnvironmentReplacement(workdirPath, replacementEnvs, true)
if err != nil {
return err
}
config.WorkingDir = ToAbsPath(resolvedWorkingDir, config.WorkingDir)
logrus.Infof("Changed working directory to %s", config.WorkingDir)
// Only create and snapshot the dir if it didn't exist already
w.snapshotFiles = []string{}
if _, err := os.Stat(config.WorkingDir); os.IsNotExist(err) {
uid, gid := int64(-1), int64(-1)
if config.User != "" {
logrus.Debugf("Fetching uid and gid for USER '%s'", config.User)
uid, gid, err = util.GetUserGroup(config.User, replacementEnvs)
if err != nil {
return errors.Wrapf(err, "identifying uid and gid for user %s", config.User)
}
}
logrus.Infof("Creating directory %s with uid %d and gid %d", config.WorkingDir, uid, gid)
w.snapshotFiles = append(w.snapshotFiles, config.WorkingDir)
if err := mkdirAllWithPermissions(config.WorkingDir, 0755, uid, gid); err != nil {
return errors.Wrapf(err, "creating workdir %s", config.WorkingDir)
}
}
return nil
}
// FilesToSnapshot returns the workingdir, which should have been created if it didn't already exist
func (w *WorkdirCommand) FilesToSnapshot() []string {
return nil
}
func (r *WorkdirCommand) ProvidesFilesToSnapshot() bool {
return false
}
// String returns some information about the command for the image config history
func (w *WorkdirCommand) String() string {
return w.cmd.String()
}
// CacheCommand returns true since this command should be cached
func (w *WorkdirCommand) CacheCommand(img v1.Image) DockerCommand {
return &CachingWorkdirCommand{
img: img,
cmd: w.cmd,
extractFn: util.ExtractFile,
}
}
func (w *WorkdirCommand) MetadataOnly() bool {
return false
}
func (r *WorkdirCommand) RequiresUnpackedFS() bool {
return true
}
func (w *WorkdirCommand) ShouldCacheOutput() bool {
return w.shdCache
}
type CachingWorkdirCommand struct {
BaseCommand
caching
img v1.Image
extractedFiles []string
cmd *instructions.WorkdirCommand
extractFn util.ExtractFunction
}
func (wr *CachingWorkdirCommand) ExecuteCommand(config *v1.Config, buildArgs *dockerfile.BuildArgs) error {
var err error
logrus.Info("Cmd: workdir")
workdirPath := wr.cmd.Path
replacementEnvs := buildArgs.ReplacementEnvs(config.Env)
resolvedWorkingDir, err := util.ResolveEnvironmentReplacement(workdirPath, replacementEnvs, true)
if err != nil {
return err
}
config.WorkingDir = ToAbsPath(resolvedWorkingDir, config.WorkingDir)
logrus.Infof("Changed working directory to %s", config.WorkingDir)
logrus.Infof("Found cached layer, extracting to filesystem")
if wr.img == nil {
return errors.New(fmt.Sprintf("command image is nil %v", wr.String()))
}
layers, err := wr.img.Layers()
if err != nil {
return errors.Wrap(err, "retrieving image layers")
}
if len(layers) != 1 {
return errors.New(fmt.Sprintf("expected %d layers but got %d", 1, len(layers)))
}
wr.layer = layers[0]
wr.extractedFiles, err = util.GetFSFromLayers(
kConfig.RootDir,
layers,
util.ExtractFunc(wr.extractFn),
util.IncludeWhiteout(),
)
if err != nil {
return errors.Wrap(err, "extracting fs from image")
}
return nil
}
// FilesToSnapshot returns the workingdir, which should have been created if it didn't already exist
func (wr *CachingWorkdirCommand) FilesToSnapshot() []string {
f := wr.extractedFiles
logrus.Debugf("%d files extracted by caching run command", len(f))
logrus.Tracef("Extracted files: %s", f)
return f
}
// String returns some information about the command for the image config history
func (wr *CachingWorkdirCommand) String() string {
if wr.cmd == nil {
return "nil command"
}
return wr.cmd.String()
}
func (wr *CachingWorkdirCommand) MetadataOnly() bool {
return false
}