Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/jonashackt/pulumi-typescript-aws-fargate

Example project showing how to run Dockerized spring-boot-vuejs with Pulumi on AWS Fargate based on AWS ECS
https://github.com/jonashackt/pulumi-typescript-aws-fargate

aws-ecs aws-fargate ecs-fargate loadbalancer nginx pulumi spring-boot-vuejs typescript

Last synced: 26 days ago
JSON representation

Example project showing how to run Dockerized spring-boot-vuejs with Pulumi on AWS Fargate based on AWS ECS

Awesome Lists containing this project

README

        

# pulumi-typescript-aws-fargate
[![Build Status](https://travis-ci.org/jonashackt/pulumi-typescript-aws-fargate.svg?branch=master)](https://travis-ci.org/jonashackt/pulumi-typescript-aws-fargate)
[![License](http://img.shields.io/:license-mit-blue.svg)](https://github.com/jonashackt/pulumi-typescript-aws-fargate/blob/master/LICENSE)
[![renovateenabled](https://img.shields.io/badge/renovate-enabled-yellow)](https://renovatebot.com)
[![versionpulumi](https://img.shields.io/github/package-json/dependency-version/jonashackt/pulumi-typescript-aws-fargate/@pulumi/pulumi?color=brightgreen&label=pulumi)](https://www.pulumi.com/)
[![versionpulumiaws](https://img.shields.io/github/package-json/dependency-version/jonashackt/pulumi-typescript-aws-fargate/@pulumi/aws?color=brightgreen)](https://www.pulumi.com/docs/reference/pkg/nodejs/pulumi/aws/)
[![versionpulumiawsx](https://img.shields.io/github/package-json/dependency-version/jonashackt/pulumi-typescript-aws-fargate/@pulumi/awsx?color=brightgreen)](https://www.pulumi.com/docs/reference/pkg/nodejs/pulumi/awsx/)
[![versiontypescript](https://img.shields.io/github/package-json/dependency-version/jonashackt/pulumi-typescript-aws-fargate/dev/@types/node?label=@types/node)](https://github.com/DefinitelyTyped/DefinitelyTyped)

Example project showing how to run Dockerized spring-boot-vuejs with Pulumi on AWS Fargate based on AWS ECS

There's some confusion about AWS ECS, EKS & Fargate - try to [read posts like this](https://cloudonaut.io/ecs-vs-fargate-whats-the-difference/).

The first contact with AWS Fargate is the AWS ECS console - access it via https://eu-central-1.console.aws.amazon.com/ecs/home?region=eu-central-1#/firstRun There's also a nice overview picture waiting:

![aws-ecs-fargate-intro](screenshots/aws-ecs-fargate-intro.png)

Pulumi also provides an example tutorial:

https://www.pulumi.com/docs/tutorials/aws/ecs-fargate/

### Pulumi AWS Crosswalk

https://www.pulumi.com/docs/guides/crosswalk/aws/

> A collection of libraries that use automatic well-architected best practices to make common infrastructure-as-code tasks in AWS easier and more secure

Import them by using the `awsx` packages:

```
import * as awsx from "@pulumi/awsx";
```

### Start with Pulumi template for Typescript & AWS

Let's create a Pulumi project by using a template again (remember to use a new directory with `mkdir dir`, otherwise pulumi won't generate the template project):

```
pulumi new aws-typescript --name pulumi-typescript-aws-fargate
```

This will result in some questions about your preferred project description, Pulumi stack name and the AWS region:

```
$ pulumi new aws-typescript --name pulumi-typescript-aws-fargate

This command will walk you through creating a new Pulumi project.

Enter a value or leave blank to accept the (default), and press .
Press ^C at any time to quit.

project description: (A minimal AWS TypeScript Pulumi program) Example project showing how to run Dockerized spring-boot-vuejs with Pulumi on AWS Fargate based on AWS ECS
Created project 'pulumi-typescript-aws-fargate'

Please enter your desired stack name.
To create a stack in an organization, use the format / (e.g. `acmecorp/dev`).
stack name: (dev) dev
Created stack 'dev'

aws:region: The AWS region to deploy into: (us-east-1) eu-central-1
Saved config

Installing dependencies...


> [email protected] install /Users/jonashecht/dev/pulumi-typescript-aws-fargate/new/node_modules/deasync
> node ./build.js

`darwin-x64-node-12` exists; testing
Binary is fine; exiting

> [email protected] install /Users/jonashecht/dev/pulumi-typescript-aws-fargate/new/node_modules/grpc
> node-pre-gyp install --fallback-to-build --library=static_library

[grpc] Success: "/Users/jonashecht/dev/pulumi-typescript-aws-fargate/new/node_modules/grpc/src/node/extension_binary/node-v72-darwin-x64-unknown/grpc_node.node" is installed via remote

> @pulumi/[email protected] install /Users/jonashecht/dev/pulumi-typescript-aws-fargate/new/node_modules/@pulumi/aws
> node scripts/install-pulumi-plugin.js resource aws v1.7.0

[resource plugin aws-1.7.0] installing

> @pulumi/[email protected] install /Users/jonashecht/dev/pulumi-typescript-aws-fargate/new/node_modules/@pulumi/docker
> node scripts/install-pulumi-plugin.js resource docker v0.17.4

[resource plugin docker-0.17.4] installing
Downloading plugin: 17.29 MiB / 17.29 MiB [========================] 100.00% 16s
Moving plugin... done.

> [email protected] postinstall /Users/jonashecht/dev/pulumi-typescript-aws-fargate/new/node_modules/protobufjs
> node scripts/postinstall

added 197 packages from 295 contributors and audited 1566 packages in 31.223s
found 0 vulnerabilities

Finished installing dependencies

Your new project is ready to go! ✨

To perform an initial deployment, run 'pulumi up'
```

Now the project should be generated by Pulumi.

### Create a AWS loadbalancer & Fargate service from our app's Docker image

As we're using Typescript, we're coding inside the [index.ts](index.ts) file:

```typescript
import * as awsx from "@pulumi/awsx";

// Create a load balancer to listen for requests and route them to the container.
let loadbalancer = new awsx.lb.ApplicationListener("nginx", { port: 8098, protocol: "HTTP" });

// Define the service using the example Spring Boot Vue.js app from https://github.com/jonashackt/spring-boot-vuejs
let service = new awsx.ecs.FargateService("spring-boot-vuejs", {
taskDefinitionArgs: {
containers: {
nginx: {
image: "jonashackt/spring-boot-vuejs",
memory: 512,
portMappings: [ loadbalancer ],
},
},
},
desiredCount: 2,
});

// Export the URL so we can easily access it.
export const url = loadbalancer.endpoint.hostname;
```

Now try to fire it up with `pulumi up`! :)

But this time Pulumi showed the power of it's AWS crosswalk libraries! Pulumi creates all needed AWS ECS cluster SecurityGroups from egress over ingress to ingress for cluster-ssh. But all of these are also needed for loadbalancer, which is based on AWS EC2. And finally Pulumi creates the AWS Fargate Service and Taskdefintion incl. IAM roles, policies and CloudWatch log group:

```
$ pulumi up

Please choose a stack, or create a new one: dev
Previewing update (dev):

Updating (dev):

Type Name Status Info
+ pulumi:pulumi:Stack pulumi-typescript-aws-fargate-dev created
+ ├─ awsx:x:ec2:SecurityGroup nginx created
+ │ └─ aws:ec2:SecurityGroup nginx created
+ ├─ awsx:x:ec2:Vpc default-vpc-accd9bc5 created
+ │ ├─ awsx:x:ec2:Subnet default-vpc-accd9bc5-public-1 created
+ │ └─ awsx:x:ec2:Subnet default-vpc-accd9bc5-public-0 created
+ ├─ awsx:x:ecs:FargateService spring-boot-vuejs created
+ │ └─ aws:ecs:Service spring-boot-vuejs created
+ ├─ awsx:x:ecs:FargateTaskDefinition spring-boot-vuejs created
+ │ ├─ aws:cloudwatch:LogGroup spring-boot-vuejs created
+ │ ├─ aws:iam:Role spring-boot-vuejs-execution created
+ │ ├─ aws:iam:Role spring-boot-vuejs-task created
+ │ ├─ aws:iam:RolePolicyAttachment spring-boot-vuejs-execution-9a42f520 created
+ │ ├─ aws:iam:RolePolicyAttachment spring-boot-vuejs-task-32be53a2 created
+ │ ├─ aws:iam:RolePolicyAttachment spring-boot-vuejs-task-fd1a00e5 created
+ │ └─ aws:ecs:TaskDefinition spring-boot-vuejs created
+ ├─ awsx:x:ecs:Cluster default-cluster created
+ │ ├─ awsx:x:ec2:SecurityGroup default-cluster created
+ │ │ ├─ awsx:x:ec2:IngressSecurityGroupRule default-cluster-ssh created
+ │ │ │ └─ aws:ec2:SecurityGroupRule default-cluster-ssh created
+ │ │ ├─ awsx:x:ec2:IngressSecurityGroupRule default-cluster-containers created
+ │ │ │ └─ aws:ec2:SecurityGroupRule default-cluster-containers created
+ │ │ ├─ awsx:x:ec2:EgressSecurityGroupRule default-cluster-egress created
+ │ │ │ └─ aws:ec2:SecurityGroupRule default-cluster-egress created
+ │ │ └─ aws:ec2:SecurityGroup default-cluster created
+ │ └─ aws:ecs:Cluster default-cluster created
+ └─ aws:lb:ApplicationLoadBalancer nginx created
+ ├─ awsx:lb:ApplicationTargetGroup nginx created
+ │ └─ aws:lb:TargetGroup nginx created
+ ├─ awsx:lb:ApplicationListener nginx created
+ │ ├─ awsx:x:ec2:IngressSecurityGroupRule nginx-external-0-ingress created
+ │ │ └─ aws:ec2:SecurityGroupRule nginx-external-0-ingress created
+ │ ├─ awsx:x:ec2:EgressSecurityGroupRule nginx-external-0-egress created
+ │ │ └─ aws:ec2:SecurityGroupRule nginx-external-0-egress created
+ │ └─ aws:lb:Listener nginx created
+ └─ aws:lb:LoadBalancer nginx created

Outputs:
url: "nginx-a935168-1026809480.eu-central-1.elb.amazonaws.com"

Resources:
+ 36 created

Duration: 4m14s

Permalink: https://app.pulumi.com/jonashackt/pulumi-typescript-aws-fargate/dev/updates/3
```

This will take a while, you can have a look into the AWS ECS console at https://eu-central-1.console.aws.amazon.com/ecs/home?region=eu-central-1#/clusters and see Pulumi "working":

![aws-ecs-fargate-console](screenshots/aws-ecs-fargate-console.png)

Also the `Task Definitions` give fine grained details for your app.

Have a look at the containers' logs, there you should see the example app [spring-boot-vuejs](https://github.com/jonashackt/spring-boot-vuejs) running:

```
pulumi logs --follow
```

Now access the provides url nginx-a935168-1026809480.eu-central-1.elb.amazonaws.com:8098 with the port defined inside the `loadbalancer` defintion and you should see our app:

![running-spring-boot-vuejs-aws-fargate](screenshots/running-spring-boot-vuejs-aws-fargate.png)

### Run Typescript-based Pulumi on TravisCI

The example project [pulumi-python-aws-ansible](https://github.com/jonashackt/pulumi-python-aws-ansible) shows already how to [run a Python-based Pulumi project on TravisCI](https://github.com/jonashackt/pulumi-python-aws-ansible#run-pulumi-with-travis).

But here we'll need the Typescript/Javascript variant! The Travis [docs provide a guide showing how to build Node.js applications](Building a JavaScript and Node.js project), so let's create a [.travis.yml](.travis.yml):

```yaml
language: node_js
node_js:
- 12.13.0

install:
# Install Pulumi SDK with the installation script from https://www.pulumi.com/docs/get-started/install/#installation-script
- curl -fsSL https://get.pulumi.com | sh
# Add Pulumi to Travis' PATH so the executable could be found
- export PATH=$PATH:/home/travis/.pulumi/bin
- pulumi version
```

First we set the Travis `language` to `node_js` incl. a current version of node (which is quite old with `node --version \ v0.10.48`, if we leave out this version definition). We also install Pulumi just as we already know it from the Python-based setup.

Then we install the Node.js dependencies needed by our Pulumi project and configure the AWS CLI:

```yaml
# Install pulumi-aws dependencies (among others like awscli) via npm dependency manager
- npm install
# Check, if Pulumi aws plugin was installed correctly
- pulumi plugin ls
```

Be sure to not forget to define the needed environment variables `AWS_ACCESS_KEY` and `AWS_SECRET_KEY` inside the TravisCI project setup at https://travis-ci.org/jonashackt/pulumi-typescript-aws-fargate/settings . The cool thing here about Pulumi is, that [those two environment variables sufficient to work with AWS](https://www.pulumi.com/docs/intro/cloud-providers/aws/setup/#environment-variables)! There's no extra AWS CLI installation needed - although the docs recommend using the Shared Credentials File created by AWS CLI installation & configuration. But with a Typescript-based Pulumi setup, we then would not only need to managed Node.js packages, but also a Python installation and PIP packages.

Don't forget to also define the `PULUMI_ACCESS_TOKEN` variable for the next step.

Finally we should be able to fire up our Pulumi AWS Fargate deployment:

```yaml
script:
# login to app.pulumi.com with the predefined PULUMI_ACCESS_TOKEN
- pulumi login

# Select your Pulumi projects' stack
- pulumi stack select dev

# Create AWS Fargate application deployment
- pulumi up --yes

# Destroy Pulumi setup after successful creation
- pulumi destroy --yes
```

Now every push to the repository will install Pulumi and it's dependencies and then spin up the complete setup - and tear it down in the end, if everything went fine.