From 0ebf1d0da9c0adcaf1b02b75215ce701cef54ddb Mon Sep 17 00:00:00 2001 From: Sander Saares Date: Wed, 2 Aug 2017 14:27:22 +0300 Subject: [PATCH 1/2] Link VHD instead of copy where allowed by OS No need to incur that copy I/O if we don't even change the file! --- post-processor/vagrant/hyperv.go | 17 +++++++++++++---- post-processor/vagrant/util.go | 17 +++++++++++++++++ 2 files changed, 30 insertions(+), 4 deletions(-) diff --git a/post-processor/vagrant/hyperv.go b/post-processor/vagrant/hyperv.go index cf71820e9..14486738b 100644 --- a/post-processor/vagrant/hyperv.go +++ b/post-processor/vagrant/hyperv.go @@ -2,10 +2,11 @@ package vagrant import ( "fmt" - "github.com/hashicorp/packer/packer" "os" "path/filepath" "strings" + + "github.com/hashicorp/packer/packer" ) type HypervProvider struct{} @@ -55,10 +56,18 @@ func (p *HypervProvider) Process(ui packer.Ui, artifact packer.Artifact, dir str dstPath := filepath.Join(dstDir, filepath.Base(path)) - if err = CopyContents(dstPath, path); err != nil { - ui.Message(fmt.Sprintf("err in copying: %s to %s", path, dstPath)) - return + // We prefer to link the files where possible because they are often very huge. + // Some filesystem configurations do not allow hardlinks. As the possibilities + // of mounting different devices in different paths are flexible, we just try to + // link the file and copy if the link fails, thereby automatically optimizing with a safe fallback. + if err = LinkFile(dstPath, path); err != nil { + // ui.Message(fmt.Sprintf("err in linking: %s to %s", path, dstPath)) + if err = CopyContents(dstPath, path); err != nil { + ui.Message(fmt.Sprintf("err in copying: %s to %s", path, dstPath)) + return + } } + ui.Message(fmt.Sprintf("Copyed %s to %s", path, dstPath)) } diff --git a/post-processor/vagrant/util.go b/post-processor/vagrant/util.go index bc154d640..8c6e09b50 100644 --- a/post-processor/vagrant/util.go +++ b/post-processor/vagrant/util.go @@ -51,6 +51,23 @@ func CopyContents(dst, src string) error { return nil } +// Creates a (hard) link to a file, ensuring that all parent directories also exist. +func LinkFile(dst, src string) error { + dstDir, _ := filepath.Split(dst) + if dstDir != "" { + err := os.MkdirAll(dstDir, os.ModePerm) + if err != nil { + return err + } + } + + if err := os.Link(src, dst); err != nil { + return err + } + + return nil +} + // DirToBox takes the directory and compresses it into a Vagrant-compatible // box. This function does not perform checks to verify that dir is // actually a proper box. This is an expected precondition. From fb4db02586432aaa7a61830a020d605ec537b8d2 Mon Sep 17 00:00:00 2001 From: Sander Saares Date: Sat, 2 Sep 2017 15:06:04 +0300 Subject: [PATCH 2/2] Copy/link permissions 0777 -> 0755 Previous permission flags were unusually permissive. Almost everything else in Packer uses 0755. --- post-processor/vagrant/util.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/post-processor/vagrant/util.go b/post-processor/vagrant/util.go index 8c6e09b50..de80912bb 100644 --- a/post-processor/vagrant/util.go +++ b/post-processor/vagrant/util.go @@ -32,7 +32,7 @@ func CopyContents(dst, src string) error { dstDir, _ := filepath.Split(dst) if dstDir != "" { - err := os.MkdirAll(dstDir, os.ModePerm) + err := os.MkdirAll(dstDir, 0755) if err != nil { return err } @@ -55,7 +55,7 @@ func CopyContents(dst, src string) error { func LinkFile(dst, src string) error { dstDir, _ := filepath.Split(dst) if dstDir != "" { - err := os.MkdirAll(dstDir, os.ModePerm) + err := os.MkdirAll(dstDir, 0755) if err != nil { return err }