pulumi/sdk/python/lib/test/test_invoke.py

282 lines
11 KiB
Python

# Copyright 2016-2023, 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.
import pytest
from typing import Any, Optional, Union
from concurrent.futures import ThreadPoolExecutor
import asyncio
import time
import pulumi
from pulumi import InvokeOptions, InvokeOutputOptions
from pulumi.output import UNKNOWN
from pulumi.resource import DependencyResource, Resource
def unknown() -> pulumi.Output[Any]:
is_known_fut: asyncio.Future[bool] = asyncio.Future()
is_secret_fut: asyncio.Future[bool] = asyncio.Future()
is_known_fut.set_result(False)
is_secret_fut.set_result(False)
value_fut: asyncio.Future[Any] = asyncio.Future()
value_fut.set_result(pulumi.UNKNOWN)
return pulumi.Output(set(), value_fut, is_known_fut, is_secret_fut)
class MyMocks(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 {} if args.args.get("empty") else {"result": "mock"}
class MockResource(pulumi.CustomResource):
def __init__(self, name: str, opts: Optional[pulumi.ResourceOptions] = None):
super().__init__("python:test:MockResource", name, {}, opts)
class MockComponentResource(pulumi.ComponentResource):
def __init__(self, name: str, opts: Optional[pulumi.ResourceOptions] = None):
super().__init__("python:test:MockComponentResource", name, {}, opts)
@pytest.mark.parametrize(
"tok,version,empty,expected",
[
("test:index:MyFunction", "", True, {}),
("test:index:MyFunction", "invalid", True, {}),
("test:index:MyFunction", "1.0.0", True, {}),
("test:index:MyFunction", "", False, {"result": "mock"}),
("test:index:MyFunction", "invalid", False, {"result": "mock"}),
("test:index:MyFunction", "1.0.0", False, {"result": "mock"}),
("kubernetes:something:new", "", True, {}),
("kubernetes:something:new", "invalid", True, {}),
("kubernetes:something:new", "1.0.0", True, {}),
("kubernetes:something:new", "4.5.3", True, {}),
("kubernetes:something:new", "4.5.4", True, {}),
("kubernetes:something:new", "4.5.5", True, {}),
("kubernetes:something:new", "4.6.0", True, {}),
("kubernetes:something:new", "5.0.0", True, {}),
("kubernetes:something:new", "", False, {"result": "mock"}),
("kubernetes:something:new", "invalid", False, {"result": "mock"}),
("kubernetes:something:new", "1.0.0", False, {"result": "mock"}),
# Expect the legacy behavior of getting None for empty results for these Kubernetes
# function tokens when the version is unspecified, invalid, or <= 4.5.4.
("kubernetes:yaml:decode", "", True, None),
("kubernetes:yaml:decode", "invalid", True, None),
("kubernetes:yaml:decode", "1.0.0", True, None),
("kubernetes:yaml:decode", "4.5.3", True, None),
("kubernetes:yaml:decode", "4.5.4", True, None),
("kubernetes:yaml:decode", "4.5.5", True, {}),
("kubernetes:yaml:decode", "4.6.0", True, {}),
("kubernetes:yaml:decode", "5.0.0", True, {}),
("kubernetes:yaml:decode", "", False, {"result": "mock"}),
("kubernetes:yaml:decode", "invalid", False, {"result": "mock"}),
("kubernetes:yaml:decode", "1.0.0", False, {"result": "mock"}),
("kubernetes:helm:template", "", True, None),
("kubernetes:helm:template", "invalid", True, None),
("kubernetes:helm:template", "1.0.0", True, None),
("kubernetes:helm:template", "4.5.3", True, None),
("kubernetes:helm:template", "4.5.4", True, None),
("kubernetes:helm:template", "4.5.5", True, {}),
("kubernetes:helm:template", "4.6.0", True, {}),
("kubernetes:helm:template", "5.0.0", True, {}),
("kubernetes:helm:template", "", False, {"result": "mock"}),
("kubernetes:helm:template", "invalid", False, {"result": "mock"}),
("kubernetes:helm:template", "1.0.0", False, {"result": "mock"}),
("kubernetes:kustomize:directory", "", True, None),
("kubernetes:kustomize:directory", "invalid", True, None),
("kubernetes:kustomize:directory", "1.0.0", True, None),
("kubernetes:kustomize:directory", "4.5.3", True, None),
("kubernetes:kustomize:directory", "4.5.4", True, None),
("kubernetes:kustomize:directory", "4.5.5", True, {}),
("kubernetes:kustomize:directory", "4.6.0", True, {}),
("kubernetes:kustomize:directory", "5.0.0", True, {}),
("kubernetes:kustomize:directory", "", False, {"result": "mock"}),
("kubernetes:kustomize:directory", "invalid", False, {"result": "mock"}),
("kubernetes:kustomize:directory", "1.0.0", False, {"result": "mock"}),
],
)
@pulumi.runtime.test
def test_invoke_empty_return(tok: str, version: str, empty: bool, expected) -> None:
pulumi.runtime.mocks.set_mocks(MyMocks())
props = {"empty": True} if empty else {}
opts = pulumi.InvokeOptions(version=version) if version else None
assert pulumi.runtime.invoke(tok, props, opts).value == expected
@pulumi.runtime.test
def test_invoke_plain_invoke_takes_output_args() -> None:
# Plain invoke calls serialize_properties on the arguments, so even though
# it is supposed to only take plain values, it still happens to work with
# Outputs. In practice, people rely on this behavior, so we test it here to
# ensure we don't break it this behavior.
pulumi.runtime.mocks.set_mocks(MyMocks())
# Construct an argument that depends on an unknown resource.
dep = MockResource(name="dep")
dep.__dict__["id"] = unknown()
arg_value = asyncio.Future[int]()
arg_value.set_result(0)
is_known = asyncio.Future[bool]()
is_known.set_result(True)
is_secret = asyncio.Future[bool]()
is_secret.set_result(False)
arg = pulumi.Output(
resources={dep},
future=arg_value,
is_known=is_known,
is_secret=is_secret,
)
result = pulumi.runtime.invoke("test::MyFunction", {"arg": arg})
assert result.value == {"result": "mock"}
@pulumi.runtime.test
async def test_invoke_depends_on() -> None:
pulumi.runtime.mocks.set_mocks(MyMocks())
dep: pulumi.Resource = MockResource(name="dep1")
dep_future = asyncio.Future()
dep_output = pulumi.Output.from_input(dep_future)
# This function will sleep for 1 second and then resolve dep_future with the
# resource, which in turn resolves the output.
def resolve():
time.sleep(1)
dep_future.set_result(dep)
# Run the resolve function in a separate thread.
with ThreadPoolExecutor(max_workers=2) as executor:
f = executor.submit(resolve)
assert not dep_future.done(), "dep_future should not be done yet"
opts = pulumi.InvokeOutputOptions(depends_on=[dep_output])
o = pulumi.runtime.invoke_output("test::MyFunction", {}, opts)
def check(invoke_result):
assert (
dep_future.done()
), "invoke_output should wait for dep_future to be done"
assert invoke_result == {"result": "mock"}
return o.apply(check)
@pulumi.runtime.test
async def test_invoke_depends_on_component() -> None:
pulumi.runtime.mocks.set_mocks(MyMocks())
dep = MockComponentResource(name="c")
MockResource(name="r", opts=pulumi.ResourceOptions(parent=dep))
dep_remote = DependencyResource("some:urn")
opts = pulumi.InvokeOutputOptions(depends_on=[dep_remote, dep])
o = pulumi.runtime.invoke_output("test::MyFunction", {}, opts)
v = await o.future()
assert v == {"result": "mock"}
deps = await o.resources()
assert len(deps) == 2
assert deps == {dep, dep_remote}
@pulumi.runtime.test
async def test_invoke_depends_on_unknown() -> None:
pulumi.runtime.mocks.set_mocks(MyMocks())
dep = MockResource(name="dep")
dep.__dict__["id"] = unknown()
opts = pulumi.InvokeOutputOptions(depends_on=[dep])
o = pulumi.runtime.invoke_output("test::MyFunction", {}, opts)
k = await o.is_known()
assert k == False
@pulumi.runtime.test
async def test_invoke_depends_on_unknown_component_child() -> None:
pulumi.runtime.mocks.set_mocks(MyMocks())
comp = MockComponentResource(name="c")
dep = MockResource(name="dep", opts=pulumi.ResourceOptions(parent=comp))
dep.__dict__["id"] = unknown()
opts = pulumi.InvokeOutputOptions(depends_on=[comp])
o = pulumi.runtime.invoke_output("test::MyFunction", {}, opts)
k = await o.is_known()
assert k == False
@pulumi.runtime.test
async def test_invoke_input_dependency() -> None:
pulumi.runtime.mocks.set_mocks(MyMocks())
# Create a resource with an unknown ID.
dep = MockResource(name="dep")
dep.__dict__["id"] = unknown()
# Create an output that depends on the resource.
arg_value = asyncio.Future[int]()
arg_value.set_result(0)
is_known = asyncio.Future[bool]()
is_known.set_result(True)
is_secret = asyncio.Future[bool]()
is_secret.set_result(False)
arg_with_resource_dependency = pulumi.Output(
resources={dep},
future=arg_value,
is_known=is_known,
is_secret=is_secret,
)
o = pulumi.runtime.invoke_output(
"test::MyFunction", {"arg": arg_with_resource_dependency}
)
k = await o.is_known()
assert k == False
@pytest.mark.parametrize(
"a,b,expected",
[
(InvokeOptions(version="1.0.0"), InvokeOptions(version="2.0.0"), "2.0.0"),
(None, InvokeOptions(version="2.0.0"), "2.0.0"),
(InvokeOptions(version="1.0.0"), None, "1.0.0"),
(InvokeOptions(version="1.0.0"), InvokeOptions(version=""), ""),
(InvokeOutputOptions(version="1.0.0"), InvokeOptions(version="2.0.0"), "2.0.0"),
(
InvokeOutputOptions(version="1.0.0"),
InvokeOutputOptions(version="2.0.0"),
"2.0.0",
),
(None, InvokeOutputOptions(version="2.0.0"), "2.0.0"),
(InvokeOutputOptions(version="1.0.0"), None, "1.0.0"),
],
)
def test_invoke_merge(
a: Union[InvokeOptions, InvokeOutputOptions],
b: Union[InvokeOptions, InvokeOutputOptions],
expected: str,
) -> None:
if a is not None:
assert a.merge(b).version == expected
if isinstance(a, InvokeOutputOptions):
assert InvokeOutputOptions.merge(a, b).version == expected
else:
assert InvokeOptions.merge(a, b).version == expected