mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-07 22:53:08 -06:00
86f0b5191c
The provider FQN is becoming our primary identifier for a provider, so it's important that we are clear about the equality rules for these addresses and what characters are valid within them. We previously had a basic regex permitting ASCII letters and digits for validation and no normalization at all. We need to do at least case folding and UTF-8 normalization because these names will appear in file and directory names in case-insensitive filesystems and in repository names such as on GitHub. Since we're already using DNS-style normalization and validation rules for the hostname part, rather than defining an entirely new set of rules here we'll just treat the provider namespace and type as if they were single labels in a DNS name. Aside from some internal consistency, that also works out nicely because systems like GitHub use organization and repository names as part of hostnames (e.g. with GitHub Pages) and so tend to apply comparable constraints themselves. This introduces the possibility of names containing letters from alphabets other than the latin alphabet, and for latin letters with diacritics. That's consistent with our introduction of similar support for identifiers in the language in Terraform 0.12, and is intended to be more friendly to Terraform users throughout the world that might prefer to name their products using a different alphabet. This is also a further justification for using the DNS normalization rules: modern companies tend to choose product names that make good domain names, and now such names will be usable as Terraform provider names too.
212 lines
6.4 KiB
Go
212 lines
6.4 KiB
Go
package terraform
|
|
|
|
import (
|
|
version "github.com/hashicorp/go-version"
|
|
|
|
"github.com/hashicorp/terraform/addrs"
|
|
"github.com/hashicorp/terraform/configs"
|
|
"github.com/hashicorp/terraform/moduledeps"
|
|
"github.com/hashicorp/terraform/plugin/discovery"
|
|
"github.com/hashicorp/terraform/states"
|
|
)
|
|
|
|
// ConfigTreeDependencies returns the dependencies of the tree of modules
|
|
// described by the given configuration and state.
|
|
//
|
|
// Both configuration and state are required because there can be resources
|
|
// implied by instances in the state that no longer exist in config.
|
|
func ConfigTreeDependencies(root *configs.Config, state *states.State) *moduledeps.Module {
|
|
// First we walk the configuration tree to build the overall structure
|
|
// and capture the explicit/implicit/inherited provider dependencies.
|
|
deps := configTreeConfigDependencies(root, nil)
|
|
|
|
// Next we walk over the resources in the state to catch any additional
|
|
// dependencies created by existing resources that are no longer in config.
|
|
// Most things we find in state will already be present in 'deps', but
|
|
// we're interested in the rare thing that isn't.
|
|
configTreeMergeStateDependencies(deps, state)
|
|
|
|
return deps
|
|
}
|
|
|
|
func configTreeConfigDependencies(root *configs.Config, inheritProviders map[string]*configs.Provider) *moduledeps.Module {
|
|
if root == nil {
|
|
// If no config is provided, we'll make a synthetic root.
|
|
// This isn't necessarily correct if we're called with a nil that
|
|
// *isn't* at the root, but in practice that can never happen.
|
|
return &moduledeps.Module{
|
|
Name: "root",
|
|
Providers: make(moduledeps.Providers),
|
|
}
|
|
}
|
|
|
|
name := "root"
|
|
if len(root.Path) != 0 {
|
|
name = root.Path[len(root.Path)-1]
|
|
}
|
|
|
|
ret := &moduledeps.Module{
|
|
Name: name,
|
|
}
|
|
|
|
module := root.Module
|
|
|
|
// Provider dependencies
|
|
{
|
|
providers := make(moduledeps.Providers)
|
|
|
|
// The main way to declare a provider dependency is explicitly inside
|
|
// the "terraform" block, which allows declaring a requirement without
|
|
// also creating a configuration.
|
|
for localName, req := range module.ProviderRequirements {
|
|
// The handling here is a bit fiddly because the moduledeps package
|
|
// was designed around the legacy (pre-0.12) configuration model
|
|
// and hasn't yet been revised to handle the new model. As a result,
|
|
// we need to do some translation here.
|
|
// FIXME: Eventually we should adjust the underlying model so we
|
|
// can also retain the source location of each constraint, for
|
|
// more informative output from the "terraform providers" command.
|
|
var rawConstraints version.Constraints
|
|
for _, constraint := range req.VersionConstraints {
|
|
rawConstraints = append(rawConstraints, constraint.Required...)
|
|
}
|
|
discoConstraints := discovery.NewConstraints(rawConstraints)
|
|
fqn := req.Type
|
|
if fqn.IsZero() {
|
|
fqn = addrs.NewLegacyProvider(localName)
|
|
}
|
|
|
|
providers[req.Type] = moduledeps.ProviderDependency{
|
|
Constraints: discoConstraints,
|
|
Reason: moduledeps.ProviderDependencyExplicit,
|
|
}
|
|
}
|
|
|
|
// Provider configurations can also include version constraints,
|
|
// allowing for more terse declaration in situations where both a
|
|
// configuration and a constraint are defined in the same module.
|
|
for _, pCfg := range module.ProviderConfigs {
|
|
fqn := module.ProviderForLocalConfig(pCfg.Addr())
|
|
|
|
discoConstraints := discovery.AllVersions
|
|
if pCfg.Version.Required != nil {
|
|
discoConstraints = discovery.NewConstraints(pCfg.Version.Required)
|
|
}
|
|
if existing, exists := providers[fqn]; exists {
|
|
constraints := existing.Constraints.Append(discoConstraints)
|
|
providers[fqn] = moduledeps.ProviderDependency{
|
|
Constraints: constraints,
|
|
Reason: moduledeps.ProviderDependencyExplicit,
|
|
}
|
|
} else {
|
|
providers[fqn] = moduledeps.ProviderDependency{
|
|
Constraints: discoConstraints,
|
|
Reason: moduledeps.ProviderDependencyExplicit,
|
|
}
|
|
}
|
|
}
|
|
|
|
// Each resource in the configuration creates an *implicit* provider
|
|
// dependency, though we'll only record it if there isn't already
|
|
// an explicit dependency on the same provider.
|
|
for _, rc := range module.ManagedResources {
|
|
addr := rc.ProviderConfigAddr()
|
|
fqn := module.ProviderForLocalConfig(addr)
|
|
|
|
if _, exists := providers[fqn]; exists {
|
|
// Explicit dependency already present
|
|
continue
|
|
}
|
|
|
|
reason := moduledeps.ProviderDependencyImplicit
|
|
if _, inherited := inheritProviders[addr.StringCompact()]; inherited {
|
|
reason = moduledeps.ProviderDependencyInherited
|
|
}
|
|
|
|
providers[fqn] = moduledeps.ProviderDependency{
|
|
Constraints: discovery.AllVersions,
|
|
Reason: reason,
|
|
}
|
|
}
|
|
for _, rc := range module.DataResources {
|
|
addr := rc.ProviderConfigAddr()
|
|
fqn := module.ProviderForLocalConfig(addr)
|
|
|
|
if _, exists := providers[fqn]; exists {
|
|
// Explicit dependency already present
|
|
continue
|
|
}
|
|
|
|
reason := moduledeps.ProviderDependencyImplicit
|
|
if _, inherited := inheritProviders[addr.String()]; inherited {
|
|
reason = moduledeps.ProviderDependencyInherited
|
|
}
|
|
|
|
providers[fqn] = moduledeps.ProviderDependency{
|
|
Constraints: discovery.AllVersions,
|
|
Reason: reason,
|
|
}
|
|
}
|
|
|
|
ret.Providers = providers
|
|
}
|
|
|
|
childInherit := make(map[string]*configs.Provider)
|
|
for k, v := range inheritProviders {
|
|
childInherit[k] = v
|
|
}
|
|
for k, v := range module.ProviderConfigs {
|
|
childInherit[k] = v
|
|
}
|
|
for _, c := range root.Children {
|
|
ret.Children = append(ret.Children, configTreeConfigDependencies(c, childInherit))
|
|
}
|
|
|
|
return ret
|
|
}
|
|
|
|
func configTreeMergeStateDependencies(root *moduledeps.Module, state *states.State) {
|
|
if state == nil {
|
|
return
|
|
}
|
|
|
|
findModule := func(path addrs.ModuleInstance) *moduledeps.Module {
|
|
module := root
|
|
for _, step := range path {
|
|
var next *moduledeps.Module
|
|
for _, cm := range module.Children {
|
|
if cm.Name == step.Name {
|
|
next = cm
|
|
break
|
|
}
|
|
}
|
|
|
|
if next == nil {
|
|
// If we didn't find a next node, we'll need to make one
|
|
next = &moduledeps.Module{
|
|
Name: step.Name,
|
|
Providers: make(moduledeps.Providers),
|
|
}
|
|
module.Children = append(module.Children, next)
|
|
}
|
|
|
|
module = next
|
|
}
|
|
return module
|
|
}
|
|
|
|
for _, ms := range state.Modules {
|
|
module := findModule(ms.Addr)
|
|
|
|
for _, rs := range ms.Resources {
|
|
fqn := rs.ProviderConfig.Provider
|
|
if _, exists := module.Providers[fqn]; !exists {
|
|
module.Providers[fqn] = moduledeps.ProviderDependency{
|
|
Constraints: discovery.AllVersions,
|
|
Reason: moduledeps.ProviderDependencyFromState,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|