Skip to content

Commit

Permalink
Expand deployment file capabilities
Browse files Browse the repository at this point in the history
  • Loading branch information
kddejong committed Jan 10, 2025
1 parent c56d4f9 commit 3e0f51a
Show file tree
Hide file tree
Showing 37 changed files with 587 additions and 383 deletions.
50 changes: 31 additions & 19 deletions src/cfnlint/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,12 +12,14 @@
import logging
import os
import sys
from copy import deepcopy
from pathlib import Path
from typing import Any, Dict, Sequence, TypedDict

from typing_extensions import Unpack

import cfnlint.decode.cfn_yaml
from cfnlint.context.parameters import ParameterSet
from cfnlint.helpers import REGIONS, format_json_string
from cfnlint.jsonschema import StandardValidator
from cfnlint.version import __version__
Expand Down Expand Up @@ -681,15 +683,15 @@ class ManualArgs(TypedDict, total=False):
non_zero_exit_code: str
output_file: str
regions: list
parameters: list[dict[str, Any]]
parameters: list[ParameterSet]
templates: list[str]


# pylint: disable=too-many-public-methods
class ConfigMixIn(TemplateArgs, CliArgs, ConfigFileArgs):

def __init__(self, cli_args: list[str] | None = None, **kwargs: Unpack[ManualArgs]):
self._manual_args = kwargs or ManualArgs()
self._templates_to_process = False
CliArgs.__init__(self, cli_args)
# configure debug as soon as we can
TemplateArgs.__init__(self, {})
Expand All @@ -716,12 +718,17 @@ def __repr__(self):
"merge_configs": self.merge_configs,
"non_zero_exit_code": self.non_zero_exit_code,
"override_spec": self.override_spec,
"regions": self.regions,
"parameters": self.parameters,
"regions": self.regions,
"templates": self.templates,
}
)

def __eq__(self, value):
if not isinstance(value, ConfigMixIn):
return False
return str(self) == str(value)

def _get_argument_value(self, arg_name, is_template, is_config_file):
cli_value = getattr(self.cli_args, arg_name)
template_value = self.template_args.get(arg_name)
Expand Down Expand Up @@ -816,7 +823,9 @@ def templates(self):
file_args = self._get_argument_value("templates", False, True)
cli_args = self._get_argument_value("templates", False, False)

if cli_alt_args:
if "templates" in self._manual_args:
filenames = self._manual_args["templates"]
elif cli_alt_args:
filenames = cli_alt_args
elif file_args:
filenames = file_args
Expand All @@ -826,10 +835,6 @@ def templates(self):
# No filenames found, could be piped in or be using the api.
return None

# If we're still haven't returned, we've got templates to lint.
# Build up list of templates to lint.
self.templates_to_process = True

if isinstance(filenames, str):
filenames = [filenames]

Expand Down Expand Up @@ -880,12 +885,21 @@ def append_rules(self):
)

@property
def parameters(self):
return self._get_argument_value("parameters", True, True)
def parameters(self) -> list[ParameterSet]:
parameter_sets = self._get_argument_value("parameters", True, True)
results: list[ParameterSet] = []
for parameter_set in parameter_sets:
if isinstance(parameter_set, ParameterSet):
results.append(parameter_set)
else:
results.append(
ParameterSet(
source=None,
parameters=parameter_set,
)
)

@parameters.setter
def parameters(self, parameters: list[dict[str, Any]]):
self._manual_args["parameters"] = parameters
return results

@property
def override_spec(self):
Expand Down Expand Up @@ -952,10 +966,8 @@ def non_zero_exit_code(self):
def force(self):
return self._get_argument_value("force", False, False)

@property
def templates_to_process(self):
return self._templates_to_process
def evolve(self, **kwargs: Unpack[ManualArgs]) -> "ConfigMixIn":

@templates_to_process.setter
def templates_to_process(self, value: bool):
self._templates_to_process = value
config = deepcopy(self)
config._manual_args.update(kwargs)
return config
9 changes: 7 additions & 2 deletions src/cfnlint/context/__init__.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,8 @@
__all__ = ["Context", "create_context_for_template"]
__all__ = ["Context", "create_context_for_template", "ParameterSet"]

