2014-10-28 18:09:22 -04:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
2014-10-28 22:29:51 -04:00
|
|
|
"io"
|
2014-12-03 15:15:48 -05:00
|
|
|
"os"
|
|
|
|
"os/signal"
|
2014-10-28 22:29:51 -04:00
|
|
|
"path/filepath"
|
2014-10-28 18:09:22 -04:00
|
|
|
"strings"
|
|
|
|
|
2014-12-01 18:20:10 -05:00
|
|
|
"github.com/hashicorp/atlas-go/archive"
|
2014-12-03 13:01:00 -05:00
|
|
|
"github.com/hashicorp/atlas-go/v1"
|
2014-10-28 18:09:22 -04:00
|
|
|
"github.com/mitchellh/packer/packer"
|
|
|
|
)
|
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
// archiveTemplateEntry is the name the template always takes within the slug.
|
2014-10-28 23:02:07 -04:00
|
|
|
const archiveTemplateEntry = ".packer-template"
|
2014-10-28 22:29:51 -04:00
|
|
|
|
2014-10-28 18:09:22 -04:00
|
|
|
type PushCommand struct {
|
|
|
|
Meta
|
2014-10-28 22:29:51 -04:00
|
|
|
|
2014-12-01 18:20:10 -05:00
|
|
|
client *atlas.Client
|
2014-10-28 22:43:41 -04:00
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
// For tests:
|
2014-12-03 13:01:00 -05:00
|
|
|
uploadFn pushUploadFn
|
2014-10-28 18:09:22 -04:00
|
|
|
}
|
|
|
|
|
2014-12-03 13:01:00 -05:00
|
|
|
// pushUploadFn is the callback type used for tests to stub out the uploading
|
|
|
|
// logic of the push command.
|
|
|
|
type pushUploadFn func(
|
|
|
|
io.Reader, *uploadOpts) (<-chan struct{}, <-chan error, error)
|
|
|
|
|
2014-10-28 18:09:22 -04:00
|
|
|
func (c *PushCommand) Run(args []string) int {
|
2014-10-28 23:02:07 -04:00
|
|
|
var create bool
|
2014-10-28 22:29:51 -04:00
|
|
|
var token string
|
|
|
|
|
2014-10-28 18:09:22 -04:00
|
|
|
f := flag.NewFlagSet("push", flag.ContinueOnError)
|
|
|
|
f.Usage = func() { c.Ui.Error(c.Help()) }
|
2014-10-28 23:02:07 -04:00
|
|
|
f.BoolVar(&create, "create", false, "create")
|
2014-10-28 22:29:51 -04:00
|
|
|
f.StringVar(&token, "token", "", "token")
|
2014-10-28 18:09:22 -04:00
|
|
|
if err := f.Parse(args); err != nil {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
args = f.Args()
|
|
|
|
if len(args) != 1 {
|
|
|
|
f.Usage()
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read the template
|
|
|
|
tpl, err := packer.ParseTemplateFile(args[0], nil)
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to parse template: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
// Validate some things
|
|
|
|
if tpl.Push.Name == "" {
|
|
|
|
c.Ui.Error(fmt.Sprintf(
|
|
|
|
"The 'push' section must be specified in the template with\n" +
|
|
|
|
"at least the 'name' option set."))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2014-11-06 11:43:31 -05:00
|
|
|
// Determine our token
|
|
|
|
if token == "" {
|
|
|
|
token = tpl.Push.Token
|
|
|
|
}
|
|
|
|
|
2014-10-28 22:43:41 -04:00
|
|
|
// Build our client
|
|
|
|
defer func() { c.client = nil }()
|
2014-12-01 18:20:10 -05:00
|
|
|
c.client = atlas.DefaultClient()
|
2014-10-28 23:06:19 -04:00
|
|
|
if tpl.Push.Address != "" {
|
2014-12-01 18:20:10 -05:00
|
|
|
c.client, err = atlas.NewClient(tpl.Push.Address)
|
2014-10-28 23:06:19 -04:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf(
|
|
|
|
"Error setting up API client: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
}
|
2014-12-01 19:54:12 -05:00
|
|
|
if token != "" {
|
|
|
|
c.client.Token = token
|
|
|
|
}
|
2014-10-28 22:43:41 -04:00
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
// Build the archiving options
|
|
|
|
var opts archive.ArchiveOpts
|
|
|
|
opts.Include = tpl.Push.Include
|
|
|
|
opts.Exclude = tpl.Push.Exclude
|
|
|
|
opts.VCS = tpl.Push.VCS
|
|
|
|
opts.Extra = map[string]string{
|
|
|
|
archiveTemplateEntry: args[0],
|
|
|
|
}
|
|
|
|
|
2014-12-01 22:49:55 -05:00
|
|
|
// Determine the path we're archiving. This logic is a bit complicated
|
|
|
|
// as there are three possibilities:
|
|
|
|
//
|
|
|
|
// 1.) BaseDir is an absolute path, just use that.
|
|
|
|
//
|
|
|
|
// 2.) BaseDir is empty, so we use the directory of the template.
|
|
|
|
//
|
|
|
|
// 3.) BaseDir is relative, so we use the path relative to the directory
|
|
|
|
// of the template.
|
|
|
|
//
|
2014-10-28 22:29:51 -04:00
|
|
|
path := tpl.Push.BaseDir
|
2014-12-01 22:49:55 -05:00
|
|
|
if path == "" || !filepath.IsAbs(path) {
|
|
|
|
tplPath, err := filepath.Abs(args[0])
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error determining path to archive: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
tplPath = filepath.Dir(tplPath)
|
|
|
|
if path != "" {
|
|
|
|
tplPath = filepath.Join(tplPath, path)
|
|
|
|
}
|
|
|
|
path, err = filepath.Abs(tplPath)
|
2014-10-28 22:29:51 -04:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error determining path to archive: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-03 13:01:00 -05:00
|
|
|
// Find the Atlas post-processors, if possible
|
|
|
|
var atlasPPs []packer.RawPostProcessorConfig
|
|
|
|
for _, list := range tpl.PostProcessors {
|
|
|
|
for _, pp := range list {
|
|
|
|
if pp.Type == "atlas" {
|
|
|
|
atlasPPs = append(atlasPPs, pp)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
// Build the upload options
|
|
|
|
var uploadOpts uploadOpts
|
|
|
|
uploadOpts.Slug = tpl.Push.Name
|
2014-12-03 13:01:00 -05:00
|
|
|
uploadOpts.Builds = make(map[string]*uploadBuildInfo)
|
2014-10-28 22:43:41 -04:00
|
|
|
for _, b := range tpl.Builders {
|
2014-12-03 13:01:00 -05:00
|
|
|
info := &uploadBuildInfo{Type: b.Type}
|
|
|
|
|
|
|
|
// Determine if we're artifacting this build
|
|
|
|
for _, pp := range atlasPPs {
|
|
|
|
if !pp.Skip(b.Name) {
|
|
|
|
info.Artifact = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uploadOpts.Builds[b.Name] = info
|
2014-10-28 22:43:41 -04:00
|
|
|
}
|
2014-10-28 22:29:51 -04:00
|
|
|
|
2014-12-03 15:04:01 -05:00
|
|
|
// Warn about builds not having post-processors.
|
|
|
|
var badBuilds []string
|
|
|
|
for name, b := range uploadOpts.Builds {
|
|
|
|
if b.Artifact {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
badBuilds = append(badBuilds, name)
|
|
|
|
}
|
|
|
|
if len(badBuilds) > 0 {
|
|
|
|
c.Ui.Error(fmt.Sprintf(
|
2014-12-03 15:15:48 -05:00
|
|
|
"Warning! One or more of the builds in this template does not\n"+
|
|
|
|
"have an Atlas post-processor. Artifacts from this template will\n"+
|
|
|
|
"not appear in the Atlas artifact registry.\n\n"+
|
|
|
|
"This is just a warning. Atlas will still build your template\n"+
|
|
|
|
"and assume other post-processors are sending the artifacts where\n"+
|
|
|
|
"they need to go.\n\n"+
|
|
|
|
"Builds: %s\n\n", strings.Join(badBuilds, ", ")))
|
2014-12-03 15:04:01 -05:00
|
|
|
}
|
|
|
|
|
2014-10-28 23:02:07 -04:00
|
|
|
// Create the build config if it doesn't currently exist.
|
|
|
|
if err := c.create(uploadOpts.Slug, create); err != nil {
|
|
|
|
c.Ui.Error(err.Error())
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
// Start the archiving process
|
2014-12-01 18:20:10 -05:00
|
|
|
r, err := archive.CreateArchive(path, &opts)
|
2014-10-28 22:29:51 -04:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error archiving: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
2014-10-28 22:34:19 -04:00
|
|
|
defer r.Close()
|
2014-10-28 22:29:51 -04:00
|
|
|
|
|
|
|
// Start the upload process
|
|
|
|
doneCh, uploadErrCh, err := c.upload(r, &uploadOpts)
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error starting upload: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2014-12-03 15:15:48 -05:00
|
|
|
// Make a ctrl-C channel
|
|
|
|
sigCh := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(sigCh, os.Interrupt)
|
|
|
|
defer signal.Stop(sigCh)
|
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
err = nil
|
|
|
|
select {
|
|
|
|
case err = <-uploadErrCh:
|
|
|
|
err = fmt.Errorf("Error uploading: %s", err)
|
2014-12-03 15:15:48 -05:00
|
|
|
case <-sigCh:
|
|
|
|
err = fmt.Errorf("Push cancelled from Ctrl-C")
|
2014-10-28 22:29:51 -04:00
|
|
|
case <-doneCh:
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(err.Error())
|
|
|
|
return 1
|
|
|
|
}
|
2014-10-28 18:09:22 -04:00
|
|
|
|
2014-12-01 18:20:10 -05:00
|
|
|
c.Ui.Output(fmt.Sprintf("Push successful to '%s'", tpl.Push.Name))
|
2014-10-28 18:09:22 -04:00
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (*PushCommand) Help() string {
|
|
|
|
helpText := `
|
|
|
|
Usage: packer push [options] TEMPLATE
|
|
|
|
|
|
|
|
Push the template and the files it needs to a Packer build service.
|
|
|
|
This will not initiate any builds, it will only update the templates
|
|
|
|
used for builds.
|
2014-10-28 22:29:51 -04:00
|
|
|
|
2014-10-28 23:02:07 -04:00
|
|
|
The configuration about what is pushed is configured within the
|
|
|
|
template's "push" section.
|
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
Options:
|
|
|
|
|
2014-10-28 23:02:07 -04:00
|
|
|
-create Create the build configuration if it doesn't exist.
|
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
-token=<token> Access token to use to upload. If blank, the
|
|
|
|
TODO environmental variable will be used.
|
2014-10-28 18:09:22 -04:00
|
|
|
`
|
|
|
|
|
|
|
|
return strings.TrimSpace(helpText)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (*PushCommand) Synopsis() string {
|
|
|
|
return "push template files to a Packer build service"
|
|
|
|
}
|
2014-10-28 22:29:51 -04:00
|
|
|
|
2014-10-28 23:02:07 -04:00
|
|
|
func (c *PushCommand) create(name string, create bool) error {
|
|
|
|
if c.uploadFn != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Separate the slug into the user and name components
|
2014-12-01 18:20:10 -05:00
|
|
|
user, name, err := atlas.ParseSlug(name)
|
2014-10-28 23:02:07 -04:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Malformed push name: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if it exists. If so, we're done.
|
|
|
|
if _, err := c.client.BuildConfig(user, name); err == nil {
|
|
|
|
return nil
|
2014-12-01 18:20:10 -05:00
|
|
|
} else if err != atlas.ErrNotFound {
|
2014-10-28 23:02:07 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, show an error if we're not creating.
|
|
|
|
if !create {
|
|
|
|
return fmt.Errorf(
|
2014-12-03 13:01:00 -05:00
|
|
|
"Push target doesn't exist: %s. Either create this online via\n"+
|
2014-11-06 11:43:31 -05:00
|
|
|
"the website or pass the -create flag.", name)
|
2014-10-28 23:02:07 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create it
|
|
|
|
if err := c.client.CreateBuildConfig(user, name); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-10-28 22:29:51 -04:00
|
|
|
func (c *PushCommand) upload(
|
2014-12-01 18:20:10 -05:00
|
|
|
r *archive.Archive, opts *uploadOpts) (<-chan struct{}, <-chan error, error) {
|
2014-10-28 22:29:51 -04:00
|
|
|
if c.uploadFn != nil {
|
|
|
|
return c.uploadFn(r, opts)
|
|
|
|
}
|
|
|
|
|
2014-10-28 22:43:41 -04:00
|
|
|
// Separate the slug into the user and name components
|
2014-12-01 18:20:10 -05:00
|
|
|
user, name, err := atlas.ParseSlug(opts.Slug)
|
2014-10-28 22:43:41 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("upload: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the app
|
|
|
|
bc, err := c.client.BuildConfig(user, name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("upload: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build the version to send up
|
2014-12-01 18:20:10 -05:00
|
|
|
version := atlas.BuildConfigVersion{
|
2014-10-28 22:43:41 -04:00
|
|
|
User: bc.User,
|
|
|
|
Name: bc.Name,
|
2014-12-01 18:20:10 -05:00
|
|
|
Builds: make([]atlas.BuildConfigBuild, 0, len(opts.Builds)),
|
2014-10-28 22:43:41 -04:00
|
|
|
}
|
2014-12-03 13:01:00 -05:00
|
|
|
for name, info := range opts.Builds {
|
2014-12-01 18:20:10 -05:00
|
|
|
version.Builds = append(version.Builds, atlas.BuildConfigBuild{
|
2014-12-03 13:01:00 -05:00
|
|
|
Name: name,
|
|
|
|
Type: info.Type,
|
|
|
|
Artifact: info.Artifact,
|
2014-10-28 22:43:41 -04:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start the upload
|
|
|
|
doneCh, errCh := make(chan struct{}), make(chan error)
|
|
|
|
go func() {
|
2014-12-01 18:20:10 -05:00
|
|
|
err := c.client.UploadBuildConfigVersion(&version, r, r.Size)
|
2014-10-28 22:43:41 -04:00
|
|
|
if err != nil {
|
|
|
|
errCh <- err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
close(doneCh)
|
|
|
|
}()
|
|
|
|
|
|
|
|
return doneCh, errCh, nil
|
2014-10-28 22:29:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
type uploadOpts struct {
|
2014-10-28 22:43:41 -04:00
|
|
|
URL string
|
|
|
|
Slug string
|
2014-12-03 13:01:00 -05:00
|
|
|
Builds map[string]*uploadBuildInfo
|
|
|
|
}
|
|
|
|
|
|
|
|
type uploadBuildInfo struct {
|
2014-12-03 15:15:48 -05:00
|
|
|
Type string
|
|
|
|
Artifact bool
|
2014-10-28 22:29:51 -04:00
|
|
|
}
|