2013-07-02 22:11:30 -04:00
|
|
|
package file
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2013-08-06 18:30:49 -04:00
|
|
|
"github.com/mitchellh/packer/common"
|
2013-07-02 22:11:30 -04:00
|
|
|
"github.com/mitchellh/packer/packer"
|
|
|
|
"os"
|
|
|
|
)
|
|
|
|
|
|
|
|
type config struct {
|
2013-08-09 17:21:31 -04:00
|
|
|
common.PackerConfig `mapstructure:",squash"`
|
|
|
|
|
2013-07-02 22:11:30 -04:00
|
|
|
// The local path of the file to upload.
|
|
|
|
Source string
|
|
|
|
|
|
|
|
// The remote path where the local file will be uploaded to.
|
|
|
|
Destination string
|
2013-08-08 19:31:34 -04:00
|
|
|
|
2013-08-15 22:17:23 -04:00
|
|
|
tpl *packer.ConfigTemplate
|
2013-07-02 22:11:30 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
type Provisioner struct {
|
|
|
|
config config
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Provisioner) Prepare(raws ...interface{}) error {
|
2013-08-06 18:30:49 -04:00
|
|
|
md, err := common.DecodeConfig(&p.config, raws...)
|
2013-07-13 20:28:56 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2013-08-15 22:17:23 -04:00
|
|
|
p.config.tpl, err = packer.NewConfigTemplate()
|
2013-08-08 19:31:34 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2013-08-09 17:21:31 -04:00
|
|
|
p.config.tpl.UserVars = p.config.PackerUserVars
|
2013-08-08 19:31:34 -04:00
|
|
|
|
2013-07-13 20:28:56 -04:00
|
|
|
// Accumulate any errors
|
2013-08-06 18:30:49 -04:00
|
|
|
errs := common.CheckUnusedConfig(md)
|
2013-07-02 22:11:30 -04:00
|
|
|
|
2013-08-08 19:31:34 -04:00
|
|
|
templates := map[string]*string{
|
|
|
|
"source": &p.config.Source,
|
|
|
|
"destination": &p.config.Destination,
|
|
|
|
}
|
|
|
|
|
|
|
|
for n, ptr := range templates {
|
|
|
|
var err error
|
|
|
|
*ptr, err = p.config.tpl.Process(*ptr, nil)
|
|
|
|
if err != nil {
|
|
|
|
errs = packer.MultiErrorAppend(
|
|
|
|
errs, fmt.Errorf("Error processing %s: %s", n, err))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-02 22:11:30 -04:00
|
|
|
if _, err := os.Stat(p.config.Source); err != nil {
|
2013-08-06 18:30:49 -04:00
|
|
|
errs = packer.MultiErrorAppend(errs,
|
2013-07-04 15:50:00 -04:00
|
|
|
fmt.Errorf("Bad source '%s': %s", p.config.Source, err))
|
2013-07-02 22:11:30 -04:00
|
|
|
}
|
|
|
|
|
2013-07-04 15:50:00 -04:00
|
|
|
if p.config.Destination == "" {
|
2013-08-06 18:30:49 -04:00
|
|
|
errs = packer.MultiErrorAppend(errs,
|
|
|
|
errors.New("Destination must be specified."))
|
2013-07-02 22:11:30 -04:00
|
|
|
}
|
|
|
|
|
2013-08-06 18:30:49 -04:00
|
|
|
if errs != nil && len(errs.Errors) > 0 {
|
|
|
|
return errs
|
2013-07-02 22:11:30 -04:00
|
|
|
}
|
2013-07-04 15:50:00 -04:00
|
|
|
|
2013-07-02 22:11:30 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Provisioner) Provision(ui packer.Ui, comm packer.Communicator) error {
|
|
|
|
ui.Say(fmt.Sprintf("Uploading %s => %s", p.config.Source, p.config.Destination))
|
2013-09-09 16:58:23 -04:00
|
|
|
info, err := os.Stat(p.config.Source)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we're uploading a directory, short circuit and do that
|
|
|
|
if info.IsDir() {
|
|
|
|
return comm.UploadDir(p.config.Destination, p.config.Source, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We're uploading a file...
|
2013-07-02 22:11:30 -04:00
|
|
|
f, err := os.Open(p.config.Source)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2013-07-04 15:50:00 -04:00
|
|
|
defer f.Close()
|
|
|
|
|
2014-05-10 00:03:35 -04:00
|
|
|
fi, err := f.Stat()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = comm.Upload(p.config.Destination, f, &fi)
|
2013-07-17 21:17:46 -04:00
|
|
|
if err != nil {
|
|
|
|
ui.Error(fmt.Sprintf("Upload failed: %s", err))
|
|
|
|
}
|
|
|
|
return err
|
2013-07-02 22:11:30 -04:00
|
|
|
}
|
2013-08-31 02:23:36 -04:00
|
|
|
|
|
|
|
func (p *Provisioner) Cancel() {
|
|
|
|
// Just hard quit. It isn't a big deal if what we're doing keeps
|
|
|
|
// running on the other side.
|
|
|
|
os.Exit(0)
|
|
|
|
}
|