from cfnlint.context.context import Context, Path, create_context_for_template
from cfnlint.context.context import (
Context,
ParameterSet,
Path,
create_context_for_template,
)
22 changes: 16 additions & 6 deletions src/cfnlint/context/context.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,10 +9,11 @@
from collections import deque
from dataclasses import InitVar, dataclass, field, fields
from functools import lru_cache
from typing import Any, Deque, Iterator, Sequence, Set, Tuple
from typing import TYPE_CHECKING, Any, Deque, Iterator, Set, Tuple

from cfnlint.context._mappings import Mappings
from cfnlint.context.conditions._conditions import Conditions
from cfnlint.context.parameters import ParameterSet
from cfnlint.helpers import (
BOOLEAN_STRINGS_TRUE,
FUNCTIONS,
Expand All @@ -22,6 +23,9 @@
)
from cfnlint.schema import PROVIDER_SCHEMA_MANAGER, AttributeDict

if TYPE_CHECKING:
from cfnlint.template import Template

_PSEUDOPARAMS_NON_REGION = ["AWS::AccountId", "AWS::NoValue", "AWS::StackName"]


Expand Down Expand Up @@ -132,12 +136,12 @@ class Context:
"""

# what regions we are processing
regions: Sequence[str] = field(
regions: list[str] = field(
init=True, default_factory=lambda: list([REGION_PRIMARY])
)

# supported functions at this point in the template
functions: Sequence[str] = field(init=True, default_factory=list)
functions: list[str] = field(init=True, default_factory=list)

path: Path = field(init=True, default_factory=Path)

Expand All @@ -153,7 +157,7 @@ class Context:
init=True, default_factory=lambda: set(PSEUDOPARAMS)
)

# Combiniation of storing any resolved ref
# Combination of storing any resolved ref
# and adds in any Refs available from things like Fn::Sub
ref_values: dict[str, Any] = field(init=True, default_factory=dict)

Expand All @@ -163,6 +167,9 @@ class Context:
is_resolved_value: bool = field(init=True, default=False)
resolve_pseudo_parameters: bool = field(init=True, default=True)

# Deployment parameters
parameter_sets: list[ParameterSet] | None = field(init=True, default_factory=list)

def evolve(self, **kwargs) -> "Context":
"""
Create a new context without merging together attributes
Expand Down Expand Up @@ -441,7 +448,9 @@ def _init_transforms(transforms: Any) -> Transforms:
return Transforms([])


def create_context_for_template(cfn):
def create_context_for_template(
cfn: Template,
) -> "Context":
parameters = {}
try:
parameters = _init_parameters(cfn.template.get("Parameters", {}))
Expand Down Expand Up @@ -476,5 +485,6 @@ def create_context_for_template(cfn):
regions=cfn.regions,
path=Path(),
functions=["Fn::Transform"],
ref_values=cfn.parameters or {},
ref_values={},
parameter_sets=[],
)
16 changes: 16 additions & 0 deletions src/cfnlint/context/parameters.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
"""
Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: MIT-0
"""

from __future__ import annotations

from dataclasses import dataclass, field
from typing import Any


@dataclass
class ParameterSet:

source: str | None = field(default=None)
parameters: dict[str, Any] = field(default_factory=dict)
3 changes: 0 additions & 3 deletions src/cfnlint/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,12 +12,9 @@
from cfnlint.exceptions import UnexpectedRuleException
from cfnlint.match import Match
from cfnlint.rules import RulesCollection

from cfnlint.runner.exceptions import UnexpectedRuleException
from cfnlint.runner.template.runner import _run_template



