Here's a fully working MVP for a file uploader that I did to learn about serverless, while taking advantage of technologies that I already know and appreciate, like TDD, Webpack, Babel & Jest. It was a bit challenging since I'm more familiar with frameworks like Sails, Feathers, Express & Koa, where you have to either provide the server and keep it running, or use Heroku. At this point, I feel confident I can also work and deliver products using Serverless! π
This file uploader has the following features:
- Allows user registration and user verification.
- Further authentication powered by JWT tokens.
- All the files are private.
- Users can list their uploads, and remove them.
- Users can also generate temporary links to access these files.
Here is a list of features that are probably only important for developers:
- It's built with NodeJS π
- It uses the latest EcmaScript features I could think of at the moment.
- It's fully unit tested πͺ
- As much documentation as possible.
- It desploys on AWS using serverless.
- It uses Prettier and Eslint, so no ugly code on sight π
- It runs lint and test validations on CircleCI on pull requests.
I know the checkIn
policy needs some love, but I want to move to
another project.
The first step is actually not with the repo, but with your AWS account:
- Get in the AWS Console.
- Go to the IAM section.
- Create a user, give it
Programmatic access
. - Give it full
List
,Read
&Write
access toCloudFormation
. - Give it
IAMFullAccess
. - Give it
AmazonDynamoDBFullAccess
. - Give it
AmazonAPIGatewayAdministrator
. - Give it
AmazonAPIGatewayAdministrator
. - Give it
CloudFrontFullAccess
. - Give it
AmazonS3FullAccess
. - Give it
AWSLambdaFullAccess
.
Now, you're ready for the terminal.
You'll need NodeJS. I like using nvm, since it gives you the freedom to move back and forward between one version and another. Just as they say in their README, you can install NVM with:
curl -o- https://raw.githubusercontent.com/creationix/nvm/v0.33.11/install.sh | bash
You'll also need serverless. The quick guide says:
npm install -g serverless
# Updating serverless from a previous version of serverless
npm install -g serverless
# Login to the serverless platform (optional)
serverless login
It will ask you the credentials of an IAM user. Put the credentials of the user you just created.
After that, I highly recommend reading the serverless CLI reference (I tend to forget commands, even the most used ones): https://serverless.com/framework/docs/providers/azure/cli-reference/
You can do it this way:
git clonse git@github.com:sadasant/uploader-js.git
Then cd uploader-js
. Followed by npm install
.
At this point, you're ready, this is yours now, play with it, break it, submit issues back, learn with me! You can start by running the tests:
npm t
Or by deploying it yourself:
serverless deploy
I have this deployed (and open to the public, for a brief period of time), in the following endpoints:
POST - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/register
POST - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/unregister
POST - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/getAuthToken
POST - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/getVerifyToken
POST - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/verify
DELETE - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/removeAccount
GET - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/listUploads
GET - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/shareUpload
POST - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/upload
GET - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/getUpload
DELETE - https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/removeUpload
Let's use CURL to start using these endpoints right now (here is a curl crash course, just in case):
POST register
: The registration process requires sending an email and a password, which will result in a JSON object with a singleverifyToken
property, which you will use to verify yourself. I'm mimicking the email verification processes.
email="lurker@hotmail.com"
password="avocadoPower!1234"
verifyToken=$(curl \
-d "{\"email\":\"$email", \"password\":\"$password\"}" \
-H "Content-Type: application/json" \
-X POST https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/register | \
python3 -c "import sys, json; print(json.load(sys.stdin)['verifyToken'])")
echo "Your verification token is: $verifyToken"
POST unregister
: As long as you haven't verified yourself, you can unregister with:
email="lurker@hotmail.com"
curl \
-d "{\"email\":\"$email"}' \
-H "Content-Type: application/json" \
-X POST https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/unregister
POST getAuthToken
: Allows you to retrieve an authorization token that you will be able to send to other endpoints on theAuthorization
header. You need to send theemail
and thepassword
, and if all goes well, you'll receive a JSON object with a singple property:token
.
token=$(curl \
-d "{\"email\":\"$email", \"password\":\"$password\"}" \
-H "Content-Type: application/json" \
-X POST https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/getAuthToken | \
python3 -c "import sys, json; print(json.load(sys.stdin)['token'])")
-
POST getVerifyToken
: Allows you to retrieve again yourverifyToken
, in case you lost it. Requires sending theuser
andpassword
again. -
POST verify
: Allows you to verify yourself. Nothing special happens afterwards. It's just a mimick of email verification.
curl \
-d "{\"verifyToken\": \"$verifyToken\"}" \
-H "Content-Type: application/json" \
-H "Authorization: $token" \
-X POST https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/verify
DELETE removeAccount
: Allows you to remove your own account.
curl -H "Authorization: $token" \
-X DELETE https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/removeAccount
GET listUploads
: Will return a list of file names that have been uploaded previously. Empty if no file names have been uploaded so far.
NOTE: You might need to re-generate another token by calling
getAuthToken
again.
curl -H "Authorization: $token" \
https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/listUploads
GET shareUpload
: Will generate a shareable link to one of your downloads. Requires you to send (in the query parameters) thefileName
you want to share, and optionally anexpiresAt
property with either a valid string date of a date in the future, or the number of seconds you want the shareable link to be active.
curl -H "Authorization: $token" \
https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/shareUpload?fileName=isThisAFunMeme.jpg
POST upload
: Allows you to upload files. You'll need to send a JSON body with two properties:fileName
, which will hold the name of the file, and abase64File
property that will hold the base64 represenation of the file contents. You can try it with an image that comes with this repository, in the foldermisc
, as the following example shows:
base64File="$(base64 -w 0 misc/isThisAFunMeme.jpg)"
curl \
-d "{\"base64File\": \"$base64File\", \"fileName\": \"isThisAFunMeme.jpg\"}" \
-H "Content-Type: application/json" \
-H "Authorization: $token" \
-X POST https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/upload
GET getUpload
: Since you have an authorized token for your own account, you can retrieve this file (base64 encoded) with a simple call to this endpoint:
Note: This was probably better called getFile
, now that I think
of it.
base64File=$(curl -H "Authorization: $token" \
https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/getUpload?fileName=isThisAFunMeme.jpg | \
python3 -c "import sys, json; print(json.load(sys.stdin)['base64File'])")
If you want to transform it to it's original form, you can do it in the console with the following command:
base64 -d <<< $base64File > isThisAFunMeme.jpg
If you are a console freak (and/or you happen to be in Debian), you can view it without
leaving the terminal by installing sudo apt-get install caca-utils
and running:
cacaview isThisAFunMeme.jpg
DELETE removeUpload
: Finally, you can remove files previously uploaded. You can call this endpoint as follows:
curl -H "Authorization: $token" \
-X DELETE https://rvpujtrb06.execute-api.us-east-1.amazonaws.com/dev/removeUpload?fileName=isThisAFunMeme.jpg
My main goal with this repository was to learn how to deploy code to AWS Lambdas using serverless. I decided to take a TDD approach, and to focus on a folder structure and utilities that would be powerful to extend for further purposes. It was very challenging to navigate through the intricacies of these technologies, but I'm happy to say that this code could be used as a stepping stone for building any other service or API.
While working on this project, I took the liberty to make some design decissions that I believe are correct but might not be for others (based on opinions and experiences). I'm going to list them so I can find them later more easily:
-
Self Contained Utilities & Small Handlers: AWS recommends the business code to be detached from the handlers. I tried to follow this approach by making multiple-purpose self-contained utilities that the handlers could use to fulfil each one of their purposes, but I left the business logic in the handlers. My position here is that each handler should do just one thing, which should be small, and should have a limited and short list of caveats also described in the handlers. Since handlers have 100% unit test coverage, I believe this approach to be decent. On the utilities side, on larger projects they might be better separated in folders (and perhaps into separate repositories if they grow enough).
-
Bad RESTful API: This API is bad at REST, but it could be better with little tweaks. I decided to put my time into making sure I could support a RESTful API using serverless and AWS Lambdas, so we have some DELETE endpoints outside of the GET and POST endpoints. The namings of these endpoints don't overlap, which is a key difference with RESTful APIs. A project like this with the proper time invested would have a RESTful API without trouble, and specially without having to change the underlying ideas: it's just about defining the proper methods at the
serverless.yml
file, and grouping the handlers that share a common resource in a separate folder. -
Piping Handlers: our
util/handler.js
allows our handlers to be piped from and to one another. This for me is a key decission point, since it allows us to compose handlers, thus permitting code reutilization. -
Meta Authentication Policies: The handlers I made allow an authentication policy that is invoked within the execution of a single endpoint, which might happen right afterwards AWS routes the request through the authenticator. Both the authorize handler and this authentication policy (called
checkIn
) call almost the same code underneath. This causes a little overhead, but the benefit is that we're able to have a centralized place where we do authentication and user retrieval, which we can just pipe at the beginning of the handlers we have to provide security and a ready to use user in the sameevent
object. It might be the case that AWS has a way to do this more accordingly to AWS designs, but I am not aware of it. -
TDD: I decided to write the unit tests as early as possible. I tried to do them before I wrote some endpoints, but more realistically I ended up writing many of them in parallel. It proved to be a bit challenging, because serverless and AWS functions have their own complications which required further reading on the AWS docs and the source code of the libraries that I'm using. I felt a bit blocked while figuring out why these libraries were screaming, and then when even though the tests passed, the endpoints didn't work. However, as soon as I figured out how to make a couple work, fixing the rest of them followed smoothly. I think it proves that environmental issues should be able to be solved by inference, where the business logic can remain as unedited as possible, and small centralized changes can solve issues accross the platform.
-
No ORM: Using DynamoDB feels hard because it's really different to other No-SQL databases that are more popular (such as Mongo or CouchDB), and also that it lacks the many ORMs that other databases have. Adding a local copy of DynamoDB to CircleCI just for unit tests felt a little bit wasteful. While looking at Dynamoose, I found out that it required a local copy of DynamoDB to run unit tests. I browsed for the alternatives and found this post: Introducing the Amazon DynamoDB DataMapper for JavaScript. A library made by Amazon to make it easier to use the DynamoDB client. It's beautiful and it works, and it pairs well with
aws-sdk-mock
, so I used it :) -
Why base64 file contents? It's just easier to read in the consle in case I dump it :( I and easier to work with over the network. I'm sorry! :) it should be easy to change!
This API feels secure and robust, I think the file structure needs work but sacrificing it to focus in the toolset was a decent compromise, since the requirements were not very elaborated. The handlers being small and 100% unit tested feels great! - I ended up writing endpoints that I didn't try live until I finished the rest of the application. I think that my approach is decent. However, I didn't put attention on several important things. Here's a list:
Perhaps obvious things that I'm not including:
- I know I could reuse the JWT token for the verification token, I don't think it adds any practical benefit to invest more time on that, the code that generates the verification token is just this. Now, if this was a real project, I would certainly change it to reuse jwt :)
- I should have discovered this earlier: https://github.com/dherault/serverless-offline
- Automatically generated documentation. I've heard it's a thing in serverless/aws lambdas, but I haven't gotten myself there.
- Currently, a user can retrieve more than one jwt token and use them independently. In some applications, it makes sense to restrict tokens so that only the last one generated is the valid one.
- The database models don't have a migration strategy.
- I'm missing configurations based on the deploy environment.
- Information about the files is not stored in a separate table.
- Multipart uploads, we don't have this yet, but I've seen code about it.
- Password recovery. If users forget their password, they're locked! π¬ π β οΈ
- I'm missing a user interface π I hope the fact that I'm using webpack highlights that I know a fair bit about building user interfaces. I'm not saying that using webpack makes you good at building user interfaces (one doesn't imply the other), I'm just saying I've built a bunch and I will keep building them, just probably not for this project π¬
And, because we're all noobs at some point, here are some of the things I haven't figured out so far:
- DynamoDB is super tricky. Setting up createdAt as a range key causes wreckage in my current way to update. I wasn't successful with dynamo-mappers or with dynamoose. I know I will be able to figure out why if I start digging up the queries that are sent directly to dynamo, and I match them with the documentation that Dynamo has, but this is time expensive.
- Why does serverless logs behaves so weird? I haven't figured out how to just get the last log consistently, and it seems to loop over older logs, then the newer ones, then somehow returns nothing a couple of times, then I get the most recent logs... Seems crazy!
Up next, I want to focus on learning about PRobot, then I'd like to revisit this to make a Golang copy of this repository. I actually started with Golang, but it has been about 3 years without having excuses to use Golang on my daily job, so I'm rusty! :(
Here is a list of things I looked at while working on this repository:
- LambdAuth
- Building a REST API in Node.js with Lambda, API Gateway, DynamoDB, and Serverless framework
- Error Handling Patterns in Amazon API Gateway and AWS Lambda
- Introducing the Amazon DynamoDB DataMapper for JavaScript
- Class: AWS.S3 on the docs for AWSJavaScriptSDK
- Serverless File Uploads
- Generate a Pre-signed Object URL Using the AWS SDK for Java
- How to upload files to Amazon s3 using NodeJs, Lambda and Api Gateway
- Nodejs AWS SDK S3 Generate Presigned URL
- Serverless Authentication with JSON Web Tokens
- Serverless JWT Auth Boilerplate