Описание

В своих проектах для управления миграциями баз данных я всегда использую Flyway, включая некоторые важные компоненты его экосистемы:

  1. Versioned migrations — это стандартные миграции, которые применяются последовательно друг за другом.

  2. Repeatable migrations — это повторяемые миграции, например, скрипты для создания процедур. Они позволяют отслеживать историю изменений с помощью Git.

  3. Callbacks — это различные хуки, которые срабатывают в определённое время. Например, можно создать скрипт beforeMigrate__init_types.sql, в котором описаны все кастомные типы, используемые в базе данных. Мигратор сначала выполняет этот скрипт, а затем остальные операции.

Я бэкенд разработчик на Node.js и стараюсь использовать консольные утилиты написанные под Node.js, в случаи с Flyway использую Node.js-обёртку — node-flywaydb.

Этот инструмент очень прост в использовании: при запуске он скачивает оригинальную Java-утилиту и запускает её, поэтому для работы требуется наличие JVM на машине.

Проблема

Когда мы создаем Docker-образы, предназначенные для запуска миграций базы данных, нам приходится включать в них JVM, что значительно увеличивает размер образа и замедляет весь CI/CD-процесс.

Возможные решения

  1. Использовать Node.js-миграторы (например: db-migrate, umzug, pg-migrate).

  2. Использовать миграторы, встроенные в ORM, применяемую в проекте (например: prisma, typeorm).

  3. Написать лёгкий клон Flyway на Node.js для одного типа базы данных (PostgreSQL).

Принятое решение

Поскольку я активно использую возможности Flyway, такие как repeatable migrations и callbacks, и не хочу отказываться от них, так как аналоги этих функций отсутствуют в других системах миграции, оптимальным решением стало создание собственного легкого клона Flyway на Node.js для работы с PostgreSQL.

Этапы разработки

  1. Versioned migrationsвыполнено.

  2. Repeatable migrationsвыполнено.

  3. Callbacksчастично выполнено, только для versioned.

  4. Flyway schema history tableчастично выполнена, только для versioned и repeatable.

  5. Migration Command Dry Runsвыполнено.

  6. Baseline migrations — не выполнено.

  7. Undo migrations — не выполнено.

  8. Script migrations — не выполнено.

  9. Migration transaction handling — не выполнено.

Там у них действительно много интересных возможностей, я перечислил только те, которые были наиболее полезными в моей практике.

Пример использования

Запуск базы данных

Для запуска базы данных выполните следующую команду:

curl -fsSL https://raw.githubusercontent.com/EndyKaufman/pg-tools/refs/heads/master/docker-compose.yml -o docker-compose.yml
docker compose up -d

Результатом будет успешный запуск контейнера PostgreSQL:

[+] Running 3/3
 ✔ Network pg-tools_default              Created                         0.1s
 ✔ Volume "pg-tools-postgre-sql-volume"  Created                         0.0s
 ✔ Container pg-tools-postgre-sql        Started                         0.2s

Создание миграции

Для создания первой миграции выполните следующую команду:

npx pg-flyway create --name=Init --version=1
echo "CREATE TABLE \"User\"(id uuid NOT NULL DEFAULT uuid_generate_v4() constraint PK_USER primary key,email varchar(20));" > migrations/V1__Init.sql

Результатом будет успешное создание миграции:

[2025-01-15T23:23:53.903] [INFO] create - Name: Init
[2025-01-15T23:23:53.904] [INFO] create - Locations: migrations
[2025-01-15T23:23:53.914] [INFO] create - Migration "migrations/V1__Init.sql" was created successfully!

Применение миграции

Для применения созданной миграции выполните следующую команду:

npx pg-flyway migrate --database-url=postgres://pgtoolsusername:pgtoolspassword@localhost:5432/pgtoolsdatabase?schema=public

Результатом будет успешное выполнение миграции:

[2025-01-16T00:08:39.052] [INFO] migrate - Locations: migrations
[2025-01-16T00:08:39.053] [INFO] migrate - HistoryTable: __migrations
[2025-01-16T00:08:39.053] [INFO] migrate - DatabaseUrl: postgres://pgtoolsusername:pgtoolspassword@localhost:5432/pgtoolsdatabase?schema=public
[2025-01-16T00:08:39.074] [INFO] migrate - Migrations: 1

Просмотр списка выполненных миграций

Для просмотра списка выполненных миграций выполните следующую команду:

npx pg-flyway info --database-url=postgres://pgtoolsusername:pgtoolspassword@localhost:5432/pgtoolsdatabase?schema=public

Результатом будет таблица с информацией о выполненных миграциях:

[2025-01-16T09:15:34.007] [INFO] info - HistoryTable: __migrations
[2025-01-16T09:15:34.008] [INFO] info - DatabaseUrl: postgres://pgtoolsusername:pgtoolspassword@localhost:5432/pgtoolsdatabase?schema=public
[2025-01-16T09:15:34.057] [INFO] info - Migrations: 1
┌───────────┬─────────┬─────────────┬──────┬─────────────────────┬─────────┬──────────┐
│  Category │ Version │ Description │ Type │        Installed On │   State │ Undoable │
├───────────┼─────────┼─────────────┼──────┼─────────────────────┼─────────┼──────────┤
│ Versioned │       1 │        Init │  SQL │ 2025-01-15 20:08:39 │ Success │       No │
└───────────┴─────────┴─────────────┴──────┴─────────────────────┴─────────┴──────────┘

Остановка базы данных

Для остановки базы данных выполните следующую команду:

docker compose down

Планы

Основные функции Flyway уже перенесены в Typescript-код, что обеспечивает необходимый функционал для большинства сценариев при работе с миграциями.

Остальные не портированные функции я планирую переносить по мере наличия свободного времени.

Буду рад pull-реквестам с новыми функциями и исправлением багов.

Ссылки

Комментарии (2)


  1. DIMANVAZ
    16.01.2025 19:25

    Вот вы энтузиаст фанат, конечно. Выражаю одобрение


    1. kaufmanendy Автор
      16.01.2025 19:25

      Спасибо большое за комментарий)