Skip to content

Latest commit

 

History

History
 
 

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 
 
 

AWS IoT Greengrass V2 Base Implementation

This is the base accelerator that deploys a single instance of an AWS IoT Greengrass core device and all supporting resources for operation, including:

  • Thing - Virtual representation of the Greengrass core device (along with certificate and private key)
  • Thing group - The thing is part of this group and the target for the Greengrass deployment
  • IoT Core role alias - Used by the credential provider to vend AWS credentials to Greengrass components
  • IAM role - Referred to by the role alias, contains the inline IAM policy of minimal permissions for the Greengrass core device
  • Greengrass component - An example component (recipe and artifacts) deployable to Greengrass cores. Part of the Greengrass deployment
  • Greengrass deployment - A list of public and private components targeting the members of the thing group

Once this is fully deployed, a fully functional AWS IoT Greengrass environment will be running as a Docker container, with the components deployed and running on the Greengrass core device.

All other version 2 accelerators leverage the resources created by this base stack, then extend them by creating additional components, a thing group specific to the accelerator, and a deployment that targets the new thing group.

Base Implementation Use Case

This accelerator demonstrates the ability to automate the creation and deployment of resources for a complete Greengrass implementation. It is a prerequisite for all other version 2 accelerators, utilizing the same Greengrass core device.

Design Pattern

The following architecture shows the process flow for deploying the accelerator.

Base Implementation Process Steps

  1. CDK command line interface (CLI) is used locally to create the CloudFormation stack and deploy to the cloud.
  2. The stack creates all the resources for the accelerator, where the Deployment targets the Thing Group that includes the Greengrass core device.
  3. Locally, the config_docker.py script is run to populate the certs and config directories from the local CDK stack and CloudFormation output in the cloud.
  4. When the local Docker container is started, it reads the configuration settings, does an initial setup and configuration of Greengrass within the container, and then receives the Deployment from the cloud. Once all components are downloaded, installed, and operational, the local Docker container continues to run.

Folder Structure

├── base
│   ├── README.md         <--- this file
│   ├── cdk               <--- builds and deploys CloudFormation to cloud
│   │   ├── bin
│   │   ├── cdk.json
│   │   ├── components
│   │   ├── jest.config.js
│   │   ├── lib
│   │   ├── package-lock.json
│   │   ├── package.json
│   │   ├── test
│   │   └── tsconfig.json
│   ├── docker            <--- configures and runs Greengrass as a container
│   └── docs
├── cdk-constructs        <--- common CDK constructs used by all accelerators

There are two main aspects to using the accelerator. First, from the cdk/ directory, the CDK CLI is used to build and deploy a complete AWS IoT Greengrass stack in the cloud. Once complete, all resources exist in the cloud and are available for the next step which is configuring and running Greengrass locally.

Once the stack has been successfully deployed, the second step is running commands from the docker/ directory . First, running python config_docker.py(or python3) will populate the initial configuration file and download the credentials for AWS IoT Greengrass to run. Then, docker compose up will start the core device and complete the deployment of resources.

Deploying the Accelerator

NOTE: All accelerators use the same structure and steps to deploy, even if the actual output of the steps differ slightly.

❗ Although not recommended, this stack can be deployed more than once--but requires changing the stack name and modifying the resources deployed. Please see the Frequently Asked Questions section for more details.

This accelerator is designed to deploy as a set of AWS CloudFormation resources as a parent stack in the cloud, and then run AWS IoT Greengrass as a Docker container on your local system or through AWS Cloud9. This provides a consistent approach to testing or investigating functionality without impacting or leaving behind unneeded artifacts locally, or in the cloud when the CloudFormation stack is deleted/destroyed. To launch this accelerator as a Docker container, there are a few prerequisites and steps to complete.

It is assumed you have basic experience with AWS IoT via the console and have familiarity with the command line interface (CLI).

Prerequisites

The following is a list of prerequisites to deploy the accelerator:

  1. Ensure you have an AWS user account with permissions to create and manage resources. An Identity and Access Management (IAM) user with the Administrator role meets this requirement.
  2. Install the AWS CLI locally and create a named profile with credentials for the AWS user account. For Cloud9, there is a profile already created named default that will have your accounts permissions.
  3. Install the AWS Cloud Development Kit and perform a bootstrap in the region you will be deploying the accelerator.
  4. Verify Docker Desktop for macOS or Microsoft Windows installed, or Docker Engine for Linux. Ensure you have the ability to create or download Docker images locally and run containers. The Docker Compose tool is required, either from Docker Desktop or installed separately.

