Sequelize and PostgreSQL with a Node client

Backend, Node, Express, Sequelize, Sql

We’ve already checked out MongoDB, now it’s time for one of the most popular database out there which is PostgreSQL, which is a free and open source relational database management system.

PostgreSQL is a powerful, open source object-relational database system with over 30 years of active development that has earned it a strong reputation for reliability, feature robustness, and performance.

We will also use Sequelize which is:

a promise-based Node.js ORM for Postgres, MySQL, MariaDB, SQLite and Microsoft SQL Server. It features solid transaction support, relations, eager and lazy loading, read replication and more.

It puts a ORM between Express and PostgreSQL to perform actions like writing SQL queries in the language of the server, it simplifies the process and it’s the reason it’s so popular to use.

To put it in another perspective, instead of writing:

SELECT * FROM users WHERE email = 'test@test.com';

We can do:

var orm = require('generic-orm-libarry');
var user = orm("users").where({ email: 'test@test.com' });

Notice how similar it’s to Mongoose.

Libraries needed

On the server besides the common ones for Express, we will also need the following:

Installing PostgreSQL and adding it to the command line

Go to their website and download the version you want, then just install and REMEMBER YOUR PASSWORD, for the username it’s usually postgres.

For adding it to the command line:

  1. Go inside your installation folder into the .bin folder and copy that address, go to control panel -> system -> advanced -> env variables -> system variables, select Path and edit -> new -> copy your URL.
  2. Do the same process but now go inside the lib folder and copy that URL instead.
  3. On your CMD do psql -U <database_name> , insert your password and now you’re connected to the command line! This will be useful for what we’re going to do next.

Setting up Sequelize

First create a .sequelizerc file inside the root and the following:

const path = require("path");

module.exports = {
  config: path.resolve("./database/config", "config.js"),
  "models-path": path.resolve("./database/models"),
  "seeders-path": path.resolve("./database/seeders"),
  "migrations-path": path.resolve("./database/migrations"),
};

Second on your terminal do sequelize init. This will create following folders

Third edit your database/config/config.js file with:

require("dotenv").config();

module.exports = {
  development: {
    url: process.env.DEV_DATABASE_URL,
    dialect: "postgres",
  },
  test: {
    url: process.env.TEST_DATABASE_URL,
    dialect: "postgres",
  },
  production: {
    url: process.env.DATABASE_URL,
    dialect: "postgres",
  },
};

Create your database

In your CMD where you’re logged in inside your postgres account, perform the following commands:

With this your connection strings are:

Now create an .env file and add these connection strings:

DATABASE_URL=
DEV_DATABASE_URL=postgres://<db_user>:<db_password>@127.0.0.1:5432/dev_db
TEST_DATABASE_URL=postgres://<db_user>:<db_password>@127.0.0.1:5432/test_db

Creating Models and Migrations

Database migration is the process of migrating data from one or more source databases to one or more target databases by using a database migration service. When a migration is finished, the dataset in the source databases resides fully, though possibly restructured, in the target databases

This will be the one we’re gonna use in the project in the next post, which consist of a user who can post and comment, so let’s use sequelize:

We’re telling sequelize to generate a model with a specific name and the attributes are the Schema basically.

Now inside the migrations edit the following models:

userId: {
  type: Sequelize.INTEGER,
  allowNull: false,
},

postId: {
  type: Sequelize.INTEGER,
  allowNull: false,
},

Next we have to change the models/index.js file to make it use the enviroment variables we setup:

const fs = require("fs");
const path = require("path");
const Sequelize = require("sequelize");
const envConfigs = require("../config/config");

const basename = path.basename(__filename);
const env = process.env.NODE_ENV || "development";
const config = envConfigs[env];
const db = {};

let sequelize;
if (config.url) {
  sequelize = new Sequelize(config.url, config);
} else {
  sequelize = new Sequelize(
    config.database,
    config.username,
    config.password,
    config
  );
}

