API server with DynamoDB
In this tutorial let's take a look at how we can use it to build a small API that has endpoints to insert and retrieve information, backed by DynamoDB.
The tutorial assumes that you have an AWS and Deno Deploy account.
- Overview
- Setup DynamoDB
- Create a Project in Deno Deploy
- Write the Application
- Deploy the Application
Overview Jump to heading
We're going to build an API with a single endpoint that accepts GET/POST requests and returns appropriate information
# A GET request to the endpoint should return the details of the song based on its title.
GET /songs?title=Song%20Title # '%20' == space
# response
{
  title: "Song Title"
  artist: "Someone"
  album: "Something",
  released: "1970",
  genres: "country rap",
}
# A POST request to the endpoint should insert the song details.
POST /songs
# post request body
{
  title: "A New Title"
  artist: "Someone New"
  album: "Something New",
  released: "2020",
  genres: "country rap",
}
Setup DynamoDB Jump to heading
Our first step in the process is to generate AWS credentials to programmatically access DynamoDB.
Generate Credentials:
- Go to https://console.aws.amazon.com/iam/ and go to "Users" section.
- Click on Create user button, fill the User name field (maybe use
denamo) and select Programmatic access type.
- Click Next
- Select Attach policies directly and search for
AmazonDynamoDBFullAccess. Check the box next to this policy in the results.
- Click Next and Create user
- On the resulting Users page, click through to the user you just created
- Click on Create access key
- Select Application running outside AWS
- Click *Create
- Click Download .csv file to download the credentials you just created.
Create database table:
- Go to https://console.aws.amazon.com/dynamodb and click on Create table button.
- Fill the Table name field with songsand Partition key withtitle.
- Scroll down and click on Create table.
- Once the table is created, click on the table name and find its General information
- Under Amazon Resource Name (ARN) take note of the region of your new table (for example us-east-1).
Write the Application Jump to heading
Create a file called index.js and insert the following:
import {
  json,
  serve,
  validateRequest,
} from "https://deno.land/x/sift@0.6.0/mod.ts";
// AWS has an official SDK that works with browsers. As most Deno Deploy's
// APIs are similar to browser's, the same SDK works with Deno Deploy.
// So we import the SDK along with some classes required to insert and
// retrieve data.
import {
  DynamoDBClient,
  GetItemCommand,
  PutItemCommand,
} from "https://esm.sh/@aws-sdk/client-dynamodb";
// Create a client instance by providing your region information.
// The credentials are obtained from environment variables which
// we set during our project creation step on Deno Deploy.
const client = new DynamoDBClient({
  region: Deno.env.get("AWS_TABLE_REGION"),
  credentials: {
    accessKeyId: Deno.env.get("AWS_ACCESS_KEY_ID"),
    secretAccessKey: Deno.env.get("AWS_SECRET_ACCESS_KEY"),
  },
});
serve({
  "/songs": handleRequest,
});
async function handleRequest(request) {
  // The endpoint allows GET and POST request. A parameter named "title"
  // for a GET request to be processed. And body with the fields defined
  // below are required to process POST request.
  // validateRequest ensures that the provided terms are met by the request.
  const { error, body } = await validateRequest(request, {
    GET: {
      params: ["title"],
    },
    POST: {
      body: ["title", "artist", "album", "released", "genres"],
    },
  });
  if (error) {
    return json({ error: error.message }, { status: error.status });
  }
  // Handle POST request.
  if (request.method === "POST") {
    try {
      // When we want to interact with DynamoDB, we send a command using the client
      // instance. Here we are sending a PutItemCommand to insert the data from the
      // request.
      const {
        $metadata: { httpStatusCode },
      } = await client.send(
        new PutItemCommand({
          TableName: "songs",
          Item: {
            // Here 'S' implies that the value is of type string
            // and 'N' implies a number.
            title: { S: body.title },
            artist: { S: body.artist },
            album: { S: body.album },
            released: { N: body.released },
            genres: { S: body.genres },
          },
        }),
      );
      // On a successful put item request, dynamo returns a 200 status code (weird).
      // So we test status code to verify if the data has been inserted and respond
      // with the data provided by the request as a confirmation.
      if (httpStatusCode === 200) {
        return json({ ...body }, { status: 201 });
      }
    } catch (error) {
      // If something goes wrong while making the request, we log
      // the error for our reference.
      console.log(error);
    }
    // If the execution reaches here it implies that the insertion wasn't successful.
    return json({ error: "couldn't insert data" }, { status: 500 });
  }
  // Handle GET request.
  try {
    // We grab the title form the request and send a GetItemCommand
    // to retrieve the information about the song.
    const { searchParams } = new URL(request.url);
    const { Item } = await client.send(
      new GetItemCommand({
        TableName: "songs",
        Key: {
          title: { S: searchParams.get("title") },
        },
      }),
    );
    // The Item property contains all the data, so if it's not undefined,
    // we proceed to returning the information about the title
    if (Item) {
      return json({
        title: Item.title.S,
        artist: Item.artist.S,
        album: Item.album.S,
        released: Item.released.S,
        genres: Item.genres.S,
      });
    }
  } catch (error) {
    console.log(error);
  }
  // We might reach here if an error is thrown during the request to database
  // or if the Item is not found in the database.
  // We reflect both conditions with a general message.
  return json(
    {
      message: "couldn't find the title",
    },
    { status: 404 },
  );
}
Initialize git in your new project and push it to GitHub.
Deploy the Application Jump to heading
Now that we have everything in place, let's deploy your new application!
- In your browser, visit Deno Deploy and link your GitHub account.
- Select the repository which contains your new application.
- You can give your project a name or allow Deno to generate one for you
- Select index.jsin the Entrypoint dropdown
- Click Deploy Project
In order for your Application to work, we will need to configure its environment variables.
On your project's success page, or in your project dashboard, click on Add environmental variables. Under Environment Variables, click + Add Variable. Create the following variables:
- AWS_ACCESS_KEY_ID- with the value from the CSV you downloaded
- AWS_SECRET_ACCESS_KEY- with the value from the CSV you downloaded.
- AWS_TABLE_REGION- with your table's region
Click to save the variables.
Let's test the API.
POST some data.
curl --request POST --data \
'{"title": "Old Town Road", "artist": "Lil Nas X", "album": "7", "released": "2019", "genres": "Country rap, Pop"}' \
--dump-header - https://<project_name>.deno.dev/songs
GET information about the title.
curl https://<project_name>.deno.dev/songs?title=Old%20Town%20Road
Congratulations on learning how to use DynamoDB with Deno Deploy!