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
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
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 Add user button, fill the User name field (maybe use
denamo
) and select Programmatic access type. - Click on Next: Permissions, then on Attach existing policies
directly, search for
AmazonDynamoDBFullAccess
and select it. - Click on Next: Tags, then on Next: Review and finally Create user.
- Click on Download .csv button to download the credentials.
Create database table:
- Go to https://console.aws.amazon.com/dynamodb and click on Create table button.
- Fill the Table name field with
Songs
and Primary key withtitle
. - Scroll down and click on Create. That's it.
Create a Project in Deno Deploy
- Go to https://dash.deno.com/new (Sign in with GitHub if you didn't already) and click on Create.
- Now click on Settings button available on the project page.
- Navigate to Environment Variables Section and add the following secrets.
AWS_ACCESS_KEY_ID
- Use the value that's available under Access key ID column in the downloaded CSV.AWS_SECRET_ACCESS_KEY
- Use the value that's available under Secret access key column in the downloaded CSV.
Now click on the project name to go back to the project dashboard. Keep this tab open as we will come back here later in the deploy step.
Write the Application
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://cdn.skypack.dev/@aws-sdk/client-dynamodb?dts";
// 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: "ap-south-1",
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 },
);
}
Deploy the Application
Now that we've everything in place, let's get to deploying the application.
The steps:
- Go to https://gist.github.com/new and create a new gist with the above code
(make sure the file ends with
.js
).For convenience, the code is also hosted at https://deno.com/examples/dynamo.js so you can skip creating a gist if you just want to try out the above example without making changes to it.
- Go to the project (that we previously created) dashboard in Deno Deploy.
- Click on Deploy URL and paste the raw link of the gist.
- Click on Deploy and copy the URL that's displayed under Domains section.
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!