fs.readdirSync(__dirname)
  .filter((file) => {
    return (
      file.indexOf(".") !== 0 && file !== basename && file.slice(-3) === ".js"
    );
  })
  .forEach((file) => {
    const model = sequelize["import"](path.join(__dirname, file));
    db[model.name] = model;
  });

Object.keys(db).forEach((modelName) => {
  if (db[modelName].associate) {
    db[modelName].associate(db);
  }
});

db.sequelize = sequelize;
db.Sequelize = Sequelize;

module.exports = db;

//models/index.js

Define the models relationships

Postgres is a relational database, so we need to specify the relationships between our three models generated (posts, comments, user):

Now let’s translate this into code editing each database/models file as the follow:

// database/models/user.js
module.exports = (sequelize, DataTypes) => {
  const User = sequelize.define(
    "User",
    {
      name: DataTypes.STRING,
      email: DataTypes.STRING,
    },
    {}
  );
  User.associate = function (models) {
    // associations can be defined here
    User.hasMany(models.Post, {
      foreignKey: "userId",
      as: "posts",
      onDelete: "CASCADE",
    });

    User.hasMany(models.Comment, {
      foreignKey: "userId",
      as: "comments",
      onDelete: "CASCADE",
    });
  };
  return User;
};
module.exports = (sequelize, DataTypes) => {
  const Post = sequelize.define(
    "Post",
    {
      title: DataTypes.STRING,
      content: DataTypes.TEXT,
      userId: DataTypes.INTEGER,
    },
    {}
  );
  Post.associate = function (models) {
    // associations can be defined here
    Post.hasMany(models.Comment, {
      foreignKey: "postId",
      as: "comments",
      onDelete: "CASCADE",
    });

    Post.belongsTo(models.User, {
      foreignKey: "userId",
      as: "author",
      onDelete: "CASCADE",
    });
  };
  return Post;
};

// database/models/post.js
module.exports = (sequelize, DataTypes) => {
  const Comment = sequelize.define(
    "Comment",
    {
      postId: DataTypes.INTEGER,
      comment: DataTypes.TEXT,
      userId: DataTypes.INTEGER,
    },
    {}
  );
  Comment.associate = function (models) {
    // associations can be defined here
    Comment.belongsTo(models.User, {
      foreignKey: "userId",
      as: "author",
    });
    Comment.belongsTo(models.Post, {
      foreignKey: "postId",
      as: "post",
    });
  };
  return Comment;
};

// database/models/comment.js

With this done let’s run sequelize db:migrate

Seeding data to the database

Database seeding is populating a database with an initial set of data. It’s common to load seed data such as initial user accounts or dummy data upon initial setup of an application

Seeding is basically generating some dummy data and push it to the database:

Those commands will generate three files xxxx-User.js, xxxx-Post.js, and xxxx-Comment.js for User, Post and Comment models respectively. Now edit them as follow:

module.exports = {
  up: (queryInterface, Sequelize) =>
    queryInterface.bulkInsert(
      "Users",
      [
        {
          name: "Jane Doe",
          email: "janedoe@example.com",
          createdAt: new Date(),
          updatedAt: new Date(),
        },
        {
          name: "Jon Doe",
          email: "jondoe@example.com",
          createdAt: new Date(),
          updatedAt: new Date(),
        },
      ],
      {}
    ),

  down: (queryInterface, Sequelize) =>
    queryInterface.bulkDelete("Users", null, {}),
};

