mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-01 11:47:07 -06:00
adb88eaa16
In order to properly migrate the contents of resource, data, provider and provisioner blocks we will need the provider's schema in order to understand what is expected, so we can resolve some ambiguities inherent in the legacy HCL AST. This includes an initial prototype of migrating the content of resource blocks just to verify that the information is being gathered correctly. As with the rest of the upgrade_native.go file, this will be reorganized significantly once the basic end-to-end flow is established and we can see how to organize this code better.
250 lines
7.6 KiB
Go
250 lines
7.6 KiB
Go
package command
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/hashicorp/hcl2/hcl"
|
|
"github.com/hashicorp/terraform/configs/configupgrade"
|
|
"github.com/hashicorp/terraform/terraform"
|
|
"github.com/hashicorp/terraform/tfdiags"
|
|
)
|
|
|
|
// ZeroTwelveUpgradeCommand is a Command implementation that can upgrade
|
|
// the configuration files for a module from pre-0.11 syntax to new 0.12
|
|
// idiom, while also flagging any suspicious constructs that will require
|
|
// human review.
|
|
type ZeroTwelveUpgradeCommand struct {
|
|
Meta
|
|
}
|
|
|
|
func (c *ZeroTwelveUpgradeCommand) Run(args []string) int {
|
|
args, err := c.Meta.process(args, true)
|
|
if err != nil {
|
|
return 1
|
|
}
|
|
|
|
var skipConfirm, force bool
|
|
|
|
flags := c.Meta.flagSet("0.12upgrade")
|
|
flags.BoolVar(&skipConfirm, "yes", false, "skip confirmation prompt")
|
|
flags.BoolVar(&force, "force", false, "override duplicate upgrade heuristic")
|
|
if err := flags.Parse(args); err != nil {
|
|
return 1
|
|
}
|
|
|
|
var diags tfdiags.Diagnostics
|
|
|
|
var dir string
|
|
args = flags.Args()
|
|
switch len(args) {
|
|
case 0:
|
|
dir = "."
|
|
case 1:
|
|
dir = args[0]
|
|
default:
|
|
diags = diags.Append(tfdiags.Sourceless(
|
|
tfdiags.Error,
|
|
"Too many arguments",
|
|
"The command 0.12upgrade expects only a single argument, giving the directory containing the module to upgrade.",
|
|
))
|
|
c.showDiagnostics(diags)
|
|
return 1
|
|
}
|
|
|
|
dir = c.normalizePath(dir)
|
|
|
|
sources, err := configupgrade.LoadModule(dir)
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
diags = diags.Append(tfdiags.Sourceless(
|
|
tfdiags.Error,
|
|
"Module directory not found",
|
|
fmt.Sprintf("The given directory %s does not exist.", dir),
|
|
))
|
|
} else {
|
|
diags = diags.Append(err)
|
|
}
|
|
c.showDiagnostics(diags)
|
|
return 1
|
|
}
|
|
|
|
if len(sources) == 0 {
|
|
diags = diags.Append(tfdiags.Sourceless(
|
|
tfdiags.Error,
|
|
"Not a module directory",
|
|
fmt.Sprintf("The given directory %s does not contain any Terraform configuration files.", dir),
|
|
))
|
|
c.showDiagnostics(diags)
|
|
return 1
|
|
}
|
|
|
|
// The config loader doesn't naturally populate our sources
|
|
// map, so we'll do it manually so our diagnostics can have
|
|
// source code snippets inside them.
|
|
// This is weird, but this whole upgrade codepath is pretty
|
|
// weird and temporary, so we'll accept it.
|
|
if loader, err := c.initConfigLoader(); err == nil {
|
|
parser := loader.Parser()
|
|
for name, src := range sources {
|
|
parser.ForceFileSource(filepath.Join(dir, name), src)
|
|
}
|
|
}
|
|
|
|
if !force {
|
|
// We'll check first if this directory already looks upgraded, so we
|
|
// don't waste the user's time dealing with an interactive prompt
|
|
// immediately followed by an error.
|
|
if already, rng := sources.MaybeAlreadyUpgraded(); already {
|
|
diags = diags.Append(&hcl.Diagnostic{
|
|
Severity: hcl.DiagError,
|
|
Summary: "Module already upgraded",
|
|
Detail: fmt.Sprintf("The module in directory %s has a version constraint that suggests it has already been upgraded for v0.12. If this is incorrect, either remove this constraint or override this heuristic with the -force argument. Upgrading a module that was already upgraded may change the meaning of that module.", dir),
|
|
Subject: rng.ToHCL().Ptr(),
|
|
})
|
|
c.showDiagnostics(diags)
|
|
return 1
|
|
}
|
|
}
|
|
|
|
if !skipConfirm {
|
|
c.Ui.Output(fmt.Sprintf(`
|
|
This command will rewrite the configuration files in the given directory so
|
|
that they use the new syntax features from Terraform v0.12, and will identify
|
|
any constructs that may need to be adjusted for correct operation with
|
|
Terraform v0.12.
|
|
|
|
We recommend using this command in a clean version control work tree, so that
|
|
you can easily see the proposed changes as a diff against the latest commit.
|
|
If you have uncommited changes already present, we recommend aborting this
|
|
command and dealing with them before running this command again.
|
|
`))
|
|
|
|
query := "Would you like to upgrade the module in the current directory?"
|
|
if dir != "." {
|
|
query = fmt.Sprintf("Would you like to upgrade the module in %s?", dir)
|
|
}
|
|
v, err := c.UIInput().Input(&terraform.InputOpts{
|
|
Id: "approve",
|
|
Query: query,
|
|
Description: `Only 'yes' will be accepted to confirm.`,
|
|
})
|
|
if err != nil {
|
|
diags = diags.Append(err)
|
|
c.showDiagnostics(diags)
|
|
return 1
|
|
}
|
|
if v != "yes" {
|
|
c.Ui.Info("Upgrade cancelled.")
|
|
return 0
|
|
}
|
|
|
|
c.Ui.Output(`-----------------------------------------------------------------------------`)
|
|
}
|
|
|
|
upgrader := &configupgrade.Upgrader{
|
|
Providers: c.providerResolver(),
|
|
Provisioners: c.provisionerFactories(),
|
|
}
|
|
newSources, upgradeDiags := upgrader.Upgrade(sources)
|
|
diags = diags.Append(upgradeDiags)
|
|
if upgradeDiags.HasErrors() {
|
|
c.showDiagnostics(diags)
|
|
return 2
|
|
}
|
|
|
|
// Now we'll write the contents of newSources into the filesystem.
|
|
for name, src := range newSources {
|
|
fn := filepath.Join(dir, name)
|
|
if src == nil {
|
|
// indicates a file to be deleted
|
|
err := os.Remove(fn)
|
|
if err != nil {
|
|
diags = diags.Append(tfdiags.Sourceless(
|
|
tfdiags.Error,
|
|
"Failed to remove file",
|
|
fmt.Sprintf("The file %s must be renamed as part of the upgrade process, but the old file could not be deleted: %s.", fn, err),
|
|
))
|
|
}
|
|
continue
|
|
}
|
|
|
|
err := ioutil.WriteFile(fn, src, 0644)
|
|
if err != nil {
|
|
diags = diags.Append(tfdiags.Sourceless(
|
|
tfdiags.Error,
|
|
"Failed to write file",
|
|
fmt.Sprintf("The file %s must be updated or created as part of the upgrade process, but there was an error while writing: %s.", fn, err),
|
|
))
|
|
}
|
|
}
|
|
|
|
c.showDiagnostics(diags)
|
|
if diags.HasErrors() {
|
|
return 2
|
|
}
|
|
|
|
if !skipConfirm {
|
|
if len(diags) != 0 {
|
|
c.Ui.Output(`-----------------------------------------------------------------------------`)
|
|
}
|
|
c.Ui.Output(c.Colorize().Color(`
|
|
[bold][green]Upgrade complete![reset]
|
|
|
|
The configuration files were upgraded successfully. Use your version control
|
|
system to review the proposed changes, make any necessary adjustments, and
|
|
then commit.
|
|
`))
|
|
if len(diags) != 0 {
|
|
// We checked for errors above, so these must be warnings.
|
|
c.Ui.Output(`Some warnings were generated during the upgrade, as shown above. These
|
|
indicate situations where Terraform could not decide on an appropriate course
|
|
of action without further human input.
|
|
|
|
Where possible, these have also been marked with TF-UPGRADE-TODO comments to
|
|
mark the locations where a decision must be made. After reviewing and adjusting
|
|
these, manually remove the TF-UPGRADE-TODO comment before continuing.
|
|
`)
|
|
}
|
|
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func (c *ZeroTwelveUpgradeCommand) Help() string {
|
|
helpText := `
|
|
Usage: terraform 0.12upgrade [module-dir]
|
|
|
|
Rewrites the .tf files for a single module that was written for a Terraform
|
|
version prior to v0.12 so that it uses new syntax features from v0.12
|
|
and later.
|
|
|
|
Also rewrites constructs that behave differently after v0.12, and flags any
|
|
suspicious constructs that require human review,
|
|
|
|
By default, 0.12upgrade rewrites the files in the current working directory.
|
|
However, a path to a different directory can be provided. The command will
|
|
prompt for confirmation interactively unless the -yes option is given.
|
|
|
|
Options:
|
|
|
|
-yes Skip the initial introduction messages and interactive
|
|
confirmation. This can be used to run this command in
|
|
batch from a script.
|
|
|
|
-force Override the heuristic that attempts to detect if a
|
|
configuration is already written for v0.12 or later.
|
|
Some of the transformations made by this command are
|
|
not idempotent, so re-running against the same module
|
|
may change the meanings expressions in the module.
|
|
`
|
|
return strings.TrimSpace(helpText)
|
|
}
|
|
|
|
func (c *ZeroTwelveUpgradeCommand) Synopsis() string {
|
|
return "Rewrites pre-0.12 module source code for v0.12"
|
|
}
|