2020-02-11 19:17:37 +01:00
|
|
|
package configs
|
|
|
|
|
|
|
|
import (
|
2020-04-24 16:54:24 +02:00
|
|
|
"strings"
|
2020-02-11 19:17:37 +01:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/hashicorp/terraform/addrs"
|
|
|
|
)
|
|
|
|
|
|
|
|
// TestNewModule_provider_fqns 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())
|
|
|
|
}
|
|
|
|
|
2020-03-10 19:43:57 +01:00
|
|
|
p := addrs.NewProvider(addrs.DefaultRegistryHost, "foo", "test")
|
2020-02-11 19:17:37 +01:00
|
|
|
if name, exists := mod.ProviderLocalNames[p]; !exists {
|
|
|
|
t.Fatal("provider FQN foo/test not found")
|
|
|
|
} else {
|
2020-02-15 03:10:03 +01:00
|
|
|
if name != "foo-test" {
|
|
|
|
t.Fatalf("provider localname mismatch: got %s, want foo-test", name)
|
2020-02-11 19:17:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ensure the reverse lookup (fqn to local name) works as well
|
|
|
|
localName := mod.LocalNameForProvider(p)
|
2020-02-15 03:10:03 +01:00
|
|
|
if localName != "foo-test" {
|
2020-02-11 19:17:37 +01:00
|
|
|
t.Fatal("provider local name not found")
|
|
|
|
}
|
2020-03-23 14:50:35 +01:00
|
|
|
|
|
|
|
// if there is not a local name for a provider, it should return the type name
|
|
|
|
localName = mod.LocalNameForProvider(addrs.NewLegacyProvider("nonexist"))
|
|
|
|
if localName != "nonexist" {
|
|
|
|
t.Error("wrong local name returned for a non-local provider")
|
|
|
|
}
|
2020-04-02 01:11:15 +02:00
|
|
|
|
|
|
|
// 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.DefaultRegistryHost, "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)
|
|
|
|
}
|
|
|
|
}
|
2020-02-11 19:17:37 +01:00
|
|
|
}
|
2020-03-12 17:00:00 +01:00
|
|
|
|
2020-03-16 19:36:16 +01:00
|
|
|
// 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
|
2020-03-31 20:03:33 +02:00
|
|
|
wantImplicit := addrs.NewDefaultProvider("test")
|
2020-03-16 19:36:16 +01:00
|
|
|
wantFoo := addrs.NewProvider(addrs.DefaultRegistryHost, "foo", "test")
|
|
|
|
wantBar := addrs.NewProvider(addrs.DefaultRegistryHost, "bar", "test")
|
|
|
|
|
|
|
|
// root module
|
|
|
|
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,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-03-18 13:58:20 +01:00
|
|
|
// 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,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-03-16 19:36:16 +01:00
|
|
|
// child module
|
|
|
|
cm := cfg.Children["child"].Module
|
|
|
|
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,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-12 17:00:00 +01:00
|
|
|
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.DefaultRegistryHost, "foo", "test")
|
|
|
|
if !got.Equals(want) {
|
|
|
|
t.Fatalf("wrong result! got %#v, want %#v\n", got, want)
|
|
|
|
}
|
|
|
|
}
|
2020-04-24 16:54:24 +02:00
|
|
|
|
|
|
|
// 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.DefaultRegistryHost, "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.DefaultRegistryHost, "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.DefaultRegistryHost, "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,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|