Update 2017-01-04 : I have now updated this repo to use the API-G proxy integration. The main reason for going for 'emulating express' instead of 'bundling express' is to not to further extend the cold-start time. Starting a new express instance is a time-taking process. This affects everytime the number of concurrent invocations is increased. Also, the bundling up of all required modules seemed like a good idea. There are many use-cases when aws-serverless-express is a better idea then using cloudhopper, but I shall stick with this for now.
Update 2016-11-09 : AWS Lambda has now a better method of Proxy Integration with API Gateway. As such, this repo is now depraced. I am working on updating it to make better use of latest features in Lambda and API-G.
Deploy your APIs with AWS Lambda
- No vendor lock-in Project can also be deployed as a normal express application.
- Fast startup time With a single function for all end-points, the API is less affected by cold-start latency.
- No intrusive persmissions The project is designed to just do the work of an application. It requires a finite set of IAM persmissions for setup. This also means you need to do some configuration on AWS for this to work
- Local testing Local testing before deploying to production
- Small deployment package The deploy tool tries to create a smaller deployment package by only including the required files
- ExpressJS APIs The request response in lambda emulates the ExpressJS API which makes it very easy to scaffold your API
- NodeJS: 4.3 or greater
Step 1: NPM init your package and add a index.js
mkdir myapp
cd myapp
touch index.js
npm init
npm i -S cloudhopper
cd node_modules/cloudhopper/
npm install
cd ../../
Step 2: Add the following scripts to your package.json
"scripts" : {
"cloudhopper": "cloudhopper"
}
Step 3: Your index.js should look something like this:
'use strict';
let Router = require('router');
let router = Router();
let cloudhopper = require('cloudhopper')
router.get('/', (req, res) => {
res.json({message: "Hello world"})
});
router.use((req, res) => {
res.status(404).json({
message: "Endpoint does not exist"
})
});
cloudhopper.use(router)
exports.handler = cloudhopper.handler
Step 4: Make sure the IAM user in ~/.aws/credentials
has the following permissions:
{
"Version": "2012-10-17",
"Statement": [
{
"Effect": "Allow",
"Action": [
"apigateway:*"
],
"Resource": [
"arn:aws:apigateway:<region>::/restapis/<api-id>/*"
]
},
{
"Effect": "Allow",
"Action": [
"lambda:UpdateFunctionCode"
],
"Resource": [
"arn:aws:lambda:<region>:<account-id>:function:<function-name>"
]
}
]
}
Step 5: Create an empty API Gateway and basic lambda function.
Step 6: Create a git-ignored file local.cloudhopper.config.json
with the following:
{
"lambda_function_name": "xxxxxx",
"apigateway_id": "xxxxx"
"restApiId" : "<rest api>",
"stageName": "<stage name>",
"apiTitle": "<name of API, this will override API name>",
"region": "<region>",
"lambdaName": "<name of lambda function>",
"tempFile": "<temp file which is in ignored list>",
"lambdaArn": "<lambda_arn>",
"stageVariables": {
"local": { },
"development": { },
"production": { }
}
}
Run your API locally at http://127.0.0.1:3000
npm run cloudhopper -- runLocal
Step 7: Prepare the API and packages the function and deploys it on Lambda
npm run cloudhopper -- setUpAPi
npm run cloudhopper -- deploy
This framework bootstraps an API Gateway which will forward everything to the lambda function. The lambda function operates from your VPC with no public DNS name. This means it can not connect to anything outside of your assigned VPC.
To connect to external network or a non-VPC endpoints or another region in your account, it requires a NAT Gateway/Instance with a private subnet.
The Request Response objects partially emulate the corresponding objects of Express Framework such that you can easily swap express instead of cloudhopper and run the project anywhere.
That said there are limited response methods that can be called: json
, status
, end
and redirect
.
These should be sufficient for an API but we can always discuss these on issues.
The api will need to connect to various aws or third party services in order to exchange data. You should have already set up a NAT instance for providing an external route to our lambda. The hostnames or the access parameters should never be hard coded in the application. It should also not be part of the version control. For this purpose, AWS provides stage variables in API Gateway. In cloudhopper you save your variables in a file stageVariables.json
which looks like this :
When you deploy your code, cloudhopper copies the production component of stageVariables.json
to the API Gateway stage variables. Cloudhopper sets up the Gateway to pass these variables to lambda.
This workflow requires a slightly different application structure. You need to require your routes and database config files only when you have obtained the variables. For this reason, we have a setInit function. Here is a sample of what it might look like. You must write your requires inside this setInit
function.
The deploy scripts goes through your node_modules folder and tries to create a zip archive for only the required files excluding the dev dependencies.
Simply run npm run cloudhopper -- runLocal
and your API is running on 127.0.0.1:3000
- Only JSON I/O
- No Cookie support
1. Why have a single function for all API end-points?
Having a single lambda function also helps to keep redundancy at a minimum and IMO keeps the project much more organized. All endpoints often need the same kind of db models and it feels quite useless to separate them into isolated blocks of code. The traditional resource based code structure feel more familiar.
Lambda works on docker containers on AWS managed ec2 instances. If you do not call a Lambda function for sometime, the docker is destroyed. A fresh start will take extra time because a new docker will be set up with your code.
2. Why not use the API Gateway end-points instead of having parameters?
Since, we are using the same function for multiple end-points, we need to have some kind of routing logic in the function itself. This makes the Gateway kind of redundant. Also, with API Gateway, non-existent end-points are returned with a wierd "Missing Authentication Error" which can not be overridden. This may not be 404 or 415 that you wish to return to your client. It is therefore better to just use wildcards and forward everything to the client.
Copyright 2016 VdoCipher Media Solutions
GPL-3.0