package lxd import ( "context" "fmt" "io" "log" "os" "os/exec" "path/filepath" "syscall" "github.com/hashicorp/packer/packer" ) type Communicator struct { ContainerName string CmdWrapper CommandWrapper } func (c *Communicator) Start(ctx context.Context, cmd *packer.RemoteCmd) error { localCmd, err := c.Execute(cmd.Command) if err != nil { return err } localCmd.Stdin = cmd.Stdin localCmd.Stdout = cmd.Stdout localCmd.Stderr = cmd.Stderr if err := localCmd.Start(); err != nil { return err } go func() { exitStatus := 0 if err := localCmd.Wait(); err != nil { if exitErr, ok := err.(*exec.ExitError); ok { exitStatus = 1 // There is no process-independent way to get the REAL // exit status so we just try to go deeper. if status, ok := exitErr.Sys().(syscall.WaitStatus); ok { exitStatus = status.ExitStatus() } } } log.Printf( "lxc exec execution exited with '%d': '%s'", exitStatus, cmd.Command) cmd.SetExited(exitStatus) }() return nil } func (c *Communicator) Upload(dst string, r io.Reader, fi *os.FileInfo) error { ctx := context.TODO() fileDestination := filepath.Join(c.ContainerName, dst) // find out if the place we are pushing to is a directory testDirectoryCommand := fmt.Sprintf(`test -d "%s"`, dst) cmd := &packer.RemoteCmd{Command: testDirectoryCommand} err := c.Start(ctx, cmd) if err != nil { log.Printf("Unable to check whether remote path is a dir: %s", err) return err } cmd.Wait() if cmd.ExitStatus() == 0 { log.Printf("path is a directory; copying file into directory.") fileDestination = filepath.Join(c.ContainerName, dst, (*fi).Name()) } cpCmd, err := c.CmdWrapper(fmt.Sprintf("lxc file push - %s", fileDestination)) if err != nil { return err } log.Printf("Running copy command: %s", cpCmd) command := ShellCommand(cpCmd) command.Stdin = r return command.Run() } func (c *Communicator) UploadDir(dst string, src string, exclude []string) error { fileDestination := filepath.Join(c.ContainerName, dst) cp, err := c.CmdWrapper(fmt.Sprintf("lxc file push -pr %s %s", src, fileDestination)) if err != nil { return err } cpCmd := ShellCommand(cp) log.Printf("Running cp command: %s", cp) err = cpCmd.Run() if err != nil { log.Printf("Error running cp command: %s", err) return err } return nil } func (c *Communicator) Download(src string, w io.Writer) error { cpCmd, err := c.CmdWrapper(fmt.Sprintf("lxc file pull %s -", filepath.Join(c.ContainerName, src))) if err != nil { return err } log.Printf("Running copy command: %s", cpCmd) command := ShellCommand(cpCmd) command.Stdout = w return command.Run() } func (c *Communicator) DownloadDir(src string, dst string, exclude []string) error { // TODO This could probably be "lxc exec -- cd && tar -czf - | tar -xzf - -C " return fmt.Errorf("DownloadDir is not implemented for lxc") } func (c *Communicator) Execute(commandString string) (*exec.Cmd, error) { log.Printf("Executing with lxc exec in container: %s %s", c.ContainerName, commandString) command, err := c.CmdWrapper( fmt.Sprintf("lxc exec %s -- /bin/sh -c \"%s\"", c.ContainerName, commandString)) if err != nil { return nil, err } localCmd := ShellCommand(command) log.Printf("Executing lxc exec: %s %#v", localCmd.Path, localCmd.Args) return localCmd, nil }