builder/amazon/chroot: mount extra paths

This commit is contained in:
Mitchell Hashimoto 2013-07-30 12:08:16 -07:00
parent 9bb9f02b99
commit 462e48cac4
3 changed files with 96 additions and 15 deletions

View File

@ -24,13 +24,13 @@ type Config struct {
common.PackerConfig `mapstructure:",squash"` common.PackerConfig `mapstructure:",squash"`
awscommon.AccessConfig `mapstructure:",squash"` awscommon.AccessConfig `mapstructure:",squash"`
AttachedDevicePath string `mapstructure:"attached_device_path"` AttachedDevicePath string `mapstructure:"attached_device_path"`
ChrootMounts []string `mapstructure:"chroot_mounts"` ChrootMounts [][]string `mapstructure:"chroot_mounts"`
DevicePath string `mapstructure:"device_path"` DevicePath string `mapstructure:"device_path"`
MountCommand string `mapstructure:"mount_command"` MountCommand string `mapstructure:"mount_command"`
MountPath string `mapstructure:"mount_path"` MountPath string `mapstructure:"mount_path"`
SourceAmi string `mapstructure:"source_ami"` SourceAmi string `mapstructure:"source_ami"`
UnmountCommand string `mapstructure:"unmount_command"` UnmountCommand string `mapstructure:"unmount_command"`
} }
type Builder struct { type Builder struct {
@ -46,16 +46,16 @@ func (b *Builder) Prepare(raws ...interface{}) error {
// Defaults // Defaults
if b.config.ChrootMounts == nil { if b.config.ChrootMounts == nil {
b.config.ChrootMounts = make([]string, 0) b.config.ChrootMounts = make([][]string, 0)
} }
if len(b.config.ChrootMounts) == 0 { if len(b.config.ChrootMounts) == 0 {
b.config.ChrootMounts = []string{ b.config.ChrootMounts = [][]string{
"{{.MountCommand}} -t proc proc {{.MountPath}}/proc", []string{"proc", "proc", "/proc"},
"{{.MountCommand}} -t sysfs sysfs {{.MountPath}}/sys", []string{"sysfs", "sysfs", "/sys"},
"{{.MountCommand}} -t bind /dev {{.MountPath}}/dev", []string{"bind", "/dev", "/dev"},
"{{.MountCommand}} -t devpts devpts {{.MountPath}}/dev/pts", []string{"devpts", "devpts", "/dev/pts"},
"{{.MountCommand}} -t binfmt_misc binfmt_misc {{.MountPath}}/proc/sys/fs/binfmt_misc", []string{"binfmt_misc", "binfmt_misc", "/proc/sys/fs/binfmt_misc"},
} }
} }
@ -126,6 +126,7 @@ func (b *Builder) Run(ui packer.Ui, hook packer.Hook, cache packer.Cache) (packe
&StepCreateVolume{}, &StepCreateVolume{},
&StepAttachVolume{}, &StepAttachVolume{},
&StepMountDevice{}, &StepMountDevice{},
&StepMountExtra{},
} }
// Run! // Run!

View File

@ -32,7 +32,7 @@ func (s *StepMountDevice) Run(state map[string]interface{}) multistep.StepAction
mountPathRaw := new(bytes.Buffer) mountPathRaw := new(bytes.Buffer)
t := template.Must(template.New("mountPath").Parse(config.MountPath)) t := template.Must(template.New("mountPath").Parse(config.MountPath))
t.Execute(mountPathRaw, &mountPathData{ t.Execute(mountPathRaw, &mountPathData{
Device: filepath.Basename(device), Device: filepath.Base(device),
}) })
mountPath := mountPathRaw.String() mountPath := mountPathRaw.String()
@ -60,6 +60,7 @@ func (s *StepMountDevice) Run(state map[string]interface{}) multistep.StepAction
// Set the mount path so we remember to unmount it later // Set the mount path so we remember to unmount it later
s.mountPath = mountPath s.mountPath = mountPath
state["mount_path"] = s.mountPath
return multistep.ActionContinue return multistep.ActionContinue
} }

View File

@ -0,0 +1,79 @@
package chroot
import (
"bytes"
"fmt"
"github.com/mitchellh/multistep"
"github.com/mitchellh/packer/packer"
"os"
"os/exec"
)
// StepMountExtra mounts the attached device.
//
// Produces:
// mount_path string - The location where the volume was mounted.
type StepMountExtra struct {
mounts []string
}
func (s *StepMountExtra) Run(state map[string]interface{}) multistep.StepAction {
config := state["config"].(*Config)
mountPath := state["mount_path"].(string)
ui := state["ui"].(packer.Ui)
s.mounts = make([]string, 0, len(config.ChrootMounts))
ui.Say("Mounting additional paths within the chroot...")
for _, mountInfo := range config.ChrootMounts {
innerPath := mountPath + mountInfo[2]
if err := os.MkdirAll(innerPath, 0755); err != nil {
err := fmt.Errorf("Error creating mount directory: %s", err)
state["error"] = err
ui.Error(err.Error())
return multistep.ActionHalt
}
ui.Message(fmt.Sprintf("Mounting: %s", mountInfo[2]))
stderr := new(bytes.Buffer)
mountCommand := fmt.Sprintf(
"%s -t %s %s %s",
config.MountCommand,
mountInfo[0],
mountInfo[1],
innerPath)
cmd := exec.Command("/bin/sh", "-c", mountCommand)
cmd.Stderr = stderr
if err := cmd.Run(); err != nil {
err := fmt.Errorf(
"Error mounting: %s\nStderr: %s", err, stderr.String())
state["error"] = err
ui.Error(err.Error())
return multistep.ActionHalt
}
s.mounts = append(s.mounts, innerPath)
}
return multistep.ActionContinue
}
func (s *StepMountExtra) Cleanup(state map[string]interface{}) {
if s.mounts == nil {
return
}
config := state["config"].(*Config)
ui := state["ui"].(packer.Ui)
for _, path := range s.mounts {
unmountCommand := fmt.Sprintf("%s %s", config.UnmountCommand, path)
cmd := exec.Command("bin/sh", "-c", unmountCommand)
if err := cmd.Run(); err != nil {
ui.Error(fmt.Sprintf(
"Error unmounting root device: %s", err))
return
}
}
}