-
Notifications
You must be signed in to change notification settings - Fork 1.2k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
test: Integration tests for remote invoke on regular lambda functions (…
…#5382) * Created base integ glass for remote invoke tests * Add integration tests for invoking lambda functions * make black * Moved tearDownClass to base class * Removed tearDown class from inherited classes and updated lambda fn timeout * Remove the check to skip appveyor tests on master branch
- Loading branch information
Showing
15 changed files
with
508 additions
and
0 deletions.
There are no files selected for viewing
Empty file.
Empty file.
104 changes: 104 additions & 0 deletions
104
tests/integration/remote/invoke/remote_invoke_integ_base.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,104 @@ | ||
from unittest import TestCase, skipIf | ||
from pathlib import Path | ||
from typing import Optional | ||
|
||
from tests.testing_utils import ( | ||
get_sam_command, | ||
run_command, | ||
) | ||
from tests.integration.deploy.deploy_integ_base import DeployIntegBase | ||
|
||
from samcli.lib.utils.boto_utils import get_boto_resource_provider_with_config, get_boto_client_provider_with_config | ||
from samcli.lib.utils.cloudformation import get_resource_summaries | ||
|
||
|
||
class RemoteInvokeIntegBase(TestCase): | ||
template: Optional[Path] = None | ||
|
||
@classmethod | ||
def setUpClass(cls): | ||
cls.cmd = get_sam_command() | ||
cls.test_data_path = cls.get_integ_dir().joinpath("testdata") | ||
if cls.template: | ||
cls.template_path = str(cls.test_data_path.joinpath("remote_invoke", cls.template)) | ||
cls.events_folder_path = cls.test_data_path.joinpath("remote_invoke", "events") | ||
|
||
@classmethod | ||
def tearDownClass(cls): | ||
# Delete the deployed stack | ||
cls.cfn_client.delete_stack(StackName=cls.stack_name) | ||
|
||
@staticmethod | ||
def get_integ_dir(): | ||
return Path(__file__).resolve().parents[2] | ||
|
||
@staticmethod | ||
def remote_invoke_deploy_stack(stack_name, template_path): | ||
|
||
deploy_cmd = DeployIntegBase.get_deploy_command_list( | ||
stack_name=stack_name, | ||
template_file=template_path, | ||
resolve_s3=True, | ||
capabilities_list=["CAPABILITY_IAM", "CAPABILITY_AUTO_EXPAND"], | ||
) | ||
|
||
run_command(deploy_cmd) | ||
|
||
@classmethod | ||
def create_resources_and_boto_clients(cls): | ||
cls.remote_invoke_deploy_stack(cls.stack_name, cls.template_path) | ||
stack_resource_summaries = get_resource_summaries( | ||
get_boto_resource_provider_with_config(), | ||
get_boto_client_provider_with_config(), | ||
cls.stack_name, | ||
) | ||
cls.stack_resources = { | ||
resource_full_path: stack_resource_summary.physical_resource_id | ||
for resource_full_path, stack_resource_summary in stack_resource_summaries.items() | ||
} | ||
cls.cfn_client = get_boto_client_provider_with_config()("cloudformation") | ||
cls.lambda_client = get_boto_client_provider_with_config()("lambda") | ||
|
||
@staticmethod | ||
def get_command_list( | ||
stack_name=None, | ||
resource_id=None, | ||
event=None, | ||
event_file=None, | ||
parameter_list=None, | ||
output=None, | ||
region=None, | ||
profile=None, | ||
beta_features=None, | ||
): | ||
command_list = [get_sam_command(), "remote", "invoke"] | ||
|
||
if stack_name: | ||
command_list = command_list + ["--stack-name", stack_name] | ||
|
||
if event: | ||
command_list = command_list + ["-e", event] | ||
|
||
if event_file: | ||
command_list = command_list + ["--event-file", event_file] | ||
|
||
if profile: | ||
command_list = command_list + ["--parameter", parameter] | ||
|
||
if output: | ||
command_list = command_list + ["--output", output] | ||
|
||
if parameter_list: | ||
for (parameter, value) in parameter_list: | ||
command_list = command_list + ["--parameter", f"{parameter}={value}"] | ||
|
||
if region: | ||
command_list = command_list + ["--region", region] | ||
|
||
if beta_features is not None: | ||
command_list = command_list + ["--beta-features" if beta_features else "--no-beta-features"] | ||
|
||
if resource_id: | ||
command_list = command_list + [resource_id] | ||
|
||
return command_list |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,279 @@ | ||
import json | ||
import uuid | ||
import base64 | ||
|
||
from parameterized import parameterized | ||
|
||
from tests.integration.remote.invoke.remote_invoke_integ_base import RemoteInvokeIntegBase | ||
from tests.testing_utils import run_command | ||
|
||
from pathlib import Path | ||
import pytest | ||
|
||
|
||
@pytest.mark.xdist_group(name="sam_remote_invoke_single_lambda_resource") | ||
class TestSingleResourceInvoke(RemoteInvokeIntegBase): | ||
template = Path("template-single-lambda.yaml") | ||
|
||
@classmethod | ||
def setUpClass(cls): | ||
super().setUpClass() | ||
cls.stack_name = f"{TestSingleResourceInvoke.__name__}-{uuid.uuid4().hex}" | ||
cls.create_resources_and_boto_clients() | ||
|
||
def test_invoke_empty_event_provided(self): | ||
command_list = self.get_command_list(stack_name=self.stack_name) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout["errorType"], "KeyError") | ||
|
||
def test_invoke_with_only_event_provided(self): | ||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, {"message": "Hello world"}) | ||
|
||
def test_invoke_with_only_event_file_provided(self): | ||
event_file_path = str(self.events_folder_path.joinpath("default_event.json")) | ||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, resource_id="HelloWorldFunction", event_file=event_file_path | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, {"message": "Hello world"}) | ||
|
||
def test_invoke_with_resource_id_provided_as_arn(self): | ||
resource_id = "HelloWorldFunction" | ||
lambda_name = self.stack_resources[resource_id] | ||
lambda_arn = self.lambda_client.get_function(FunctionName=lambda_name)["Configuration"]["FunctionArn"] | ||
|
||
command_list = self.get_command_list( | ||
resource_id=lambda_arn, | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, {"message": "Hello world"}) | ||
|
||
def test_invoke_asynchronous_using_boto_parameter(self): | ||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
parameter_list=[("InvocationType", "Event"), ("LogType", "None")], | ||
output="json", | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout["Payload"], "") | ||
self.assertEqual(remote_invoke_result_stdout["StatusCode"], 202) | ||
|
||
def test_invoke_dryrun_using_boto_parameter(self): | ||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
parameter_list=[("InvocationType", "DryRun"), ("Qualifier", "$LATEST")], | ||
output="json", | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout["Payload"], "") | ||
self.assertEqual(remote_invoke_result_stdout["StatusCode"], 204) | ||
|
||
def test_invoke_response_json_output_format(self): | ||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
output="json", | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
|
||
response_payload = json.loads(remote_invoke_result_stdout["Payload"]) | ||
self.assertEqual(response_payload, {"message": "Hello world"}) | ||
self.assertEqual(remote_invoke_result_stdout["StatusCode"], 200) | ||
|
||
|
||
@pytest.mark.xdist_group(name="sam_remote_invoke_multiple_resources") | ||
class TestMultipleResourcesInvoke(RemoteInvokeIntegBase): | ||
template = Path("template-multiple-resources.yaml") | ||
|
||
@classmethod | ||
def tearDownClass(cls): | ||
# Delete the deployed stack | ||
cls.cfn_client.delete_stack(StackName=cls.stack_name) | ||
|
||
@classmethod | ||
def setUpClass(cls): | ||
super().setUpClass() | ||
cls.stack_name = f"{TestMultipleResourcesInvoke.__name__}-{uuid.uuid4().hex}" | ||
cls.create_resources_and_boto_clients() | ||
|
||
def test_invoke_empty_event_provided(self): | ||
command_list = self.get_command_list(stack_name=self.stack_name, resource_id="EchoEventFunction") | ||
|
||
remote_invoke_result = run_command(command_list) | ||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, {}) | ||
|
||
@parameterized.expand( | ||
[ | ||
("HelloWorldServerlessFunction", {"message": "Hello world"}), | ||
("EchoCustomEnvVarFunction", "MyOtherVar"), | ||
("EchoEventFunction", {"key1": "Hello", "key2": "serverless", "key3": "world"}), | ||
] | ||
) | ||
def test_invoke_with_only_event_provided(self, resource_id, expected_response): | ||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, | ||
resource_id=resource_id, | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, expected_response) | ||
|
||
@parameterized.expand( | ||
[ | ||
("HelloWorldServerlessFunction", {"message": "Hello world"}), | ||
("EchoCustomEnvVarFunction", "MyOtherVar"), | ||
("EchoEventFunction", {"key1": "Hello", "key2": "serverless", "key3": "world"}), | ||
] | ||
) | ||
def test_invoke_with_resource_id_provided_as_arn(self, resource_id, expected_response): | ||
lambda_name = self.stack_resources[resource_id] | ||
lambda_arn = self.lambda_client.get_function(FunctionName=lambda_name)["Configuration"]["FunctionArn"] | ||
|
||
command_list = self.get_command_list( | ||
resource_id=lambda_arn, | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, expected_response) | ||
|
||
def test_lambda_writes_to_stderr_invoke(self): | ||
command_list = RemoteInvokeIntegBase.get_command_list( | ||
stack_name=self.stack_name, | ||
resource_id="WriteToStderrFunction", | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = remote_invoke_result.stdout.strip().decode() | ||
remote_invoke_result_stderr = remote_invoke_result.stderr.strip().decode() | ||
self.assertIn("Lambda Function is writing to stderr", remote_invoke_result_stderr) | ||
self.assertEqual('"wrote to stderr"', remote_invoke_result_stdout) | ||
|
||
def test_lambda_raises_exception_invoke(self): | ||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, | ||
resource_id="RaiseExceptionFunction", | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stderr = remote_invoke_result.stderr.strip().decode() | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
|
||
self.assertIn("Lambda is raising an exception", remote_invoke_result_stderr) | ||
self.assertEqual("Lambda is raising an exception", remote_invoke_result_stdout["errorMessage"]) | ||
|
||
def test_lambda_invoke_client_context_boto_parameter(self): | ||
custom_json_str = {"custom": {"foo": "bar", "baz": "quzz"}} | ||
client_context_base64_str = base64.b64encode(json.dumps(custom_json_str).encode()).decode("utf-8") | ||
|
||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, | ||
resource_id="EchoClientContextData", | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
parameter_list=[("ClientContext", client_context_base64_str)], | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, custom_json_str["custom"]) | ||
|
||
|
||
@pytest.mark.xdist_group(name="sam_remote_invoke_nested_resources") | ||
class TestNestedTemplateResourcesInvoke(RemoteInvokeIntegBase): | ||
template = Path("nested_templates/template.yaml") | ||
|
||
@classmethod | ||
def tearDownClass(cls): | ||
# Delete the deployed stack | ||
cls.cfn_client.delete_stack(StackName=cls.stack_name) | ||
|
||
@classmethod | ||
def setUpClass(cls): | ||
super().setUpClass() | ||
cls.stack_name = f"{TestNestedTemplateResourcesInvoke.__name__}-{uuid.uuid4().hex}" | ||
cls.create_resources_and_boto_clients() | ||
|
||
def test_invoke_empty_event_provided(self): | ||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, {"message": "Hello world"}) | ||
|
||
def test_invoke_with_only_event_provided(self): | ||
command_list = self.get_command_list( | ||
stack_name=self.stack_name, | ||
resource_id="ChildStack/HelloWorldFunction", | ||
event='{"key1": "Hello", "key2": "serverless", "key3": "world"}', | ||
) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, {"message": "Hello world"}) | ||
|
||
def test_invoke_default_lambda_function(self): | ||
event_file_path = str(self.events_folder_path.joinpath("default_event.json")) | ||
command_list = self.get_command_list(stack_name=self.stack_name, event_file=event_file_path) | ||
|
||
remote_invoke_result = run_command(command_list) | ||
|
||
self.assertEqual(0, remote_invoke_result.process.returncode) | ||
remote_invoke_result_stdout = json.loads(remote_invoke_result.stdout.strip().decode()) | ||
self.assertEqual(remote_invoke_result_stdout, {"message": "Hello world"}) |
Empty file.
5 changes: 5 additions & 0 deletions
5
tests/integration/testdata/remote_invoke/events/default_event.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
{ | ||
"key1": "Hello", | ||
"key2": "serverless", | ||
"key3": "world" | ||
} |
Empty file.
Oops, something went wrong.