231 lines
6.0 KiB
Go
231 lines
6.0 KiB
Go
package packer
|
|
|
|
import (
|
|
"fmt"
|
|
"sort"
|
|
|
|
"github.com/hashicorp/go-multierror"
|
|
"github.com/mitchellh/packer/template"
|
|
"github.com/mitchellh/packer/template/interpolate"
|
|
)
|
|
|
|
// Core is the main executor of Packer. If Packer is being used as a
|
|
// library, this is the struct you'll want to instantiate to get anything done.
|
|
type Core struct {
|
|
components ComponentFinder
|
|
template *template.Template
|
|
variables map[string]string
|
|
builds map[string]*template.Builder
|
|
}
|
|
|
|
// CoreConfig is the structure for initializing a new Core. Once a CoreConfig
|
|
// is used to initialize a Core, it shouldn't be re-used or modified again.
|
|
type CoreConfig struct {
|
|
Components ComponentFinder
|
|
Template *template.Template
|
|
Variables map[string]string
|
|
}
|
|
|
|
// The function type used to lookup Builder implementations.
|
|
type BuilderFunc func(name string) (Builder, error)
|
|
|
|
// The function type used to lookup Hook implementations.
|
|
type HookFunc func(name string) (Hook, error)
|
|
|
|
// The function type used to lookup PostProcessor implementations.
|
|
type PostProcessorFunc func(name string) (PostProcessor, error)
|
|
|
|
// The function type used to lookup Provisioner implementations.
|
|
type ProvisionerFunc func(name string) (Provisioner, error)
|
|
|
|
// ComponentFinder is a struct that contains the various function
|
|
// pointers necessary to look up components of Packer such as builders,
|
|
// commands, etc.
|
|
type ComponentFinder struct {
|
|
Builder BuilderFunc
|
|
Hook HookFunc
|
|
PostProcessor PostProcessorFunc
|
|
Provisioner ProvisionerFunc
|
|
}
|
|
|
|
// NewCore creates a new Core.
|
|
func NewCore(c *CoreConfig) (*Core, error) {
|
|
// Go through and interpolate all the build names. We shuld be able
|
|
// to do this at this point with the variables.
|
|
builds := make(map[string]*template.Builder)
|
|
for _, b := range c.Template.Builders {
|
|
v, err := interpolate.Render(b.Name, &interpolate.Context{
|
|
UserVariables: c.Variables,
|
|
})
|
|
if err != nil {
|
|
return nil, fmt.Errorf(
|
|
"Error interpolating builder '%s': %s",
|
|
b.Name, err)
|
|
}
|
|
|
|
builds[v] = b
|
|
}
|
|
|
|
return &Core{
|
|
components: c.Components,
|
|
template: c.Template,
|
|
variables: c.Variables,
|
|
builds: builds,
|
|
}, nil
|
|
}
|
|
|
|
// BuildNames returns the builds that are available in this configured core.
|
|
func (c *Core) BuildNames() []string {
|
|
r := make([]string, 0, len(c.builds))
|
|
for n, _ := range c.builds {
|
|
r = append(r, n)
|
|
}
|
|
sort.Strings(r)
|
|
|
|
return r
|
|
}
|
|
|
|
// Build returns the Build object for the given name.
|
|
func (c *Core) Build(n string) (Build, error) {
|
|
// Setup the builder
|
|
configBuilder, ok := c.builds[n]
|
|
if !ok {
|
|
return nil, fmt.Errorf("no such build found: %s", n)
|
|
}
|
|
builder, err := c.components.Builder(configBuilder.Type)
|
|
if err != nil {
|
|
return nil, fmt.Errorf(
|
|
"error initializing builder '%s': %s",
|
|
configBuilder.Type, err)
|
|
}
|
|
if builder == nil {
|
|
return nil, fmt.Errorf(
|
|
"builder type not found: %s", configBuilder.Type)
|
|
}
|
|
|
|
// rawName is the uninterpolated name that we use for various lookups
|
|
rawName := configBuilder.Name
|
|
|
|
// Setup the provisioners for this build
|
|
provisioners := make([]coreBuildProvisioner, 0, len(c.template.Provisioners))
|
|
for _, rawP := range c.template.Provisioners {
|
|
// If we're skipping this, then ignore it
|
|
if rawP.Skip(rawName) {
|
|
continue
|
|
}
|
|
|
|
// Get the provisioner
|
|
provisioner, err := c.components.Provisioner(rawP.Type)
|
|
if err != nil {
|
|
return nil, fmt.Errorf(
|
|
"error initializing provisioner '%s': %s",
|
|
rawP.Type, err)
|
|
}
|
|
if provisioner == nil {
|
|
return nil, fmt.Errorf(
|
|
"provisioner type not found: %s", rawP.Type)
|
|
}
|
|
|
|
// Get the configuration
|
|
config := make([]interface{}, 1, 2)
|
|
config[0] = rawP.Config
|
|
if rawP.Override != nil {
|
|
if override, ok := rawP.Override[rawName]; ok {
|
|
config = append(config, override)
|
|
}
|
|
}
|
|
|
|
// If we're pausing, we wrap the provisioner in a special pauser.
|
|
if rawP.PauseBefore > 0 {
|
|
provisioner = &PausedProvisioner{
|
|
PauseBefore: rawP.PauseBefore,
|
|
Provisioner: provisioner,
|
|
}
|
|
}
|
|
|
|
provisioners = append(provisioners, coreBuildProvisioner{
|
|
provisioner: provisioner,
|
|
config: config,
|
|
})
|
|
}
|
|
|
|
// Setup the post-processors
|
|
postProcessors := make([][]coreBuildPostProcessor, 0, len(c.template.PostProcessors))
|
|
for _, rawPs := range c.template.PostProcessors {
|
|
current := make([]coreBuildPostProcessor, 0, len(rawPs))
|
|
for _, rawP := range rawPs {
|
|
// If we skip, ignore
|
|
if rawP.Skip(rawName) {
|
|
continue
|
|
}
|
|
|
|
// Get the post-processor
|
|
postProcessor, err := c.components.PostProcessor(rawP.Type)
|
|
if err != nil {
|
|
return nil, fmt.Errorf(
|
|
"error initializing post-processor '%s': %s",
|
|
rawP.Type, err)
|
|
}
|
|
if postProcessor == nil {
|
|
return nil, fmt.Errorf(
|
|
"post-processor type not found: %s", rawP.Type)
|
|
}
|
|
|
|
current = append(current, coreBuildPostProcessor{
|
|
processor: postProcessor,
|
|
processorType: rawP.Type,
|
|
config: rawP.Config,
|
|
keepInputArtifact: rawP.KeepInputArtifact,
|
|
})
|
|
}
|
|
|
|
// If we have no post-processors in this chain, just continue.
|
|
if len(current) == 0 {
|
|
continue
|
|
}
|
|
|
|
postProcessors = append(postProcessors, current)
|
|
}
|
|
|
|
// TODO hooks one day
|
|
|
|
return &coreBuild{
|
|
name: n,
|
|
builder: builder,
|
|
builderConfig: configBuilder.Config,
|
|
builderType: configBuilder.Type,
|
|
postProcessors: postProcessors,
|
|
provisioners: provisioners,
|
|
variables: c.variables,
|
|
}, nil
|
|
}
|
|
|
|
// Validate does a full validation of the template.
|
|
//
|
|
// This will automatically call template.Validate() in addition to doing
|
|
// richer semantic checks around variables and so on.
|
|
func (c *Core) Validate() error {
|
|
// First validate the template in general, we can't do anything else
|
|
// unless the template itself is valid.
|
|
if err := c.template.Validate(); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Validate variables are set
|
|
var err error
|
|
for n, v := range c.template.Variables {
|
|
if v.Required {
|
|
if _, ok := c.variables[n]; !ok {
|
|
err = multierror.Append(err, fmt.Errorf(
|
|
"required variable not set: %s", n))
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO: validate all builders exist
|
|
// TODO: ^^ provisioner
|
|
// TODO: ^^ post-processor
|
|
|
|
return err
|
|
}
|