// database/seeds/xxxx-User.js
module.exports = {
  up: (queryInterface, Sequelize) =>
    queryInterface.bulkInsert(
      "Posts",
      [
        {
          userId: 1,
          title: "hispotan de nu",
          content:
            "Nulla mollis molestie lorem. Quisque ut erat. Curabitur gravida nisi at nibh.",
          createdAt: new Date(),
          updatedAt: new Date(),
        },
        {
          userId: 2,
          title: "some dummy title",
          content:
            "Maecenas tincidunt lacus at velit. Vivamus vel nulla eget eros elementum pellentesque. Quisque porta volutpat erat.",
          createdAt: new Date(),
          updatedAt: new Date(),
        },
      ],

      {}
    ),

  down: (queryInterface, Sequelize) =>
    queryInterface.bulkDelete("Posts", null, {}),
};
// database/seeds/xxxx-Post.js
module.exports = {
  up: (queryInterface, Sequelize) =>
    queryInterface.bulkInsert(
      "Comments",
      [
        {
          userId: 1,
          postId: 2,
          comment:
            "Nulla mollis molestie lorem. Quisque ut erat. Curabitur gravida nisi at nibh.",
          createdAt: new Date(),
          updatedAt: new Date(),
        },
        {
          userId: 2,
          postId: 1,
          comment:
            "Maecenas tincidunt lacus at velit. Vivamus vel nulla eget eros elementum pellentesque. Quisque porta volutpat erat.",
          createdAt: new Date(),
          updatedAt: new Date(),
        },
      ],
      {}
    ),

  down: (queryInterface, Sequelize) =>
    queryInterface.bulkDelete("Comments", null, {}),
};

// database/seeds/xxxx-Comment.js

Now let’s seed it using sequelize db:seed:all, and now we have added dummy data for a Client to consume!

Summary

Creating a PERN App server

This is the link of the project we will be using to learn about this stack.

A controller, in a computing context, is a hardware device or a software program that manages or directs the flow of data between two entities. In computing, controllers may be cards, microchips or separate hardware devices for the control of a peripheral device.

Folder Structure

Our server will look like this when we’re done.

|── controllers
|   └── index.js
|── database
|   └── config
|       └── config.js
|── migrations
|   |── xxx-create-user.js
|   |── xxx-create-post.js
|   └── xxx-create-comment.js
|── models
|   |── comment.js
|   |── index.js
|   |── post.js
|   └── user.js
|── seeders
|   |── xxx-User.js
|   |── xxx-Post.js
|   └── xxx-Comment.js
|── routes
|   └── index.js
|── server
|   └── index.js
|── .env
|── .sequelizerc
└── index.js

Endpoints

These are the endpoints for reference for CRUD operations on posts, we will add later the CRUDs for Users and Comments.

└── /api/posts                  [GET] [POST]
              /:post_id         [GET]
              /:post_id         [PUT]
              /:post_id         [DELETE]

Install dependencies

yarn add express body-parser cors sequelize sequelize-cli pg pg-hstore dotenv

yarn add nodemon -D

Create your server main file

Let’s create a usual express app:

// server/index.js
const express = require("express");
const cors = require("cors");
const routes = require("../routes");
const bodyParser = require("body-parser");

const server = express();
server.use(express.json());
server.use(bodyParser.urlencoded({ extended: false }));
server.use(bodyParser.json());
server.use(function (req, res, next) {
  res.header("Access-Control-Allow-Origin", "*");
  res.header(
    "Access-Control-Allow-Headers",
    "Origin, X-Requested-With, Content-Type, Accept"
  );
  next();
});
server.use(cors());
server.options("*", cors());

server.use("/api", routes);

module.exports = server;

And add this to:

require("dotenv").config();

const server = require("./server");

const PORT = process.env.PORT || 5000;

server.listen(PORT, () => console.log(`Server is live at localhost:${PORT}`));

Create your database

We will follow what we did on the last post, here is a summary:

Add our Endpoints

const { Router } = require("express");
const controllers = require("../controllers");

const router = Router();

router.get("/", (req, res) => res.send("Welcome"));

// @route  GET /api/posts
// @desc   Get all posts available
// @access Public
router.get("/posts", controllers.getAllPosts);

