builder/docker: change commit to the opt-in, special case build steps

/cc @andytson - Wanted to CC you in here so you could see some changes I
made.

First, I changed commit to opt-in, so you set "commit": true to get
commit behavior. This simplifies the logic a bit. Then, I removed the
skipping for StepExport and StepCommit and put that into the Builder
itself. This simplifies those steps (limits abstraction leakage).
Otherwise, everything looks great!
This commit is contained in:
Mitchell Hashimoto 2014-09-04 18:03:15 -07:00
parent 5f126dc154
commit 5cb7355814
7 changed files with 33 additions and 77 deletions

View File

@ -8,6 +8,7 @@ import (
) )
const BuilderId = "packer.docker" const BuilderId = "packer.docker"
const BuilderIdImport = "packer.post-processor.docker-import"
type Builder struct { type Builder struct {
config *Config config *Config
@ -35,8 +36,12 @@ func (b *Builder) Run(ui packer.Ui, hook packer.Hook, cache packer.Cache) (packe
&StepPull{}, &StepPull{},
&StepRun{}, &StepRun{},
&StepProvision{}, &StepProvision{},
&StepCommit{}, }
&StepExport{},
if b.config.Commit {
steps = append(steps, new(StepCommit))
} else {
steps = append(steps, new(StepExport))
} }
// Setup the state bag and initial state for the steps // Setup the state bag and initial state for the steps
@ -67,14 +72,14 @@ func (b *Builder) Run(ui packer.Ui, hook packer.Hook, cache packer.Cache) (packe
var artifact packer.Artifact var artifact packer.Artifact
// No errors, must've worked // No errors, must've worked
if b.config.Export { if b.config.Commit {
artifact = &ExportArtifact{path: b.config.ExportPath}
} else {
artifact = &ImportArtifact{ artifact = &ImportArtifact{
IdValue: state.Get("image_id").(string), IdValue: state.Get("image_id").(string),
BuilderIdValue: "packer.post-processor.docker-import", BuilderIdValue: BuilderIdImport,
Driver: driver, Driver: driver,
} }
} else {
artifact = &ExportArtifact{path: b.config.ExportPath}
} }
return artifact, nil return artifact, nil
} }

View File

@ -9,8 +9,8 @@ import (
type Config struct { type Config struct {
common.PackerConfig `mapstructure:",squash"` common.PackerConfig `mapstructure:",squash"`
Commit bool
ExportPath string `mapstructure:"export_path"` ExportPath string `mapstructure:"export_path"`
Export bool
Image string Image string
Pull bool Pull bool
RunCommand []string `mapstructure:"run_command"` RunCommand []string `mapstructure:"run_command"`
@ -72,13 +72,16 @@ func NewConfig(raws ...interface{}) (*Config, []string, error) {
} }
} }
c.Export = c.ExportPath != ""
if c.Image == "" { if c.Image == "" {
errs = packer.MultiErrorAppend(errs, errs = packer.MultiErrorAppend(errs,
fmt.Errorf("image must be specified")) fmt.Errorf("image must be specified"))
} }
if c.ExportPath != "" && c.Commit {
errs = packer.MultiErrorAppend(errs,
fmt.Errorf("both commit and export_path cannot be set"))
}
if errs != nil && len(errs.Errors) > 0 { if errs != nil && len(errs.Errors) > 0 {
return nil, nil, errs return nil, nil, errs
} }

View File

