VOIDKAT

Using Serverless on AWS - Saving to S3 and DynamoDb - Part 1/2

October 21, 2019

In a previous post Using AWS Lambda to save files to AWS S3 using Node.js we covered using AWS Lambda to create functions to execute a task that usually would be used within a server. Today we will cover the same process but use the popular Serverless framework.

The Serverless framework allows the creation, monitoring and deployment of serverless functions on pretty much all leading cloud providers from AWS to Google to Alibaba.

Due to the authors familiarity with AWS we will be using AWS using Node.js. From here on we will refer to serverless as sls which is the command line shortcut name.

Why use Serverless?

If you can write AWS Lambda functions why would you what to use Serverless? Well simply it cuts down the amount of time spent on configurations and bouncing between multiple screens within the AWS console. Permissions and API gateways are all handled within the sls configuration file. This will become more clear down the line.

Our Example Application

Our test application will take in data from the user: name, email and a base64 image. The intent is to save the base64 image to AWS S3, the user data to AWS DynamoDb. The API will check if the email is unique and refuse to save the data is the email has been used before. This is the basic setup for something like a id card store. The example is trivial but complex enough to fully use serverless.

As such we will need an AWS S3 bucket and a AWS DynamoDb database. Be sure to create both resources in the same region, here we will be use us-east-1.

S3 Bucket configuration

The bucket must be configured to be publicly accessible. This is best done by using a policy:

{
    "Version": "2012-10-17",
    "Id": "MakePublic",
    "Statement": [
        {
            "Sid": "MakePublic",
            "Effect": "Allow",
            "Principal": "*",
            "Action": "s3:GetObject",
            "Resource": "arn:aws:s3:::<YOUR-BUCKET-NAME>/*"
        }
    ]
}

This will allow anyone to get objects (view our images).

DynamoDb configuration

Create an AWS DynamoDb, the only requirement here would be to use email as a Partition Key (to my SQL background this is our primary key).

Writing our Serverless Function.

Now that we have created our resources on AWS, let’s continue to configure our local environment.

Setup prerequisites

To follow this tutorial, you will need the following:

  • Install aws command line installed and configured with your admin rights. SLS still needs access to your AWS stack.
  • Install sls from https://serverless.com/
  • Install node.js v8.10 and up.

Important Concepts & Serverless Commands

When starting out creating serverless functions you will create 2 files, one will be your .js file for the function and the other file will be the serverless.yml.

Your JS file will mimic the same functional code you would write when creating AWS Lambda functions. The more important file is the serverless.yml which will hold your basic function name, resources and API endpoints. Just like permissions with AWS a lot of problems cna be traced back to an improperly configured serverless.yml.

Note: YML is space sensitive the way Python is. Pay attention to your indentations.

More on this as we write out our function.

Useful commands

  • Deploy: sls deploy -v (with verbose flag)
  • Logging: sls logs -f <YOUR_FUNCTION_NAME> -t (this is helpful for testing when deploying on AWS)
  • Testing: sls invoke local -f <YOUR_FUNCTION_NAME> -p mocks/<YOUR_TEST_JSON>.json (helpful to make sure your function works)

Processing incoming events and responding

It is important to remember that incoming requests to our functions are JSON based, but the body is always a JSON string. Like wise when we respond we need to JSON.stringify our body responses.

So requests incoming into the API should be formatted like so:

{
    "body": "{
        \"name\": \"test\",
        \"email\": \"test@test.com\",
        \"data\": \"\"
    }",
}

This is helpful when creating our mock tests. Like wise when responding it is important to respond with a statusCode and JSON.stringify() body message. The hello world SLS example for example is:

  return {
    statusCode: 200,
    body: JSON.stringify(
      {
        message: 'Go Serverless v1.0! Your function executed successfully!',
        input: event,
      },
      null,
      2
    ),
  };

Configure SLS and Hello World example

Run sls command, this will ask if you want to create a new application. Use the prompts to select AWS Node.js, name your project, you can use the Serverless account for monitoring but it is not necessary for this example. Once complete you will have a named directory that holds a index.js and a serverless.yml.

For now let’s just execute this function, run sls deploy. Serverless will now create your function on AWS Lambda. But it will not create any endpoints. Run sls invoke local -f hello, this will locally execute your function which is called hello by default. You should see the following:

{
    "statusCode": 200,
    "body": "{\n  \"message\": \"Go Serverless v1.0! Your function executed successfully!\",\n  \"input\": \"\"\n}"
}

Let’s add an endpoint for our function. Open serverless.yml there is a lot of information here about configuring the YML file. Most of it will be comments, but we will see our service name, our providers (AWS under node) and finally our function and handler. Let’s rename these.

We will change the name of the handler.js to api.js and change the function name to save with that JS file. So module.exports.hello will now be module.exports.save and the function name in the YML will now be:

functions:
  save:
    handler: api.save

Run sls invoke local -f save to check that everything is correct. If you get a Function "save" doesn't exist in this Service error that means you forgot to rename the function and or handler in the .js or .yml files.

Adding an endpoint

To add an endpoint to our small API you will need to add it to your function, so update your YML with:

functions:
  save:
    handler: api.save
    events:
      - http:
          path: users/create
          method: get

The path is the url endpoint and the method is what HTTP protocol you will be using. Run sls deploy again. You should see an endpoint created:

endpoints:
  GET - https://<YOUR_ENDPOINT>.execute-api.us-east-1.amazonaws.com/dev/users/create

