mirror of
https://github.com/hashicorp/packer.git
synced 2026-02-20 00:10:04 -05:00
* Updating the license from MPL to Business Source License Going forward, this project will be licensed under the Business Source License v1.1. Please see our blog post for more details at https://hashi.co/bsl-blog, FAQ at https://hashi.co/license-faq, and details of the license at www.hashicorp.com/bsl. * Update copyright file headers to BUSL-1.1 --------- Co-authored-by: hashicorp-copywrite[bot] <110428419+hashicorp-copywrite[bot]@users.noreply.github.com>
217 lines
5.9 KiB
Go
217 lines
5.9 KiB
Go
// Copyright (c) HashiCorp, Inc.
|
|
// SPDX-License-Identifier: BUSL-1.1
|
|
|
|
package command
|
|
|
|
import (
|
|
"context"
|
|
"crypto/sha256"
|
|
"fmt"
|
|
"log"
|
|
"runtime"
|
|
"strings"
|
|
|
|
pluginsdk "github.com/hashicorp/packer-plugin-sdk/plugin"
|
|
"github.com/hashicorp/packer/packer"
|
|
plugingetter "github.com/hashicorp/packer/packer/plugin-getter"
|
|
"github.com/hashicorp/packer/packer/plugin-getter/github"
|
|
"github.com/hashicorp/packer/version"
|
|
"github.com/posener/complete"
|
|
)
|
|
|
|
type InitCommand struct {
|
|
Meta
|
|
}
|
|
|
|
func (c *InitCommand) Run(args []string) int {
|
|
ctx, cleanup := handleTermInterrupt(c.Ui)
|
|
defer cleanup()
|
|
|
|
cfg, ret := c.ParseArgs(args)
|
|
if ret != 0 {
|
|
return ret
|
|
}
|
|
|
|
return c.RunContext(ctx, cfg)
|
|
}
|
|
|
|
func (c *InitCommand) ParseArgs(args []string) (*InitArgs, int) {
|
|
var cfg InitArgs
|
|
flags := c.Meta.FlagSet("init", 0)
|
|
flags.Usage = func() { c.Ui.Say(c.Help()) }
|
|
cfg.AddFlagSets(flags)
|
|
if err := flags.Parse(args); err != nil {
|
|
return &cfg, 1
|
|
}
|
|
|
|
args = flags.Args()
|
|
if len(args) != 1 {
|
|
flags.Usage()
|
|
return &cfg, 1
|
|
}
|
|
cfg.Path = args[0]
|
|
return &cfg, 0
|
|
}
|
|
|
|
func (c *InitCommand) RunContext(buildCtx context.Context, cla *InitArgs) int {
|
|
packerStarter, ret := c.GetConfig(&cla.MetaArgs)
|
|
if ret != 0 {
|
|
return ret
|
|
}
|
|
|
|
// Get plugins requirements
|
|
reqs, diags := packerStarter.PluginRequirements()
|
|
ret = writeDiags(c.Ui, nil, diags)
|
|
if ret != 0 {
|
|
return ret
|
|
}
|
|
|
|
opts := plugingetter.ListInstallationsOptions{
|
|
FromFolders: c.Meta.CoreConfig.Components.PluginConfig.KnownPluginFolders,
|
|
BinaryInstallationOptions: plugingetter.BinaryInstallationOptions{
|
|
OS: runtime.GOOS,
|
|
ARCH: runtime.GOARCH,
|
|
APIVersionMajor: pluginsdk.APIVersionMajor,
|
|
APIVersionMinor: pluginsdk.APIVersionMinor,
|
|
Checksummers: []plugingetter.Checksummer{
|
|
{Type: "sha256", Hash: sha256.New()},
|
|
},
|
|
},
|
|
}
|
|
|
|
if runtime.GOOS == "windows" && opts.Ext == "" {
|
|
opts.BinaryInstallationOptions.Ext = ".exe"
|
|
}
|
|
|
|
log.Printf("[TRACE] init: %#v", opts)
|
|
|
|
getters := []plugingetter.Getter{
|
|
&github.Getter{
|
|
// In the past some terraform plugins downloads were blocked from a
|
|
// specific aws region by s3. Changing the user agent unblocked the
|
|
// downloads so having one user agent per version will help mitigate
|
|
// that a little more. Especially in the case someone forks this
|
|
// code to make it more aggressive or something.
|
|
// TODO: allow to set this from the config file or an environment
|
|
// variable.
|
|
UserAgent: "packer-getter-github-" + version.String(),
|
|
},
|
|
}
|
|
|
|
ui := &packer.ColoredUi{
|
|
Color: packer.UiColorCyan,
|
|
Ui: c.Ui,
|
|
}
|
|
|
|
for _, pluginRequirement := range reqs {
|
|
// Get installed plugins that match requirement
|
|
|
|
installs, err := pluginRequirement.ListInstallations(opts)
|
|
if err != nil {
|
|
c.Ui.Error(err.Error())
|
|
return 1
|
|
}
|
|
|
|
log.Printf("[TRACE] for plugin %s found %d matching installation(s)", pluginRequirement.Identifier, len(installs))
|
|
|
|
if len(installs) > 0 && cla.Upgrade == false {
|
|
continue
|
|
}
|
|
|
|
newInstall, err := pluginRequirement.InstallLatest(plugingetter.InstallOptions{
|
|
InFolders: opts.FromFolders,
|
|
BinaryInstallationOptions: opts.BinaryInstallationOptions,
|
|
Getters: getters,
|
|
})
|
|
if err != nil {
|
|
if pluginRequirement.Implicit {
|
|
msg := fmt.Sprintf(`
|
|
Warning! At least one component used in your config file(s) has moved out of
|
|
Packer into the %q plugin.
|
|
For that reason, Packer init tried to install the latest version of the %s
|
|
plugin. Unfortunately, this failed :
|
|
%s`,
|
|
pluginRequirement.Identifier,
|
|
pluginRequirement.Identifier.Type,
|
|
err)
|
|
c.Ui.Say(msg)
|
|
} else {
|
|
c.Ui.Error(fmt.Sprintf("Failed getting the %q plugin:", pluginRequirement.Identifier))
|
|
c.Ui.Error(err.Error())
|
|
ret = 1
|
|
}
|
|
}
|
|
if newInstall != nil {
|
|
if pluginRequirement.Implicit {
|
|
msg := fmt.Sprintf("Installed implicitly required plugin %s %s in %q", pluginRequirement.Identifier, newInstall.Version, newInstall.BinaryPath)
|
|
ui.Say(msg)
|
|
|
|
warn := fmt.Sprintf(`
|
|
Warning, at least one component used in your config file(s) has moved out of
|
|
Packer into the %[2]q plugin and is now being implicitly required.
|
|
For more details on implicitly required plugins see https://packer.io/docs/commands/init#implicit-required-plugin
|
|
|
|
To avoid any backward incompatible changes with your
|
|
config file you may want to lock the plugin version by pasting the following to your config:
|
|
|
|
packer {
|
|
required_plugins {
|
|
%[1]s = {
|
|
source = "%[2]s"
|
|
version = "~> %[3]s"
|
|
}
|
|
}
|
|
}
|
|
`,
|
|
pluginRequirement.Identifier.Type,
|
|
pluginRequirement.Identifier,
|
|
newInstall.Version,
|
|
)
|
|
ui.Error(warn)
|
|
continue
|
|
}
|
|
msg := fmt.Sprintf("Installed plugin %s %s in %q", pluginRequirement.Identifier, newInstall.Version, newInstall.BinaryPath)
|
|
ui.Say(msg)
|
|
|
|
}
|
|
}
|
|
return ret
|
|
}
|
|
|
|
func (*InitCommand) Help() string {
|
|
helpText := `
|
|
Usage: packer init [options] TEMPLATE
|
|
|
|
Install all the missing plugins required in a Packer config. Note that Packer
|
|
does not have a state.
|
|
|
|
This is the first command that should be executed when working with a new
|
|
or existing template.
|
|
|
|
This command is always safe to run multiple times. Though subsequent runs may
|
|
give errors, this command will never delete anything.
|
|
|
|
Options:
|
|
-upgrade On top of installing missing plugins, update
|
|
installed plugins to the latest available
|
|
version, if there is a new higher one. Note that
|
|
this still takes into consideration the version
|
|
constraint of the config.
|
|
`
|
|
|
|
return strings.TrimSpace(helpText)
|
|
}
|
|
|
|
func (*InitCommand) Synopsis() string {
|
|
return "Install missing plugins or upgrade plugins"
|
|
}
|
|
|
|
func (*InitCommand) AutocompleteArgs() complete.Predictor {
|
|
return complete.PredictNothing
|
|
}
|
|
|
|
func (*InitCommand) AutocompleteFlags() complete.Flags {
|
|
return complete.Flags{
|
|
"-upgrade": complete.PredictNothing,
|
|
}
|
|
}
|