2013-06-26 21:55:11 -04:00
|
|
|
package vagrant
|
|
|
|
|
|
|
|
import (
|
|
|
|
"archive/tar"
|
|
|
|
"compress/gzip"
|
|
|
|
"encoding/json"
|
2013-08-20 01:38:32 -04:00
|
|
|
"fmt"
|
|
|
|
"github.com/mitchellh/packer/packer"
|
2013-06-26 21:55:11 -04:00
|
|
|
"io"
|
2013-06-27 17:06:14 -04:00
|
|
|
"log"
|
2013-06-26 21:55:11 -04:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
)
|
|
|
|
|
2013-06-27 13:51:13 -04:00
|
|
|
// OutputPathTemplate is the structure that is availalable within the
|
|
|
|
// OutputPath variables.
|
|
|
|
type OutputPathTemplate struct {
|
|
|
|
ArtifactId string
|
2013-07-01 18:07:09 -04:00
|
|
|
BuildName string
|
2013-06-27 13:51:13 -04:00
|
|
|
Provider string
|
|
|
|
}
|
|
|
|
|
2013-07-07 20:44:13 -04:00
|
|
|
// Copies a file by copying the contents of the file to another place.
|
|
|
|
func CopyContents(dst, src string) error {
|
|
|
|
srcF, err := os.Open(src)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer srcF.Close()
|
|
|
|
|
|
|
|
dstF, err := os.Create(dst)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer dstF.Close()
|
|
|
|
|
|
|
|
if _, err := io.Copy(dstF, srcF); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-06-26 21:55:11 -04:00
|
|
|
// 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.
|
2013-08-20 01:38:32 -04:00
|
|
|
func DirToBox(dst, dir string, ui packer.Ui) error {
|
2013-07-05 14:11:54 -04:00
|
|
|
log.Printf("Turning dir into box: %s => %s", dir, dst)
|
2013-06-26 21:55:11 -04:00
|
|
|
dstF, err := os.Create(dst)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer dstF.Close()
|
|
|
|
|
|
|
|
gzipWriter := gzip.NewWriter(dstF)
|
|
|
|
defer gzipWriter.Close()
|
|
|
|
|
|
|
|
tarWriter := tar.NewWriter(gzipWriter)
|
|
|
|
defer tarWriter.Close()
|
|
|
|
|
|
|
|
// This is the walk func that tars each of the files in the dir
|
|
|
|
tarWalk := func(path string, info os.FileInfo, prevErr error) error {
|
2013-07-02 11:46:00 -04:00
|
|
|
// If there was a prior error, return it
|
|
|
|
if prevErr != nil {
|
|
|
|
return prevErr
|
|
|
|
}
|
|
|
|
|
2013-06-27 17:06:14 -04:00
|
|
|
// Skip directories
|
|
|
|
if info.IsDir() {
|
2013-07-05 14:00:18 -04:00
|
|
|
log.Printf("Skipping directory '%s' for box '%s'", path, dst)
|
2013-06-27 17:06:14 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Box add: '%s' to '%s'", path, dst)
|
2013-06-26 21:55:11 -04:00
|
|
|
f, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
header, err := tar.FileInfoHeader(info, "")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have to set the Name explicitly because it is supposed to
|
|
|
|
// be a relative path to the root. Otherwise, the tar ends up
|
|
|
|
// being a bunch of files in the root, even if they're actually
|
|
|
|
// nested in a dir in the original "dir" param.
|
|
|
|
header.Name, err = filepath.Rel(dir, path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2013-08-20 01:38:32 -04:00
|
|
|
if ui != nil {
|
|
|
|
ui.Message(fmt.Sprintf("Compressing: %s", header.Name))
|
|
|
|
}
|
|
|
|
|
2013-06-26 21:55:11 -04:00
|
|
|
if err := tarWriter.WriteHeader(header); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := io.Copy(tarWriter, f); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tar.gz everything up
|
|
|
|
return filepath.Walk(dir, tarWalk)
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteMetadata writes the "metadata.json" file for a Vagrant box.
|
|
|
|
func WriteMetadata(dir string, contents interface{}) error {
|
|
|
|
f, err := os.Create(filepath.Join(dir, "metadata.json"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
enc := json.NewEncoder(f)
|
|
|
|
return enc.Encode(contents)
|
|
|
|
}
|