serverless-graphql v0.0.4

Serverless GraphQL Boilerplate
This is a web application boilerplate for a remarkably efficent Graph API. This Graph API is powered by a single AWS Lambda function containing GraphQL, accessible via one HTTP endpoint. Through this endpoint, you can request any custom shape of data, across records stored in multiple DynamoDB tables, and GraphQL will return it. The result is a very low total cost of ownership (i.e., least amount of code, administration, cost).
Users CRUD has been implemented, as well as authentication and authorization via JSON webtokens. Also provided is an entire front-end built in React. You can deploy everything easily via the Serverless Framework. A plugin for the Framework allowing you to easily deploy your front-end assets to S3 is also included.
For more information, please read the FAQ at the bottom, and be sure to enjoy! - serverless.com
Setup
If you haven't yet installed serverless on your machine, run:
npm install -g serverlessthen install the GraphQL Boilerplate in the CWD by running:
sls project install serverless-boilerplate-graphql
cd serverless-boilerplate-graphqlBack
Add the authTokenSecret variable to _meta/variables/s-variables-STAGE-REGION.json and give it a strong value. This is the secret used to generate JSON web tokens. Then:
cd back/api
npm install
sls function deploy --all
sls endpoint deploy --allClient
Set API_URL in client/src/app/js/actions/index.js. Then:
cd ../../client/src
npm install
npm startThis will run the client locally. You can then deploy the client to an S3 bucket with:
npm run build
sls client deployDeploying to S3 bucket
Make sure you're still in the client/src folder mentioned above, then run:
npm run build
sls client deployTesting With GraphiQL
If you're running OSX, you can use the GraphiQL Electron App to test the GraphQL backend without a client:
- Install brew cask for easy installation: brew tap caskroom/cask
- Install GraphiQL App: brew cask install graphiql
- Open GraphiQL application. Just search for GraphiQLusing OSX Spotlight Search!
- Add your dataendpoint to the "GraphQL Endpoint" text field, and make sure the "Method" is set toPOST.
- Try this mutation to create the first user:
mutation createUserTest {
  createUser (username: "serverless", name: "Serverless Inc.", email: "hello@serverless.com", password: "secret") {
    id 
    username 
    name 
    email  
  }
}- Now list all users using the following query:
query getUsersTest { 
  users {
    id
    username
    name
    email
  } 
}- You should get the user you just created:
{
  "data": {
    "users": [
      {
        "id": "aca42ee0-f509-11e5-bc11-0d8b1f79b4b9",
        "username": "serverless",
        "name": "Serverless Inc.",
        "email": "hello@serverless.com"
      }
    ]
  }
}Team Workflow with Meta Sync Plugin
This boilerplate includes the Meta Sync Plugin. To start using it you need to add the following serverless variables to _meta/variables/s-variables-common.json:
"meta_bucket" : "SOME_BUCKET_NAME",
"meta_bucket_region" : "us-east-1" // or any other regionTesting With A Local DynamoDB Instance
- Install Docker
- Run docker-compose upto install and run DynamoDB.
- Add the localDynamoDbEndpointvariable with the valuehttp://<DOCKER-MACHINE-IP>:8000to_meta/variables/s-variables-common.json. Example value:http://192.168.99.100:8000.
- Run sls setup db -s <stage> -r <region>to create tables in the local DynamoDB instance.
- Run sls offline startto start the offline server.
FAQ
Why use GraphQL + Lambda?
Lambda is a revolutionary compute service that makes it really easy to build and maintain microservices at a fraction of the cost. GraphQL is equally revolutionary in its ability to receive and return requests for custom data shapes from records across different data sources. We thought these two technologies would work well together.
GraphQL reduces the amount of endpoints required in a traditional REST API back-end, to 1-2 graph API endpoints, significantly reducing development and maintenance time, as well as network overhead. Further, multiple client applications (and their many versions) can request any data they need to populate their views via these same API endpoints. It's incredible efficiency.
Putting GraphQL in AWS Lambda means simple Graph APIs, requiring zero administration, and you only get charged when that API is used.
How to add more data records?
In this boilerplate, we're managing all data records using GraphQL. Currently the boilerplate only has a Users record/collection. But you can easily add any other data collection (i.e.. Posts) in the back/api/lib/graphql/collections directory. Just follow the same pattern in the Users collection.
How does validation work with GraphQL?
Each data collection has a validate.js file. This is where you should keep your validation logic, and call the validation functions on the data received from GraphQL before you resolve them. GraphQL has its own validation implementation, but it's at a very early stage at this point.
How does authentication work with GraphQL?
In this boilerplate, we're using JSON Web Token for authentication. You can find this logic in the back/api/lib/auth.js file. You can simply switch to another authentication mechanism by editing this file.
Where to add more business logic?
This is a new architecture and there are still some questions left unanswered. The biggest question is where should further business logic go (e.g., sending transactional email after a user is created)? We have two answers for that:
- Monolithic - Add business logic into the single AWS Lambda function and build out a monolithic Lambda.
- Microservices - Add business logic in separate Lambda functions which you can invoke asynchronously.  In the GraphQL resolvefunctions is where you should make these calls. And you can store these extra Lambda functions inapi/events. Doing synchronous calls to other Lambda functions will likely result in too much latency. But logic that can be done in the background can easily be isolated in separate Lambda functions. We've included some code to help you invoke separate Lambda functions and we've even added it already to theusers createresolve function.
There are some challenges here. However, building and maintaining another large REST API requires a lot of time and effort. This requires significantly less.
What plugins are included with this boilerplate?
- serverless-client-s3: To deploy front end assets to S3
- serverless-cors-plugin: To enable CORS for your data endpoint and give the client access to your backend.
- serverless-meta-sync: To collaborate with your teammates and sync your _metafolder with them securely using an S3 bucket.
- serverless-offline: To test your project locally during development. That also includes a local dynamoDB instance.
What's the difference between installing vs cloning & initing the boilerplate?
They both achieve the same goal. However, Installing the boilerplate with sls project install is simpler because it also handles installing the project dependencies. We recommend using sls project init only if you clone the project to contribute/make a PR.
Why do we have to npm install 2-3 times?
Because we're dealing with isolated micro services architecture, we have some separation of concerns around different areas of the project. So dependencies are managed at three levels:
- Project Dependencies: by running npm installin the root of the project. This is done for your automatically when you runsls project install. This mostly handles installing the plugins.
- Backend Dependencies: by running npm installin the root of thedata/apidirectory. This makes all thenode_modulesrequired by the boilerplate available for deployment with your functions.
- Frontend Dependencies: by running npm installin the root of theclient/srcdirectory. This installs all the client side dependencies to make your React application work.
Why can't we deploy with sls dash deploy?
You can deploy with sls dash deploy, however the Serverless CORS Plugin requires that you deploy your endpoints with sls endpoint deploy so that it can fire the necessary pre hooks that will enable CORS.
How to connect the client to the Serverless backend?
By setting the API_URL variable in client/src/app/js/actions/index.js Please keep in mind that this is the root of your API not the endpoint url you get from sls endpoint deploy
Team
10 years ago