Skip to content
This repository has been archived by the owner on May 4, 2021. It is now read-only.

Make ADD untar tar.gz files #309

Merged
merged 3 commits into from
Mar 11, 2020
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 7 additions & 3 deletions lib/builder/step/add_copy_step.go
Original file line number Diff line number Diff line change
Expand Up @@ -58,12 +58,13 @@ type addCopyStep struct {
toPath string
chown string
preserveOwner bool
isAdd bool
jockech marked this conversation as resolved.
Show resolved Hide resolved
}

// newAddCopyStep returns a BuildStep from given arguments.
func newAddCopyStep(
directive Directive, args, chown, fromStage string,
fromPaths []string, toPath string, commit, preserveOwner bool) (*addCopyStep, error) {
fromPaths []string, toPath string, commit, preserveOwner, isAdd bool) (*addCopyStep, error) {

toPath = strings.Trim(toPath, "\"'")
for i := range fromPaths {
Expand All @@ -80,6 +81,7 @@ func newAddCopyStep(
toPath: toPath,
chown: chown,
preserveOwner: preserveOwner,
isAdd: isAdd,
}, nil
}

Expand Down Expand Up @@ -136,14 +138,16 @@ func (s *addCopyStep) Execute(ctx *context.BuildContext, modifyFS bool) (err err

internal := s.fromStage != ""
blacklist := append(pathutils.DefaultBlacklist, ctx.ImageStore.RootDir)
copyOp, err := snapshot.NewCopyOperation(
relPaths, sourceRoot, s.workingDir, s.toPath, s.chown, blacklist, internal, s.preserveOwner)
copyOp, err := snapshot.NewCopyOperation(relPaths, sourceRoot, s.workingDir, s.toPath, s.chown,
s.preserveOwner, internal, blacklist, s.isAdd)
if err != nil {
return fmt.Errorf("invalid copy operation: %s", err)
}

ctx.CopyOps = append(ctx.CopyOps, copyOp)
if modifyFS {
// Note: We still want to keep this copyop in the list, just in case there
// is no RUN afterwards, and layer needs to be created from copyops.
return copyOp.Execute()
}
return nil
Expand Down
8 changes: 4 additions & 4 deletions lib/builder/step/add_copy_step_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,21 +24,21 @@ func TestContextDirs(t *testing.T) {
require := require.New(t)

srcs := []string{}
ac, err := newAddCopyStep(Copy, "", "", "", srcs, "", false, false)
ac, err := newAddCopyStep(Copy, "", "", "", srcs, "", false, false, false)
require.NoError(err)
stage, paths := ac.ContextDirs()
require.Equal("", stage)
require.Len(paths, 0)

srcs = []string{"src"}
ac, err = newAddCopyStep(Copy, "", "", "", srcs, "", false, false)
ac, err = newAddCopyStep(Copy, "", "", "", srcs, "", false, false, false)
require.NoError(err)
stage, paths = ac.ContextDirs()
require.Equal("", stage)
require.Len(paths, 0)

srcs = []string{"src"}
ac, err = newAddCopyStep(Copy, "", "", "stage", srcs, "", false, false)
ac, err = newAddCopyStep(Copy, "", "", "stage", srcs, "", false, false, false)
require.NoError(err)
stage, paths = ac.ContextDirs()
require.Equal("stage", stage)
Expand All @@ -48,7 +48,7 @@ func TestContextDirs(t *testing.T) {
func TestTrimmingPaths(t *testing.T) {
require := require.New(t)

ac, err := newAddCopyStep(Copy, "", "", "", []string{"\"/from/path\""}, "\"/to/path\"", false, false)
ac, err := newAddCopyStep(Copy, "", "", "", []string{"\"/from/path\""}, "\"/to/path\"", false, false, false)
require.NoError(err)

require.Equal("/from/path", ac.fromPaths[0])
Expand Down
3 changes: 2 additions & 1 deletion lib/builder/step/add_step.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,8 @@ type AddStep struct {

// NewAddStep creates a new AddStep
func NewAddStep(args, chown string, fromPaths []string, toPath string, commit, preserverOwner bool) (*AddStep, error) {
s, err := newAddCopyStep(Add, args, chown, "", fromPaths, toPath, commit, preserverOwner)
s, err := newAddCopyStep(
Add, args, chown, "", fromPaths, toPath, commit, preserverOwner, true)
if err != nil {
return nil, fmt.Errorf("new add/copy step: %s", err)
}
Expand Down
3 changes: 2 additions & 1 deletion lib/builder/step/copy_step.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,8 @@ func NewCopyStep(
args, chown, fromStage string, fromPaths []string, toPath string, commit, preserveOwner bool,
) (*CopyStep, error) {

s, err := newAddCopyStep(Copy, args, chown, fromStage, fromPaths, toPath, commit, preserveOwner)
s, err := newAddCopyStep(
Copy, args, chown, fromStage, fromPaths, toPath, commit, preserveOwner, false)
if err != nil {
return nil, fmt.Errorf("new add/copy step: %s", err)
}
Expand Down
44 changes: 37 additions & 7 deletions lib/snapshot/copy_op.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,10 +22,10 @@ import (

"github.com/uber/makisu/lib/fileio"
"github.com/uber/makisu/lib/pathutils"
"github.com/uber/makisu/lib/tario"
"github.com/uber/makisu/lib/utils"
)

// CopyOperation defines a copy operation that occurred to generate a layer from.
type CopyOperation struct {
srcRoot string
srcs []string
Expand All @@ -34,16 +34,20 @@ type CopyOperation struct {
gid int
preserveOwner bool

blacklist []string
// Indicates if the copy op is used for copying from previous stages.
internal bool
// Blacklist is ignored in that case.
internal bool
blacklist []string

// Set to true if the copy op was created by ADD step.
// Some functionalities are only available to ADD.
fromAdd bool
}

// NewCopyOperation initializes and validates a CopyOperation. Use "internal" to
// specify if the copy op is used for copying from previous stages.
func NewCopyOperation(
srcs []string, srcRoot, workDir, dst, chown string,
blacklist []string, internal, preserveOwner bool) (*CopyOperation, error) {
func NewCopyOperation(srcs []string, srcRoot, workDir, dst, chown string,
preserveOwner, internal bool, blacklist []string, fromAdd bool) (*CopyOperation, error) {

if err := checkCopyParams(srcs, workDir, dst); err != nil {
return nil, fmt.Errorf("check copy param: %s", err)
Expand All @@ -70,6 +74,7 @@ func NewCopyOperation(
preserveOwner: preserveOwner,
blacklist: blacklist,
internal: internal,
fromAdd: fromAdd,
}, nil
}

Expand All @@ -86,6 +91,32 @@ func (c *CopyOperation) Execute() error {
if err != nil {
return fmt.Errorf("lstat %s: %s", src, err)
}

if strings.HasSuffix(src, ".tar.gz") && c.fromAdd {
// Special feature for ADD - Extract tar.gz into dst directory.
// If dst is an existing directory, untar.
// If dst doesn't exist, create it with root.
// If dst exists and is not a directory, fail.
reader, err := os.Open(src)
if err != nil {
return fmt.Errorf("open tar file: %s", err)
}
defer reader.Close()

dstFi, err := os.Stat(c.dst)
if os.IsNotExist(err) {
if err := os.MkdirAll(c.dst, 0755); err != nil {
return fmt.Errorf("create untar directory: %s", err)
}
}
if !dstFi.IsDir() {
return fmt.Errorf("target untar path exists and is not directory: %s", c.dst)
}
if err := tario.Untar(reader, c.dst); err != nil {
return fmt.Errorf("untar tar: %s", err)
}
}

var copier fileio.Copier
if c.internal {
copier = fileio.NewInternalCopier()
Expand Down Expand Up @@ -115,7 +146,6 @@ func (c *CopyOperation) Execute() error {
return fmt.Errorf("copy file %s to dir %s: %s", src, targetFilePath, err)
}
}

} else {
// File to file
if c.preserveOwner {
Expand Down
20 changes: 10 additions & 10 deletions lib/snapshot/copy_op_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,28 +45,28 @@ func TestNewCopyOperation(t *testing.T) {
workDir := ""
dst := "/test2/test.txt"
_, err = NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.Error(err)

srcs = []string{"file", "dir/"}
workDir = ""
dst = "/target/test"
_, err = NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.Error(err)

srcs = []string{"file", "dir/"}
workDir = ""
dst = "target/test"
_, err = NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.Error(err)

srcs = []string{"file", "dir/"}
workDir = "wrk/"
dst = "target/test/"
_, err = NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.Error(err)
}

Expand All @@ -87,7 +87,7 @@ func TestExecuteCopyOperation(t *testing.T) {
srcRoot := tmpRoot1
dst := filepath.Join(tmpRoot2, "test2/test.txt")
c, err := NewCopyOperation(
srcs, srcRoot, "", dst, validChown, pathutils.DefaultBlacklist, false, true)
srcs, srcRoot, "", dst, validChown, true, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
require.NoError(c.Execute())
b, err := ioutil.ReadFile(dst)
Expand All @@ -107,7 +107,7 @@ func TestExecuteCopyOperation(t *testing.T) {
workDir := tmpRoot2
dst := "test2/test.txt"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
require.NoError(c.Execute())
b, err := ioutil.ReadFile(filepath.Join(tmpRoot2, dst))
Expand All @@ -129,7 +129,7 @@ func TestExecuteCopyOperation(t *testing.T) {
workDir := tmpRoot2
dst := "test2/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
require.NoError(c.Execute())
b, err := ioutil.ReadFile(filepath.Join(tmpRoot2, dst, "test.txt"))
Expand All @@ -154,7 +154,7 @@ func TestExecuteCopyOperation(t *testing.T) {
workDir := filepath.Join(tmpRoot2, "test2")
dst := "."
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, true)
srcs, srcRoot, workDir, dst, validChown, true, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
require.NoError(c.Execute())
b, err := ioutil.ReadFile(filepath.Join(tmpRoot2, "test2", "test.txt"))
Expand All @@ -181,7 +181,7 @@ func TestExecuteCopyOperation(t *testing.T) {
workDir := tmpRoot2
dst := "test2/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
require.NoError(c.Execute())
b, err := ioutil.ReadFile(filepath.Join(tmpRoot2, dst, "test.txt"))
Expand All @@ -207,7 +207,7 @@ func TestExecuteCopyOperation(t *testing.T) {
workDir := tmpRoot2
dst := "test2/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
require.NoError(c.Execute())
b, err := ioutil.ReadFile(filepath.Join(tmpRoot2, dst, "test.txt"))
Expand Down
16 changes: 8 additions & 8 deletions lib/snapshot/mem_fs_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -710,7 +710,7 @@ func TestCreateLayerByCopy(t *testing.T) {
workDir := ""
dst := "/test2/test.txt"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
err = fs.addToLayer(newMemLayer(), c)
require.NoError(err)
Expand Down Expand Up @@ -750,7 +750,7 @@ func TestCreateLayerByCopy(t *testing.T) {
workDir := ""
dst := "/dst/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
err = fs.addToLayer(newMemLayer(), c)
require.NoError(err)
Expand Down Expand Up @@ -792,7 +792,7 @@ func TestCreateLayerByCopy(t *testing.T) {
workDir := ""
dst := "/dst/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
err = fs.addToLayer(newMemLayer(), c)
require.NoError(err)
Expand Down Expand Up @@ -839,7 +839,7 @@ func TestCreateLayerByCopy(t *testing.T) {
workDir := ""
dst := "/dst/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
err = fs.addToLayer(newMemLayer(), c)
require.NoError(err)
Expand Down Expand Up @@ -887,7 +887,7 @@ func TestCreateLayerByCopy(t *testing.T) {
workDir := ""
dst := "/dst/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
err = fs.addToLayer(newMemLayer(), c)
require.NoError(err)
Expand Down Expand Up @@ -945,7 +945,7 @@ func TestCreateLayerByCopy(t *testing.T) {
workDir := ""
dst := "/dst/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
err = fs.addToLayer(newMemLayer(), c)
require.NoError(err)
Expand Down Expand Up @@ -1003,7 +1003,7 @@ func TestCreateLayerByCopy(t *testing.T) {
workDir := "/wrk"
dst := "dst/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
err = fs.addToLayer(newMemLayer(), c)
require.NoError(err)
Expand Down Expand Up @@ -1159,7 +1159,7 @@ func TestAddLayersEqual(t *testing.T) {
workDir := "/wrk"
dst := "dst/"
c, err := NewCopyOperation(
srcs, srcRoot, workDir, dst, validChown, pathutils.DefaultBlacklist, false, false)
srcs, srcRoot, workDir, dst, validChown, false, false, pathutils.DefaultBlacklist, false)
require.NoError(err)
err = fs1.AddLayerByCopyOps([]*CopyOperation{c}, w1)
require.NoError(err)
Expand Down