def get_rules(
append_rules: list[str],
ignore_rules: list[str],
Expand Down
9 changes: 9 additions & 0 deletions src/cfnlint/decode/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,15 @@
SPDX-License-Identifier: MIT-0
"""

__all__ = [
"create_match_file_error",
"create_match_json_parser_error",
"create_match_yaml_parser_error",
"decode",
"decode_str",
"convert_dict",
]

from cfnlint.decode.decode import (
create_match_file_error,
create_match_json_parser_error,
Expand Down
11 changes: 7 additions & 4 deletions src/cfnlint/decode/decode.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@

import logging
from json.decoder import JSONDecodeError
from typing import Any, Callable, Dict, List, Tuple, Union
from typing import Any, Callable, List, Tuple, Union

from yaml import YAMLError
from yaml.parser import ParserError
Expand All @@ -19,7 +19,7 @@
LOGGER = logging.getLogger(__name__)

Matches = List[Match]
Decode = Tuple[Union[Dict[str, Any], None], Matches]
Decode = Tuple[Union[dict[str, Any], None], Matches]


def decode_str(s: str) -> Decode:
Expand All @@ -37,7 +37,7 @@ def _decode(
) -> Decode:
"""Decode payload using yaml_f and json_f, using filename for log output."""
template = None
matches = []
matches: Matches = []
try:
template = yaml_f(payload)
except IOError as e:
Expand Down Expand Up @@ -145,7 +145,10 @@ def _decode(
message="Template needs to be an object.",
)
]
return (template, matches)

if matches:
return None, matches
return template, matches


def create_match_yaml_parser_error(parser_error, filename):
Expand Down
24 changes: 20 additions & 4 deletions src/cfnlint/exceptions.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,11 @@
"""
Copyright 2019 Amazon.com, Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: MIT-0
"""

from __future__ import annotations


class CfnLintExitException(Exception):
"""
An exception that is raised to indicate that the CloudFormation linter should exit.
Expand All @@ -10,11 +18,11 @@ class CfnLintExitException(Exception):
exit_code (int): The exit code to be used when the linter exits.
Methods:
__init__(self, exit_code: int) -> None:
__init__(self, msg: str | None=None, exit_code: int=1) -> None:
Initialize a new CfnLintExitException instance with the specified exit code.
"""

def __init__(self, msg=None, exit_code=1):
def __init__(self, msg: str | None = None, exit_code: int = 1):
"""
Initialize a new CfnLintExitException instance with the specified exit code.
Expand Down Expand Up @@ -49,9 +57,17 @@ class UnexpectedRuleException(CfnLintExitException):

class DuplicateRuleError(CfnLintExitException):
"""
The data associated with a particular path could not be loaded.
:ivar data_path: The data path that the user attempted to load.
An exception that is raised when an unexpected error occurs while loading rules.
This exception is raised when the CloudFormation linter encounters a rule with a
duplicate ID.
"""

def __init__(self, rule_id: str):
"""
Initialize a new CfnLintExitException instance with the specified exit code.
Args:
rule_id (str): The rule ID that a duplicate was found for.
"""
super().__init__(f"Rule already included: {rule_id}")
3 changes: 3 additions & 0 deletions src/cfnlint/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@

from __future__ import annotations

import dataclasses
import datetime
import fnmatch
import gzip
Expand Down Expand Up @@ -627,6 +628,8 @@ def converter(o): # pylint: disable=R1710
"""Help convert date/time into strings"""
if isinstance(o, datetime.datetime):
return o.__str__() # pylint: disable=unnecessary-dunder-call
elif dataclasses.is_dataclass(o):
return dataclasses.asdict(o)

return json.dumps(
json_string, indent=1, sort_keys=True, separators=(",", ": "), default=converter
Expand Down
4 changes: 2 additions & 2 deletions src/cfnlint/jsonschema/validators.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@
from typing import Any, Callable

from cfnlint.conditions import UnknownSatisfisfaction
from cfnlint.context import Context, create_context_for_template
from cfnlint.context import Context
from cfnlint.helpers import is_function
from cfnlint.jsonschema import _keywords, _keywords_cfn, _resolvers_cfn
from cfnlint.jsonschema._filter import FunctionFilter
Expand Down Expand Up @@ -92,7 +92,7 @@ class Validator:

def __post_init__(self):
if self.context is None:
self.context = create_context_for_template(self.cfn)
self.context = self.cfn.context.evolve()
if self.resolver is None:
self.resolver = RefResolver.from_schema(
schema=self.schema,
Expand Down
2 changes: 2 additions & 0 deletions src/cfnlint/match.py
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,8 @@ def create(
if linenumberend is None:
linenumberend = linenumber

filename = getattr(rulematch_obj, "filename", filename)

return cls(
linenumber=linenumber,
columnnumber=columnnumber,
Expand Down
Loading

0 comments on commit 3e0f51a

Please sign in to comment.