file provisioner improvements
* allow specify source/destination as dir * allow specify many files as source Signed-off-by: Vasiliy Tolstov <v.tolstov@selfip.ru>
This commit is contained in:
parent
6496a25d08
commit
feee19e4ed
|
@ -3,7 +3,6 @@ package chroot
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/mitchellh/packer/packer"
|
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
"log"
|
||||||
|
@ -12,6 +11,8 @@ import (
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
|
||||||
|
"github.com/mitchellh/packer/packer"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Communicator is a special communicator that works by executing
|
// Communicator is a special communicator that works by executing
|
||||||
|
@ -114,6 +115,10 @@ func (c *Communicator) UploadDir(dst string, src string, exclude []string) error
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Communicator) DownloadDir(src string, dst string, exclude []string) error {
|
||||||
|
return fmt.Errorf("DownloadDir is not implemented for amazon-chroot")
|
||||||
|
}
|
||||||
|
|
||||||
func (c *Communicator) Download(src string, w io.Writer) error {
|
func (c *Communicator) Download(src string, w io.Writer) error {
|
||||||
src = filepath.Join(c.Chroot, src)
|
src = filepath.Join(c.Chroot, src)
|
||||||
log.Printf("Downloading from chroot dir: %s", src)
|
log.Printf("Downloading from chroot dir: %s", src)
|
||||||
|
|
|
@ -233,6 +233,10 @@ func (c *Communicator) Download(src string, dst io.Writer) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Communicator) DownloadDir(src string, dst string, exclude []string) error {
|
||||||
|
return fmt.Errorf("DownloadDir is not implemented for docker")
|
||||||
|
}
|
||||||
|
|
||||||
// canExec tells us whether `docker exec` is supported
|
// canExec tells us whether `docker exec` is supported
|
||||||
func (c *Communicator) canExec() bool {
|
func (c *Communicator) canExec() bool {
|
||||||
execConstraint, err := version.NewConstraint(">= 1.4.0")
|
execConstraint, err := version.NewConstraint(">= 1.4.0")
|
||||||
|
|
|
@ -12,6 +12,7 @@ import (
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -156,12 +157,73 @@ func (c *comm) UploadDir(dst string, src string, excl []string) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *comm) DownloadDir(src string, dst string, excl []string) error {
|
||||||
|
log.Printf("Download dir '%s' to '%s'", src, dst)
|
||||||
|
scpFunc := func(w io.Writer, stdoutR *bufio.Reader) error {
|
||||||
|
for {
|
||||||
|
fmt.Fprint(w, "\x00")
|
||||||
|
|
||||||
|
// read file info
|
||||||
|
fi, err := stdoutR.ReadString('\n')
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(fi) < 0 {
|
||||||
|
return fmt.Errorf("empty response from server")
|
||||||
|
}
|
||||||
|
|
||||||
|
switch fi[0] {
|
||||||
|
case '\x01', '\x02':
|
||||||
|
return fmt.Errorf("%s", fi[1:len(fi)])
|
||||||
|
case 'C', 'D':
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
return fmt.Errorf("unexpected server response (%x)", fi[0])
|
||||||
|
}
|
||||||
|
|
||||||
|
var mode string
|
||||||
|
var size int64
|
||||||
|
var name string
|
||||||
|
log.Printf("Download dir str:%s", fi)
|
||||||
|
n, err := fmt.Sscanf(fi, "%6s %d %s", &mode, &size, &name)
|
||||||
|
if err != nil || n != 3 {
|
||||||
|
return fmt.Errorf("can't parse server response (%s)", fi)
|
||||||
|
}
|
||||||
|
if size < 0 {
|
||||||
|
return fmt.Errorf("negative file size")
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Printf("Download dir mode:%s size:%d name:%s", mode, size, name)
|
||||||
|
switch fi[0] {
|
||||||
|
case 'D':
|
||||||
|
err = os.MkdirAll(filepath.Join(dst, name), os.FileMode(0755))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fmt.Fprint(w, "\x00")
|
||||||
|
return nil
|
||||||
|
case 'C':
|
||||||
|
fmt.Fprint(w, "\x00")
|
||||||
|
err = scpDownloadFile(filepath.Join(dst, name), stdoutR, size, os.FileMode(0644))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := checkSCPStatus(stdoutR); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return c.scpSession("scp -vrf "+src, scpFunc)
|
||||||
|
}
|
||||||
|
|
||||||
func (c *comm) Download(path string, output io.Writer) error {
|
func (c *comm) Download(path string, output io.Writer) error {
|
||||||
if c.config.UseSftp {
|
if c.config.UseSftp {
|
||||||
return c.sftpDownloadSession(path, output)
|
return c.sftpDownloadSession(path, output)
|
||||||
} else {
|
|
||||||
return c.scpDownloadSession(path, output)
|
|
||||||
}
|
}
|
||||||
|
return c.scpDownloadSession(path, output)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *comm) newSession() (session *ssh.Session, err error) {
|
func (c *comm) newSession() (session *ssh.Session, err error) {
|
||||||
|
@ -563,6 +625,9 @@ func (c *comm) scpDownloadSession(path string, output io.Writer) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if strings.Index(path, " ") == -1 {
|
||||||
|
return c.scpSession("scp -vf "+path, scpFunc)
|
||||||
|
}
|
||||||
return c.scpSession("scp -vf "+strconv.Quote(path), scpFunc)
|
return c.scpSession("scp -vf "+strconv.Quote(path), scpFunc)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -668,6 +733,18 @@ func checkSCPStatus(r *bufio.Reader) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func scpDownloadFile(dst string, src io.Reader, size int64, mode os.FileMode) error {
|
||||||
|
f, err := os.OpenFile(dst, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, mode)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
if _, err := io.CopyN(f, src, size); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func scpUploadFile(dst string, src io.Reader, w io.Writer, r *bufio.Reader, fi *os.FileInfo) error {
|
func scpUploadFile(dst string, src io.Reader, w io.Writer, r *bufio.Reader, fi *os.FileInfo) error {
|
||||||
var mode os.FileMode
|
var mode os.FileMode
|
||||||
var size int64
|
var size int64
|
||||||
|
|
|
@ -140,6 +140,10 @@ func (c *Communicator) Download(src string, dst io.Writer) error {
|
||||||
return fmt.Errorf("WinRM doesn't support download.")
|
return fmt.Errorf("WinRM doesn't support download.")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Communicator) DownloadDir(src string, dst string, exclude []string) error {
|
||||||
|
return fmt.Errorf("WinRM doesn't support download dir.")
|
||||||
|
}
|
||||||
|
|
||||||
func (c *Communicator) newCopyClient() (*winrmcp.Winrmcp, error) {
|
func (c *Communicator) newCopyClient() (*winrmcp.Winrmcp, error) {
|
||||||
addr := fmt.Sprintf("%s:%d", c.endpoint.Host, c.endpoint.Port)
|
addr := fmt.Sprintf("%s:%d", c.endpoint.Host, c.endpoint.Port)
|
||||||
return winrmcp.New(addr, &winrmcp.Config{
|
return winrmcp.New(addr, &winrmcp.Config{
|
||||||
|
|
|
@ -75,6 +75,8 @@ type Communicator interface {
|
||||||
// with the contents writing to the given writer. This method will
|
// with the contents writing to the given writer. This method will
|
||||||
// block until it completes.
|
// block until it completes.
|
||||||
Download(string, io.Writer) error
|
Download(string, io.Writer) error
|
||||||
|
|
||||||
|
DownloadDir(src string, dst string, exclude []string) error
|
||||||
}
|
}
|
||||||
|
|
||||||
// StartWithUi runs the remote command and streams the output to any
|
// StartWithUi runs the remote command and streams the output to any
|
||||||
|
|
|
@ -25,6 +25,10 @@ type MockCommunicator struct {
|
||||||
UploadDirSrc string
|
UploadDirSrc string
|
||||||
UploadDirExclude []string
|
UploadDirExclude []string
|
||||||
|
|
||||||
|
DownloadDirDst string
|
||||||
|
DownloadDirSrc string
|
||||||
|
DownloadDirExclude []string
|
||||||
|
|
||||||
DownloadCalled bool
|
DownloadCalled bool
|
||||||
DownloadPath string
|
DownloadPath string
|
||||||
DownloadData string
|
DownloadData string
|
||||||
|
@ -98,3 +102,11 @@ func (c *MockCommunicator) Download(path string, w io.Writer) error {
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *MockCommunicator) DownloadDir(src string, dst string, excl []string) error {
|
||||||
|
c.DownloadDirDst = dst
|
||||||
|
c.DownloadDirSrc = src
|
||||||
|
c.DownloadDirExclude = excl
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
|
@ -52,6 +52,12 @@ type CommunicatorUploadDirArgs struct {
|
||||||
Exclude []string
|
Exclude []string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type CommunicatorDownloadDirArgs struct {
|
||||||
|
Dst string
|
||||||
|
Src string
|
||||||
|
Exclude []string
|
||||||
|
}
|
||||||
|
|
||||||
func Communicator(client *rpc.Client) *communicator {
|
func Communicator(client *rpc.Client) *communicator {
|
||||||
return &communicator{client: client}
|
return &communicator{client: client}
|
||||||
}
|
}
|
||||||
|
@ -135,6 +141,22 @@ func (c *communicator) UploadDir(dst string, src string, exclude []string) error
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *communicator) DownloadDir(src string, dst string, exclude []string) error {
|
||||||
|
args := &CommunicatorDownloadDirArgs{
|
||||||
|
Dst: dst,
|
||||||
|
Src: src,
|
||||||
|
Exclude: exclude,
|
||||||
|
}
|
||||||
|
|
||||||
|
var reply error
|
||||||
|
err := c.client.Call("Communicator.DownloadDir", args, &reply)
|
||||||
|
if err == nil {
|
||||||
|
err = reply
|
||||||
|
}
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
func (c *communicator) Download(path string, w io.Writer) (err error) {
|
func (c *communicator) Download(path string, w io.Writer) (err error) {
|
||||||
// Serve a single connection and a single copy
|
// Serve a single connection and a single copy
|
||||||
streamId := c.mux.NextId()
|
streamId := c.mux.NextId()
|
||||||
|
@ -253,6 +275,10 @@ func (c *CommunicatorServer) UploadDir(args *CommunicatorUploadDirArgs, reply *e
|
||||||
return c.c.UploadDir(args.Dst, args.Src, args.Exclude)
|
return c.c.UploadDir(args.Dst, args.Src, args.Exclude)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *CommunicatorServer) DownloadDir(args *CommunicatorUploadDirArgs, reply *error) error {
|
||||||
|
return c.c.DownloadDir(args.Src, args.Dst, args.Exclude)
|
||||||
|
}
|
||||||
|
|
||||||
func (c *CommunicatorServer) Download(args *CommunicatorDownloadArgs, reply *interface{}) (err error) {
|
func (c *CommunicatorServer) Download(args *CommunicatorDownloadArgs, reply *interface{}) (err error) {
|
||||||
writerC, err := c.mux.Dial(args.WriterStreamId)
|
writerC, err := c.mux.Dial(args.WriterStreamId)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -57,3 +57,7 @@ func (c *Communicator) UploadDir(string, string, []string) error {
|
||||||
func (c *Communicator) Download(string, io.Writer) error {
|
func (c *Communicator) Download(string, io.Writer) error {
|
||||||
return fmt.Errorf("download not supported")
|
return fmt.Errorf("download not supported")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Communicator) DownloadDir(src string, dst string, exclude []string) error {
|
||||||
|
return fmt.Errorf("downloadDir not supported")
|
||||||
|
}
|
||||||
|
|
|
@ -4,6 +4,7 @@ import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/mitchellh/packer/common"
|
"github.com/mitchellh/packer/common"
|
||||||
"github.com/mitchellh/packer/helper/config"
|
"github.com/mitchellh/packer/helper/config"
|
||||||
|
@ -15,7 +16,8 @@ type Config struct {
|
||||||
common.PackerConfig `mapstructure:",squash"`
|
common.PackerConfig `mapstructure:",squash"`
|
||||||
|
|
||||||
// The local path of the file to upload.
|
// The local path of the file to upload.
|
||||||
Source string
|
Source string
|
||||||
|
Sources []string
|
||||||
|
|
||||||
// The remote path where the local file will be uploaded to.
|
// The remote path where the local file will be uploaded to.
|
||||||
Destination string
|
Destination string
|
||||||
|
@ -52,14 +54,24 @@ func (p *Provisioner) Prepare(raws ...interface{}) error {
|
||||||
errs = packer.MultiErrorAppend(errs,
|
errs = packer.MultiErrorAppend(errs,
|
||||||
errors.New("Direction must be one of: download, upload."))
|
errors.New("Direction must be one of: download, upload."))
|
||||||
}
|
}
|
||||||
|
if p.config.Source != "" {
|
||||||
|
p.config.Sources = append(p.config.Sources, p.config.Source)
|
||||||
|
}
|
||||||
|
|
||||||
if p.config.Direction == "upload" {
|
if p.config.Direction == "upload" {
|
||||||
if _, err := os.Stat(p.config.Source); err != nil {
|
for _, src := range p.config.Sources {
|
||||||
errs = packer.MultiErrorAppend(errs,
|
if _, err := os.Stat(src); err != nil {
|
||||||
fmt.Errorf("Bad source '%s': %s", p.config.Source, err))
|
errs = packer.MultiErrorAppend(errs,
|
||||||
|
fmt.Errorf("Bad source '%s': %s", src, err))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if len(p.config.Sources) < 1 {
|
||||||
|
errs = packer.MultiErrorAppend(errs,
|
||||||
|
errors.New("Source must be specified."))
|
||||||
|
}
|
||||||
|
|
||||||
if p.config.Destination == "" {
|
if p.config.Destination == "" {
|
||||||
errs = packer.MultiErrorAppend(errs,
|
errs = packer.MultiErrorAppend(errs,
|
||||||
errors.New("Destination must be specified."))
|
errors.New("Destination must be specified."))
|
||||||
|
@ -81,50 +93,65 @@ func (p *Provisioner) Provision(ui packer.Ui, comm packer.Communicator) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Provisioner) ProvisionDownload(ui packer.Ui, comm packer.Communicator) error {
|
func (p *Provisioner) ProvisionDownload(ui packer.Ui, comm packer.Communicator) error {
|
||||||
ui.Say(fmt.Sprintf("Downloading %s => %s", p.config.Source, p.config.Destination))
|
for _, src := range p.config.Sources {
|
||||||
|
ui.Say(fmt.Sprintf("Downloading %s => %s", src, p.config.Destination))
|
||||||
|
|
||||||
f, err := os.OpenFile(p.config.Destination, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0644)
|
if strings.HasSuffix(p.config.Destination, "/") {
|
||||||
if err != nil {
|
err := os.MkdirAll(p.config.Destination, os.FileMode(0755))
|
||||||
return err
|
if err != nil {
|
||||||
}
|
return err
|
||||||
defer f.Close()
|
}
|
||||||
|
return comm.DownloadDir(src, p.config.Destination, nil)
|
||||||
|
}
|
||||||
|
|
||||||
err = comm.Download(p.config.Source, f)
|
f, err := os.OpenFile(p.config.Destination, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ui.Error(fmt.Sprintf("Download failed: %s", err))
|
return err
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
|
err = comm.Download(src, f)
|
||||||
|
if err != nil {
|
||||||
|
ui.Error(fmt.Sprintf("Download failed: %s", err))
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return err
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Provisioner) ProvisionUpload(ui packer.Ui, comm packer.Communicator) error {
|
func (p *Provisioner) ProvisionUpload(ui packer.Ui, comm packer.Communicator) error {
|
||||||
ui.Say(fmt.Sprintf("Uploading %s => %s", p.config.Source, p.config.Destination))
|
for _, src := range p.config.Sources {
|
||||||
info, err := os.Stat(p.config.Source)
|
ui.Say(fmt.Sprintf("Uploading %s => %s", src, p.config.Destination))
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we're uploading a directory, short circuit and do that
|
info, err := os.Stat(src)
|
||||||
if info.IsDir() {
|
if err != nil {
|
||||||
return comm.UploadDir(p.config.Destination, p.config.Source, nil)
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// We're uploading a file...
|
// If we're uploading a directory, short circuit and do that
|
||||||
f, err := os.Open(p.config.Source)
|
if info.IsDir() {
|
||||||
if err != nil {
|
return comm.UploadDir(p.config.Destination, src, nil)
|
||||||
return err
|
}
|
||||||
}
|
|
||||||
defer f.Close()
|
|
||||||
|
|
||||||
fi, err := f.Stat()
|
// We're uploading a file...
|
||||||
if err != nil {
|
f, err := os.Open(src)
|
||||||
return err
|
if err != nil {
|
||||||
}
|
return err
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
err = comm.Upload(p.config.Destination, f, &fi)
|
fi, err := f.Stat()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ui.Error(fmt.Sprintf("Upload failed: %s", err))
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = comm.Upload(p.config.Destination, f, &fi)
|
||||||
|
if err != nil {
|
||||||
|
ui.Error(fmt.Sprintf("Upload failed: %s", err))
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return err
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Provisioner) Cancel() {
|
func (p *Provisioner) Cancel() {
|
||||||
|
|
|
@ -76,6 +76,10 @@ func (c *Communicator) Download(string, io.Writer) error {
|
||||||
return fmt.Errorf("download not supported")
|
return fmt.Errorf("download not supported")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Communicator) DownloadDir(string, string, []string) error {
|
||||||
|
return fmt.Errorf("downloadDir not supported")
|
||||||
|
}
|
||||||
|
|
||||||
type ExecuteCommandTemplate struct {
|
type ExecuteCommandTemplate struct {
|
||||||
Command string
|
Command string
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue