Skip to main content

Managing Sequelize Schemas in Script Mode

This document describes how to set up Atlas to load your Sequelize schema in Script Mode. This mode is suitable for cases requiring finer control when loading Sequelize models.

If your Sequelize models are all exposed via a single Node module, Standalone Mode might be more suitable. Refer to Standalone Mode for more details.

Installation

  1. Install Atlas from macOS or Linux by running:

To download and install the latest release of the Atlas CLI, simply run the following in your terminal:

curl -sSf https://atlasgo.sh | sh
  1. Install the provider by running:
npm install @ariga/atlas-provider-sequelize

Extract Sequelize model via SQL

Create a new file named load.js with the following contents:

#!/usr/bin/env node

// require sequelize models you want to load
const user = require("./models/user");
const task = require("./models/task");
const loadModels = require("@ariga/atlas-provider-sequelize");

console.log(loadModels("mysql", user, task));

Setup Atlas

In your project directory, create a file named atlas.hcl with the following contents:

data "external_schema" "sequelize" {
program = [
"node",
"load.js"
]
}

env "sequelize" {
src = data.external_schema.sequelize.url
dev = "docker://mysql/8/dev"
migration {
dir = "file://migrations"
}
format {
migrate {
diff = "{{ sql . \" \" }}"
}
}
}
note

Be sure to update the dialect in the config file, as well as the selected dev configuration, depending on which database you are using. For more information, see Dev Database Concept.

Next, verify Atlas is able to read our desired schema, by running the schema inspect command.

atlas schema inspect --env sequelize --url "env://src"
info

For in-depth details on the atlas schema inspect command, covering aspects like inspecting specific schemas, handling multiple schemas concurrently, excluding tables, and more, refer to our documentation here.

Example

Consider the following Sequelize models directory, which includes two models: user and task:

'use strict';
module.exports = (sequelize, DataTypes) => {
const Task = sequelize.define('Task', {
complete: {
type: DataTypes.BOOLEAN,
defaultValue: false,
}
});

Task.associate = (models) => {
Task.belongsTo(models.User, {
foreignKey: {
name: 'userID',
allowNull: false
},
as: 'tasks'
});
};

return Task;
};

We should get the following output after running the inspect command above:

table "Tasks" {
schema = schema.dev
column "id" {
null = false
type = int
auto_increment = true
}
column "complete" {
null = true
type = bool
default = 0
}
column "createdAt" {
null = false
type = datetime
}
column "updatedAt" {
null = false
type = datetime
}
column "userID" {
null = false
type = int
}
primary_key {
columns = [column.id]
}
foreign_key "Tasks_ibfk_1" {
columns = [column.userID]
ref_columns = [table.Users.column.id]
on_update = CASCADE
on_delete = NO_ACTION
}
index "userID" {
columns = [column.userID]
}
}
table "Users" {
schema = schema.dev
column "id" {
null = false
type = int
auto_increment = true
}
column "name" {
null = false
type = varchar(255)
}
column "email" {
null = false
type = varchar(255)
}
column "createdAt" {
null = false
type = datetime
}
column "updatedAt" {
null = false
type = datetime
}
primary_key {
columns = [column.id]
}
}
schema "dev" {
charset = "utf8mb4"
collate = "utf8mb4_0900_ai_ci"
}

Generating SQL migration files from a Sequelize Schema

We can now generate a migration file by running this command:

atlas migrate diff --env sequelize

Running this command will generate files similar to this in the migrations directory:

migrations
|-- 20241205100006.sql
`-- atlas.sum

0 directories, 2 files

Examining the contents of 20241205100006.sql:

-- Create "Users" table
CREATE TABLE `Users` (
`id` int NOT NULL AUTO_INCREMENT,
`name` varchar(255) NOT NULL,
`email` varchar(255) NOT NULL,
`createdAt` datetime NOT NULL,
`updatedAt` datetime NOT NULL,
PRIMARY KEY (`id`)
) CHARSET utf8mb4 COLLATE utf8mb4_0900_ai_ci;
-- Create "Tasks" table
CREATE TABLE `Tasks` (
`id` int NOT NULL AUTO_INCREMENT,
`complete` bool NULL DEFAULT 0,
`createdAt` datetime NOT NULL,
`updatedAt` datetime NOT NULL,
`userID` int NOT NULL,
PRIMARY KEY (`id`),
INDEX `userID` (`userID`),
CONSTRAINT `Tasks_ibfk_1` FOREIGN KEY (`userID`) REFERENCES `Users` (`id`) ON UPDATE CASCADE ON DELETE NO ACTION
) CHARSET utf8mb4 COLLATE utf8mb4_0900_ai_ci;

Applying the migration directory to a database

Let's apply our generated migration files to a development database.

First, create a MySQL development database with Docker:

docker run --name mysql -e MYSQL_ROOT_PASSWORD=pass -e MYSQL_DATABASE=example -p 3306:3306 -d mysql:latest

Then, run the following command to apply the schema to the database:

atlas migrate apply --env sequelize --url "mysql://root:pass@localhost:3306/example"

Atlas should print an output similar to this:

Migrating to version 20241205100006 (1 migrations in total):

-- migrating version 20241205100006
-> CREATE TABLE `Users` (
`id` int NOT NULL AUTO_INCREMENT,
`name` varchar(255) NOT NULL,
`email` varchar(255) NOT NULL,
`createdAt` datetime NOT NULL,
`updatedAt` datetime NOT NULL,
PRIMARY KEY (`id`)
) CHARSET utf8mb4 COLLATE utf8mb4_0900_ai_ci;
-> CREATE TABLE `Tasks` (
`id` int NOT NULL AUTO_INCREMENT,
`complete` bool NULL DEFAULT 0,
`createdAt` datetime NOT NULL,
`updatedAt` datetime NOT NULL,
`userID` int NOT NULL,
PRIMARY KEY (`id`),
INDEX `userID` (`userID`),
CONSTRAINT `Tasks_ibfk_1` FOREIGN KEY (`userID`) REFERENCES `Users` (`id`) ON UPDATE CASCADE ON DELETE NO ACTION
) CHARSET utf8mb4 COLLATE utf8mb4_0900_ai_ci;
-- ok (65.529168ms)

-------------------------
-- 102.386668ms
-- 1 migration
-- 2 sql statements
Apply the Schema Directly on the Database

Sometimes, there is a need to apply the schema directly to the database without generating a migration file. For example, when experimenting with schema changes, spinning up a database for testing, etc. In such cases, you can use the command below to apply the schema directly to the database:

atlas schema apply \
--env local \
--url "mysql://root:pass@localhost:3306/example"

Next Steps

Now that your project is set up, start by choosing between the two workflows offered by Atlas for generating and planning migrations. Select the one you prefer that works best for you:

  • Declarative Migrations: Set up a Terraform-like workflow where each migration is calculated as the diff between your desired state and the current state of the database. See Declarative Schema Migrations

  • Versioned Migrations: Set up a migration directory for your project, creating a version-controlled source of truth of your database schema. See Quick Introduction