From 271da22d5c41dd7957fe847a68331165686cca84 Mon Sep 17 00:00:00 2001 From: Rajesh Duraisamy Date: Tue, 21 May 2024 11:20:07 -0700 Subject: [PATCH 1/3] Implement canary file generation functionality from contract test inputs files --- src/rpdk/core/generate.py | 2 +- src/rpdk/core/generate_stack_templates.py | 233 ++++++++++++++++++++++ src/rpdk/core/project.py | 57 +++++- tests/test_generate_stack_templates.py | 231 +++++++++++++++++++++ tests/test_project.py | 194 ++++++++++++++++++ 5 files changed, 715 insertions(+), 2 deletions(-) create mode 100644 src/rpdk/core/generate_stack_templates.py create mode 100644 tests/test_generate_stack_templates.py diff --git a/src/rpdk/core/generate.py b/src/rpdk/core/generate.py index 58383055..3f99eb78 100644 --- a/src/rpdk/core/generate.py +++ b/src/rpdk/core/generate.py @@ -20,7 +20,7 @@ def generate(args): args.profile, ) project.generate_docs() - + project.generate_canary_files() LOG.warning("Generated files for %s", project.type_name) diff --git a/src/rpdk/core/generate_stack_templates.py b/src/rpdk/core/generate_stack_templates.py new file mode 100644 index 00000000..4f412e8c --- /dev/null +++ b/src/rpdk/core/generate_stack_templates.py @@ -0,0 +1,233 @@ +import json +import re +import shutil +from pathlib import Path +from typing import Any, Dict +from uuid import uuid4 + +import yaml + +CONTRACT_TEST_FOLDER = "contract-tests-artifacts" +CONTRACT_TEST_INPUT_PREFIX = "inputs_*" +CONTRACT_TEST_DEPENDENCY_FILE_NAME = "dependencies.yml" +FILE_GENERATION_ENABLED = "file_generation_enabled" +TYPE_NAME = "typeName" +CONTRACT_TEST_FILE_NAMES = "contract_test_file_names" +INPUT1_FILE_NAME = "inputs_1.json" +FN_SUB = "Fn::Sub" +FN_IMPORT_VALUE = "Fn::ImportValue" +UUID = "uuid" +DYNAMIC_VALUES_MAP = { + "region": "${AWS::Region}", + "partition": "${AWS::Partition}", + "account": "${AWS::AccountId}", +} + + +class StackTemplateGenerator: + def __init__( + self, + type_name: str, + stack_template_config: dict, + contract_test_file_names: list, + root=None, + ): + self.root = Path(root) if root else Path.cwd() + self.stack_template_config = stack_template_config + self.type_name = type_name + self.contract_test_file_names = contract_test_file_names + + @property + def file_generation_enabled(self): + return self.stack_template_config["file_generation_enabled"] + + @property + def file_prefix(self): + return self.stack_template_config["file_prefix"] + + @property + def stack_template_root_folder_path(self): + return self.stack_template_config["root_folder_path"] + + @property + def target_stack_template_folder_path(self): + return self.stack_template_config["target_folder_path"] + + @property + def stack_template_dependency_file_name(self): + return self.stack_template_config["dependency_file_name"] + + def generate_stack_templates(self) -> None: + """ + Generate stack_template files based on the contract test input files. + + This method checks if file generation is enabled and if the target contract test folder exists. + If both conditions are met, it creates the stack_template folder, copies the contract test dependencies, + and generates stack_template files for each contract test input file up to the specified count. + """ + if not self.file_generation_enabled or not self.contract_test_folder_exists(): + return + self._setup_stack_template_environment() + self._generate_stack_template_files() + + def contract_test_folder_exists(self) -> bool: + return Path(self.target_contract_test_folder_path).exists() + + def _setup_stack_template_environment(self) -> None: + stack_template_root = Path(self.stack_template_root_folder_path) + stack_template_folder = Path(self.target_stack_template_folder_path) + self.clean_and_create_stack_template_folder( + stack_template_root, stack_template_folder + ) + self.create_stack_template_bootstrap( + Path(self.target_contract_test_folder_path), stack_template_root + ) + + def _generate_stack_template_files(self) -> None: + resource_name = self.type_name + stack_template_folder = Path(self.target_stack_template_folder_path) + contract_test_files = self._get_sorted_contract_test_files() + for count, ct_file in enumerate(contract_test_files, start=1): + self.create_stack_template_file( + resource_name, ct_file, stack_template_folder, self.file_prefix, count + ) + + def _get_sorted_contract_test_files(self) -> list: + contract_test_folder = Path(self.target_contract_test_folder_path) + contract_test_files = [ + file + for file in contract_test_folder.glob(CONTRACT_TEST_INPUT_PREFIX) + if file.is_file() and file.name in self.contract_test_file_names + ] + return sorted(contract_test_files) + + def clean_and_create_stack_template_folder( + self, stack_template: Path, stack_template_folder: Path + ) -> None: + """ + Clean and create the stack_template folder. + + This method removes the existing stack_template root folder and creates a new stack_template folder. + + Args: + stack_template (Path): The path to the stack_template root folder. + stack_template_folder (Path): The path to the stack_template folder. + """ + stack_template_folder.mkdir(parents=True, exist_ok=True) + + def create_stack_template_bootstrap( + self, file_location: Path, stack_template: Path + ) -> None: + """ + Copy the contract test dependencies to the stack_template root folder. + + This method copies the contract test dependency file to the stack_template root folder + as the stack_template bootstrap file. + + Args: + file_location (Path): The path to the contract test folder. + stack_template (Path): The path to the stack_template root folder. + """ + dependencies_file = file_location / CONTRACT_TEST_DEPENDENCY_FILE_NAME + bootstrap_file = stack_template / self.stack_template_dependency_file_name + if dependencies_file.exists(): + shutil.copy(str(dependencies_file), str(bootstrap_file)) + + def create_stack_template_file( + self, + resource_type: str, + ct_file: Path, + stack_template_folder: Path, + stack_template_file_name_prefix: str, + count: int, + ) -> None: + """ + Create a stack_template file based on the contract test input file. + + This method generates a stack_template file in YAML format based on the provided contract test input file. + The stack_template file contains the resource configuration with dynamic values replaced. + + Args: + resource_type (str): The type of the resource being tested. + ct_file (Path): The path to the contract test input file. + stack_template_folder (Path): The path to the stack_template folder. + stack_template_file_name_prefix (str): The prefix for the stack_template file name. + count (int): The count of the stack_template file being generated. + """ + with ct_file.open("r") as f: + json_data = json.load(f) + resource_name = resource_type.split("::")[2] + stack_template_data = { + "Description": f"Template for {resource_type}", + "Resources": { + f"{resource_name}": { + "Type": resource_type, + "Properties": self.replace_dynamic_values( + json_data["CreateInputs"] + ), + } + }, + } + stack_template_file_name = f"{stack_template_file_name_prefix}{count}_001.yaml" + stack_template_file_path = stack_template_folder / stack_template_file_name + + with stack_template_file_path.open("w") as stack_template_file: + yaml.dump(stack_template_data, stack_template_file, indent=2) + + def replace_dynamic_values(self, properties: Dict[str, Any]) -> Dict[str, Any]: + """ + Replace dynamic values in the resource properties. + + This method recursively replaces dynamic values in the resource properties dictionary. + It handles nested dictionaries, lists, and strings with dynamic value placeholders. + + Args: + properties (Dict[str, Any]): The resource properties dictionary. + + Returns: + Dict[str, Any]: The resource properties dictionary with dynamic values replaced. + """ + for key, value in properties.items(): + if isinstance(value, dict): + properties[key] = self.replace_dynamic_values(value) + elif isinstance(value, list): + properties[key] = [self._replace_dynamic_value(item) for item in value] + else: + return_value = self._replace_dynamic_value(value) + properties[key] = return_value + return properties + + def _replace_dynamic_value(self, original_value: Any) -> Any: + """ + Replace a dynamic value with its corresponding value. + + This method replaces dynamic value placeholders in a string with their corresponding values. + It handles UUID generation, partition replacement, and Fn::ImportValue function. + + Args: + original_value (Any): The value to be replaced. + + Returns: + Any: The replaced value. + """ + pattern = r"\{\{(.*?)\}\}" + + def replace_token(match): + token = match.group(1) + if UUID in token: + return str(uuid4()) + if token in DYNAMIC_VALUES_MAP: + return DYNAMIC_VALUES_MAP[token] + return f'{{"{FN_IMPORT_VALUE}": "{token.strip()}"}}' + + replaced_value = re.sub(pattern, replace_token, str(original_value)) + + if any(value in replaced_value for value in DYNAMIC_VALUES_MAP.values()): + replaced_value = {FN_SUB: replaced_value} + if FN_IMPORT_VALUE in replaced_value: + replaced_value = json.loads(replaced_value) + return replaced_value + + @property + def target_contract_test_folder_path(self): + return self.root / CONTRACT_TEST_FOLDER diff --git a/src/rpdk/core/project.py b/src/rpdk/core/project.py index 1649bdd8..14bfade0 100644 --- a/src/rpdk/core/project.py +++ b/src/rpdk/core/project.py @@ -29,6 +29,7 @@ SpecValidationError, ) from .fragment.module_fragment_reader import _get_fragment_file +from .generate_stack_templates import StackTemplateGenerator from .jsonutils.pointer import fragment_decode, fragment_encode from .jsonutils.utils import traverse from .plugin_registry import load_plugin @@ -56,7 +57,17 @@ ARTIFACT_TYPE_RESOURCE = "RESOURCE" ARTIFACT_TYPE_MODULE = "MODULE" ARTIFACT_TYPE_HOOK = "HOOK" - +TARGET_CANARY_ROOT_FOLDER = "canary-bundle" +TARGET_CANARY_FOLDER = "canary-bundle/canary" +RPDK_CONFIG_FILE = ".rpdk-config" +CANARY_FILE_PREFIX = "canary" +CONTRACT_TEST_DEPENDENCY_FILE_NAME = "dependencies.yml" +CANARY_DEPENDENCY_FILE_NAME = "bootstrap.yaml" +CANARY_SETTINGS = "canarySettings" +TYPE_NAME = "typeName" +CONTRACT_TEST_FILE_NAMES = "contract_test_file_names" +INPUT1_FILE_NAME = "inputs_1.json" +FILE_GENERATION_ENABLED = "file_generation_enabled" DEFAULT_ROLE_TIMEOUT_MINUTES = 120 # 2 hours # min and max are according to CreateRole API restrictions # https://docs.aws.amazon.com/IAM/latest/APIReference/API_CreateRole.html @@ -145,6 +156,7 @@ def __init__(self, overwrite_enabled=False, root=None): self.test_entrypoint = None self.executable_entrypoint = None self.fragment_dir = None + self.canary_settings = {} self.target_info = {} self.env = Environment( @@ -207,6 +219,18 @@ def target_schemas_path(self): def target_info_path(self): return self.root / TARGET_INFO_FILENAME + @property + def target_canary_root_path(self): + return self.root / TARGET_CANARY_ROOT_FOLDER + + @property + def target_canary_folder_path(self): + return self.root / TARGET_CANARY_FOLDER + + @property + def rpdk_config(self): + return self.root / RPDK_CONFIG_FILE + @staticmethod def _raise_invalid_project(msg, e): LOG.debug(msg, exc_info=e) @@ -277,6 +301,7 @@ def validate_and_load_resource_settings(self, raw_settings): self.executable_entrypoint = raw_settings.get("executableEntrypoint") self._plugin = load_plugin(raw_settings["language"]) self.settings = raw_settings.get("settings", {}) + self.canary_settings = raw_settings.get("canarySettings", {}) def _write_example_schema(self): self.schema = resource_json( @@ -338,6 +363,7 @@ def _write_resource_settings(f): "testEntrypoint": self.test_entrypoint, "settings": self.settings, **executable_entrypoint_dict, + "canarySettings": self.canary_settings, }, f, indent=4, @@ -391,6 +417,10 @@ def init(self, type_name, language, settings=None): self.language = language self._plugin = load_plugin(language) self.settings = settings or {} + self.canary_settings = { + FILE_GENERATION_ENABLED: True, + CONTRACT_TEST_FILE_NAMES: [INPUT1_FILE_NAME], + } self._write_example_schema() self._write_example_inputs() self._plugin.init(self) @@ -1251,3 +1281,28 @@ def _load_target_info( ) return type_info + + def generate_canary_files(self) -> None: + """ + Generate canary files based on the contract test input files. + + This method checks if file generation is enabled and if the target contract test folder exists. + If both conditions are met, it creates the canary folder, copies the contract test dependencies, + and generates canary files for each contract test input file up to the specified count. + """ + stack_template_config = { + "root_folder_path": self.target_canary_root_path, + "target_folder_path": self.target_canary_folder_path, + "dependency_file_name": CANARY_DEPENDENCY_FILE_NAME, + "file_prefix": CANARY_FILE_PREFIX, + "file_generation_enabled": self.canary_settings.get( + FILE_GENERATION_ENABLED, False + ), + } + generate_stack_templates = StackTemplateGenerator( + self.type_name, + stack_template_config, + self.canary_settings.get(CONTRACT_TEST_FILE_NAMES, [INPUT1_FILE_NAME]), + self.root, + ) + generate_stack_templates.generate_stack_templates() diff --git a/tests/test_generate_stack_templates.py b/tests/test_generate_stack_templates.py new file mode 100644 index 00000000..0795f906 --- /dev/null +++ b/tests/test_generate_stack_templates.py @@ -0,0 +1,231 @@ +import json +import os +import re +import shutil +import uuid +from pathlib import Path +from unittest.mock import ANY, patch + +import pytest + +from rpdk.core.generate_stack_templates import ( + CONTRACT_TEST_DEPENDENCY_FILE_NAME, + CONTRACT_TEST_FOLDER, + StackTemplateGenerator, +) + + +@pytest.fixture(name="stack_template_generator") +def setup_fixture(tmp_path): + root_path = tmp_path + type_name = "AWS::Example::Resource" + stack_template_config = { + "root_folder_path": root_path / "stack-templates", + "target_folder_path": root_path / "stack-templates" / "target", + "dependency_file_name": CONTRACT_TEST_DEPENDENCY_FILE_NAME, + "file_prefix": "stack-template-", + "file_generation_enabled": True, + } + contract_test_file_names = ["inputs_1.json", "inputs_2.json"] + contract_test_folder = root_path / CONTRACT_TEST_FOLDER + contract_test_folder.mkdir(parents=True, exist_ok=True) + # Create a dummy JSON file in the contract_test_folder directory + create_dummy_json_file(contract_test_folder, "inputs_1.json") + create_dummy_json_file(contract_test_folder, "inputs_2.json") + (contract_test_folder / CONTRACT_TEST_DEPENDENCY_FILE_NAME).touch() + + return StackTemplateGenerator( + type_name, + stack_template_config, + contract_test_file_names, + root_path, + ) + + +def create_dummy_json_file(directory: Path, file_name: str): + """Create a dummy JSON file in the given directory.""" + dummy_json_file = directory / file_name + dummy_data = { + "CreateInputs": { + "Property1": "Value1", + "Property2": "Value1", + } + } + with dummy_json_file.open("w") as f: + json.dump(dummy_data, f) + + +def create_folder(folder: Path): + if os.path.exists(folder): + shutil.rmtree(folder) + folder.mkdir() + + +def test_is_file_generation_enabled(stack_template_generator): + assert stack_template_generator.file_generation_enabled is True + + +def test_contract_test_folder_exists(stack_template_generator): + assert stack_template_generator.contract_test_folder_exists() is True + + +def test_generate_stack_template_files(stack_template_generator, tmp_path): + stack_template_generator.generate_stack_templates() + + stack_template_root_path = stack_template_generator.stack_template_root_folder_path + stack_template_folder_path = ( + stack_template_generator.target_stack_template_folder_path + ) + assert stack_template_root_path.exists() + assert stack_template_folder_path.exists() + + template_files = list( + stack_template_folder_path.glob(f"{stack_template_generator.file_prefix}*") + ) + assert len(template_files) == 2 + template_files.sort() + assert template_files[0].name == f"{stack_template_generator.file_prefix}1_001.yaml" + assert template_files[1].name == f"{stack_template_generator.file_prefix}2_001.yaml" + + bootstrap_file = ( + stack_template_root_path + / stack_template_generator.stack_template_dependency_file_name + ) + assert bootstrap_file.exists() + + +@pytest.mark.usefixtures("stack_template_generator") +def test_clean_and_create_template_folder(stack_template_generator, tmp_path): + template_root_path = stack_template_generator.stack_template_root_folder_path + template_folder_path = stack_template_generator.target_stack_template_folder_path + template_root_path.mkdir() + (template_root_path / "existing_file.txt").touch() + stack_template_generator.clean_and_create_stack_template_folder( + template_root_path, template_folder_path + ) + assert template_root_path.exists() + assert not list(template_folder_path.glob("*")) + assert template_folder_path.exists() + + +def test_create_template_bootstrap(stack_template_generator, tmp_path): + contract_test_folder = tmp_path / CONTRACT_TEST_FOLDER + create_folder(contract_test_folder) + (contract_test_folder / CONTRACT_TEST_DEPENDENCY_FILE_NAME).touch() + + template_root_path = stack_template_generator.stack_template_root_folder_path + create_folder(template_root_path) + + stack_template_generator.create_stack_template_bootstrap( + contract_test_folder, template_root_path + ) + + bootstrap_file = ( + template_root_path + / stack_template_generator.stack_template_dependency_file_name + ) + assert bootstrap_file.exists() + + +@patch("rpdk.core.generate_stack_templates.yaml.dump") +def test_create_template_file(mock_yaml_dump, stack_template_generator, tmp_path): + contract_test_folder = tmp_path / CONTRACT_TEST_FOLDER + create_folder(contract_test_folder) + contract_test_file = contract_test_folder / "inputs_1.json" + contract_test_data = { + "CreateInputs": { + "Property1": "Value1", + "Property2": "{{test123}}", + "Property3": {"Nested": "{{partition}}"}, + "Property4": ["{{region}}", "Value2"], + "Property5": "{{uuid}}", + "Property6": "{{account}}", + } + } + with contract_test_file.open("w") as f: + json.dump(contract_test_data, f) + + template_folder_path = stack_template_generator.target_stack_template_folder_path + template_folder_path.mkdir(parents=True, exist_ok=True) + + stack_template_generator.create_stack_template_file( + "AWS::Example::Resource", + contract_test_file, + template_folder_path, + stack_template_generator.file_prefix, + 1, + ) + + expected_template_data = { + "Description": "Template for AWS::Example::Resource", + "Resources": { + "Resource": { + "Type": "AWS::Example::Resource", + "Properties": { + "Property1": "Value1", + "Property2": {"Fn::ImportValue": ANY}, + "Property3": {"Nested": {"Fn::Sub": "${AWS::Partition}"}}, + "Property4": [{"Fn::Sub": "${AWS::Region}"}, "Value2"], + "Property5": ANY, + "Property6": {"Fn::Sub": "${AWS::AccountId}"}, + }, + } + }, + } + args, kwargs = mock_yaml_dump.call_args + assert args[0] == expected_template_data + assert kwargs + # Assert UUID generation + replaced_properties = args[0]["Resources"]["Resource"]["Properties"] + assert isinstance(replaced_properties["Property5"], str) + assert len(replaced_properties["Property5"]) == 36 # Standard UUID length + assert re.match( + r"[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}", + replaced_properties["Property5"], + ) + + # Assert the generated UUID is a valid UUID + generated_uuid = replaced_properties["Property5"] + assert uuid.UUID(generated_uuid) + + +def test_replace_dynamic_values(stack_template_generator): + properties = { + "Property1": "Value1", + "Property2": "{{uuid}}", + "Property3": {"Nested": "{{partition}}"}, + "Property4": ["{{region}}", "Value2"], + "Property5": [{"Key": "{{uuid}}"}], + "Property6": "{{account}}", + "Property7": "prefix-{{uuid}}-sufix", + "Property8": "{{value8}}", + } + replaced_properties = stack_template_generator.replace_dynamic_values(properties) + assert replaced_properties["Property1"] == "Value1" + assert isinstance(replaced_properties["Property2"], str) + assert len(replaced_properties["Property2"]) == 36 + assert re.match( + r"[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}", + replaced_properties["Property2"], + ) + assert replaced_properties["Property3"]["Nested"] == { + "Fn::Sub": "${AWS::Partition}" + } + assert replaced_properties["Property4"][0] == {"Fn::Sub": "${AWS::Region}"} + assert replaced_properties["Property4"][1] == "Value2" + assert replaced_properties["Property6"] == {"Fn::Sub": "${AWS::AccountId}"} + property7_value = replaced_properties["Property7"] + # Assert the replaced value + assert isinstance(property7_value, str) + assert "prefix-" in property7_value + assert "-sufix" in property7_value + # Extract the UUID part + property7_value = property7_value.replace("prefix-", "").replace("-sufix", "") + # Assert the UUID format + assert len(property7_value) == 36 # Standard UUID length + assert re.match( + r"[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}", property7_value + ) + # Assert the UUID is a valid UUID + assert uuid.UUID(property7_value) + assert replaced_properties["Property8"] == {"Fn::ImportValue": "value8"} diff --git a/tests/test_project.py b/tests/test_project.py index dbbb5eda..26e9fdb0 100644 --- a/tests/test_project.py +++ b/tests/test_project.py @@ -5,6 +5,7 @@ import logging import os import random +import shutil import string import sys import zipfile @@ -26,13 +27,21 @@ InvalidProjectError, SpecValidationError, ) +from rpdk.core.generate_stack_templates import CONTRACT_TEST_FOLDER from rpdk.core.plugin_base import LanguagePlugin from rpdk.core.project import ( + CANARY_DEPENDENCY_FILE_NAME, + CANARY_FILE_PREFIX, CFN_METADATA_FILENAME, CONFIGURATION_SCHEMA_UPLOAD_FILENAME, + CONTRACT_TEST_DEPENDENCY_FILE_NAME, + CONTRACT_TEST_FILE_NAMES, + FILE_GENERATION_ENABLED, OVERRIDES_FILENAME, SCHEMA_UPLOAD_FILENAME, SETTINGS_FILENAME, + TARGET_CANARY_FOLDER, + TARGET_CANARY_ROOT_FOLDER, TARGET_INFO_FILENAME, Project, escape_markdown, @@ -2733,3 +2742,188 @@ def test__load_target_info_for_hooks_local_only(project): sorted(test_type_info.keys()), local_schemas=ANY, local_info=test_type_info ) assert len(mock_loader.call_args[1]["local_schemas"]) == 4 + + +def setup_contract_test_data(tmp_path): + root_path = tmp_path + contract_test_folder = root_path / CONTRACT_TEST_FOLDER + contract_test_folder.mkdir(parents=True, exist_ok=True) + assert contract_test_folder.exists() + # Create a dummy JSON file in the canary_root_path directory + create_dummy_json_file(contract_test_folder, "inputs_1.json") + create_dummy_json_file(contract_test_folder, "inputs_2.json") + (contract_test_folder / CONTRACT_TEST_DEPENDENCY_FILE_NAME).touch() + assert contract_test_folder.exists() + return Project(str(root_path)) + + +def create_dummy_json_file(directory: Path, file_name: str): + """Create a dummy JSON file in the given directory.""" + dummy_json_file = directory / file_name + dummy_data = { + "CreateInputs": { + "Property1": "Value1", + "Property2": "Value1", + } + } + with dummy_json_file.open("w") as f: + json.dump(dummy_data, f) + + +def create_folder(folder: Path): + if os.path.exists(folder): + shutil.rmtree(folder) + folder.mkdir() + + +def test_generate_canary_files(project): + setup_contract_test_data(project.root) + tmp_path = project.root + plugin = object() + data = json.dumps( + { + "artifact_type": "RESOURCE", + "language": LANGUAGE, + "runtime": RUNTIME, + "entrypoint": None, + "testEntrypoint": None, + "futureProperty": "value", + "typeName": "AWS::Example::Resource", + "canarySettings": { + FILE_GENERATION_ENABLED: True, + CONTRACT_TEST_FILE_NAMES: ["inputs_1.json", "inputs_2.json"], + }, + } + ) + patch_load = patch( + "rpdk.core.project.load_plugin", autospec=True, return_value=plugin + ) + + with patch_settings(project, data) as mock_open, patch_load as mock_load: + project.load_settings() + project.generate_canary_files() + mock_open.assert_called_once_with("r", encoding="utf-8") + mock_load.assert_called_once_with(LANGUAGE) + canary_root_path = tmp_path / TARGET_CANARY_ROOT_FOLDER + canary_folder_path = tmp_path / TARGET_CANARY_FOLDER + assert canary_root_path.exists() + assert canary_folder_path.exists() + + canary_files = list(canary_folder_path.glob(f"{CANARY_FILE_PREFIX}*")) + assert len(canary_files) == 2 + canary_files.sort() + assert canary_files[0].name == f"{CANARY_FILE_PREFIX}1_001.yaml" + assert canary_files[1].name == f"{CANARY_FILE_PREFIX}2_001.yaml" + + bootstrap_file = canary_root_path / CANARY_DEPENDENCY_FILE_NAME + assert bootstrap_file.exists() + + +def setup_rpdk_config(project, rpdk_config): + root_path = project.root + plugin = object() + data = json.dumps(rpdk_config) + patch_load = patch( + "rpdk.core.project.load_plugin", autospec=True, return_value=plugin + ) + + with patch_settings(project, data) as mock_open, patch_load as mock_load: + project.load_settings() + mock_open.assert_called_once_with("r", encoding="utf-8") + mock_load.assert_called_once_with(LANGUAGE) + contract_test_folder = root_path / CONTRACT_TEST_FOLDER + contract_test_folder.mkdir(parents=True, exist_ok=True) + # Create a dummy JSON file in the canary_root_path directory + create_dummy_json_file(contract_test_folder, "inputs_1.json") + create_dummy_json_file(contract_test_folder, "inputs_2.json") + (contract_test_folder / CONTRACT_TEST_DEPENDENCY_FILE_NAME).touch() + + +def test_generate_canary_files_when_not_enabled(project): + rpdk_config = { + ARTIFACT_TYPE_RESOURCE: "RESOURCE", + "language": LANGUAGE, + "runtime": RUNTIME, + "entrypoint": None, + "testEntrypoint": None, + "futureProperty": "value", + "typeName": "AWS::Example::Resource", + "canarySettings": { + FILE_GENERATION_ENABLED: False, + "contract_test_file_names": ["inputs_1.json", "inputs_2.json"], + }, + } + tmp_path = project.root + setup_rpdk_config(project, rpdk_config) + project.generate_canary_files() + + canary_root_path = tmp_path / TARGET_CANARY_ROOT_FOLDER + canary_folder_path = tmp_path / TARGET_CANARY_FOLDER + assert not canary_root_path.exists() + assert not canary_folder_path.exists() + + +def test_generate_canary_files_no_canary_settings(project): + rpdk_config = { + ARTIFACT_TYPE_RESOURCE: "RESOURCE", + "language": LANGUAGE, + "runtime": RUNTIME, + "entrypoint": None, + "testEntrypoint": None, + "futureProperty": "value", + "typeName": "AWS::Example::Resource", + } + tmp_path = project.root + setup_rpdk_config(project, rpdk_config) + project.generate_canary_files() + + canary_root_path = tmp_path / TARGET_CANARY_ROOT_FOLDER + canary_folder_path = tmp_path / TARGET_CANARY_FOLDER + assert not canary_root_path.exists() + assert not canary_folder_path.exists() + + +def test_generate_canary_files_empty_input_files(project): + rpdk_config = { + ARTIFACT_TYPE_RESOURCE: "RESOURCE", + "language": LANGUAGE, + "runtime": RUNTIME, + "entrypoint": None, + "testEntrypoint": None, + "futureProperty": "value", + "typeName": "AWS::Example::Resource", + "canarySettings": { + FILE_GENERATION_ENABLED: True, + "contract_test_file_names": [], + }, + } + tmp_path = project.root + setup_rpdk_config(project, rpdk_config) + project.generate_canary_files() + + canary_root_path = tmp_path / TARGET_CANARY_ROOT_FOLDER + canary_folder_path = tmp_path / TARGET_CANARY_FOLDER + assert canary_root_path.exists() + assert canary_folder_path.exists() + canary_files = list(canary_folder_path.glob(f"{CANARY_FILE_PREFIX}*")) + assert not canary_files + + +def test_generate_canary_files_empty_canary_settings(project): + rpdk_config = { + ARTIFACT_TYPE_RESOURCE: "RESOURCE", + "language": LANGUAGE, + "runtime": RUNTIME, + "entrypoint": None, + "testEntrypoint": None, + "futureProperty": "value", + "typeName": "AWS::Example::Resource", + "canarySettings": {}, + } + tmp_path = project.root + setup_rpdk_config(project, rpdk_config) + project.generate_canary_files() + canary_root_path = tmp_path / TARGET_CANARY_ROOT_FOLDER + canary_folder_path = tmp_path / TARGET_CANARY_FOLDER + assert not canary_root_path.exists() + assert not canary_folder_path.exists() From fcb3fa466f4b625ae61fe6e1317d755765a64d30 Mon Sep 17 00:00:00 2001 From: Rajesh Duraisamy Date: Wed, 29 May 2024 14:53:22 -0700 Subject: [PATCH 2/3] Implement canary file generation functionality from contract test inputs files --- src/rpdk/core/generate_stack_templates.py | 233 ---------------------- src/rpdk/core/project.py | 130 +++++++++--- tests/test_generate_stack_templates.py | 231 --------------------- tests/test_project.py | 110 +++++++++- 4 files changed, 207 insertions(+), 497 deletions(-) delete mode 100644 src/rpdk/core/generate_stack_templates.py delete mode 100644 tests/test_generate_stack_templates.py diff --git a/src/rpdk/core/generate_stack_templates.py b/src/rpdk/core/generate_stack_templates.py deleted file mode 100644 index 4f412e8c..00000000 --- a/src/rpdk/core/generate_stack_templates.py +++ /dev/null @@ -1,233 +0,0 @@ -import json -import re -import shutil -from pathlib import Path -from typing import Any, Dict -from uuid import uuid4 - -import yaml - -CONTRACT_TEST_FOLDER = "contract-tests-artifacts" -CONTRACT_TEST_INPUT_PREFIX = "inputs_*" -CONTRACT_TEST_DEPENDENCY_FILE_NAME = "dependencies.yml" -FILE_GENERATION_ENABLED = "file_generation_enabled" -TYPE_NAME = "typeName" -CONTRACT_TEST_FILE_NAMES = "contract_test_file_names" -INPUT1_FILE_NAME = "inputs_1.json" -FN_SUB = "Fn::Sub" -FN_IMPORT_VALUE = "Fn::ImportValue" -UUID = "uuid" -DYNAMIC_VALUES_MAP = { - "region": "${AWS::Region}", - "partition": "${AWS::Partition}", - "account": "${AWS::AccountId}", -} - - -class StackTemplateGenerator: - def __init__( - self, - type_name: str, - stack_template_config: dict, - contract_test_file_names: list, - root=None, - ): - self.root = Path(root) if root else Path.cwd() - self.stack_template_config = stack_template_config - self.type_name = type_name - self.contract_test_file_names = contract_test_file_names - - @property - def file_generation_enabled(self): - return self.stack_template_config["file_generation_enabled"] - - @property - def file_prefix(self): - return self.stack_template_config["file_prefix"] - - @property - def stack_template_root_folder_path(self): - return self.stack_template_config["root_folder_path"] - - @property - def target_stack_template_folder_path(self): - return self.stack_template_config["target_folder_path"] - - @property - def stack_template_dependency_file_name(self): - return self.stack_template_config["dependency_file_name"] - - def generate_stack_templates(self) -> None: - """ - Generate stack_template files based on the contract test input files. - - This method checks if file generation is enabled and if the target contract test folder exists. - If both conditions are met, it creates the stack_template folder, copies the contract test dependencies, - and generates stack_template files for each contract test input file up to the specified count. - """ - if not self.file_generation_enabled or not self.contract_test_folder_exists(): - return - self._setup_stack_template_environment() - self._generate_stack_template_files() - - def contract_test_folder_exists(self) -> bool: - return Path(self.target_contract_test_folder_path).exists() - - def _setup_stack_template_environment(self) -> None: - stack_template_root = Path(self.stack_template_root_folder_path) - stack_template_folder = Path(self.target_stack_template_folder_path) - self.clean_and_create_stack_template_folder( - stack_template_root, stack_template_folder - ) - self.create_stack_template_bootstrap( - Path(self.target_contract_test_folder_path), stack_template_root - ) - - def _generate_stack_template_files(self) -> None: - resource_name = self.type_name - stack_template_folder = Path(self.target_stack_template_folder_path) - contract_test_files = self._get_sorted_contract_test_files() - for count, ct_file in enumerate(contract_test_files, start=1): - self.create_stack_template_file( - resource_name, ct_file, stack_template_folder, self.file_prefix, count - ) - - def _get_sorted_contract_test_files(self) -> list: - contract_test_folder = Path(self.target_contract_test_folder_path) - contract_test_files = [ - file - for file in contract_test_folder.glob(CONTRACT_TEST_INPUT_PREFIX) - if file.is_file() and file.name in self.contract_test_file_names - ] - return sorted(contract_test_files) - - def clean_and_create_stack_template_folder( - self, stack_template: Path, stack_template_folder: Path - ) -> None: - """ - Clean and create the stack_template folder. - - This method removes the existing stack_template root folder and creates a new stack_template folder. - - Args: - stack_template (Path): The path to the stack_template root folder. - stack_template_folder (Path): The path to the stack_template folder. - """ - stack_template_folder.mkdir(parents=True, exist_ok=True) - - def create_stack_template_bootstrap( - self, file_location: Path, stack_template: Path - ) -> None: - """ - Copy the contract test dependencies to the stack_template root folder. - - This method copies the contract test dependency file to the stack_template root folder - as the stack_template bootstrap file. - - Args: - file_location (Path): The path to the contract test folder. - stack_template (Path): The path to the stack_template root folder. - """ - dependencies_file = file_location / CONTRACT_TEST_DEPENDENCY_FILE_NAME - bootstrap_file = stack_template / self.stack_template_dependency_file_name - if dependencies_file.exists(): - shutil.copy(str(dependencies_file), str(bootstrap_file)) - - def create_stack_template_file( - self, - resource_type: str, - ct_file: Path, - stack_template_folder: Path, - stack_template_file_name_prefix: str, - count: int, - ) -> None: - """ - Create a stack_template file based on the contract test input file. - - This method generates a stack_template file in YAML format based on the provided contract test input file. - The stack_template file contains the resource configuration with dynamic values replaced. - - Args: - resource_type (str): The type of the resource being tested. - ct_file (Path): The path to the contract test input file. - stack_template_folder (Path): The path to the stack_template folder. - stack_template_file_name_prefix (str): The prefix for the stack_template file name. - count (int): The count of the stack_template file being generated. - """ - with ct_file.open("r") as f: - json_data = json.load(f) - resource_name = resource_type.split("::")[2] - stack_template_data = { - "Description": f"Template for {resource_type}", - "Resources": { - f"{resource_name}": { - "Type": resource_type, - "Properties": self.replace_dynamic_values( - json_data["CreateInputs"] - ), - } - }, - } - stack_template_file_name = f"{stack_template_file_name_prefix}{count}_001.yaml" - stack_template_file_path = stack_template_folder / stack_template_file_name - - with stack_template_file_path.open("w") as stack_template_file: - yaml.dump(stack_template_data, stack_template_file, indent=2) - - def replace_dynamic_values(self, properties: Dict[str, Any]) -> Dict[str, Any]: - """ - Replace dynamic values in the resource properties. - - This method recursively replaces dynamic values in the resource properties dictionary. - It handles nested dictionaries, lists, and strings with dynamic value placeholders. - - Args: - properties (Dict[str, Any]): The resource properties dictionary. - - Returns: - Dict[str, Any]: The resource properties dictionary with dynamic values replaced. - """ - for key, value in properties.items(): - if isinstance(value, dict): - properties[key] = self.replace_dynamic_values(value) - elif isinstance(value, list): - properties[key] = [self._replace_dynamic_value(item) for item in value] - else: - return_value = self._replace_dynamic_value(value) - properties[key] = return_value - return properties - - def _replace_dynamic_value(self, original_value: Any) -> Any: - """ - Replace a dynamic value with its corresponding value. - - This method replaces dynamic value placeholders in a string with their corresponding values. - It handles UUID generation, partition replacement, and Fn::ImportValue function. - - Args: - original_value (Any): The value to be replaced. - - Returns: - Any: The replaced value. - """ - pattern = r"\{\{(.*?)\}\}" - - def replace_token(match): - token = match.group(1) - if UUID in token: - return str(uuid4()) - if token in DYNAMIC_VALUES_MAP: - return DYNAMIC_VALUES_MAP[token] - return f'{{"{FN_IMPORT_VALUE}": "{token.strip()}"}}' - - replaced_value = re.sub(pattern, replace_token, str(original_value)) - - if any(value in replaced_value for value in DYNAMIC_VALUES_MAP.values()): - replaced_value = {FN_SUB: replaced_value} - if FN_IMPORT_VALUE in replaced_value: - replaced_value = json.loads(replaced_value) - return replaced_value - - @property - def target_contract_test_folder_path(self): - return self.root / CONTRACT_TEST_FOLDER diff --git a/src/rpdk/core/project.py b/src/rpdk/core/project.py index 14bfade0..0c907a5d 100644 --- a/src/rpdk/core/project.py +++ b/src/rpdk/core/project.py @@ -2,13 +2,16 @@ import json import logging import os +import re import shutil import sys import zipfile from pathlib import Path from tempfile import TemporaryFile +from typing import Any, Dict from uuid import uuid4 +import yaml from botocore.exceptions import ClientError, WaiterError from jinja2 import Environment, PackageLoader, select_autoescape from jsonschema import Draft7Validator @@ -29,7 +32,6 @@ SpecValidationError, ) from .fragment.module_fragment_reader import _get_fragment_file -from .generate_stack_templates import StackTemplateGenerator from .jsonutils.pointer import fragment_decode, fragment_encode from .jsonutils.utils import traverse from .plugin_registry import load_plugin @@ -68,6 +70,21 @@ CONTRACT_TEST_FILE_NAMES = "contract_test_file_names" INPUT1_FILE_NAME = "inputs_1.json" FILE_GENERATION_ENABLED = "file_generation_enabled" +CONTRACT_TEST_FOLDER = "contract-tests-artifacts" +CONTRACT_TEST_INPUT_PREFIX = "inputs_*" +CONTRACT_TEST_DEPENDENCY_FILE_NAME = "dependencies.yml" +FILE_GENERATION_ENABLED = "file_generation_enabled" +TYPE_NAME = "typeName" +CONTRACT_TEST_FILE_NAMES = "contract_test_file_names" +INPUT1_FILE_NAME = "inputs_1.json" +FN_SUB = "Fn::Sub" +FN_IMPORT_VALUE = "Fn::ImportValue" +UUID = "uuid" +DYNAMIC_VALUES_MAP = { + "region": "${AWS::Region}", + "partition": "${AWS::Partition}", + "account": "${AWS::AccountId}", +} DEFAULT_ROLE_TIMEOUT_MINUTES = 120 # 2 hours # min and max are according to CreateRole API restrictions # https://docs.aws.amazon.com/IAM/latest/APIReference/API_CreateRole.html @@ -231,6 +248,18 @@ def target_canary_folder_path(self): def rpdk_config(self): return self.root / RPDK_CONFIG_FILE + @property + def file_generation_enabled(self): + return self.canary_settings.get(FILE_GENERATION_ENABLED, False) + + @property + def contract_test_file_names(self): + return self.canary_settings.get(CONTRACT_TEST_FILE_NAMES, [INPUT1_FILE_NAME]) + + @property + def target_contract_test_folder_path(self): + return self.root / CONTRACT_TEST_FOLDER + @staticmethod def _raise_invalid_project(msg, e): LOG.debug(msg, exc_info=e) @@ -1283,26 +1312,81 @@ def _load_target_info( return type_info def generate_canary_files(self) -> None: - """ - Generate canary files based on the contract test input files. - - This method checks if file generation is enabled and if the target contract test folder exists. - If both conditions are met, it creates the canary folder, copies the contract test dependencies, - and generates canary files for each contract test input file up to the specified count. - """ - stack_template_config = { - "root_folder_path": self.target_canary_root_path, - "target_folder_path": self.target_canary_folder_path, - "dependency_file_name": CANARY_DEPENDENCY_FILE_NAME, - "file_prefix": CANARY_FILE_PREFIX, - "file_generation_enabled": self.canary_settings.get( - FILE_GENERATION_ENABLED, False - ), - } - generate_stack_templates = StackTemplateGenerator( - self.type_name, - stack_template_config, - self.canary_settings.get(CONTRACT_TEST_FILE_NAMES, [INPUT1_FILE_NAME]), - self.root, + if ( + not self.file_generation_enabled + or not Path(self.target_contract_test_folder_path).exists() + ): + return + self._setup_stack_template_environment() + self._generate_stack_template_files() + + def _setup_stack_template_environment(self) -> None: + stack_template_root = Path(self.target_canary_root_path) + stack_template_folder = Path(self.target_canary_folder_path) + stack_template_folder.mkdir(parents=True, exist_ok=True) + dependencies_file = ( + Path(self.target_contract_test_folder_path) + / CONTRACT_TEST_DEPENDENCY_FILE_NAME ) - generate_stack_templates.generate_stack_templates() + bootstrap_file = stack_template_root / CANARY_DEPENDENCY_FILE_NAME + if dependencies_file.exists(): + shutil.copy(str(dependencies_file), str(bootstrap_file)) + + def _generate_stack_template_files(self) -> None: + stack_template_folder = Path(self.target_canary_folder_path) + contract_test_folder = Path(self.target_contract_test_folder_path) + contract_test_files = [ + file + for file in contract_test_folder.glob(CONTRACT_TEST_INPUT_PREFIX) + if file.is_file() and file.name in self.contract_test_file_names + ] + contract_test_files = sorted(contract_test_files) + for count, ct_file in enumerate(contract_test_files, start=1): + with ct_file.open("r") as f: + json_data = json.load(f) + resource_name = self.type_info[2] + stack_template_data = { + "Description": f"Template for {self.type_name}", + "Resources": { + f"{resource_name}": { + "Type": self.type_name, + "Properties": self._replace_dynamic_values( + json_data["CreateInputs"] + ), + } + }, + } + stack_template_file_name = f"{CANARY_FILE_PREFIX}{count}_001.yaml" + stack_template_file_path = stack_template_folder / stack_template_file_name + with stack_template_file_path.open("w") as stack_template_file: + yaml.dump(stack_template_data, stack_template_file, indent=2) + + def _replace_dynamic_values(self, properties: Dict[str, Any]) -> Dict[str, Any]: + for key, value in properties.items(): + if isinstance(value, dict): + properties[key] = self._replace_dynamic_values(value) + elif isinstance(value, list): + properties[key] = [self._replace_dynamic_value(item) for item in value] + else: + return_value = self._replace_dynamic_value(value) + properties[key] = return_value + return properties + + def _replace_dynamic_value(self, original_value: Any) -> Any: + pattern = r"\{\{(.*?)\}\}" + + def replace_token(match): + token = match.group(1) + if UUID in token: + return str(uuid4()) + if token in DYNAMIC_VALUES_MAP: + return DYNAMIC_VALUES_MAP[token] + return f'{{"{FN_IMPORT_VALUE}": "{token.strip()}"}}' + + replaced_value = re.sub(pattern, replace_token, str(original_value)) + + if any(value in replaced_value for value in DYNAMIC_VALUES_MAP.values()): + replaced_value = {FN_SUB: replaced_value} + if FN_IMPORT_VALUE in replaced_value: + replaced_value = json.loads(replaced_value) + return replaced_value diff --git a/tests/test_generate_stack_templates.py b/tests/test_generate_stack_templates.py deleted file mode 100644 index 0795f906..00000000 --- a/tests/test_generate_stack_templates.py +++ /dev/null @@ -1,231 +0,0 @@ -import json -import os -import re -import shutil -import uuid -from pathlib import Path -from unittest.mock import ANY, patch - -import pytest - -from rpdk.core.generate_stack_templates import ( - CONTRACT_TEST_DEPENDENCY_FILE_NAME, - CONTRACT_TEST_FOLDER, - StackTemplateGenerator, -) - - -@pytest.fixture(name="stack_template_generator") -def setup_fixture(tmp_path): - root_path = tmp_path - type_name = "AWS::Example::Resource" - stack_template_config = { - "root_folder_path": root_path / "stack-templates", - "target_folder_path": root_path / "stack-templates" / "target", - "dependency_file_name": CONTRACT_TEST_DEPENDENCY_FILE_NAME, - "file_prefix": "stack-template-", - "file_generation_enabled": True, - } - contract_test_file_names = ["inputs_1.json", "inputs_2.json"] - contract_test_folder = root_path / CONTRACT_TEST_FOLDER - contract_test_folder.mkdir(parents=True, exist_ok=True) - # Create a dummy JSON file in the contract_test_folder directory - create_dummy_json_file(contract_test_folder, "inputs_1.json") - create_dummy_json_file(contract_test_folder, "inputs_2.json") - (contract_test_folder / CONTRACT_TEST_DEPENDENCY_FILE_NAME).touch() - - return StackTemplateGenerator( - type_name, - stack_template_config, - contract_test_file_names, - root_path, - ) - - -def create_dummy_json_file(directory: Path, file_name: str): - """Create a dummy JSON file in the given directory.""" - dummy_json_file = directory / file_name - dummy_data = { - "CreateInputs": { - "Property1": "Value1", - "Property2": "Value1", - } - } - with dummy_json_file.open("w") as f: - json.dump(dummy_data, f) - - -def create_folder(folder: Path): - if os.path.exists(folder): - shutil.rmtree(folder) - folder.mkdir() - - -def test_is_file_generation_enabled(stack_template_generator): - assert stack_template_generator.file_generation_enabled is True - - -def test_contract_test_folder_exists(stack_template_generator): - assert stack_template_generator.contract_test_folder_exists() is True - - -def test_generate_stack_template_files(stack_template_generator, tmp_path): - stack_template_generator.generate_stack_templates() - - stack_template_root_path = stack_template_generator.stack_template_root_folder_path - stack_template_folder_path = ( - stack_template_generator.target_stack_template_folder_path - ) - assert stack_template_root_path.exists() - assert stack_template_folder_path.exists() - - template_files = list( - stack_template_folder_path.glob(f"{stack_template_generator.file_prefix}*") - ) - assert len(template_files) == 2 - template_files.sort() - assert template_files[0].name == f"{stack_template_generator.file_prefix}1_001.yaml" - assert template_files[1].name == f"{stack_template_generator.file_prefix}2_001.yaml" - - bootstrap_file = ( - stack_template_root_path - / stack_template_generator.stack_template_dependency_file_name - ) - assert bootstrap_file.exists() - - -@pytest.mark.usefixtures("stack_template_generator") -def test_clean_and_create_template_folder(stack_template_generator, tmp_path): - template_root_path = stack_template_generator.stack_template_root_folder_path - template_folder_path = stack_template_generator.target_stack_template_folder_path - template_root_path.mkdir() - (template_root_path / "existing_file.txt").touch() - stack_template_generator.clean_and_create_stack_template_folder( - template_root_path, template_folder_path - ) - assert template_root_path.exists() - assert not list(template_folder_path.glob("*")) - assert template_folder_path.exists() - - -def test_create_template_bootstrap(stack_template_generator, tmp_path): - contract_test_folder = tmp_path / CONTRACT_TEST_FOLDER - create_folder(contract_test_folder) - (contract_test_folder / CONTRACT_TEST_DEPENDENCY_FILE_NAME).touch() - - template_root_path = stack_template_generator.stack_template_root_folder_path - create_folder(template_root_path) - - stack_template_generator.create_stack_template_bootstrap( - contract_test_folder, template_root_path - ) - - bootstrap_file = ( - template_root_path - / stack_template_generator.stack_template_dependency_file_name - ) - assert bootstrap_file.exists() - - -@patch("rpdk.core.generate_stack_templates.yaml.dump") -def test_create_template_file(mock_yaml_dump, stack_template_generator, tmp_path): - contract_test_folder = tmp_path / CONTRACT_TEST_FOLDER - create_folder(contract_test_folder) - contract_test_file = contract_test_folder / "inputs_1.json" - contract_test_data = { - "CreateInputs": { - "Property1": "Value1", - "Property2": "{{test123}}", - "Property3": {"Nested": "{{partition}}"}, - "Property4": ["{{region}}", "Value2"], - "Property5": "{{uuid}}", - "Property6": "{{account}}", - } - } - with contract_test_file.open("w") as f: - json.dump(contract_test_data, f) - - template_folder_path = stack_template_generator.target_stack_template_folder_path - template_folder_path.mkdir(parents=True, exist_ok=True) - - stack_template_generator.create_stack_template_file( - "AWS::Example::Resource", - contract_test_file, - template_folder_path, - stack_template_generator.file_prefix, - 1, - ) - - expected_template_data = { - "Description": "Template for AWS::Example::Resource", - "Resources": { - "Resource": { - "Type": "AWS::Example::Resource", - "Properties": { - "Property1": "Value1", - "Property2": {"Fn::ImportValue": ANY}, - "Property3": {"Nested": {"Fn::Sub": "${AWS::Partition}"}}, - "Property4": [{"Fn::Sub": "${AWS::Region}"}, "Value2"], - "Property5": ANY, - "Property6": {"Fn::Sub": "${AWS::AccountId}"}, - }, - } - }, - } - args, kwargs = mock_yaml_dump.call_args - assert args[0] == expected_template_data - assert kwargs - # Assert UUID generation - replaced_properties = args[0]["Resources"]["Resource"]["Properties"] - assert isinstance(replaced_properties["Property5"], str) - assert len(replaced_properties["Property5"]) == 36 # Standard UUID length - assert re.match( - r"[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}", - replaced_properties["Property5"], - ) - - # Assert the generated UUID is a valid UUID - generated_uuid = replaced_properties["Property5"] - assert uuid.UUID(generated_uuid) - - -def test_replace_dynamic_values(stack_template_generator): - properties = { - "Property1": "Value1", - "Property2": "{{uuid}}", - "Property3": {"Nested": "{{partition}}"}, - "Property4": ["{{region}}", "Value2"], - "Property5": [{"Key": "{{uuid}}"}], - "Property6": "{{account}}", - "Property7": "prefix-{{uuid}}-sufix", - "Property8": "{{value8}}", - } - replaced_properties = stack_template_generator.replace_dynamic_values(properties) - assert replaced_properties["Property1"] == "Value1" - assert isinstance(replaced_properties["Property2"], str) - assert len(replaced_properties["Property2"]) == 36 - assert re.match( - r"[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}", - replaced_properties["Property2"], - ) - assert replaced_properties["Property3"]["Nested"] == { - "Fn::Sub": "${AWS::Partition}" - } - assert replaced_properties["Property4"][0] == {"Fn::Sub": "${AWS::Region}"} - assert replaced_properties["Property4"][1] == "Value2" - assert replaced_properties["Property6"] == {"Fn::Sub": "${AWS::AccountId}"} - property7_value = replaced_properties["Property7"] - # Assert the replaced value - assert isinstance(property7_value, str) - assert "prefix-" in property7_value - assert "-sufix" in property7_value - # Extract the UUID part - property7_value = property7_value.replace("prefix-", "").replace("-sufix", "") - # Assert the UUID format - assert len(property7_value) == 36 # Standard UUID length - assert re.match( - r"[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}", property7_value - ) - # Assert the UUID is a valid UUID - assert uuid.UUID(property7_value) - assert replaced_properties["Property8"] == {"Fn::ImportValue": "value8"} diff --git a/tests/test_project.py b/tests/test_project.py index 26e9fdb0..1d52fef0 100644 --- a/tests/test_project.py +++ b/tests/test_project.py @@ -5,9 +5,11 @@ import logging import os import random +import re import shutil import string import sys +import uuid import zipfile from contextlib import contextmanager from io import StringIO @@ -27,7 +29,6 @@ InvalidProjectError, SpecValidationError, ) -from rpdk.core.generate_stack_templates import CONTRACT_TEST_FOLDER from rpdk.core.plugin_base import LanguagePlugin from rpdk.core.project import ( CANARY_DEPENDENCY_FILE_NAME, @@ -36,6 +37,7 @@ CONFIGURATION_SCHEMA_UPLOAD_FILENAME, CONTRACT_TEST_DEPENDENCY_FILE_NAME, CONTRACT_TEST_FILE_NAMES, + CONTRACT_TEST_FOLDER, FILE_GENERATION_ENABLED, OVERRIDES_FILENAME, SCHEMA_UPLOAD_FILENAME, @@ -2744,28 +2746,29 @@ def test__load_target_info_for_hooks_local_only(project): assert len(mock_loader.call_args[1]["local_schemas"]) == 4 -def setup_contract_test_data(tmp_path): +def setup_contract_test_data(tmp_path, contract_test_data=None): root_path = tmp_path contract_test_folder = root_path / CONTRACT_TEST_FOLDER contract_test_folder.mkdir(parents=True, exist_ok=True) assert contract_test_folder.exists() # Create a dummy JSON file in the canary_root_path directory - create_dummy_json_file(contract_test_folder, "inputs_1.json") - create_dummy_json_file(contract_test_folder, "inputs_2.json") + create_dummy_json_file(contract_test_folder, "inputs_1.json", contract_test_data) + create_dummy_json_file(contract_test_folder, "inputs_2.json", contract_test_data) (contract_test_folder / CONTRACT_TEST_DEPENDENCY_FILE_NAME).touch() assert contract_test_folder.exists() return Project(str(root_path)) -def create_dummy_json_file(directory: Path, file_name: str): +def create_dummy_json_file(directory: Path, file_name: str, dummy_data=None): """Create a dummy JSON file in the given directory.""" dummy_json_file = directory / file_name - dummy_data = { - "CreateInputs": { - "Property1": "Value1", - "Property2": "Value1", + if not dummy_data: + dummy_data = { + "CreateInputs": { + "Property1": "Value1", + "Property2": "Value1", + } } - } with dummy_json_file.open("w") as f: json.dump(dummy_data, f) @@ -2819,6 +2822,93 @@ def test_generate_canary_files(project): assert bootstrap_file.exists() +@patch("rpdk.core.project.yaml.dump") +def test_create_template_file(mock_yaml_dump, project): + contract_test_data = { + "CreateInputs": { + "Property1": "Value1", + "Property2": "{{test123}}", + "Property3": {"Nested": "{{partition}}"}, + "Property4": ["{{region}}", "Value2"], + "Property5": "{{uuid}}", + "Property6": "{{account}}", + "Property7": "prefix-{{uuid}}-sufix", + } + } + setup_contract_test_data(project.root, contract_test_data) + plugin = object() + data = json.dumps( + { + "artifact_type": "RESOURCE", + "language": LANGUAGE, + "runtime": RUNTIME, + "entrypoint": None, + "testEntrypoint": None, + "futureProperty": "value", + "typeName": "AWS::Example::Resource", + "canarySettings": { + FILE_GENERATION_ENABLED: True, + CONTRACT_TEST_FILE_NAMES: ["inputs_1.json", "inputs_2.json"], + }, + } + ) + patch_load = patch( + "rpdk.core.project.load_plugin", autospec=True, return_value=plugin + ) + + with patch_settings(project, data) as mock_open, patch_load as mock_load: + project.load_settings() + project.generate_canary_files() + mock_open.assert_called_once_with("r", encoding="utf-8") + mock_load.assert_called_once_with(LANGUAGE) + expected_template_data = { + "Description": "Template for AWS::Example::Resource", + "Resources": { + "Resource": { + "Type": "AWS::Example::Resource", + "Properties": { + "Property1": "Value1", + "Property2": {"Fn::ImportValue": ANY}, + "Property3": {"Nested": {"Fn::Sub": "${AWS::Partition}"}}, + "Property4": [{"Fn::Sub": "${AWS::Region}"}, "Value2"], + "Property5": ANY, + "Property6": {"Fn::Sub": "${AWS::AccountId}"}, + "Property7": ANY, + }, + } + }, + } + args, kwargs = mock_yaml_dump.call_args + assert args[0] == expected_template_data + assert kwargs + # Assert UUID generation + replaced_properties = args[0]["Resources"]["Resource"]["Properties"] + assert isinstance(replaced_properties["Property5"], str) + assert len(replaced_properties["Property5"]) == 36 # Standard UUID length + assert re.match( + r"[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}", + replaced_properties["Property5"], + ) + + # Assert the generated UUID is a valid UUID + generated_uuid = replaced_properties["Property5"] + assert uuid.UUID(generated_uuid) + property7_value = replaced_properties["Property7"] + # Assert the replaced value + assert isinstance(property7_value, str) + assert "prefix-" in property7_value + assert "-sufix" in property7_value + # Extract the UUID part + property7_value = property7_value.replace("prefix-", "").replace("-sufix", "") + # Assert the UUID format + assert len(property7_value) == 36 # Standard UUID length + assert re.match( + r"[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}", property7_value + ) + # Assert the UUID is a valid UUID + assert uuid.UUID(property7_value) + + def setup_rpdk_config(project, rpdk_config): root_path = project.root plugin = object() From fa716110da1ea446d53b08c96bc2fb67f3e7e823 Mon Sep 17 00:00:00 2001 From: Rajesh Duraisamy Date: Thu, 30 May 2024 19:12:58 -0700 Subject: [PATCH 3/3] Bump version for release --- src/rpdk/core/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/rpdk/core/__init__.py b/src/rpdk/core/__init__.py index d1855829..dd25bee8 100644 --- a/src/rpdk/core/__init__.py +++ b/src/rpdk/core/__init__.py @@ -1,5 +1,5 @@ import logging -__version__ = "0.2.36" +__version__ = "0.2.37" logging.getLogger(__name__).addHandler(logging.NullHandler())