@ -46,19 +46,27 @@ func TestConfigPrepare_exportPath(t *testing.T) {
// No export path // No export path
delete(raw, "export_path") delete(raw, "export_path")
c, warns, errs := NewConfig(raw) _, warns, errs := NewConfig(raw)
testConfigOk(t, warns, errs) testConfigOk(t, warns, errs)
if c.Export {
t.Fatal("should not export")
}
// Good export path // Good export path
raw["export_path"] = "good" raw["export_path"] = "good"
c, warns, errs = NewConfig(raw) _, warns, errs = NewConfig(raw)
testConfigOk(t, warns, errs)
}
func TestConfigPrepare_exportPathAndCommit(t *testing.T) {
raw := testConfig()
raw["commit"] = true
// No export path
_, warns, errs := NewConfig(raw)
testConfigErr(t, warns, errs)
// No commit
raw["commit"] = false
_, warns, errs = NewConfig(raw)
testConfigOk(t, warns, errs) testConfigOk(t, warns, errs)
if !c.Export {
t.Fatal("should export")
}
} }
func TestConfigPrepare_image(t *testing.T) { func TestConfigPrepare_image(t *testing.T) {

View File

@ -12,15 +12,10 @@ type StepCommit struct {
} }
func (s *StepCommit) Run(state multistep.StateBag) multistep.StepAction { func (s *StepCommit) Run(state multistep.StateBag) multistep.StepAction {
config := state.Get("config").(*Config)
driver := state.Get("driver").(Driver) driver := state.Get("driver").(Driver)
containerId := state.Get("container_id").(string) containerId := state.Get("container_id").(string)
ui := state.Get("ui").(packer.Ui) ui := state.Get("ui").(packer.Ui)
if config.Export {
return multistep.ActionContinue
}
ui.Say("Committing the container") ui.Say("Committing the container")
imageId, err := driver.Commit(containerId) imageId, err := driver.Commit(containerId)
if err != nil { if err != nil {

View File

@ -21,8 +21,6 @@ func TestStepCommit(t *testing.T) {
step := new(StepCommit) step := new(StepCommit)
defer step.Cleanup(state) defer step.Cleanup(state)
config := state.Get("config").(*Config)
config.Export = false
driver := state.Get("driver").(*MockDriver) driver := state.Get("driver").(*MockDriver)
driver.CommitImageId = "bar" driver.CommitImageId = "bar"
@ -48,38 +46,11 @@ func TestStepCommit(t *testing.T) {
} }
} }
func TestStepCommit_skip(t *testing.T) {
state := testStepCommitState(t)
step := new(StepCommit)
defer step.Cleanup(state)
config := state.Get("config").(*Config)
config.Export = true
driver := state.Get("driver").(*MockDriver)
// run the step
if action := step.Run(state); action != multistep.ActionContinue {
t.Fatalf("bad action: %#v", action)
}
// verify we did the right thing
if driver.CommitCalled {
t.Fatal("shouldn't have called")
}
// verify the ID is not saved
if _, ok := state.GetOk("image_id"); ok {
t.Fatal("shouldn't save image ID")
}
}
func TestStepCommit_error(t *testing.T) { func TestStepCommit_error(t *testing.T) {
state := testStepCommitState(t) state := testStepCommitState(t)
step := new(StepCommit) step := new(StepCommit)
defer step.Cleanup(state) defer step.Cleanup(state)
config := state.Get("config").(*Config)
config.Export = false
driver := state.Get("driver").(*MockDriver) driver := state.Get("driver").(*MockDriver)
driver.CommitErr = errors.New("foo") driver.CommitErr = errors.New("foo")

View File

@ -13,10 +13,6 @@ type StepExport struct{}
func (s *StepExport) Run(state multistep.StateBag) multistep.StepAction { func (s *StepExport) Run(state multistep.StateBag) multistep.StepAction {
config := state.Get("config").(*Config) config := state.Get("config").(*Config)
if !config.Export {
return multistep.ActionContinue
}
driver := state.Get("driver").(Driver) driver := state.Get("driver").(Driver)
containerId := state.Get("container_id").(string) containerId := state.Get("container_id").(string)
ui := state.Get("ui").(packer.Ui) ui := state.Get("ui").(packer.Ui)

View File

@ -34,7 +34,6 @@ func TestStepExport(t *testing.T) {
config := state.Get("config").(*Config) config := state.Get("config").(*Config)
config.ExportPath = tf.Name() config.ExportPath = tf.Name()
config.Export = true
driver := state.Get("driver").(*MockDriver) driver := state.Get("driver").(*MockDriver)
driver.ExportReader = bytes.NewReader([]byte("data!")) driver.ExportReader = bytes.NewReader([]byte("data!"))
@ -62,26 +61,6 @@ func TestStepExport(t *testing.T) {
} }
} }
func TestStepExport_skip(t *testing.T) {
state := testStepExportState(t)
step := new(StepExport)
defer step.Cleanup(state)
config := state.Get("config").(*Config)
config.Export = false
driver := state.Get("driver").(*MockDriver)
// run the step
if action := step.Run(state); action != multistep.ActionContinue {
t.Fatalf("bad action: %#v", action)
}
// verify we did the right thing
if driver.ExportCalled {
t.Fatal("shouldn't have exported")
}
}
func TestStepExport_error(t *testing.T) { func TestStepExport_error(t *testing.T) {
state := testStepExportState(t) state := testStepExportState(t)
step := new(StepExport) step := new(StepExport)
@ -100,7 +79,6 @@ func TestStepExport_error(t *testing.T) {
config := state.Get("config").(*Config) config := state.Get("config").(*Config)
config.ExportPath = tf.Name() config.ExportPath = tf.Name()
config.Export = true
driver := state.Get("driver").(*MockDriver) driver := state.Get("driver").(*MockDriver)
driver.ExportError = errors.New("foo") driver.ExportError = errors.New("foo")