There are two installation and deployment methods outlines below:

Create and Launch the Accelerator Locally

This approach uses your local system for installation and running the accelerator. It requires certain pre-requisites to be installed. If you like to run from a consistent environment, see the next section for deploying using AWS Cloud9.

Click here to show/hide the steps to run locally

💡 These steps assume familiarity with installation of NPM/NPX, Nodejs packages, Python, and working from the command line interface (CLI).

  1. Install and bootstrap the CDK:

    npm install -g aws-cdk
    # replace PROFILE_NAME with your specific AWS CLI profile that has username and region defined
    export CDK_DEPLOY_ACCOUNT=$(aws sts get-caller-identity --profile PROFILE_NAME --query Account --output text)
    # Set REGION to where the bootstrap resources will be deployed
    export CDK_DEPLOY_REGION=us-east-1
    # replace PROFILE_NAME with your specific AWS CLI profile that has username and region defined
    cdk --profile PROFILE_NAME bootstrap aws://$CDK_DEPLOY_ACCOUNT/$CDK_DEPLOY_REGION
  2. Clone the repository, change into the cdk/ directory for the accelerator, then build and deploy the CloudFormation stack:

    NOTE: The first time cdk deploy is run, it will take longer as there are Docker images required to build some of the resources. You will see Docker-related messages in your terminal session. Subsequent runs will only take a few additional seconds.

    git clone https://github.com/awslabs/aws-iot-greengrass-accelerators.git
    cd aws-iot-greengrass-accelerators/v2/base/cdk
    npm install
    npm run build
    # replace PROFILE_NAME with your specific AWS CLI profile that has username and region defined
    cdk --profile PROFILE_NAME deploy

    The result of a successful deployment will look similar to this:

     ✅  gg-accel-base
    
    
    Outputs:
    gg-accel-base.CertificateArn = arn:aws:iot:us-west-2:123456789012:cert/d6df4cea2a08c73e041e62850b6f5cb7884e7ca4ec0e80f8383539504a8690b4
    gg-accel-base.CertificatePemParameter = /gg-accel-base/gg-accel-base-greengrass-core-kzH8xKQc/certificate_pem
    gg-accel-base.ComponentBucketArn = arn:aws:s3:::gg-accel-base-ggcomponents-123456789012-us-west-2
    gg-accel-base.CredentialProviderEndpointAddress = c3XXXXXXXXf1qf.credentials.iot.us-west-2.amazonaws.com
    gg-accel-base.DataAtsEndpointAddress = a2l4icd3ffczky-ats.iot.us-west-2.amazonaws.com
    gg-accel-base.IamRoleArn = arn:aws:iam::123456789012:role/gg-accel-base-RoleForIoTRoleAlias-kzH8xKQc
    gg-accel-base.IotPolicyArn = arn:aws:iot:us-west-2:123456789012:policy/gg-accel-base-greengrass-minimal-policy-kzH8xKQc
    gg-accel-base.IotRoleAliasName = gg-accel-base-GreengrassV2TokenExchangeRole-kzH8xKQc
    gg-accel-base.PrivateKeySecretParameter = /gg-accel-base/gg-accel-base-greengrass-core-kzH8xKQc/private_key
    gg-accel-base.RoleAliasArn = arn:aws:iot:us-west-2:123456789012:rolealias/gg-accel-base-GreengrassV2TokenExchangeRole-kzH8xKQc
    gg-accel-base.ThingArn = arn:aws:iot:us-west-2:123456789012:thing/gg-accel-base-greengrass-core-kzH8xKQc
    gg-accel-base.ThingName = gg-accel-base-greengrass-core-kzH8xKQc
    Stack ARN:
    arn:aws:cloudformation:us-west-2:123456789012:stack/gg-accel-base/c358ee30-db73-11ec-b8f0-06957ff291a7
  3. At this point the CloudFormation stack is successfully deployed. Next, change to the docker/ directory and run the configuration script python3 config_docker.py (or python3), which will:

    1. Read the local CDK output to determine the CloudFormation stack name
    2. From the CloudFormation stack output, read the values to:
      1. Download and save the X.509 certificate, private key, and Amazon CA1 certificate to the volumes/certs directory
      2. Using the stack outputs, generate and save an initial configuration file to volumes/config
    cd ../docker
    # Use the same AWS CLI profile from above
    python3 docker_config.py --profile PROFILE_NAME

    NOTE: The first time python3 config_docker.py is run, the script will download the Greengrass nucleus assets and create a Dockerfile to build the Greengrass image locally. The Greengrass images are no longer published to public repositories. It will take additional time to run depending upon network speed and local system performance.

  4. Next, start the Greengrass core device. If you intend to run Greengrass on a physical device, copy the contents of the volumes/certs and the volumes/config directories to your physical device and provision and install Greengrass per the public documentation.

    NOTE: The first time docker compose up is run, it will take longer to startup as Docker will need to download the AWS IoT Greengrass image, components, and then complete the installation and configuration. Subsequent runs will only take a few seconds to complete.

    # Depending upon docker version, this may also be docker-compose
    docker compose up

