Commit c1a97284 authored by Mitchell Hashimoto's avatar Mitchell Hashimoto

post-processor/vagrant: transition aws over

parent 3dd4c08f
package vagrant package vagrant
import ( import (
"compress/flate" "bytes"
"fmt" "fmt"
"github.com/mitchellh/packer/common"
"github.com/mitchellh/packer/packer" "github.com/mitchellh/packer/packer"
"io/ioutil"
"log"
"os"
"path/filepath"
"strconv"
"strings" "strings"
"text/template"
) )
type AWSBoxConfig struct { type AWSProvider struct{}
common.PackerConfig `mapstructure:",squash"`
OutputPath string `mapstructure:"output"` func (p *AWSProvider) Process(ui packer.Ui, artifact packer.Artifact, dir string) (vagrantfile string, metadata map[string]interface{}, err error) {
VagrantfileTemplate string `mapstructure:"vagrantfile_template"` // Create the metadata
CompressionLevel string `mapstructure:"compression_level"` metadata = map[string]interface{}{"provider": "aws"}
tpl *packer.ConfigTemplate
}
type AWSVagrantfileTemplate struct {
Images map[string]string
}
type AWSBoxPostProcessor struct {
config AWSBoxConfig
}
func (p *AWSBoxPostProcessor) Configure(raws ...interface{}) error {
md, err := common.DecodeConfig(&p.config, raws...)
if err != nil {
return err
}
p.config.tpl, err = packer.NewConfigTemplate()
if err != nil {
return err
}
p.config.tpl.UserVars = p.config.PackerUserVars
// Accumulate any errors
errs := common.CheckUnusedConfig(md)
validates := map[string]*string{
"output": &p.config.OutputPath,
"vagrantfile_template": &p.config.VagrantfileTemplate,
"compression_level": &p.config.CompressionLevel,
}
for n, ptr := range validates {
if err := p.config.tpl.Validate(*ptr); err != nil {
errs = packer.MultiErrorAppend(
errs, fmt.Errorf("Error parsing %s: %s", n, err))
}
}
if errs != nil && len(errs.Errors) > 0 {
return errs
}
return nil
}
func (p *AWSBoxPostProcessor) PostProcess(ui packer.Ui, artifact packer.Artifact) (packer.Artifact, bool, error) { // Build up the template data to build our Vagrantfile
// Determine the regions... tplData := &awsVagrantfileTemplate{
tplData := &AWSVagrantfileTemplate{
Images: make(map[string]string), Images: make(map[string]string),
} }
for _, regions := range strings.Split(artifact.Id(), ",") { for _, regions := range strings.Split(artifact.Id(), ",") {
parts := strings.Split(regions, ":") parts := strings.Split(regions, ":")
if len(parts) != 2 { if len(parts) != 2 {
return nil, false, fmt.Errorf("Poorly formatted artifact ID: %s", artifact.Id()) err = fmt.Errorf("Poorly formatted artifact ID: %s", artifact.Id())
return
} }
tplData.Images[parts[0]] = parts[1] tplData.Images[parts[0]] = parts[1]
} }
// Compile the output path // Build up the contents
outputPath, err := p.config.tpl.Process(p.config.OutputPath, &OutputPathTemplate{ var contents bytes.Buffer
ArtifactId: artifact.Id(), t := template.Must(template.New("vf").Parse(defaultAWSVagrantfile))
BuildName: p.config.PackerBuildName, err = t.Execute(&contents, tplData)
Provider: "aws", vagrantfile = contents.String()
}) return
if err != nil { }
return nil, false, err
}
// Create a temporary directory for us to build the contents of the box in
dir, err := ioutil.TempDir("", "packer")
if err != nil {
return nil, false, err
}
defer os.RemoveAll(dir)
// Create the Vagrantfile from the template
vf, err := os.Create(filepath.Join(dir, "Vagrantfile"))
if err != nil {
return nil, false, err
}
defer vf.Close()
vagrantfileContents := defaultAWSVagrantfile
if p.config.VagrantfileTemplate != "" {
log.Printf("Using vagrantfile template: %s", p.config.VagrantfileTemplate)
f, err := os.Open(p.config.VagrantfileTemplate)
if err != nil {
err = fmt.Errorf("error opening vagrantfile template: %s", err)
return nil, false, err
}
defer f.Close()
contents, err := ioutil.ReadAll(f)
if err != nil {
err = fmt.Errorf("error reading vagrantfile template: %s", err)
return nil, false, err
}
vagrantfileContents = string(contents)
}
vagrantfileContents, err = p.config.tpl.Process(vagrantfileContents, tplData)
if err != nil {
return nil, false, fmt.Errorf("Error writing Vagrantfile: %s", err)
}
vf.Write([]byte(vagrantfileContents))
vf.Close()
var level int = flate.DefaultCompression
if p.config.CompressionLevel != "" {
level, err = strconv.Atoi(p.config.CompressionLevel)
if err != nil {
return nil, false, err
}
}
// Create the metadata
metadata := map[string]string{"provider": "aws"}
if err := WriteMetadata(dir, metadata); err != nil {
return nil, false, err
}
// Compress the directory to the given output path
if err := DirToBox(outputPath, dir, ui, level); err != nil {
err = fmt.Errorf("error creating box: %s", err)
return nil, false, err
}
return NewArtifact("aws", outputPath), true, nil type awsVagrantfileTemplate struct {
Images map[string]string
} }
var defaultAWSVagrantfile = ` var defaultAWSVagrantfile = `
......
package vagrant package vagrant
import ( import (
"github.com/mitchellh/packer/packer"
"testing" "testing"
) )
func TestAWSBoxPostProcessor_ImplementsPostProcessor(t *testing.T) { func TestAWSProvider_impl(t *testing.T) {
var raw interface{} var _ Provider = new(AWSProvider)
raw = &AWSBoxPostProcessor{}
if _, ok := raw.(packer.PostProcessor); !ok {
t.Fatalf("AWS PostProcessor should be a PostProcessor")
}
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment