No software is perfect, here is a list of known issues /gotchas that is worth noting with potential workarounds when you do encounter them (and for some, where you don’t really have a solution)
- Queries assigned to variables (see Guard: Variable, Projections and Interpolations) can be accessed using two forms when defining clauses, E.g.
let api_gws = Resources.*[ Type == 'AWS::ApiGateway::RestApi' ]
%api_gws.Properties.EndpointConfiguration.Types[*] == "PRIVATE"`
or
%api_gws {
Properties.EndpointConfiguration.Types[*] == "PRIVATE"
}
The block form iterates over all AWS::ApiGateway::RestApi
resources found in the input. The first form short circuits and returns immediately after the first resource failure.
Workaround: use the block form to traverse all values to show all resource failures and not just the first one that failed. We are tracking to resolve this issue. 2. Need
when
guards with filter expressions- When a query uses filters likeResources.*[ Type == 'AWS::ApiGateway::RestApi' ]
, if there are noApiGateway
resources, then Guard will fail the clause today when performing the check
%api_gws.Properties.EndpointConfiguration.Types[*] == "PRIVATE"
Workaround: assign filters to variables and use
when
condition check e.g.
let api_gws = Resources.*[ Type == 'AWS::ApiGateway::RestApi' ]
when %api_gws !empty { ...}
-
When performing
!=
comparison, if the values are incompatible like comparing astring
toint
, an error is thrown internally but currently suppressed and converted tofalse
to satisfy the requirements of Rust’s PartialEq. We are tracking to release a fix for this issue soon. -
exists
andempty
checks do not display the JSON pointer path inside the document in the error messages. Both these clauses often have retrieval errors which does not maintain this traversal information today. We are tracking to resolve this issue. -
No support for calling functions inline on the LHS of an operator
We do not support inline usage of functions on the LHS of operators at the moment. The support for built-in functions when being used on the LHS of an operator is currently limited to assignment of the return value to a variable.
Consider an example wherein our template has a node named
Instances
which is a collection. We need to author a rule that checks to ensure this collection contains a certain number of minimum items, say 2.These following examples are currently NOT SUPPORTED:
# Not supported at the moment rule INSTANCES_COUNT_CHECK { count(Instances.*) < 2 << Violation: We should have at least 2 instances >> } # OR rule VERIFY_COUNT_RETURNS_INT { count(Instances.*) is_int << Violation: We should have at least 2 instances >> }
While the above code snippet might be tempting to use we haven't made the changes required to support it in our grammar yet.
Workaround: When working with unary operators. assign the result of the function to a variable. When working with binary operators, you have the choice to have the function call on the RHS of the operator, or assign the result of the function call to a variable, and then you are free to have this variable on either side of the operator.
So, our example rule now becomes:
# Use this instead rule INSTANCES_COUNT_CHECK { let no_of_instances = count(Instances.*) # all of the following options are valid ways to write this clause %no_of_instances < 2 2 > %no_of_instances 2 > count(Instances.*) << Violation: We should have at least 2 instances >> } rule VERIFY_COUNT_RETURNS_INT { let no_of_instances = count(Instances.*) %no_of_instances is_int }
- Key names containing dashes Currently dashes in key names are treated as special characters. This means that if you're trying to access a key that has a dash in it, you must wrap that key in quotes.
eg: given the following sample template
Resources: bucket: Type: AWS::S3::Bucket Properties: some-key: true
if we wanted to check the
some-key
key here we would need to write a rule like solet root = Resources.*[ Type == "AWS::S3::Bucket" ] rule example_with_dash_in_key when %root !empty { %root.Properties."some-key" == true }