NextJS, Serverless Framework and Terraform monorepo
How Wrapper.js combines NextJS, Serverless Framework and Terraform together within one code repository
I recently created a code base called Wrapper.js, which combines NextJS, Serverless Framwork and Terraform within a Monorepo.
After writing a post about the benefits of using javascript to automate NextJS, Serverless Framework and Terraform Devops processes, I wrote this post which will show you how I have put together, in a manner that you can use in your own projects!
What is Wrapper.js and why you should care about it
A TLDR from my previous blog post:
Full Stack Dynamic Integration Wrapper.js is a monorepo, that seamlessly and dynamically integrates the NextJS, Serverless Framework and Terraform together.
Wrapper.JS orchestrates these technologies into one neat solution, which would normally be difficult due to the complexities of each specialism (e.g front end vs back end vs cloud infrastructure).
Localhost and Cloud Orchestration
It orchestrates all technologies through npm commands to:
- run a local dev environment for both front and back ends
- deploy and destroy an environment
- generate environment specific variable files based on aws secrets for NextJS, SLS and Terraform
Simple Management of Multiple Environments
It reduces complexity of using these technologies to manage multiple environments within a continous deployment and integration pipeline
How Wrapper.js combines these technologies
Here is a view of the codebase, looking only at the index directories.
Looking at the code base in its most basic form, lets start by looking at the index directory files:
// index directory files
index.js
bitbucket-pipelines.yml
.env
package.json
// tech stack directories
devops/
backend/
frontend/
- index.js: The nodejs wrapper script, that triggers logic based on the npm command that is executed.
- bitbucket-pipelines.yml: The bitbucket config file for continous deployment and integration pipeline
- .env: AWS credentials that allow the wrapper script to automate processes that interact with your aws account.
- package.json: NPM dependencies that allow the wrapper script to work.
The Devops Infrastructure
The devops folder enables the wrapper script (the root level index.js) and is also the home of the terraform code.
One of the core concepts of Wrapper.js, is the creation and use of environment variables throughout the stack. This, as well as running a local dev environment and cloud deployment commands, is what the scripts folder enables.
Terraform is configured in this codebase to dynamically generate and manage all cloud resources (apart from lambdas) based on these environment files.
index.js
bitbucket-pipelines.yml
.env
package.json
devops
|
└─── continuous-deployment
| └─── Dockerfile
└─── scripts
| └─── next.js
| └─── serverless.js
| └─── terraform.js
| └─── utils.js
└─── terraform
└─── main.tf
└─── backend.tf
└─── variables.tf
└─── terraform.tfvars.json
└─── modules
└─── acm
| └─── ...
└─── apiGateway
| └─── ...
└─── cloudfront
| └─── ...
└─── s3
| └─── ...
└─── ...
backend/
frontend/
When opening the devops folder, you see where a lot of the magic happens:
- continuous-deployment: Contains a Dockerfile for the configuration of the ci/cd server that enables the bitbucket pipeline.
- scripts: A folder that contains helper functions to automate deployment and development processes.
- terraform: A folder that contains all infrastructure as code Terraform configurations necessary to deploy the cloud services.
The Backend Services
The backend folder contains all the backend logic that is deployed to an environment’s API gateway.
Unlike Terraform, which is being used to create all architectural cloud resources, Serverless Framework is being used to deploy lambda functions only.
This means that within Wrapper.js, Terraform is creating a blank ApiGateway and Serverless Framework is modifying that API Gateway with end points that trigger lambda’s — further details on what is deployed in Terraform vs Serverless can be found here.
Back end code is organised into backend services, each service contains lambda functions based on the business logic that needs to be be performed.
index.js
bitbucket-pipelines.yml
.env
package.json
devops/
backend
|
└─── index.js
└─── utils.js
└─── serverless.common.yml
└─── serverless.env.json
└─── package.json
└─── services
└─── serviceOne
| └─── serverless.yml
| └─── package.json
| └─── lambdaOne
| | └─ lambdaOne.yml
| | └─ lambdaOne.js
| └─── lambdaTwo
| └─ lambdaTwo.yml
| └─ lambdaTwo.js
|
└─── serviceTwo
└─── ...
frontend/
Opening the backend folder, shows you how the backend is managed:
- index.js: A script to orchestrate the local host development, cloud deployment and removal of multiple Serverless Framework services through npm commands.
- utils.js: A file containing helper functions that enables the index.js script.
- serverless.common.yml: A YAML file containing configurations that are shared across all services, these depend on variables that are retrieved from serverless.env.json.
- serverless.env.json: A file that is dynamically created by the wrapper script, which contains environment specific variables that will feed into the back end services.
- package.json: A file that contains all the npm dependencies to run the index.js file.
- services/: The Services directory contains each backend service as a directory.
The Frontend Application
Last but not least, the front end folder contains the front end application which is written with ReactJS and uses NextJS as a framework for structuring the codebase.
index.js
bitbucket-pipelines.yml
.env
package.json
devops/
backend/
frontend
|
└─── .next
└─── .babelrc
└─── components
| └─── ...
└─── pages
| └─── ...
└─── public
| └─── ...
└─── stores
| └─── ...
└─── next.config.js
└─── package.json
The frontend folder is essentially a standard NextJS application, made up of:
- .next: Standard NextJS directory that contains dev and build configurations.
- .babelrc: Standard Babel file that contains all dev configurations for ES6 transpiling settings.
- components/: A folder that contains ReactJS components.
- pages/: A folder that contains special ReactJS components, which render as pages if placed within this folder — this is a special NextJS feature.
- public/: A folder that contains all assets (e.g images and videos etc) for the front end. stores/: A folder containing global state stores that are created using Zustand.
- next.config.js: A file that is dynamically generated by the wrapper script, which contains all environment specific variables for the front end.
- package.json: A file that contains all npm dependencies and commands required to run the frontend.
In conclusion
This post has detailed Wrapper.js’s approach to how it practically structures the technologies in the same codebase.
I hope this has been helpful and that someone out there is able to use this (or parts of this) to help them!!
Hope this was fun to read — enjoy automating :D