mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-16 11:42:58 -06:00
1a8da65314
It's been a long while since we gave close attention to the codepaths for module source address parsing and external module package installation. Due to their age, these codepaths often diverged from our modern practices such as representing address types in the addrs package, and encapsulating package installation details only in a particular location. In particular, this refactor makes source address parsing a separate step from module installation, which therefore makes the result of that parsing available to other Terraform subsystems which work with the configuration representation objects. This also presented the opportunity to better encapsulate our use of go-getter into a new package "getmodules" (echoing "getproviders"), which is intended to be the only part of Terraform that directly interacts with go-getter. This is largely just a refactor of the existing functionality into a new code organization, but there is one notable change in behavior here: the source address parsing now happens during configuration loading rather than module installation, which may cause errors about invalid addresses to be returned in different situations than before. That counts as backward compatible because we only promise to remain compatible with configurations that are _valid_, which means that they can be initialized, planned, and applied without any errors. This doesn't introduce any new error cases, and instead just makes a pre-existing error case be detected earlier. Our module registry client is still using its own special module address type from registry/regsrc for now, with a small shim from the new addrs.ModuleSourceRegistry type. Hopefully in a later commit we'll also rework the registry client to work with the new address type, but this commit is already big enough as it is.
157 lines
5.2 KiB
Go
157 lines
5.2 KiB
Go
package earlyconfig
|
|
|
|
import (
|
|
"fmt"
|
|
"sort"
|
|
"strings"
|
|
|
|
version "github.com/hashicorp/go-version"
|
|
"github.com/hashicorp/terraform-config-inspect/tfconfig"
|
|
"github.com/hashicorp/terraform/internal/addrs"
|
|
"github.com/hashicorp/terraform/internal/tfdiags"
|
|
)
|
|
|
|
// BuildConfig constructs a Config from a root module by loading all of its
|
|
// descendent modules via the given ModuleWalker.
|
|
func BuildConfig(root *tfconfig.Module, walker ModuleWalker) (*Config, tfdiags.Diagnostics) {
|
|
var diags tfdiags.Diagnostics
|
|
cfg := &Config{
|
|
Module: root,
|
|
}
|
|
cfg.Root = cfg // Root module is self-referential.
|
|
cfg.Children, diags = buildChildModules(cfg, walker)
|
|
return cfg, diags
|
|
}
|
|
|
|
func buildChildModules(parent *Config, walker ModuleWalker) (map[string]*Config, tfdiags.Diagnostics) {
|
|
var diags tfdiags.Diagnostics
|
|
ret := map[string]*Config{}
|
|
calls := parent.Module.ModuleCalls
|
|
|
|
// We'll sort the calls by their local names so that they'll appear in a
|
|
// predictable order in any logging that's produced during the walk.
|
|
callNames := make([]string, 0, len(calls))
|
|
for k := range calls {
|
|
callNames = append(callNames, k)
|
|
}
|
|
sort.Strings(callNames)
|
|
|
|
for _, callName := range callNames {
|
|
call := calls[callName]
|
|
path := make([]string, len(parent.Path)+1)
|
|
copy(path, parent.Path)
|
|
path[len(path)-1] = call.Name
|
|
|
|
var vc version.Constraints
|
|
if strings.TrimSpace(call.Version) != "" {
|
|
var err error
|
|
vc, err = version.NewConstraint(call.Version)
|
|
if err != nil {
|
|
diags = diags.Append(wrapDiagnostic(tfconfig.Diagnostic{
|
|
Severity: tfconfig.DiagError,
|
|
Summary: "Invalid version constraint",
|
|
Detail: fmt.Sprintf("Module %q (declared at %s line %d) has invalid version constraint %q: %s.", callName, call.Pos.Filename, call.Pos.Line, call.Version, err),
|
|
}))
|
|
continue
|
|
}
|
|
}
|
|
|
|
sourceAddr, err := addrs.ParseModuleSource(call.Source)
|
|
if err != nil {
|
|
diags = diags.Append(wrapDiagnostic(tfconfig.Diagnostic{
|
|
Severity: tfconfig.DiagError,
|
|
Summary: "Invalid module source address",
|
|
Detail: fmt.Sprintf("Module %q (declared at %s line %d) has invalid source address %q: %s.", callName, call.Pos.Filename, call.Pos.Line, call.Source, err),
|
|
}))
|
|
// If we didn't have a valid source address then we can't continue
|
|
// down the module tree with this one.
|
|
continue
|
|
}
|
|
|
|
req := ModuleRequest{
|
|
Name: call.Name,
|
|
Path: path,
|
|
SourceAddr: sourceAddr,
|
|
VersionConstraints: vc,
|
|
Parent: parent,
|
|
CallPos: call.Pos,
|
|
}
|
|
|
|
mod, ver, modDiags := walker.LoadModule(&req)
|
|
diags = append(diags, modDiags...)
|
|
if mod == nil {
|
|
// nil can be returned if the source address was invalid and so
|
|
// nothing could be loaded whatsoever. LoadModule should've
|
|
// returned at least one error diagnostic in that case.
|
|
continue
|
|
}
|
|
|
|
child := &Config{
|
|
Parent: parent,
|
|
Root: parent.Root,
|
|
Path: path,
|
|
Module: mod,
|
|
CallPos: call.Pos,
|
|
SourceAddr: sourceAddr,
|
|
Version: ver,
|
|
}
|
|
|
|
child.Children, modDiags = buildChildModules(child, walker)
|
|
diags = diags.Append(modDiags)
|
|
|
|
ret[call.Name] = child
|
|
}
|
|
|
|
return ret, diags
|
|
}
|
|
|
|
// ModuleRequest is used as part of the ModuleWalker interface used with
|
|
// function BuildConfig.
|
|
type ModuleRequest struct {
|
|
// Name is the "logical name" of the module call within configuration.
|
|
// This is provided in case the name is used as part of a storage key
|
|
// for the module, but implementations must otherwise treat it as an
|
|
// opaque string. It is guaranteed to have already been validated as an
|
|
// HCL identifier and UTF-8 encoded.
|
|
Name string
|
|
|
|
// Path is a list of logical names that traverse from the root module to
|
|
// this module. This can be used, for example, to form a lookup key for
|
|
// each distinct module call in a configuration, allowing for multiple
|
|
// calls with the same name at different points in the tree.
|
|
Path addrs.Module
|
|
|
|
// SourceAddr is the source address string provided by the user in
|
|
// configuration.
|
|
SourceAddr addrs.ModuleSource
|
|
|
|
// VersionConstraint is the version constraint applied to the module in
|
|
// configuration.
|
|
VersionConstraints version.Constraints
|
|
|
|
// Parent is the partially-constructed module tree node that the loaded
|
|
// module will be added to. Callers may refer to any field of this
|
|
// structure except Children, which is still under construction when
|
|
// ModuleRequest objects are created and thus has undefined content.
|
|
// The main reason this is provided is so that full module paths can
|
|
// be constructed for uniqueness.
|
|
Parent *Config
|
|
|
|
// CallRange is the source position for the header of the "module" block
|
|
// in configuration that prompted this request.
|
|
CallPos tfconfig.SourcePos
|
|
}
|
|
|
|
// ModuleWalker is an interface used with BuildConfig.
|
|
type ModuleWalker interface {
|
|
LoadModule(req *ModuleRequest) (*tfconfig.Module, *version.Version, tfdiags.Diagnostics)
|
|
}
|
|
|
|
// ModuleWalkerFunc is an implementation of ModuleWalker that directly wraps
|
|
// a callback function, for more convenient use of that interface.
|
|
type ModuleWalkerFunc func(req *ModuleRequest) (*tfconfig.Module, *version.Version, tfdiags.Diagnostics)
|
|
|
|
func (f ModuleWalkerFunc) LoadModule(req *ModuleRequest) (*tfconfig.Module, *version.Version, tfdiags.Diagnostics) {
|
|
return f(req)
|
|
}
|