// @route  GET /api/posts/:post_id
// @desc   Get post by id
// @access Public
router.get("/posts/:postId", controllers.getPostById);

// @route  POST /api/posts
// @desc   Create a post from a user
// @access Public
router.post("/posts", controllers.createPost);

// @route  PUT /api/posts/:post_id
// @desc   Update post using the id
// @access Public
router.put("/posts/:postId", controllers.updatePost);

// @route  DELETE /api/posts/:post_id
// @desc   Delete post by id
// @access Public
router.delete("/posts/:postId", controllers.deletePost);

module.exports = router;

And we need to add our controllers:

const models = require("../database/models");

const createPost = async (req, res) => {
  try {
    const post = await models.Post.create(req.body);
    return res.status(201).json({
      post,
    });
  } catch (error) {
    return res.status(500).json({ error: error.message });
  }
};

const getAllPosts = async (req, res) => {
  try {
    const posts = await models.Post.findAll({
      include: [
        {
          model: models.Comment,
          as: "comments",
        },
        {
          model: models.User,
          as: "author",
        },
      ],
    });
    return res.status(200).json({ posts });
  } catch (error) {
    return res.status(500).send(error.message);
  }
};

const getPostById = async (req, res) => {
  try {
    const { postId } = req.params;
    const post = await models.Post.findOne({
      where: { id: postId },
      include: [
        {
          model: models.Comment,
          as: "comments",
          include: [
            {
              model: models.User,
              as: "author",
            },
          ],
        },
        {
          model: models.User,
          as: "author",
        },
      ],
    });
    if (post) {
      return res.status(200).json({ post });
    }
    return res.status(404).send("Post with the specified ID does not exists");
  } catch (error) {
    return res.status(500).send(error.message);
  }
};

const updatePost = async (req, res) => {
  try {
    const { postId } = req.params;
    const [updated] = await models.Post.update(req.body, {
      where: { id: postId },
    });
    if (updated) {
      const updatedPost = await models.Post.findOne({ where: { id: postId } });
      return res.status(200).json({ post: updatedPost });
    }
    throw new Error("Post not found");
  } catch (error) {
    return res.status(500).send(error.message);
  }
};

const deletePost = async (req, res) => {
  try {
    const { postId } = req.params;
    const deleted = await models.Post.destroy({
      where: { id: postId },
    });
    if (deleted) {
      return res.status(204).send("Post deleted");
    }
    throw new Error("Post not found");
  } catch (error) {
    return res.status(500).send(error.message);
  }
};

module.exports = {
  createPost,
  getAllPosts,
  getPostById,
  updatePost,
  deletePost,
};

Summary

Conclusion

Using PostgreSQL is indeed more complicated than MongoDB, has more steps to setup for a new project, so why use it? Why spend 20 minutes setting up with Sequelize when you could easily do it in less than 10 minutes with Mongoose and MongoDB?

NoSQL databases are good for document storage. Data that doesn’t relate to anything else. Relational databases are for - you guessed it, relational data. Turns out, data in most applications are relational. A user has orders. Orders have items. Items have options, on and on. Relational databases have been around for decades for good reason. They’re highly optimized for querying that data efficiently.

Usually most modern applications will have relations between each other, a user can have orders, which can have items, which can have companies, and so on, imagine doing this with MongoDB and the JSON it would generate? So as a norm you should be using MongoDB when you don’t have any relational data, for every other case PostgreSQL is the go to.

Something to note is that we can use MongoDB for some relations, in our previous project we had a User that could create a Post and Comment others as well, they were related, so it’s not like you have to use PostgreSQL for simple relational data, but keep in mind how your app will scale and the size of it.

The hardest part was the setting up the database, but once you make it a step by step recipe it’s much easier to deal with. So what comes next is then creating your routes, your express code and add your controllers specifying how the data will be handled.

This was a great experience filled with plenty of learning!

See you on the next post.

Sincerely,

Eng. Adrian Beria