Call your API by using curl -GET https://<YOUR_ENDPOINT>.execute-api.us-east-1.amazonaws.com/dev/users/create. You will see a response, the hello world function returns a body with a message and the entire event as it came into the function.

Saving to S3: Permissions

Now let’s try saving to S3, we want to save a base64 image to S3. Let’s do that first before building the rest of the functionality. AWS uses IAM permssions to allow using resources on AWS. Let’s add our S3 bucket. Under provider in the YML add the following:

provider:
  name: aws
  runtime: nodejs8.10
  region: ${self:custom.region}
  iamRoleStatements:
    - Effect: Allow
      Action:
        - s3:PutObject
        - s3:GetObjectAcl
        - s3:PutObjectAcl
      Resource: "arn:aws:s3:::${self:custom.bucket}/*"

The IAM statements translate to, allow to put objects, get objects and put objects with access rights. Pay attention to the resource arn:aws:s3:::${self:custom.bucket}/*, Serverless allows you to use variables. Add above the provider

custom:
  bucket: <YOUR-BUCKET-NAME>
  region: us-east-1

This is every much like ES6 strings using $, so arn:aws:s3:::${self:custom.bucket}/* is arn:aws:s3:::<YOUR-BUCKET-NAME>/*. Likewise for the region variable.

Saving to S3: Saving to S3

Now that permissions are set up. Let’s use our resources, since we are on AWS Lambda we have access to all of the libraries available within the AWS stack.

Let’s add S3 to our api.js, we will need to invoke AWS so add the following at the top.

const AWS = require('aws-sdk');
const S3 = new AWS.S3();

and add the following after the function, this will just log out the event and parse any body into JSON.

console.log("EVENT: \n" + JSON.stringify(event, null, 2));
const data = JSON.parse(event.body);

Since we will be receiving data via POST, change the HTTP method in the serverless.yml to post from get.

For testing what we will do now is recieving a message coming in and save that message to a text file on S3. As such its best to think of your incoming request. Which I think we will do the following, let’s just test echoing the incoming data for now.

curl -POST -d '{"name": "test user","email": "test@email.com"}' https://<YOUR_ENDPOINT>.execute-api.us-east-1.amazonaws.com/dev/users/create

You should see the same event, however now in the body we get:

"body": "{\"name\": \"test user\",\"email\": \"test@email.com\"}",

So we are receiving data correctly.

Saving to S3: Creating a mock test

While we will be developing this function we will need to test it regularly, and calling curl each time will get tedious. Let’s create a mock test. Create a mock directory and create a JSON file that just contains:

{
    "body": "{\"email\": \"test@test.com\"}"
}

Now you can call sls invoke local -f save -p mock/test.json which is a great way to work on the function, instead of deploying and testing every time. You should see:

EVENT: 
{
  "body": "{\"email\": \"test@test.com\"}"
}
{
  "statusCode": 200,
  "body": "{
    \n  \"message\": 
        \"Go Serverless v1.0! Your function executed successfully!\",
    \n  \"input\": 
    {
    \n  
        \"body\": 
        \"{\\\"email\\\": \\\"test@test.com\\\"}\"
    \n 
    }
    \n}"
}

Since we are not saving anything and just echoing whatever body is incoming. Let’s now save.

Saving to S3: Writing to S3

The module exports is an async function, and writing to S3 takes some computational time. As such you should use async / await otherwise your function can execute correctly but fail to actually save. Place this code to save to S3:

// Payload key is the final name
let s3payload = {
    Bucket: '<YOUR-BUCKET-NAME>',
    Key: data.email + '.json',
    Body: data.email,
};

// Try S3 save.
const s3Response = await S3.upload(s3payload).promise();

Finally it’s best to write the function in a try / catch block to catch any errors. Below is the full code so far:

const AWS = require('aws-sdk');
const S3 = new AWS.S3();

module.exports.save = async event => {
  // Log incoming Event
  console.log("EVENT: \n" + JSON.stringify(event, null, 2));

  // parse event.body to JSON
  const data = JSON.parse(event.body);

  try {
    
    // Payload key is the final name
    let s3payload = {
      Bucket: '<YOUR-BUCKET-NAME>',
      Key: data.email + '.json',
      Body: data.email,
    };

    // Try S3 save.
    const s3Response = await S3.upload(s3payload).promise();

    return {
      statusCode: 200,
      body: JSON.stringify(
        {
          message: 'saved!',
          saved: s3Response
        },
        null,
        2
      ),
    };
    
  } catch (error) {
    return {
      statusCode: 500,
      body: JSON.stringify(
        {
          message: 'error!',
          error: error
        },
        null,
        2
      ),
    };

  }
};

Call sls invoke local -f save -p mock/test.json and you should see the following response:

EVENT: 
{
  "body": "{\"email\": \"test@test.com\"}"
}
{
    "statusCode": 200,
    "body": "{
      \n  \"message\": \"saved!\",
      \n  \"saved\": {
      \n  \"ETag\": \"\\\"b642b4217b34b1e8d3bd915fc65c4452\\\"\",
      \n  \"Location\": 
          \"https://<YOUR-BUCKET-NAME>.s3.amazonaws.com/test%40test.com.json\",
      \n  \"key\": 
          \"test@test.com.json\",
      \n    
          \"Key\": \"test@test.com.json\",
      \n  \"Bucket\": 
          \"<YOUR-BUCKET-NAME>\"
      \n  }
      \n}"
}

End of Part 1

This concludes part one of this tutorial. Part 2 will focus on saving to DynamoDb.


Farhad Agzamov

Written by Farhad Agzamov who lives and works in London building things. You can follow him on Twitter and check out his github here