Web APIs with AdonisJS and PostgreSQL
06 September 2020
In this post, we'll take a look at building an API using the AdonisJS framework. We'll specifically be looking at the AdonisJS v5 Release Preview and how we can create a simple REST API that handles some CRUD operations on a Database
Because we're going to be creating a REST API, you should have some experience making HTTP Requests, for our purposes we can use a website called Postwoman to interact with our API, but you can also use any other tool you prefer
About AdonisJS
AdonisJS is an opinionated, Typescript-first web framework that provides a lot more "out of the box" functionality than the traditional framework or library in the Node.js ecosystem and is more comparable to something like .NET's WebAPI or MVC Framework than things like Express or Next in Node.js
Some of the built-in features and decisions that stand out for me are:
- First-class Typescript support
- Class-based controllers
- Class-Based SQL ORM
- Dependency Injection
- Request Body Validation
- Authentication
- Server-side View Rendering
There are a lot more features, and it would be impractical for me to talk about all of them in a single post. Overall, the framework seems to be very complete at this point
Prerequisites
We'll be using a Node.js and a SQL Database for this post, so if you're going to follow along with this post you will need to have a couple of things installed:
- Node.js and NPM
- Any Supported SQL Database:
- MySQL
- SQLite
- Microsoft SQL Server
- PostgreSQL
- MariaDB
- OracleDB
Alternatively, you can run a Docker image for the database which may be easier, I'll be using Postgres via Docker with VSCode
To learn more about VSCode Dev Containers you can look at my previous post on how to use Dev Containers
Initialize an Application
Now that we've got all our necessary dependencies installed, we can initialize an application using the npm init
command:
During the initialization the CLI will ask you what type of project to initialize, be sure to select
API Server
npm init adonis-ts-app my-api
The above command is made up of the following parts:
npm init
which is the npm command that will run an initialization script from the provided packageadonis-ts-app
which is the package to be used for initializationmy-api
is the name of the folder/project we want to create
Once the project has been configured, navigate into the project directory:
cd my-api
And start the app:
npm start
The Ace CLI
AdonisJS makes use of a command-line application called ace
, ace
can be used to do common tasks as well as custom tasks that we define. By default, it can scaffold controllers, commands, and a bunch of other things as well as run and build an AdonisJS application
Environment Variables
AdonisJS Makes use of Environment Variables do set the application configuration, in your generated files you should see a file named .env
, this file contains the environment variables used by the application. Looking at this file will give us an idea of our current configuration:
.env
PORT=3333
HOST=0.0.0.0
NODE_ENV=development
APP_KEY=mE8zN8V_7PzazKfv9_ds-8CGjVRLA2wo
At the moment, we can see that our application will be listening on host 0.0.0.0
and port 3333
, this means that we can access our application from our browser at http://localhost:3333
If we open this page on our browser we will see the adonis logs kicked off in our command line, something like this:
ℹ info cleaning up build directory build
ℹ info copy .env,ace build
☐ pending compiling typescript source files
✔ success built successfully
ℹ info copy .adonisrc.json build
… watch watching file system for changes
ℹ info starting http server
✔ create ace-manifest.json
[1595755199986] INFO (my-app/1531 on 47f057d8722c): started server on 0.0.0.0:3333
The first request may take some time, this is because the server is still starting itself up. In the meantime, however, let's discuss how we'll be accessing our API
Making Requests
You will need to use Postwoman or something similar when making requests
Now that our application is running, we can start making requests to our API. By default, AdonisJS sets up a hello-world
route at the base of our application (the /
route`)
We can reach this endpoint by simply making a GET
request to http://localhost:3333
which will return the following:
{
"hello": "world"
}
AdonisJS can define routes using a method similar to libraries like Express.js, the "Hello World" route is defined in the start/routes.ts
file and has the following:
routes.ts
import Route from '@ioc:Adonis/Core/Route'
Route.get('/', async () => {
return { hello: 'world' }
})
The Route.get
portion states that this is a GET
route on the /
path with an async
handler function that returns an object
However, for this post, we won't be defining our route's handler functions like this. We're going to make use of controllers
Controllers
AdonisJS uses controllers
to structure our API. This makes use of a class
which contains functions intended to work as an interface between the HTTP request and the work we want to do via a provider
We will simply state which controller
and method
we want to use in our routes.ts
file instead of containing all the handler logic in that file
To generate a controller we can use ace
. We will use ace
to create a UsersContoller
.
First, stop your running application with ctrl + c
in the terminal, then use the following ace
command to generate a User
controller
:
node ace make:controller User
This will generate an app/Controllers/Http/UsersController.ts
file with the following contents:
UsersController.ts
export default class UsersController {}
Which exports a class called UsersController
with no default functionality
We can create any functions inside of here that we want to, and then map these functions to routes. For now, let's add a get
function that will put some placeholder data into. We'll update this later to use our database
A few things to note about the function we're going to create:
- The function is asynchronous, so if there are any long-running operations it won't block other things from running
- The name of the function is
get
and it has no parameters - The function returns an array of objects that represents a
User
UsersController.ts
export default class UsersController {
public async get() {
return [
{
id: 1,
name: 'Bob Smith',
email: 'bob@smithmail.com',
},
]
}
}
Now that we have defined our function, we can add a route
that will cause this function to be called. We do this by adding the following in the routes.ts
file:
Route.get('users', 'UsersController.get')
We can then run npm start
from the command line and make a GET
request to http://localhost:3333/users
which should return our user:
[
{
"id": 1,
"name": "Bob Smith",
"email": "bob@smithmail.com"
}
]
Database
Now that we've got some basic understanding of how AdonisJS maps routes to functionality, we can connect our application to a Database
To add database functionality, we first want to install the adonisjs/lucid
package to our application. Behind the scenes, AdonisJS makes use of Lucid
for connecting to and working with databases
npm i @adonisjs/lucid@alpha
And then run the following command to initialize it:
node ace invoke @adonisjs/lucid
Which should give the following output:
create config/database.ts
update .env
update tsconfig.json { types += @adonisjs/lucid }
update .adonisrc.json { commands += @adonisjs/lucid/build/commands }
update .adonisrc.json { providers += @adonisjs/lucid }
✔ create ace-manifest.json
If you open your .env
file you will see the configuration for a sqlite
database
.env
# ... other config
DB_CONNECTION=sqlite
DB_HOST=127.0.0.1
DB_USER=lucid
DB_PASSWORD=lucid
DB_NAME=lucid
You can follow the general database setup information on the AdonisJS Docs to set yours up, but I'll be using Postgres as I've mentioned before
To use Postgress, you need to install the pg
package from npm:
npm i pg
Then configure your application to use Postgres. We will first update our .env
file to have our database credentials:
.env
DB_CONNECTION=pg
DB_HOST=db
DB_USER=user
DB_PASSWORD=pass
DB_NAME=data
DB_PORT=5432
Once you've configured your database connection information in the .env
file, our database connection information will be taken care of by AdonisJS
These environment variables are used in the config/database.ts
file, we can see them in the pg
part of the file
database.ts
...
pg: {
client: 'pg',
connection: {
host: Env.get('DB_HOST', '127.0.0.1') as string,
port: Number(Env.get('DB_PORT', 5432)),
user: Env.get('DB_USER', 'lucid') as string,
password: Env.get('DB_PASSWORD', 'lucid') as string,
database: Env.get('DB_NAME', 'lucid') as string,
},
healthCheck: true,
},
...
In the database.ts
file above, look for the section for your relevant database, and set the healthCheck
property to true
Now that we've set things up, we can create a health-check route that will enable us to view the current status of our database connection. We will add a handler for this in the routes.ts
file:
import HealthCheck from '@ioc:Adonis/Core/HealthCheck'
// ... existing file content
Route.get('health', async ({ response }) => {
const report = await HealthCheck.getReport()
return report.healthy ? response.ok(report) : response.badRequest(report)
})
Once you've done all the above, start the development server again with npm start
Then make a GET
request to http://localhost:3333/health
to view your health-check information
{
"healthy": true,
"report": {
"env": {
"displayName": "Node Env Check",
"health": {
"healthy": true
}
},
"appKey": {
"displayName": "App Key Check",
"health": {
"healthy": true
}
},
"lucid": {
"displayName": "Database",
"health": {
"healthy": true,
"message": "All connections are healthy"
},
"meta": [
{
"connection": "pg",
"message": "Connection is healthy",
"error": null
}
]
}
}
}
In the lucid
section we will see if our database connection is working or any applicable error information.
Defining Models
Once we've configured our database, we will want to interact with the data in it. Lucid makes use of models
essentially as proxies for database tables. We can generate a model
for our User
with ace
as follows:
Stop your development server with
ctrl + c
before running this:
node ace make:model User
The above script will have generated a User
model class in the app/Models/User.ts
file that extends BaseModel
. All models must do this. The generated User.ts
file looks like this:
User.ts
import { DateTime } from 'luxon'
import { BaseModel, column } from '@ioc:Adonis/Lucid/Orm'
export default class User extends BaseModel {
@column({ isPrimary: true })
public id: number
@column.dateTime({ autoCreate: true })
public createdAt: DateTime
@column.dateTime({ autoCreate: true, autoUpdate: true })
public updatedAt: DateTime
}
When viewing the above file your code editor may give you a warning saying that decorators are not supported, if you see this then set the
experimentalDecorators
property totrue
in yourtsconfig.json
file:
...
"compilerOptions": {
"experimentalDecorators": true,
...
Note the @column
decorators, these are used to map columns in our database to our model fields. We'll add a field for our user's name
and email
as follows:
User.ts
export default class User extends BaseModel {
// ... other stuff in class
@column()
public name: String
@column()
public email: String
}
Any properties or methods defined in a class without the
@column
decorator will not be mapped to the database, we can just use these as normal functions in the class and implement utilities from them
Migrating the Database
At this point, our database does not contain a user
table, which will be used to store our data for the User
model. We need to create a Database Migration which will add the required table and fields.
ace
provides us with a method to scaffold a migration file. To create this file we will run the following command:
node ace make:migration users
Which will have created a database/migrations/SOME_ID_users.ts
file with the following:
*_users.ts
import BaseSchema from '@ioc:Adonis/Lucid/Schema'
export default class Users extends BaseSchema {
protected tableName = 'users'
public async up() {
this.schema.createTable(this.tableName, (table) => {
table.increments('id')
table.timestamps(true)
})
}
public async down() {
this.schema.dropTable(this.tableName)
}
}
The generated file (above) contains an up
function which will create a users
table with an id
as well as createdAt
and updatedAt
fields. We will need to modify the up
function to add our new fields as well:
export default class Users extends BaseSchema {
...
public async up () {
this.schema.createTable(this.tableName, (table) => {
table.increments('id')
table.timestamps(true)
// our added fields
table.string('name').notNullable()
table.string('email').unique().notNullable()
})
}
...
Once we've defined our migration script we need to build the application with:
node ace build
And then we can run the migration using ace
as follows:
node ace migration:run
If there is an error in a migration script, we can rollback the migration with
node ace migration:rollback
which will run thedown
function in your migration
Interact with the Database
Now that we've got our database, we can interact with it using the User
model we defined earlier
The first change we'll make is to modify the get
function to return all users. To do this we need to import App/Models/User
and use the User.all
method:
import User from 'App/Models/User'
export default class UsersController {
public async get() {
// get all users
return await User.all()
}
}
We use
await
because theUser.all
function is asynchronous
Next, we'll add a function to create a User
. We will call it post
. To make this function work we need to do a couple of things:
- Import
HttpContextContract
from@ioc:Adonis/Core/HttpContext
- Retrieve the
request
from theHttpContextContract
- Create a
User
- Return the created user
import { HttpContextContract } from '@ioc:Adonis/Core/HttpContext'
// ... other imports
export default class UsersController {
// ... get user code above
public async post({ request }: HttpContextContract) {
// get the user from the request body
const newUser = request.all() as Partial<User>
// create a user using the object we received
const user = await User.create(newUser)
// return the created user object
return user
}
}
The
request.all
function combines the data from the request body and query string into a single object
When we've added that, our completed UsersController.ts
file should look like this:
UsersController.ts
import { HttpContextContract } from '@ioc:Adonis/Core/HttpContext'
import User from 'App/Models/User'
export default class UsersController {
public async get() {
return await User.all()
}
public async post({ request }: HttpContextContract) {
const newUser = request.all() as Partial<User>
const user = await User.create(newUser)
return user
}
}
Now that we've added a new function to our controller, we need to expose a route to it in the routes.ts
file:
routes.ts
...
Route.post('users', 'UsersController.post')
...
Consume the API
Now that we've created API endpoints for listing all users and creating a user we can restart our development server with npm start
and consume our API from wherever we want to make some HTTP requests
Create a User
To create a User we need to make a
POSTrequest to
http://localhost:3333/usersand a Content-Type of
application/jsonwith the following as the
body` for our request:
{
"name": "Bob Smith",
"email": "bob@smithmail.com"
}
Which should return a created user like:
{
"name": "Bob Smith",
"email": "bob@smithmail.com",
"created_at": "2020-07-26T14:35:25.987-00:00",
"updated_at": "2020-07-26T14:35:25.988-00:00",
"id": 4
}
We can also try to create a
User
with the same information but we will see that this fails due to the database constraints we added in our migration script
Next, we can get a list of all Users by making a GET
request to http://localhost:333/users
which should give us back our created users:
[
{
"id": 4,
"created_at": "2020-07-26T14:35:25.987-00:00",
"updated_at": "2020-07-26T14:35:25.988-00:00",
"name": "Bob Smith",
"email": "bob@smithmail.com"
}
]
Summary
My overall impression of AdonisJS is pretty good. The framework feels very stable and I had much fewer issues in the process of learning it and writing this post than I have had using other more popular frameworks.
AdonisJS is very well documented with code samples for most traditional tasks and is backed by some solid libraries for things like database integration
Working with SQL using the framework has been fairly straightforward, and the ability to write database migrations using functionality provided by the framework instead of just tossing some wild SQL together makes it more approachable
Personally, however, I prefer no-SQL databases like MongoDB and tend to use them more often when using JavaScript or TypeScript, but I feel like if the need arises for a SQL database then AdonisJS is a really good option, especially if you're a JavaScript developer and don't want to have to learn Java or C# for this type of functionality
There is also a lot more functionality than what I've gone through in this post, so I'd recommend browsing the AdonisJS docs to get a broader sense of what the framework entails
If you feel like playing around with the code I've gone through in this post without having to write it all then I've got it all on GitHub and it should just be plug-and-play using Visual Studio Code Remote Containers. If you want to learn more about that, then you can check out my previous blog post on developing within a container