packer-cn/packer/build.go

110 lines
3.0 KiB
Go
Raw Normal View History

2013-03-25 19:29:26 -04:00
package packer
2013-05-09 16:26:40 -04:00
import "log"
2013-05-03 23:45:38 -04:00
// A Build represents a single job within Packer that is responsible for
// building some machine image artifact. Builds are meant to be parallelized.
type Build interface {
2013-06-14 15:15:51 -04:00
// Name is the name of the build. This is unique across a single template,
// but not absolutely unique. This is meant more to describe to the user
// what is being built rather than being a unique identifier.
2013-05-09 14:32:03 -04:00
Name() string
2013-06-14 15:15:51 -04:00
// Prepare configures the various components of this build and reports
// any errors in doing so (such as syntax errors, validation errors, etc.)
Prepare() error
2013-06-14 15:15:51 -04:00
// Run runs the actual builder, returning an artifact implementation
// of what is built. If anything goes wrong, an error is returned.
Run(Ui, Cache) (Artifact, error)
2013-06-14 15:15:51 -04:00
// Cancel will cancel a running build. This will block until the build
// is actually completely cancelled.
2013-06-03 19:03:08 -04:00
Cancel()
2013-05-03 23:45:38 -04:00
}
2013-03-25 19:29:26 -04:00
// A build struct represents a single build job, the result of which should
// be a single machine image artifact. This artifact may be comprised of
// multiple files, of course, but it should be for only a single provider
// (such as VirtualBox, EC2, etc.).
2013-05-03 23:45:38 -04:00
type coreBuild struct {
name string
builder Builder
builderConfig interface{}
hooks map[string][]Hook
provisioners []coreBuildProvisioner
2013-04-20 22:03:53 -04:00
prepareCalled bool
2013-03-25 19:29:26 -04:00
}
// Keeps track of the provisioner and the configuration of the provisioner
// within the build.
type coreBuildProvisioner struct {
provisioner Provisioner
config []interface{}
}
2013-05-09 14:32:03 -04:00
// Returns the name of the build.
func (b *coreBuild) Name() string {
return b.name
}
2013-04-20 21:55:02 -04:00
// Prepare prepares the build by doing some initialization for the builder
// and any hooks. This _must_ be called prior to Run.
func (b *coreBuild) Prepare() (err error) {
// TODO: lock
2013-04-20 22:03:53 -04:00
b.prepareCalled = true
// Prepare the builder
err = b.builder.Prepare(b.builderConfig)
2013-05-09 16:26:40 -04:00
if err != nil {
log.Printf("Build '%s' prepare failure: %s\n", b.name, err)
return
2013-05-09 16:26:40 -04:00
}
// Prepare the provisioners
for _, coreProv := range b.provisioners {
if err = coreProv.provisioner.Prepare(coreProv.config...); err != nil {
return
}
}
2013-05-09 16:26:40 -04:00
return
2013-04-20 21:55:02 -04:00
}
// Runs the actual build. Prepare must be called prior to running this.
func (b *coreBuild) Run(ui Ui, cache Cache) (Artifact, error) {
2013-04-20 22:03:53 -04:00
if !b.prepareCalled {
panic("Prepare must be called first")
}
// Copy the hooks
hooks := make(map[string][]Hook)
for hookName, hookList := range b.hooks {
hooks[hookName] = make([]Hook, len(hookList))
copy(hooks[hookName], hookList)
}
// Add a hook for the provisioners if we have provisioners
if len(b.provisioners) > 0 {
provisioners := make([]Provisioner, len(b.provisioners))
for i, p := range b.provisioners {
provisioners[i] = p.provisioner
}
if _, ok := hooks[HookProvision]; !ok {
hooks[HookProvision] = make([]Hook, 0, 1)
}
hooks[HookProvision] = append(hooks[HookProvision], &ProvisionHook{provisioners})
}
hook := &DispatchHook{hooks}
2013-06-10 01:00:47 -04:00
return b.builder.Run(ui, hook, cache)
2013-04-20 21:55:02 -04:00
}
2013-06-03 19:03:08 -04:00
// Cancels the build if it is running.
func (b *coreBuild) Cancel() {
b.builder.Cancel()
2013-06-03 19:03:08 -04:00
}