Skip to content

Commit

Permalink
Merge pull request #439 from mrinaudo-aws/add-getfromjson-custom-reso…
Browse files Browse the repository at this point in the history
…urce

Add getfromjson code for Lambda-backed custom resource consumers.
  • Loading branch information
ericzbeard authored Jun 26, 2024
2 parents 86d9d1a + 80fe003 commit eef4a48
Show file tree
Hide file tree
Showing 22 changed files with 1,101 additions and 0 deletions.
11 changes: 11 additions & 0 deletions CloudFormation/CustomResources/getfromjson/.coveragerc
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
[report]
fail_under = 100
show_missing = True

[run]
branch = True
include =
src/*.py
omit =
src/__init__.py
src/tests/*
3 changes: 3 additions & 0 deletions CloudFormation/CustomResources/getfromjson/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
venv/
__pycache__/
.coverage
266 changes: 266 additions & 0 deletions CloudFormation/CustomResources/getfromjson/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,266 @@
# getfromjson


## Overview

`getfromjson` is a module for Python that is meant to run in an [AWS
Lambda](https://aws.amazon.com/lambda/) function that, in turn, backs
one (or more) [AWS
CloudFormation](https://aws.amazon.com/cloudformation/) [custom
resource](https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/template-custom-resources.html)
that you declare and use to get a given value out of an input JSON
data structure and an input search argument you both provide.

For more information on Lambda-backed custom resources, see
[Lambda-backed custom
resources](https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/template-custom-resources-lambda.html).

There are two parts you'll need to set up. First, you set up the
infrastructure needed to support `getfromjson`: you do this with the
`src/getfromjson.yml` CloudFormation template, that describes the
following resources:

- the Lambda function and the `getfromjson.py` module for Python, that
will back custom resource consumers; this function will be
responsible for returning values from an input JSON data structure
and search argument, that you both provide;

- the [AWS Identity and Access Management
(IAM)](https://aws.amazon.com/iam/) [execution
role](https://docs.aws.amazon.com/lambda/latest/dg/lambda-intro-execution-role.html)
for the Lambda function;

- the [Amazon CloudWatch
Logs](https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/WhatIsCloudWatchLogs.html)
log group for the Lambda function.

The _Setup_ section, further down in this document, shows you how to
create resources above, in a given AWS account and AWS region, by
using the template mentioned earlier to create a CloudFormation stack.

Next, you consume the Lambda function, that you previously created in
a given AWS account and AWS region, with custom resources that you
declare in other CloudFormation templates (that you'll use to create
new stacks) where you'll pass in both JSON data and a search argument
for the data. The `example-templates/` directory contains samples that
illustrate how to consume `getfromjson` with Lambda-backed custom
resources; the following snippet shows you an overview on how to
consume, in another template, the Lambda function (the example below
consumes custom resources' values in the `Outputs` section of the
template, but you can consume such values also from properties of
other resources you describe in the `Resources` section of the
template):

```
Resources:
GetFromJsonCustomResourceSampleGetFromList:
Type: Custom::GetFromJson
Properties:
ServiceTimeout: 1
ServiceToken: !ImportValue Custom-GetFromJson
json_data: '["test0", "test1", "test2"]'
search: '[2]'
GetFromJsonCustomResourceSampleGetFromMap:
Type: Custom::GetFromJson
Properties:
ServiceTimeout: 1
ServiceToken: !ImportValue Custom-GetFromJson
json_data: '{"test": {"test1": ["x", "y"]}}'
search: '["test"]["test1"][1]'
Outputs:
GetFromJsonCustomResourceSampleGetFromListValue:
Value: !GetAtt GetFromJsonCustomResourceSampleGetFromList.Data
GetFromJsonCustomResourceSampleGetFromMapValue:
Value: !GetAtt GetFromJsonCustomResourceSampleGetFromMap.Data
```

whereas the `GetFromJsonCustomResourceSampleGetFromListValue` and
`GetFromJsonCustomResourceSampleGetFromMapValue` outputs, once you'll
create the stack, will show `test2` and `y` respectively as the
returned values.

Note also the `ServiceToken: !ImportValue Custom-GetFromJson` line,
that the example above uses to tell the custom resource what is the
[Amazon Resource Name
(ARN)](https://docs.aws.amazon.com/IAM/latest/UserGuide/reference-arns.html)
of the Lambda function that backs the custom resource itself. The
Lambda function's ARN is
[exported](https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/using-cfn-stack-exports.html)
in the `getfromjson` template with the `Custom-GetFromJson` export
name: in the example above, you use the `Fn::ImportValue` [intrinsic
function](https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/intrinsic-function-reference-importvalue.html)
to reference the ARN from the export.

For more information on CloudFormation custom resources in a
CloudFormation template, see the `AWS::CloudFormation::CustomResource`
[reference](https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-cloudformation-customresource.html).


## Input and output: values and limits

The following are supported input and output values:

- input:

- `json_data`:

- `json_data` maximum length: 4,096 bytes;

- map keys can contain alphanumeric characters, dashes, and
underscore characters;

- map values can contain Unicode characters;

- `search`:

- `search` maximum length: 256 bytes;

- map keys can contain alphanumeric characters, dashes, and
underscore characters;

- list indexes must be integers (such as, `[0]` instead of
`["0"]`);

- output:

- custom resource response: 4,096 bytes maximum; this is a
CloudFormation quota - for more information, see _Custom
resource response_ in [Understand CloudFormation
quotas](https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/cloudformation-limits.html);

- returned value can contain Unicode characters;

- returned value type is represented as a Unicode string.


## Setup

Install [rain](https://github.com/aws-cloudformation/rain), that
you'll use to create CloudFormation stacks to manage resources. When
ready, create the Lambda function, its execution role and log group,
in a given AWS account and AWS region (the example below uses
`us-east-1` as the AWS region; change this value as needed); such
resources will be backing the custom resource consumer:

```
rain deploy src/getfromjson.yml getfromjson \
--region us-east-1
```

Note that the `TagName` parameter for the `getfromjson.yml` template
is optional, and you can omit it if needed: its value defaults to
`GetFromJson`. `TagName` is used to add a tag called `Name`, with the
value for the `TagName` parameter, to resources that the template
describes (the Lambda function that backs relevant custom resources,
the function's execution role, and the function's log group).


## Usage

Please make sure to follow the _Setup_ section above before
continuing. When ready, create a CloudFormation stack that uses an
example template showing you how to invoke the Lambda function (that
you created earlier) that backs up the custom resources you'll use to
extract values from example JSON input (extracted values will be
available in the `Outputs` section for the example stack you'll
create; note that you can also choose to consume such values from
properties of other resources you describe in the `Resources` section
of the template):

```
rain deploy example-templates/getfromjson-consumer.yml getfromjson-consumer \
--region us-east-1
```


## Development

Install the [AWS Serverless Application Model Command Line Interface
(AWS SAM CLI)
](https://docs.aws.amazon.com/serverless-application-model/latest/developerguide/using-sam-cli.html)
in your workstation. When done, refer to the documentation on
[Installing Docker to use with the AWS SAM
CLI](https://docs.aws.amazon.com/serverless-application-model/latest/developerguide/install-docker.html).

Install [rain](https://github.com/aws-cloudformation/rain), that
you'll use to create CloudFormation stacks to manage, on your behalf,
the resources you'll need.

Next, create and activate a [virtual
environment](https://docs.python.org/3/library/venv.html) for Python
using the following commands:

```
python -m venv venv
source venv/bin/activate
```

Next, install the following Python module(s) in your activated
environment:

```
python -m pip install --upgrade -r requirements-dev.txt
```

To run unit tests for `getfromjson` on your machine, use the following
command:

```
pytest --cov
```

To speed up the development lifecycle, you can locally invoke the
Lambda function code, and pass input events for your test use cases
(the unit tests for `getfromjson` do something similar as well, and by
invoking the Lambda function locally you add an integration testing
flavor to your development lifecycle). To do so, run the following
command from the root level of the project:

```
./run-local-invoke
```

The script above uses the SAM CLI to invoke the Lambda function code
locally on your machine; you'll need to have Docker installed and
running. The SAM CLI uses the content of the `template.yml` file in
the `src` directory to determine which settings to use for aspects
that include which `Runtime` to use, and the `MemorySize`: if you'll
need to adjust some of these values, make sure you reflect your
changes also in relevant parts of the `src/getfromjson.yml`
CloudFormation template; note that the value for `Handler` though
needs to have a different prefix depending on the file you use (it
should be `Handler: getfromjson.lambda_handler` in the SAM template,
and `Handler: index.lambda_handler` in the CloudFormation template).

Note: the code for `getfromjson`, by default, uses the `INFO` logging
level - you'll need to update the following line and use a different
logging level (such as, `logging.DEBUG`) when developing and
troubleshooting the code):

```
LOGGER.setLevel(logging.INFO)
```

When ready to create the infrastructure for `getfromjson`, use the
following commands to do so (the example below uses `us-east-1` as the
AWS region; change this value as needed):

```
pylint src/ \
&& mypy \
&& pytest --cov \
&& bandit -c bandit.yaml -r src/ \
&& rain deploy src/getfromjson.yml getfromjson \
--region us-east-1
```

To create a stack with example custom resource consumers, run the
following command:

```
rain deploy example-templates/getfromjson-consumer.yml getfromjson-consumer \
--region us-east-1
```
3 changes: 3 additions & 0 deletions CloudFormation/CustomResources/getfromjson/bandit.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
# For more information, see https://bandit.readthedocs.io/en/latest/config.html

exclude_dirs: ['tests']
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
AWSTemplateFormatVersion: "2010-09-09"

Description: This AWS CloudFormation template describes a sample CloudFormation custom resource consumer for the GetFromJson Lambda-backed custom resource provider.

Parameters:
GetFromListJsonData:
Description: Example JSON data representing a list of values.
Type: String
Default: '["test0", "test1", "test2"]'

GetFromListJsonDataQuery:
Description: Example query for JSON data representing a list of values.
Type: String
Default: '[2]'

GetFromMapJsonData:
Description: Example JSON data representing a map data structure.
Type: String
Default: '{"test": {"test1": ["x", "y"]}}'

GetFromMapJsonDataQuery:
Description: Example query for JSON data representing a map data structure.
Type: String
Default: '["test"]["test1"][1]'

Resources:
GetFromJsonCustomResourceSampleGetFromList:
Type: Custom::GetFromJson
Properties:
ServiceTimeout: 1
ServiceToken: !ImportValue Custom-GetFromJson
json_data: !Ref GetFromListJsonData
search: !Ref GetFromListJsonDataQuery

GetFromJsonCustomResourceSampleGetFromMap:
Type: Custom::GetFromJson
Properties:
ServiceTimeout: 1
ServiceToken: !ImportValue Custom-GetFromJson
json_data: !Ref GetFromMapJsonData
search: !Ref GetFromMapJsonDataQuery

Outputs:
GetFromJsonCustomResourceSampleGetFromListValue:
Value: !GetAtt GetFromJsonCustomResourceSampleGetFromList.Data

GetFromJsonCustomResourceSampleGetFromMapValue:
Value: !GetAtt GetFromJsonCustomResourceSampleGetFromMap.Data
4 changes: 4 additions & 0 deletions CloudFormation/CustomResources/getfromjson/mypy.ini
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
[mypy]
follow_imports = silent
strict = True
files = src/
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
bandit>=1.7.9
cfn-lint>=1.3.6
cfnresponse>=1.1.4
mypy>=1.10.1
pip>=24.1
pylint>=3.2.3
pytest-cov>=5.0.0
setuptools>=70.1.1
44 changes: 44 additions & 0 deletions CloudFormation/CustomResources/getfromjson/run-local-invoke
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
#!/bin/bash


EVENTS_DIR='events'

EVENT_FILES=(
'event-consume-from-map.json'
'event-consume-from-list.json'
'event-consume-from-map-retrieval-error.json'
'event-consume-from-list-retrieval-error.json'
'event-empty-json-data-input.json'
'event-empty-search-input.json'
'event-invalid-json-data-input.json'
'event-invalid-search-input.json'
)


run_local_invoke() {
echo "**** Invoking event from file: $event_file ****"
sam local invoke --event $EVENTS_DIR/$event_file
echo '-----------------------------'
}


cat <<EOF
Note: unless you've done this already, you might want to temporarily
set:
LOGGER.setLevel(logging.DEBUG)
in getfromjson.py for testing with this script in DEBUG mode.
Current logging settings:
EOF
grep 'LOGGER.setLevel' src/getfromjson.py
echo


cd src/
for event_file in "${EVENT_FILES[@]}"; do
run_local_invoke
read -p 'Press RETURN to continue, or CONTROL-c to end: '
done
Empty file.
Loading

0 comments on commit eef4a48

Please sign in to comment.