mirror of https://github.com/pulumi/pulumi.git
376 lines
12 KiB
Python
376 lines
12 KiB
Python
# Copyright 2016-2021, Pulumi Corporation.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
from typing import Optional, TypeVar, Awaitable, List, Any
|
|
import asyncio
|
|
import os
|
|
import pytest
|
|
import unittest
|
|
|
|
from pulumi.resource import DependencyProviderResource
|
|
from pulumi.runtime import settings, mocks
|
|
from pulumi.runtime.proto import resource_pb2
|
|
from pulumi import ResourceOptions
|
|
from pulumi.runtime.rpc import ERROR_ON_DEPENDENCY_CYCLES_VAR
|
|
import pulumi
|
|
|
|
|
|
T = TypeVar("T")
|
|
|
|
|
|
class DependencyProviderResourceTests(unittest.TestCase):
|
|
def test_get_package(self):
|
|
res = DependencyProviderResource(
|
|
"urn:pulumi:stack::project::pulumi:providers:aws::default_4_13_0"
|
|
)
|
|
self.assertEqual("aws", res.package)
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def clean_up_env_vars():
|
|
try:
|
|
del os.environ[ERROR_ON_DEPENDENCY_CYCLES_VAR]
|
|
except KeyError:
|
|
pass
|
|
|
|
|
|
@pulumi.runtime.test
|
|
def test_depends_on_accepts_outputs(dep_tracker):
|
|
dep1 = MockResource(name="dep1")
|
|
dep2 = MockResource(name="dep2")
|
|
out = output_depending_on_resource(dep1, isKnown=True).apply(lambda _: dep2)
|
|
res = MockResource(name="res", opts=pulumi.ResourceOptions(depends_on=[out]))
|
|
|
|
def check(urns):
|
|
(dep1_urn, dep2_urn, res_urn) = urns
|
|
res_deps = dep_tracker.dependencies[res_urn]
|
|
assert (
|
|
dep1_urn in res_deps
|
|
), "Failed to propagate indirect dependencies via depends_on"
|
|
assert (
|
|
dep2_urn in res_deps
|
|
), "Failed to propagate direct dependencies via depends_on"
|
|
|
|
return pulumi.Output.all(dep1.urn, dep2.urn, res.urn).apply(check)
|
|
|
|
|
|
@pulumi.runtime.test
|
|
def test_depends_on_outputs_works_in_presence_of_unknowns(dep_tracker_preview):
|
|
dep1 = MockResource(name="dep1")
|
|
dep2 = MockResource(name="dep2")
|
|
known = output_depending_on_resource(dep1, isKnown=True).apply(lambda _: dep2)
|
|
unknown = output_depending_on_resource(dep2, isKnown=False).apply(lambda _: dep2)
|
|
res = MockResource(
|
|
name="res", opts=pulumi.ResourceOptions(depends_on=[known, unknown])
|
|
)
|
|
|
|
def check(urns):
|
|
(dep1_urn, res_urn) = urns
|
|
assert dep1_urn in dep_tracker_preview.dependencies[res_urn]
|
|
|
|
return pulumi.Output.all(dep1.urn, res.urn).apply(check)
|
|
|
|
|
|
@pulumi.runtime.test
|
|
def test_depends_on_respects_top_level_implicit_dependencies(dep_tracker):
|
|
dep1 = MockResource(name="dep1")
|
|
dep2 = MockResource(name="dep2")
|
|
out = output_depending_on_resource(dep1, isKnown=True).apply(lambda _: [dep2])
|
|
res = MockResource(name="res", opts=pulumi.ResourceOptions(depends_on=out))
|
|
|
|
def check(urns):
|
|
(dep1_urn, dep2_urn, res_urn) = urns
|
|
assert set(dep_tracker.dependencies[res_urn]) == set([dep1_urn, dep2_urn])
|
|
|
|
return pulumi.Output.all(dep1.urn, dep2.urn, res.urn).apply(check)
|
|
|
|
|
|
def promise(x: T) -> Awaitable[T]:
|
|
fut: asyncio.Future[T] = asyncio.Future()
|
|
fut.set_result(x)
|
|
return fut
|
|
|
|
|
|
def out(x: T) -> pulumi.Output[T]:
|
|
return pulumi.Output.from_input(x)
|
|
|
|
|
|
def depends_on_variations(dep: pulumi.Resource) -> List[pulumi.ResourceOptions]:
|
|
return [
|
|
pulumi.ResourceOptions(depends_on=None),
|
|
pulumi.ResourceOptions(depends_on=dep),
|
|
pulumi.ResourceOptions(depends_on=[dep]),
|
|
pulumi.ResourceOptions(depends_on=promise(dep)),
|
|
pulumi.ResourceOptions(depends_on=out(dep)),
|
|
pulumi.ResourceOptions(depends_on=promise([dep])),
|
|
pulumi.ResourceOptions(depends_on=out([dep])),
|
|
pulumi.ResourceOptions(depends_on=promise([promise(dep)])),
|
|
pulumi.ResourceOptions(depends_on=promise([out(dep)])),
|
|
pulumi.ResourceOptions(depends_on=out([promise(dep)])),
|
|
pulumi.ResourceOptions(depends_on=out([out(dep)])),
|
|
]
|
|
|
|
|
|
@pulumi.runtime.test
|
|
def test_depends_on_typing_variations(dep_tracker) -> None:
|
|
dep: pulumi.Resource = MockResource(name="dep1")
|
|
|
|
def check(i, urns):
|
|
(dep_urn, res_urn) = urns
|
|
|
|
if i == 0:
|
|
assert dep_tracker.dependencies[res_urn] == set([])
|
|
else:
|
|
assert dep_urn in dep_tracker.dependencies[res_urn]
|
|
|
|
def check_opts(i, name, opts):
|
|
res = MockResource(name, opts)
|
|
return pulumi.Output.all(dep.urn, res.urn).apply(lambda urns: check(i, urns))
|
|
|
|
pulumi.Output.all(
|
|
[
|
|
check_opts(i, f"res{i}", opts)
|
|
for i, opts in enumerate(depends_on_variations(dep))
|
|
]
|
|
)
|
|
|
|
|
|
@pulumi.runtime.test
|
|
def test_depends_on_typechecks_sync():
|
|
# https://github.com/pulumi/pulumi/issues/13917
|
|
try:
|
|
res = MockResource(
|
|
name="res", opts=pulumi.ResourceOptions(depends_on=["hello"])
|
|
)
|
|
assert False, "should have failed"
|
|
except TypeError as e:
|
|
assert (
|
|
str(e) == "'depends_on' was passed a value hello that was not a Resource."
|
|
)
|
|
|
|
|
|
def test_depends_on_typechecks_async():
|
|
if not hasattr(asyncio, "to_thread"):
|
|
# Old versions of Python don't have asyncio.to_thread, just skip the test in that case.
|
|
return
|
|
|
|
@pulumi.runtime.test
|
|
def test():
|
|
# https://github.com/pulumi/pulumi/issues/13917
|
|
dep = asyncio.to_thread(lambda: "goodbye")
|
|
res = MockResource(name="res", opts=pulumi.ResourceOptions(depends_on=[dep]))
|
|
|
|
try:
|
|
test()
|
|
assert False, "should have failed"
|
|
except TypeError as e:
|
|
assert (
|
|
str(e) == "'depends_on' was passed a value goodbye that was not a Resource."
|
|
)
|
|
|
|
|
|
@pulumi.runtime.test
|
|
def test_component_resource_propagates_provider() -> None:
|
|
mocks.set_mocks(MinimalMocks())
|
|
|
|
provider = pulumi.ProviderResource("test", "prov", {})
|
|
component = pulumi.ComponentResource(
|
|
"custom:foo:Component",
|
|
"comp",
|
|
opts=pulumi.ResourceOptions(provider=provider),
|
|
)
|
|
custom = pulumi.CustomResource(
|
|
"test:index:Resource",
|
|
"res",
|
|
opts=pulumi.ResourceOptions(parent=component),
|
|
)
|
|
|
|
assert (
|
|
provider == custom._provider
|
|
), "Failed to propagate provider to child resource"
|
|
|
|
|
|
@pulumi.runtime.test
|
|
def test_component_resource_propagates_providers_list() -> None:
|
|
mocks.set_mocks(MinimalMocks())
|
|
|
|
provider = pulumi.ProviderResource("test", "prov", {})
|
|
component = pulumi.ComponentResource(
|
|
"custom:foo:Component",
|
|
"comp",
|
|
opts=pulumi.ResourceOptions(providers=[provider]),
|
|
)
|
|
custom = pulumi.CustomResource(
|
|
"test:index:Resource",
|
|
"res",
|
|
opts=pulumi.ResourceOptions(parent=component),
|
|
)
|
|
|
|
assert (
|
|
provider == custom._provider
|
|
), "Failed to propagate provider to child resource"
|
|
|
|
|
|
def output_depending_on_resource(
|
|
r: pulumi.Resource, isKnown: bool
|
|
) -> pulumi.Output[None]:
|
|
"""Returns an output that depends on the given resource."""
|
|
o = pulumi.Output.from_input(None)
|
|
is_known_fut: asyncio.Future[bool] = asyncio.Future()
|
|
is_known_fut.set_result(isKnown)
|
|
|
|
return pulumi.Output(resources=set([r]), is_known=is_known_fut, future=o.future())
|
|
|
|
|
|
@pytest.fixture
|
|
def dep_tracker():
|
|
for dt in build_dep_tracker():
|
|
yield dt
|
|
|
|
|
|
@pytest.fixture
|
|
def dep_tracker_preview():
|
|
for dt in build_dep_tracker(preview=True):
|
|
yield dt
|
|
|
|
|
|
def build_dep_tracker(preview: bool = False):
|
|
old_settings = settings.SETTINGS
|
|
mm = MinimalMocks()
|
|
mocks.set_mocks(mm, preview=preview)
|
|
dt = DependencyTrackingMonitorWrapper(settings.SETTINGS.monitor)
|
|
settings.SETTINGS.monitor = dt
|
|
|
|
try:
|
|
yield dt
|
|
finally:
|
|
settings.configure(old_settings)
|
|
|
|
|
|
class MinimalMocks(pulumi.runtime.Mocks):
|
|
|
|
def new_resource(self, args: pulumi.runtime.MockResourceArgs):
|
|
return [args.name + "_id", args.inputs]
|
|
|
|
def call(self, args: pulumi.runtime.MockCallArgs):
|
|
return {}
|
|
|
|
|
|
class DependencyTrackingMonitorWrapper:
|
|
|
|
def __init__(self, inner):
|
|
self.inner = inner
|
|
self.dependencies = {}
|
|
|
|
def RegisterResource(self, req: Any):
|
|
resp = self.inner.RegisterResource(req)
|
|
self.dependencies[resp.urn] = self.dependencies.get(resp.urn, set()) | set(
|
|
req.dependencies
|
|
)
|
|
return resp
|
|
|
|
def __getattr__(self, attr):
|
|
return getattr(self.inner, attr)
|
|
|
|
|
|
class MockResource(pulumi.CustomResource):
|
|
def __init__(self, name: str, opts: Optional[pulumi.ResourceOptions] = None):
|
|
super().__init__("python:test:MockResource", name, {}, opts)
|
|
|
|
|
|
class MergeResourceOptions(unittest.TestCase):
|
|
def test_parent(self):
|
|
opts1 = ResourceOptions()
|
|
assert opts1.protect is None
|
|
opts2 = ResourceOptions.merge(opts1, ResourceOptions(protect=True))
|
|
assert opts2.protect is True
|
|
opts3 = ResourceOptions.merge(opts2, ResourceOptions())
|
|
assert opts3.protect is True
|
|
|
|
|
|
# Regression test for https://github.com/pulumi/pulumi/issues/12032
|
|
@pulumi.runtime.test
|
|
def test_parent_and_depends_on_are_the_same_12032():
|
|
os.environ[ERROR_ON_DEPENDENCY_CYCLES_VAR] = "false"
|
|
mocks.set_mocks(MinimalMocks())
|
|
|
|
parent = pulumi.ComponentResource("pkg:index:first", "first")
|
|
child = pulumi.ComponentResource(
|
|
"pkg:index:second",
|
|
"second",
|
|
opts=pulumi.ResourceOptions(parent=parent, depends_on=[parent]),
|
|
)
|
|
|
|
# This would freeze before the fix.
|
|
pulumi.CustomResource(
|
|
"foo:bar:baz",
|
|
"myresource",
|
|
opts=pulumi.ResourceOptions(parent=child),
|
|
)
|
|
|
|
|
|
# Regression test for https://github.com/pulumi/pulumi/issues/12736
|
|
@pulumi.runtime.test
|
|
def test_complex_parent_child_dependencies():
|
|
os.environ[ERROR_ON_DEPENDENCY_CYCLES_VAR] = "false"
|
|
mocks.set_mocks(MinimalMocks())
|
|
|
|
class A(pulumi.ComponentResource):
|
|
def __init__(self, name: str, opts=None):
|
|
super().__init__("my:modules:A", name, {}, opts)
|
|
self.b = B("a-b", opts=ResourceOptions(parent=self))
|
|
self.c = C("a-c", opts=ResourceOptions(parent=self.b, depends_on=[self.b]))
|
|
|
|
class B(pulumi.ComponentResource):
|
|
def __init__(self, name: str, opts=None):
|
|
super().__init__("my:modules:B", name, {}, opts)
|
|
pulumi.CustomResource(
|
|
"my:module:Child", "b-child", opts=ResourceOptions(parent=self)
|
|
)
|
|
|
|
class C(pulumi.ComponentResource):
|
|
def __init__(self, name: str, opts=None):
|
|
super().__init__("my:modules:C", name, {}, opts)
|
|
pulumi.CustomResource(
|
|
"my:module:Child", "c-child", opts=ResourceOptions(parent=self)
|
|
)
|
|
|
|
class D(pulumi.ComponentResource):
|
|
def __init__(self, name: str, opts=None):
|
|
super().__init__("my:modules:D", name, {}, opts)
|
|
pulumi.CustomResource(
|
|
"my:module:Child", "d-child", opts=ResourceOptions(parent=self)
|
|
)
|
|
|
|
a = A("a")
|
|
|
|
D("d", opts=ResourceOptions(parent=a.b, depends_on=[a.b]))
|
|
|
|
|
|
# Regression test for https://github.com/pulumi/pulumi/issues/13997
|
|
def test_bad_component_super_call():
|
|
class C(pulumi.ComponentResource):
|
|
def __init__(self, name: str, arg: int, opts=None):
|
|
super().__init__("my:module:C", name, arg, opts)
|
|
|
|
@pulumi.runtime.test
|
|
def test():
|
|
C("test", 4, None)
|
|
|
|
try:
|
|
test()
|
|
assert False, "should have failed"
|
|
except TypeError as e:
|
|
assert str(e) == "Expected resource properties to be a mapping"
|