package lxd import ( "context" "fmt" "log" "strconv" "time" "github.com/hashicorp/packer/helper/multistep" "github.com/hashicorp/packer/packer" ) type stepLxdLaunch struct{} func (s *stepLxdLaunch) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction { config := state.Get("config").(*Config) ui := state.Get("ui").(packer.Ui) name := config.ContainerName image := config.Image profile := fmt.Sprintf("--profile=%s", config.Profile) launch_args := []string{ "launch", "--ephemeral=false", profile, image, name, } for k, v := range config.LaunchConfig { launch_args = append(launch_args, "--config", fmt.Sprintf("%s=%s", k, v)) } ui.Say("Creating container...") _, err := LXDCommand(launch_args...) if err != nil { err := fmt.Errorf("Error creating container: %s", err) state.Put("error", err) ui.Error(err.Error()) return multistep.ActionHalt } sleep_seconds, err := strconv.Atoi(config.InitSleep) if err != nil { err := fmt.Errorf("Error parsing InitSleep into int: %s", err) state.Put("error", err) ui.Error(err.Error()) return multistep.ActionHalt } // TODO: Should we check `lxc info ` for "Running"? // We have to do this so /tmp doesn't get cleared and lose our provisioner scripts. time.Sleep(time.Duration(sleep_seconds) * time.Second) log.Printf("Sleeping for %d seconds...", sleep_seconds) return multistep.ActionContinue } func (s *stepLxdLaunch) Cleanup(state multistep.StateBag) { config := state.Get("config").(*Config) ui := state.Get("ui").(packer.Ui) cleanup_args := []string{ "delete", "--force", config.ContainerName, } ui.Say("Unregistering and deleting deleting container...") if _, err := LXDCommand(cleanup_args...); err != nil { ui.Error(fmt.Sprintf("Error deleting container: %s", err)) } }