Production-ready Serverless

on AWS

James Allardice

orangejellyfish

James Allardice

The Serverless framework

James Allardice

The Serverless framework

James Allardice

Production-ready?

James Allardice

Maintainable

James Allardice

Performant

James Allardice

Scalable

James Allardice

Secure

James Allardice

> npm i -g serverless

James Allardice

> npm i -g serverless
> serverless create --template aws-nodejs
# Welcome to Serverless!
#
# This file is the main config file for your service.
# It's very minimal at this point and uses default values.
# You can always add more config options for more control.
# We've included some commented out config examples here.
# Just uncomment any of them to get that config option.
#
# For full config options, check the docs:
#    docs.serverless.com
#
# Happy Coding!

service: aws-nodejs # NOTE: update this with your service name

# You can pin your service to only deploy with a specific Serverless version
# Check out our docs for more details
# frameworkVersion: "=X.X.X"

James Allardice

# Welcome to Serverless!
#
# This file is the main config file for your service.
# It's very minimal at this point and uses default values.
# You can always add more config options for more control.
# We've included some commented out config examples here.
# Just uncomment any of them to get that config option.
#
# For full config options, check the docs:
#    docs.serverless.com
#
# Happy Coding!

service: aws-nodejs # NOTE: update this with your service name

# You can pin your service to only deploy with a specific Serverless version
# Check out our docs for more details
frameworkVersion: "=1.37.0"

James Allardice

James Allardice

⚙️

🚀

⚖️

🔒

# serverless.yml
provider:
  name: aws
  runtime: nodejs8.10

James Allardice

James Allardice

⚙️

🚀

⚖️

🔒

# serverless.yml
plugins:
  - serverless-webpack

James Allardice

James Allardice

⚙️

🚀

⚖️

🔒

// .babelrc
{
  "presets": [
    [
      "@babel/preset-env",
      {
        "targets": {
          "node": "8.10"
        }
      }
    ]
  ]
}

James Allardice

const nodeExternals = require('webpack-node-externals');
const slsw = require('serverless-webpack');

module.exports = {
  entry: slsw.lib.entries,
  target: 'node',
  mode: 'production',
  externals: [nodeExternals()],
  module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /node_modules/,
        use: {
          loader: 'babel-loader',
        },
      },
    ],
  },
};

James Allardice

James Allardice

⚙️

🚀

⚖️

🔒

James Allardice

plugins:
  - serverless-webpack
  - serverless-iam-roles-per-function

James Allardice

functions:
  hello:
    handler: handler.hello
    events:
      - http:
          path: hello
          method: get

    iamRoleStatements:
      - Effect: Allow
        Action:
          - dynamodb:PutItem
        Resource:
          - Fn::GetAtt: [ MyTable, Arn ]
      - Effect: Allow
        Action:
          - sns:publish
        Resource:
          - Ref: MyTopic

James Allardice

⚙️

🚀

⚖️

🔒

James Allardice

Resources:

  # A custom API Gateway account to allow us to enable request logging.
  ApiGatewayAccount:
    Type: AWS::ApiGateway::Account
    Properties:
      CloudWatchRoleArn:
        Fn::GetAtt: [ RoleAPIGatewayCloudWatch, Arn ]

  # A custom API Gateway stage (usually handled automatically by the Serverless
  # framework) so that we can turn on CloudWatch request logging. The
  # "MethodSettings" of a stage apply to all methods (effectively all endpoints)
  # within that stage. The ApiGatewayRestApi resource is generated automatically
  # by the Serverless framework.
  ApiGatewayStage:
    Type: AWS::ApiGateway::Stage
    Properties:
      RestApiId:
        Ref: ApiGatewayRestApi
      MethodSettings:
        - DataTraceEnabled: true
          HttpMethod: "*"
          LoggingLevel: INFO
          ResourcePath: /*
          MetricsEnabled: true

Stage

Account

Resources:

  # An IAM role to allow API Gateway to log requests to CloudWatch.
  RoleAPIGatewayCloudWatch:
    Type: AWS::IAM::Role
    Properties:
      AssumeRolePolicyDocument:
        Version: "2012-10-17"
        Statement:
          -
            Effect: Allow
            Principal:
              Service:
                - apigateway.amazonaws.com
            Action:
              - sts:AssumeRole
      ManagedPolicyArns:
        - arn:aws:iam::aws:policy/service-role/AmazonAPIGatewayPushToCloudWatchLogs

James Allardice

Apply to API Gateway

James Allardice

⚙️

🚀

⚖️

🔒

functions:
  getTweets:
    handler: tweets.get
    events:
      - schedule: rate(5 minutes)
    environment:
      TWITTER_ACCESS_TOKEN: "xxxxx"

James Allardice

:(

functions:
  getTweets:
    handler: tweets.get
    events:
      - schedule: rate(5 minutes)
    environment:
      TWITTER_ACCESS_TOKEN: ${ssm:TWITTER_ACCESS_TOKEN}

James Allardice

James Allardice

⚙️

🚀

⚖️

🔒

# Lambda function definitions. Each function defines itself in a separate YAML
# file that resides alongside the function implementation. Unfortunately we have
# to duplicate the function name here due to the way Serverless YAML importing
# and parsing works.
functions:
  hello: ${file(src/functions/hello/index.yml):hello}

James Allardice

# src/functions/hello/handler.js
hello:
  handler: src/functions/hello/handler.hello
  events:
    - http:
        path: hello
        method: get

  iamRoleStatements:
    - Effect: Allow
      Action:
        - dynamodb:PutItem
      Resource:
        - Fn::GetAtt: [ MyTable, Arn ]
    - Effect: Allow
      Action:
        - sns:publish
      Resource:
        - Ref: MyTopic

James Allardice

James Allardice

⚙️

🚀

⚖️

🔒

# Custom AWS resource definitions. This includes anything that the Serverless
# framework doesn't handle for us, such as database setup, and overrides of
# Serverless defaults.
resources:
  - ${file(src/resources/dynamodb.yml)}
  - ${file(src/resources/sns.yml)}

James Allardice

Resources:

  MyTable:
    Type: AWS::DynamoDB::Table
    Properties:
      TableName: ${self:service}-${self:custom.stage}-my-table
      AttributeDefinitions:
        - AttributeName: id
          AttributeType: S
      KeySchema:
        - AttributeName: id
          KeyType: HASH
      BillingMode: PAY_PER_REQUEST

James Allardice

Resources:

  MyTable:
    Type: AWS::DynamoDB::Table
    Properties:
      TableName: ${self:service}-${self:custom.stage}-my-table
      AttributeDefinitions:
        - AttributeName: id
          AttributeType: S
      KeySchema:
        - AttributeName: id
          KeyType: HASH
      BillingMode: PAY_PER_REQUEST

James Allardice

James Allardice

⚙️

🚀

⚖️

🔒

⚙️

🚀

⚖️

🔒

James Allardice

GitHub!

James Allardice

What next?

Thank you!

James Allardice

orangejellyfish

Slides: https://slides.com/jamesallardice/production-ready-serverless-on-aws

Production-ready Serverless on AWS (Serverless London, Feb 2019)

By James Allardice

Production-ready Serverless on AWS (Serverless London, Feb 2019)

  • 928