mirror of
https://github.com/opentffoundation/opentf.git
synced 2026-03-22 19:00:35 -04:00
Previously we interpreted a "required_version" argument in a "terraform" block as if it were specifying an OpenTofu version constraint, when in reality most modules use this to represent a version constraint for OpenTofu's predecessor instead. The primary effect of this commit is to introduce a new top-level block type called "language" which describes language and implementation compatibility metadata in a way that intentionally differs from what's used by OpenTofu's predecessor. This also causes OpenTofu to ignore the required_version argument unless it appears in an OpenTofu-specific file with a ".tofu" suffix, and makes OpenTofu completely ignore the language edition and experimental feature opt-in options from OpenTofu's predecessor on the assumption that those could continue to evolve independently of changes in OpenTofu. We retain support for using required_versions in .tofu files as a bridge solution for modules that need to remain compatible with OpenTofu versions prior to v1.12. Module authors should keep following the strategy of having both a versions.tf and a versions.tofu file for now, and wait until the OpenTofu v1.11 series is end-of-life before adopting the new "language" block type. I also took this opportunity to simplify how we handle these parts of the configuration, since the OpenTofu project has no immediate plans to use either multiple language editions or language experiments and so for now we can reduce our handling of those language features to just enough that we'd return reasonable error messages if today's OpenTofu is exposed to a module that was written for a newer version of OpenTofu that extends these language features. The cross-cutting plumbing for representing the active experiments for a module is still present so that we can reactivate it later if we need to, but for now that set will always be empty. Signed-off-by: Martin Atkins <mart@degeneration.co.uk>
551 lines
19 KiB
Go
551 lines
19 KiB
Go
// Copyright (c) The OpenTofu Authors
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
// Copyright (c) 2023 HashiCorp, Inc.
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
package configs
|
|
|
|
import (
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
"github.com/hashicorp/hcl/v2"
|
|
"github.com/opentofu/opentofu/internal/addrs"
|
|
"github.com/zclconf/go-cty/cty"
|
|
)
|
|
|
|
// TestNewModule_provider_local_name exercises module.gatherProviderLocalNames()
|
|
func TestNewModule_provider_local_name(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/providers-explicit-fqn")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
p := addrs.NewProvider(addrs.DefaultProviderRegistryHost, "foo", "test")
|
|
if name, exists := mod.ProviderLocalNames[p]; !exists {
|
|
t.Fatal("provider FQN foo/test not found")
|
|
} else {
|
|
if name != "foo-test" {
|
|
t.Fatalf("provider localname mismatch: got %s, want foo-test", name)
|
|
}
|
|
}
|
|
|
|
// ensure the reverse lookup (fqn to local name) works as well
|
|
localName := mod.LocalNameForProvider(p)
|
|
if localName != "foo-test" {
|
|
t.Fatal("provider local name not found")
|
|
}
|
|
|
|
// if there is not a local name for a provider, it should return the type name
|
|
localName = mod.LocalNameForProvider(addrs.NewDefaultProvider("nonexist"))
|
|
if localName != "nonexist" {
|
|
t.Error("wrong local name returned for a non-local provider")
|
|
}
|
|
|
|
// can also look up the "terraform" provider and see that it sources is
|
|
// allowed to be overridden, even though there is a builtin provider
|
|
// called "terraform".
|
|
p = addrs.NewProvider(addrs.DefaultProviderRegistryHost, "not-builtin", "not-terraform")
|
|
if name, exists := mod.ProviderLocalNames[p]; !exists {
|
|
t.Fatal("provider FQN not-builtin/not-terraform not found")
|
|
} else {
|
|
if name != "terraform" {
|
|
t.Fatalf("provider localname mismatch: got %s, want terraform", name)
|
|
}
|
|
}
|
|
}
|
|
|
|
// This test validates the provider FQNs set in each Resource
|
|
func TestNewModule_resource_providers(t *testing.T) {
|
|
cfg, diags := testNestedModuleConfigFromDir(t, "testdata/valid-modules/nested-providers-fqns")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
// both the root and child module have two resources, one which should use
|
|
// the default implied provider and one explicitly using a provider set in
|
|
// required_providers
|
|
wantImplicit := addrs.NewDefaultProvider("test")
|
|
wantFoo := addrs.NewProvider(addrs.DefaultProviderRegistryHost, "foo", "test")
|
|
wantBar := addrs.NewProvider(addrs.DefaultProviderRegistryHost, "bar", "test")
|
|
|
|
// root module
|
|
if got, want := len(cfg.Module.ManagedResources), 2; got != want {
|
|
t.Fatalf("expected to have %d managed resources in the root module but got %d", want, got)
|
|
}
|
|
if got, want := len(cfg.Module.DataResources), 1; got != want {
|
|
t.Fatalf("expected to have %d data sources in the root module but got %d", want, got)
|
|
}
|
|
if got, want := len(cfg.Module.EphemeralResources), 2; got != want {
|
|
t.Fatalf("expected to have %d ephemeral resources in the root module but got %d", want, got)
|
|
}
|
|
if !cfg.Module.ManagedResources["test_instance.explicit"].Provider.Equals(wantFoo) {
|
|
t.Fatalf("wrong provider for \"test_instance.explicit\"\ngot: %s\nwant: %s",
|
|
cfg.Module.ManagedResources["test_instance.explicit"].Provider,
|
|
wantFoo,
|
|
)
|
|
}
|
|
if !cfg.Module.ManagedResources["test_instance.implicit"].Provider.Equals(wantImplicit) {
|
|
t.Fatalf("wrong provider for \"test_instance.implicit\"\ngot: %s\nwant: %s",
|
|
cfg.Module.ManagedResources["test_instance.implicit"].Provider,
|
|
wantImplicit,
|
|
)
|
|
}
|
|
|
|
// a data source
|
|
if !cfg.Module.DataResources["data.test_resource.explicit"].Provider.Equals(wantFoo) {
|
|
t.Fatalf("wrong provider for \"module.child.test_instance.explicit\"\ngot: %s\nwant: %s",
|
|
cfg.Module.ManagedResources["test_instance.explicit"].Provider,
|
|
wantBar,
|
|
)
|
|
}
|
|
|
|
// ephemeral resources test
|
|
if !cfg.Module.EphemeralResources["ephemeral.test_ephemeral.explicit"].Provider.Equals(wantFoo) {
|
|
t.Fatalf("wrong provider for \"test_ephemeral.explicit\"\ngot: %s\nwant: %s",
|
|
cfg.Module.EphemeralResources["test_ephemeral.explicit"].Provider,
|
|
wantFoo,
|
|
)
|
|
}
|
|
if !cfg.Module.EphemeralResources["ephemeral.test_ephemeral.implicit"].Provider.Equals(wantImplicit) {
|
|
t.Fatalf("wrong provider for \"test_ephemeral.implicit\"\ngot: %s\nwant: %s",
|
|
cfg.Module.EphemeralResources["test_instance.implicit"].Provider,
|
|
wantImplicit,
|
|
)
|
|
}
|
|
|
|
// child module
|
|
cm := cfg.Children["child"].Module
|
|
if got, want := len(cm.ManagedResources), 3; got != want {
|
|
t.Fatalf("expected to have %d managed resources in the child module but got %d", want, got)
|
|
}
|
|
if got, want := len(cm.DataResources), 0; got != want {
|
|
t.Fatalf("expected to have %d data sources in the child module but got %d", want, got)
|
|
}
|
|
if got, want := len(cm.EphemeralResources), 2; got != want {
|
|
t.Fatalf("expected to have %d ephemeral resources in the child module but got %d", want, got)
|
|
}
|
|
if !cm.ManagedResources["test_instance.explicit"].Provider.Equals(wantBar) {
|
|
t.Fatalf("wrong provider for \"module.child.test_instance.explicit\"\ngot: %s\nwant: %s",
|
|
cfg.Module.ManagedResources["test_instance.explicit"].Provider,
|
|
wantBar,
|
|
)
|
|
}
|
|
if !cm.ManagedResources["test_instance.implicit"].Provider.Equals(wantImplicit) {
|
|
t.Fatalf("wrong provider for \"module.child.test_instance.implicit\"\ngot: %s\nwant: %s",
|
|
cfg.Module.ManagedResources["test_instance.implicit"].Provider,
|
|
wantImplicit,
|
|
)
|
|
}
|
|
// ephemeral
|
|
if !cm.EphemeralResources["ephemeral.test_ephemeral.other_explicit"].Provider.Equals(wantFoo) {
|
|
t.Fatalf("wrong provider for \"module.child.ephemeral.test_ephemeral.other_explicit\"\ngot: %s\nwant: %s",
|
|
cfg.Module.EphemeralResources["ephemeral.test_ephemeral.other_explicit"].Provider,
|
|
wantFoo,
|
|
)
|
|
}
|
|
if !cm.EphemeralResources["ephemeral.test_ephemeral.other_implicit"].Provider.Equals(wantImplicit) {
|
|
t.Fatalf("wrong provider for \"module.child.ephemeral.test_ephemeral.other_implicit\"\ngot: %s\nwant: %s",
|
|
cfg.Module.EphemeralResources["ephemeral.test_ephemeral.other_implicit"].Provider,
|
|
wantFoo,
|
|
)
|
|
}
|
|
}
|
|
|
|
func TestProviderForLocalConfig(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/providers-explicit-fqn")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
lc := addrs.LocalProviderConfig{LocalName: "foo-test"}
|
|
got := mod.ProviderForLocalConfig(lc)
|
|
want := addrs.NewProvider(addrs.DefaultProviderRegistryHost, "foo", "test")
|
|
if !got.Equals(want) {
|
|
t.Fatalf("wrong result! got %#v, want %#v\n", got, want)
|
|
}
|
|
}
|
|
|
|
// At most one required_providers block per module is permitted.
|
|
func TestModule_required_providers_multiple(t *testing.T) {
|
|
_, diags := testModuleFromDir("testdata/invalid-modules/multiple-required-providers")
|
|
if !diags.HasErrors() {
|
|
t.Fatal("module should have error diags, but does not")
|
|
}
|
|
|
|
want := `Duplicate required providers configuration`
|
|
if got := diags.Error(); !strings.Contains(got, want) {
|
|
t.Fatalf("expected error to contain %q\nerror was:\n%s", want, got)
|
|
}
|
|
}
|
|
|
|
// A module may have required_providers configured in files loaded later than
|
|
// resources. These provider settings should still be reflected in the
|
|
// resources' configuration.
|
|
func TestModule_required_providers_after_resource(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/valid-modules/required-providers-after-resource")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
want := addrs.NewProvider(addrs.DefaultProviderRegistryHost, "foo", "test")
|
|
|
|
req, exists := mod.ProviderRequirements.RequiredProviders["test"]
|
|
if !exists {
|
|
t.Fatal("no provider requirements found for \"test\"")
|
|
}
|
|
if req.Type != want {
|
|
t.Errorf("wrong provider addr for \"test\"\ngot: %s\nwant: %s",
|
|
req.Type, want,
|
|
)
|
|
}
|
|
|
|
if got := mod.ManagedResources["test_instance.my-instance"].Provider; !got.Equals(want) {
|
|
t.Errorf("wrong provider addr for \"test_instance.my-instance\"\ngot: %s\nwant: %s",
|
|
got, want,
|
|
)
|
|
}
|
|
}
|
|
|
|
// We support overrides for required_providers blocks, which should replace the
|
|
// entire block for each provider localname, leaving other blocks unaffected.
|
|
// This should also be reflected in any resources in the module using this
|
|
// provider.
|
|
func TestModule_required_provider_overrides(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/valid-modules/required-providers-overrides")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
// The foo provider and resource should be unaffected
|
|
want := addrs.NewProvider(addrs.DefaultProviderRegistryHost, "acme", "foo")
|
|
req, exists := mod.ProviderRequirements.RequiredProviders["foo"]
|
|
if !exists {
|
|
t.Fatal("no provider requirements found for \"foo\"")
|
|
}
|
|
if req.Type != want {
|
|
t.Errorf("wrong provider addr for \"foo\"\ngot: %s\nwant: %s",
|
|
req.Type, want,
|
|
)
|
|
}
|
|
if got := mod.ManagedResources["foo_thing.ft"].Provider; !got.Equals(want) {
|
|
t.Errorf("wrong provider addr for \"foo_thing.ft\"\ngot: %s\nwant: %s",
|
|
got, want,
|
|
)
|
|
}
|
|
|
|
// The bar provider and resource should be using the override config
|
|
want = addrs.NewProvider(addrs.DefaultProviderRegistryHost, "blorp", "bar")
|
|
req, exists = mod.ProviderRequirements.RequiredProviders["bar"]
|
|
if !exists {
|
|
t.Fatal("no provider requirements found for \"bar\"")
|
|
}
|
|
if req.Type != want {
|
|
t.Errorf("wrong provider addr for \"bar\"\ngot: %s\nwant: %s",
|
|
req.Type, want,
|
|
)
|
|
}
|
|
if gotVer, wantVer := req.Requirement.Required.String(), "~>2.0.0"; gotVer != wantVer {
|
|
t.Errorf("wrong provider version constraint for \"bar\"\ngot: %s\nwant: %s",
|
|
gotVer, wantVer,
|
|
)
|
|
}
|
|
if got := mod.ManagedResources["bar_thing.bt"].Provider; !got.Equals(want) {
|
|
t.Errorf("wrong provider addr for \"bar_thing.bt\"\ngot: %s\nwant: %s",
|
|
got, want,
|
|
)
|
|
}
|
|
}
|
|
|
|
// When having multiple required providers defined, and one with syntax error,
|
|
// ensure that the diagnostics are returned correctly for each and every validation.
|
|
// In case a required_provider is containing syntax errors, we are returning an empty one just to allow the
|
|
// later validations to add their results.
|
|
func TestModule_required_providers_multiple_one_with_syntax_error(t *testing.T) {
|
|
_, diags := testModuleFromDir("testdata/invalid-modules/multiple-required-providers-with-syntax-error")
|
|
if !diags.HasErrors() {
|
|
t.Fatal("module should have error diags, but does not")
|
|
}
|
|
|
|
want := []string{
|
|
`Missing attribute value; Expected an attribute value`,
|
|
`Unexpected "resource" block; Blocks are not allowed here`,
|
|
`Duplicate required providers configuration`,
|
|
}
|
|
if wantLen, gotLen := len(want), len(diags.Errs()); wantLen != gotLen {
|
|
t.Fatalf("expected %d errors but got %d", wantLen, gotLen)
|
|
}
|
|
for i, e := range diags.Errs() {
|
|
if got := e.Error(); !strings.Contains(got, want[i]) {
|
|
t.Errorf("expected error to contain %q\nerror was: \n\t%q\n", want[i], got)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Resources without explicit provider configuration are assigned a provider
|
|
// implied based on the resource type. For example, this resource:
|
|
//
|
|
// resource "foo_instance" "test" {}
|
|
//
|
|
// ...is assigned to whichever provider has local name "foo" in the current
|
|
// module.
|
|
//
|
|
// To find the correct provider, we first look in the module's provider
|
|
// requirements map for a local name matching the resource type, and fall back
|
|
// to a default provider if none is found. This applies to both managed and
|
|
// data resources.
|
|
func TestModule_implied_provider(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/valid-modules/implied-providers")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
// The three providers used in the config resources
|
|
foo := addrs.NewProvider("registry.acme.corp", "acme", "foo")
|
|
whatever := addrs.NewProvider(addrs.DefaultProviderRegistryHost, "acme", "something")
|
|
bar := addrs.NewDefaultProvider("bar")
|
|
|
|
// Verify that the registry.acme.corp/acme/foo provider is defined in the
|
|
// module provider requirements with local name "foo"
|
|
req, exists := mod.ProviderRequirements.RequiredProviders["foo"]
|
|
if !exists {
|
|
t.Fatal("no provider requirements found for \"foo\"")
|
|
}
|
|
if req.Type != foo {
|
|
t.Errorf("wrong provider addr for \"foo\"\ngot: %s\nwant: %s",
|
|
req.Type, foo,
|
|
)
|
|
}
|
|
|
|
// Verify that the acme/something provider is defined in the
|
|
// module provider requirements with local name "whatever"
|
|
req, exists = mod.ProviderRequirements.RequiredProviders["whatever"]
|
|
if !exists {
|
|
t.Fatal("no provider requirements found for \"foo\"")
|
|
}
|
|
if req.Type != whatever {
|
|
t.Errorf("wrong provider addr for \"whatever\"\ngot: %s\nwant: %s",
|
|
req.Type, whatever,
|
|
)
|
|
}
|
|
|
|
// Check that resources are assigned the correct providers: foo_* resources
|
|
// should have the custom foo provider, bar_* resources the default bar
|
|
// provider.
|
|
tests := []struct {
|
|
Address string
|
|
Provider addrs.Provider
|
|
}{
|
|
{"foo_resource.a", foo},
|
|
{"data.foo_resource.b", foo},
|
|
{"ephemeral.foo_resource.c", foo},
|
|
{"bar_resource.d", bar},
|
|
{"data.bar_resource.e", bar},
|
|
{"ephemeral.bar_resource.f", bar},
|
|
{"whatever_resource.g", whatever},
|
|
{"data.whatever_resource.h", whatever},
|
|
{"ephemeral.whatever_resource.i", whatever},
|
|
}
|
|
for _, test := range tests {
|
|
resources := mod.ManagedResources
|
|
switch test.Address[:strings.Index(test.Address, ".")+1] {
|
|
case "data.":
|
|
resources = mod.DataResources
|
|
case "ephemeral.":
|
|
resources = mod.EphemeralResources
|
|
}
|
|
resource, exists := resources[test.Address]
|
|
if !exists {
|
|
t.Errorf("could not find resource %q in %#v", test.Address, resources)
|
|
continue
|
|
}
|
|
if got := resource.Provider; !got.Equals(test.Provider) {
|
|
t.Errorf("wrong provider addr for %q\ngot: %s\nwant: %s",
|
|
test.Address, got, test.Provider,
|
|
)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestImpliedProviderForUnqualifiedType(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/valid-modules/implied-providers")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
foo := addrs.NewProvider("registry.acme.corp", "acme", "foo")
|
|
whatever := addrs.NewProvider(addrs.DefaultProviderRegistryHost, "acme", "something")
|
|
bar := addrs.NewDefaultProvider("bar")
|
|
tf := addrs.NewBuiltInProvider("terraform")
|
|
|
|
tests := []struct {
|
|
Type string
|
|
Provider addrs.Provider
|
|
}{
|
|
{"foo", foo},
|
|
{"whatever", whatever},
|
|
{"bar", bar},
|
|
{"terraform", tf},
|
|
}
|
|
for _, test := range tests {
|
|
got := mod.ImpliedProviderForUnqualifiedType(test.Type)
|
|
if !got.Equals(test.Provider) {
|
|
t.Errorf("wrong result for %q: got %#v, want %#v\n", test.Type, got, test.Provider)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestModule_backend_override(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/valid-modules/override-backend")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
gotType := mod.Backend.Type
|
|
wantType := "bar"
|
|
|
|
if gotType != wantType {
|
|
t.Errorf("wrong result for backend type: got %#v, want %#v\n", gotType, wantType)
|
|
}
|
|
|
|
attrs, _ := mod.Backend.Config.JustAttributes()
|
|
|
|
gotAttr, diags := attrs["path"].Expr.Value(nil)
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
wantAttr := cty.StringVal("CHANGED/relative/path/to/terraform.tfstate")
|
|
|
|
if !gotAttr.RawEquals(wantAttr) {
|
|
t.Errorf("wrong result for backend 'path': got %#v, want %#v\n", gotAttr, wantAttr)
|
|
}
|
|
}
|
|
|
|
// Unlike most other overrides, backend blocks do not require a base configuration in a primary
|
|
// configuration file, as an omitted backend there implies the local backend.
|
|
func TestModule_backend_override_no_base(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/valid-modules/override-backend-no-base")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
if mod.Backend == nil {
|
|
t.Errorf("expected module Backend not to be nil")
|
|
}
|
|
}
|
|
|
|
func TestModule_cloud_override_backend(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/valid-modules/override-backend-with-cloud")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
if mod.Backend != nil {
|
|
t.Errorf("expected module Backend to be nil")
|
|
}
|
|
|
|
if mod.CloudConfig == nil {
|
|
t.Errorf("expected module CloudConfig not to be nil")
|
|
}
|
|
}
|
|
|
|
// Unlike most other overrides, cloud blocks do not require a base configuration in a primary
|
|
// configuration file, as an omitted backend there implies the local backend and cloud blocks
|
|
// override backends.
|
|
func TestModule_cloud_override_no_base(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/valid-modules/override-cloud-no-base")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
if mod.CloudConfig == nil {
|
|
t.Errorf("expected module CloudConfig not to be nil")
|
|
}
|
|
}
|
|
|
|
func TestModule_cloud_override(t *testing.T) {
|
|
mod, diags := testModuleFromDir("testdata/valid-modules/override-cloud")
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
attrs, _ := mod.CloudConfig.Config.JustAttributes()
|
|
|
|
gotAttr, diags := attrs["organization"].Expr.Value(nil)
|
|
if diags.HasErrors() {
|
|
t.Fatal(diags.Error())
|
|
}
|
|
|
|
wantAttr := cty.StringVal("CHANGED")
|
|
|
|
if !gotAttr.RawEquals(wantAttr) {
|
|
t.Errorf("wrong result for Cloud 'organization': got %#v, want %#v\n", gotAttr, wantAttr)
|
|
}
|
|
|
|
// The override should have completely replaced the cloud block in the primary file, no merging
|
|
if attrs["should_not_be_present_with_override"] != nil {
|
|
t.Errorf("expected 'should_not_be_present_with_override' attribute to be nil")
|
|
}
|
|
}
|
|
|
|
func TestModule_cloud_duplicate_overrides(t *testing.T) {
|
|
_, diags := testModuleFromDir("testdata/invalid-modules/override-cloud-duplicates")
|
|
want := `Duplicate cloud configurations`
|
|
if got := diags.Error(); !strings.Contains(got, want) {
|
|
t.Fatalf("expected module error to contain %q\nerror was:\n%s", want, got)
|
|
}
|
|
}
|
|
|
|
func TestModuleFromTheFuture(t *testing.T) {
|
|
_, diags := testModuleFromDir("testdata/invalid-modules/unsupported-version-and-other-error")
|
|
if !diags.HasErrors() {
|
|
t.Fatal("unexpected success; want 'incompatible module' error")
|
|
}
|
|
|
|
var gotSummaries []string
|
|
for _, diag := range diags {
|
|
if diag.Severity == hcl.DiagError {
|
|
gotSummaries = append(gotSummaries, diag.Summary)
|
|
}
|
|
}
|
|
wantSummaries := []string{
|
|
// The configuration fixture used here includes both a mismatching version
|
|
// constraint _and_ an unrecognized block type, but we should've reported
|
|
// only the mismatching version constraint because we assume the
|
|
// unrecognized block type became valid in a future version of OpenTofu.
|
|
"Incompatible module",
|
|
}
|
|
|
|
if diff := cmp.Diff(wantSummaries, gotSummaries); diff != "" {
|
|
t.Error("wrong error diagnostics\n" + diff)
|
|
}
|
|
}
|
|
|
|
func TestResourceByAddr(t *testing.T) {
|
|
managedResource := &Resource{Mode: addrs.ManagedResourceMode, Name: "name", Type: "test_resource"}
|
|
dataResource := &Resource{Mode: addrs.DataResourceMode, Name: "name", Type: "test_data"}
|
|
ephemeralResource := &Resource{Mode: addrs.EphemeralResourceMode, Name: "name", Type: "test_ephemeral"}
|
|
m := Module{
|
|
ManagedResources: map[string]*Resource{
|
|
managedResource.Addr().String(): managedResource,
|
|
},
|
|
DataResources: map[string]*Resource{
|
|
dataResource.Addr().String(): dataResource,
|
|
},
|
|
EphemeralResources: map[string]*Resource{
|
|
ephemeralResource.Addr().String(): ephemeralResource,
|
|
},
|
|
}
|
|
if got, want := m.ResourceByAddr(managedResource.Addr()), managedResource; got != want {
|
|
t.Fatalf("expected resource %+v but got %+v", want, got)
|
|
}
|
|
if got, want := m.ResourceByAddr(dataResource.Addr()), dataResource; got != want {
|
|
t.Fatalf("expected resource %+v but got %+v", want, got)
|
|
}
|
|
if got, want := m.ResourceByAddr(ephemeralResource.Addr()), ephemeralResource; got != want {
|
|
t.Fatalf("expected resource %+v but got %+v", want, got)
|
|
}
|
|
|
|
}
|