At this point, the CloudFormation stack has been deployed and the AWS IoT Greengrass container is running. Continue to the Investigating the Accelerator section for your accelerator for additional details.

The volumes/gg_root will contain the various Greengrass core device files. Of interest will be the log files located in volumes/gg_root/logs.

Step-by-Step: Create and Launch the Accelerator via AWS Cloud9

Click here to show/hide the steps to run via AWS Cloud9

💡 All steps below use a Cloud9 IDE in the same account and region where the accelerator will be run.

Prior to launching the accelerator as a container on AWS Cloud9, the AWS CDK is used to generate a CloudFormation template and deploy it. From the Cloud9 console, follow these steps to create and launch the stack via CDK.

  1. Create a new Cloud9 IDE and make sure that Amazon Linux 2 and t3.medium or larger is selected. At least 4GiB of RAM is needed for CDK to complete installation and building of the assets for deployment.

    ❗The following steps will not work if Ubuntu is selected.

  2. Once the Cloud9 environment starts, follow these steps to resize the disk. Create the resize.sh file and run bash resize.sh 40 to extend the disk to 40GiB.

  3. Pre-requisites (only needs be run once and the Cloud9 environment will reboot) - From the Cloud9 IDE, open a new Window->New Terminal window and run these commands:

    # Cloud9 Commands - change as needed for local development environment
    # Install pre-requisites, bootstrap CDK for use in account/region, and reboot
    npm uninstall -g cdk
    npm install -g aws-cdk@latest
    npm install -g npm
    sudo yum install iptables-services -y
    # Bootstrap CDK for current AWS account and region where Cloud9 runs
    ACCOUNT=$(aws sts get-caller-identity --query Account --output text)
    REGION=$(aws configure get region)
    cdk bootstrap aws://$ACCOUNT/$REGION
    sudo curl -L "https://github.com/docker/compose/releases/download/1.25.3/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose
    sudo chmod +x /usr/local/bin/docker-compose
    pip3 install --user boto3
    # Enable soft/hard links
    sudo cat <<EOF | sudo tee /etc/sysctl.d/98-cloud9-greengrass.conf
    fs.protected_hardlinks = 1
    fs.protected_symlinks = 1
    EOF
    # Complete, reboot environment
    sudo reboot
  4. Build and deploy. Once Cloud9 has restarted, issue these commands. If the terminal window is unresponsive, open a new one.

    NOTE: The first time cdk deploy is run, it will take longer as there are Docker images required to build some of the resources. You will see a lot of Docker-related messages in your terminal session as it downloads and builds the initial resources. Subsequent runs will only take a few additional seconds.

    Run each command separately to get an understanding

    # After reboot open a new terminal window and issue these commands
    # NOTE: If terminal window spins when restarted, close the terminal window and launch a new one
    cd ~/environment
    # Clone the repository
    git clone https://github.com/awslabs/aws-iot-greengrass-accelerators.git
    # Change path to accelerator you wish to run. This accelerator, "base",
    # is shown below
    cd ~/environment/aws-iot-greengrass-accelerators/v2/base/cdk
    
    # Build and deploy the CDK (CloudFormation stack)
    npm install
    npm run build
    cdk deploy
    
    # Acknowledge the creation above, then change to docker and configure initial parameters
    cd ../docker
    python3 config_docker.py --profile default
    
    # Build and start the Greengrass docker container. First time will take longer to build and deploy resources
    docker-compose up
  5. At this point, the CloudFormation stack has been deployed and the Greengrass container is running as a foreground process in the terminal. The CloudFormation stack will also trigger an initial deployment of all resources to the Greengrass Core, so the Lambda functions, Stream Manager, and docker containers are also running.

  6. ❗ The Docker containers run as the root process in Cloud9 (and other Linux environments). If you wish to look at log or deployment files locally, it is easiest to launch another terminal tab and set the user to root:

    sudo su -
    cd /home/ec2-user/environment/aws-iot-greengrass-accelerators/v2/base/docker/volumes/gg_root/logs
    # You can now cat|more|less|tail files from here
    tail -F greengrass.log
    ...

Investigating the Accelerator

As a base accelerator, the main interaction with this accelerator is to ensure it is running correctly, understand the directory structure, and access various log files. You can review the output of the Hello World example component by looking at the contents of the log file in docker/volumes/gg_root/logs/ggAccel.example.HelloWorld.log.

Other accelerators use this stack's resources to build additional Greengrass components, deployments, and a thing group. When other accelerators are deployed, the Greengrass core device will merge all deployments into a single one.

Accelerator Cleanup

To stop and completely remove this accelerator, follow these steps:

  1. From the command line where Greengrass is running (the docker-compose command was started), either locally on in Cloud 9, stop the Greengrass container byt entering CTRL+C and then:

    docker-compose down
  2. With the container stopped, change to the CDK directory and issue the command to destroy the CloudFormation stack:

    cd ../cdk
    # For Cloud9
    cdk destroy
    # For locally running (replace PROFILE_NAME with one used to create stack)
    cdk --profile PROFILE_NAME destroy

    If there are any errors abut not being able to completely destroy the stack, review the error messages and resolve. In most cases it will be related to assets that may have been modified. Once resolve, run the cdk destroy command again, or delete the stack from the CloudFormation console.

    Also, at this point the certificate and private key are no longer valid in the Greengrass docker/volumes/certs directory. If you wish to redeploy the stack, you can run the python3 config_docker.py --clean command to remove all configuration and Greengrass files.

  3. Review any CloudWatch Logs log groups and delete these if needed.

  4. Finally, change out of the GitHub repository and fully delete the directory.

That's it! Fully deployed, ran, and cleaned up!

Frequently Asked Questions

How can I use this accelerator beyond the "Hello World" example component?

By creating a single core device, thing group, and deployment, you can create additional components external to the accelerator and then revise the Greengrass deployment that targets the thing group. This could be for cloud deployment testing or to investigate operationally how Greengrass handles deployment updates.

How can I view the log file /tmp/Greengrass_HelloWorld.log referred to in the "Hello World" application code?

The full container file system can be accessed by executing a shell within the running container. To do so, with the container running via docker compose up, open a new terminal window locally or in Cloud9 and run these commands:

# Verify container name (default: greengrass-accel)
docker ps
CONTAINER ID   IMAGE          COMMAND                  CREATED      STATUS          PORTS                                       NAMES
fd53c272f65d   dafe85b8555c   "/greengrass-entrypo…"   2 days ago   Up 11 minutes   0.0.0.0:8883->8883/tcp, :::8883->8883/tcp   greengrass-accel

# Execute shell within the container ("bash-4.2#" is the shell in the container)
docker exec -it greengrass-accel /bin/bash
bash-4.2# cd /tmp
bash-4.2# cat Greengrass_HelloWorld.log
Hello, Welcome from the Greengrass accelerator stack! Current time: 2021-08-03 00:49:59.275648.
Hello, Welcome from the Greengrass accelerator stack! Current time: 2021-08-05 13:33:53.487968.
# Exit command will exit from the container back to your local terminal
bash-4.2# exit
exit

How can I deploy the base stack more than once?

It is not recommended to deploy a second stack as-is without making changes first. Although most resources have unique names, the Greengrass component, ggAccel.example.HelloWorld is not unique across deployments of the stack. If a second instance of the base accelerator is required, follow these steps:

  1. Clone the repository and run the new deployment from that directory structure. Doing this will keep the deployed stack details separate and not require passing in the stack name for any other CDK commands.

  2. In the lib/BaseImplementationStack.ts file, remove the creation of the helloWorldComponent component, and adjust the greengrassDeployment construct. Failing to do so will result in the stack failing due to duplicate components.

  3. By default, the AWS CloudFormation stack name of gg-accel-base is used. Deploy the copied stack with a unique stack name by setting the environment variable STACK_NAME to a new name to use for the stack.

    $ STACK_NAME="gg-accel-base-unique" cdk deploy --profile PROFILE_NAME
    >>> build steps <<<
    (NOTE: There may be security-related changes not in this list. See https://github.com/aws/aws-cdk/issues/1299)
    
    Do you wish to deploy these changes (y/n)? y
    ...

Once completed, the new stack will be available. To destroy the stack, pass in the same STACK_NAME value and run the command STACK_NAME="gg-accel-unique" cdk destroy --profile PROFILE_NAME

Implementation Notes

The Hello World component is a minimal python script to demonstrate component creation and deployment. In a normal operation, it will run once upon deployment or core device startup and create a log file in /tmp along with entries in the log file folder.