Spring Boot 2 实战:使用 Flyway 管理你数据库的版本变动

1. 前言

随着项目的不断迭代,数据库表结构、数据都在发生着变化。甚至有的业务在多环境版本并行运行。数据为王的时代,管理好数据库的版本也成为了迫切的须要。如何能作到像 Git 之类的版本控制工具来管理数据库?Java 项目中经常使用 FlywayLiquibase 来管理数据库版本。其中 Flyway 相对来讲比较受欢迎。html

2. Flyway 的特色

Flyway 大受欢迎是由于它具备如下优势:mysql

  • 简单 很是容易安装和学习,同时迁移的方式也很容易被开发者接受。
  • 专注 Flyway 专一于搞数据库迁移、版本控制而并无其它反作用。
  • 强大 专为连续交付而设计。让Flyway在应用程序启动时迁移数据库。

3. Flyway 的工做机制

Flyway 须要在 DB 中先建立一个 metadata 表 (缺省表名为 flyway_schema_history), 在该表中保存着每次 migration (迁移)的记录, 记录包含 migration 脚本的版本号和 SQL 脚本的 checksum 值。下图表示了多个数据库版本。web

对应的 metadata 表记录:spring

installed_rank version description type script checksum installed_by installed_on execution_time success
1 1 Initial Setup SQL V1__Initial_Setup.sql 1996767037 axel 2016-02-04 22:23:00.0 546 true
2 2 First Changes SQL V2__First_Changes.sql 1279644856 axel 2016-02-06 09:18:00.0 127 true

Flyway 扫描文件系统或应用程序的类路径读取 DDLDML 以进行迁移。根据metadata 表进行检查迁移。若是脚本声明的版本号小于或等于标记为当前版本的版本号之一,将忽略它们。其他迁移是待处理迁移:可用,但未应用。最后按版本号对它们进行排序并按顺序执行 并将执行结果写入 metadata 表。sql

对应的 metadata 表记录:数据库

installed_rank version description type script checksum installed_by installed_on execution_time success
1 1 Initial Setup SQL V1__Initial_Setup.sql 1996767037 axel 2016-02-04 22:23:00.0 546 true
2 2 First Changes SQL V2__First_Changes.sql 1279644856 axel 2016-02-06 09:18:00.0 127 true
3 2.1 Refactoring JDBC V2_1__Refactoring axel 2016-02-10 17:45:05.4 251 true

Flyway 支持命令行(须要下载命令行工具)和 Java Api ,也支持构建工具 MavenGradle 。这里咱们将目光放在 Java Api 上。app

3. Flyway 的规则

Flyway 是如何比较两个 SQL 文件的前后顺序呢?它采用 采用左对齐原则, 缺位用 0 代替 。举几个例子:spring-boot

1.0.1.1 比 1.0.1 版本高。工具

1.0.10 比 1.0.9.4 版本高。oop

1.0.10 和 1.0.010 版本号同样高, 每一个版本号部分的前导 0 会被忽略。

FlywaySQL 文件分为 VersionedRepeatableUndo 三种:

  • Versioned 用于版本升级, 每一个版本有惟一的版本号并只能执行一次.
  • Repeatable 可重复执行, 当 Flyway检测到 Repeatable 类型的 SQL 脚本的 checksum 有变更, Flyway 就会从新应用该脚本. 它并不用于版本更新, 这类的 migration 老是在 Versioned 执行以后才被执行。
  • Undo 用于撤销具备相同版本的版本化迁移带来的影响。可是该回滚过于粗暴,过于机械化,通常不推荐使用。通常建议使用 Versioned 模式来解决。

这三种的命名规则以下图:

naming.png

  • Prefix 可配置,前缀标识,默认值 V 表示 Versioned, R 表示 Repeatable, U 表示 Undo
  • Version 标识版本号, 由一个或多个数字构成, 数字之间的分隔符可用点 . 或下划线 _
  • Separator 可配置, 用于分隔版本标识与描述信息, 默认为两个下划线 __
  • Description 描述信息, 文字之间能够用下划线 _ 或空格 分隔
  • Suffix 可配置, 后续标识, 默认为 .sql

4. Spring Boot 集成 Flyway

Spring Boot 提供了对 Flyway 的自动配置 。使咱们能够开箱即用 Flyway 进行数据库版本控制。

4.1 Flyway 依赖

你只须要引入依赖:

<!-- 无需版本号 -->
 <dependency>
      <groupId>org.flywaydb</groupId>
      <artifactId>flyway-core</artifactId>
 </dependency>      
复制代码

固然你要集成你的相关数据库环境。这里咱们采用 H2 数据库来演示,其它数据库同理只不过方言不一样。不熟悉 H2 数据库的可参阅个人专题文章 Spring Boot 2 实战:H2数据库集成以及使用

4.2 Flyway 配置

为了直观的讲解配置,首先在 Spring Boot 配置文件 application.yml 咱们配置 H2 数据库为:

 spring:
 datasource:
     # h2 驱动
 driver-class-name: org.h2.Driver
     # h2 数据库 持久化到磁盘D:/h2 库名: flyway mysql模式
 url: jdbc:h2:file:D:/h2/flyway;MODE=MySQL;DATABASE_TO_LOWER=TRUE
 h2:
     # 开启console 访问 默认false
 console:
 enabled: true
 settings:
         # 开启h2 console 跟踪 方便调试 默认 false
 trace: true
         # 容许console 远程访问 默认false
 web-allow-others: true
       # h2 访问路径上下文
 path: /h2-console
复制代码

对应Flyway的配置为:

# flyway 配置
spring:
 flyway:
    # 启用或禁用 flyway
 enabled: true
    # flyway 的 clean 命令会删除指定 schema 下的全部 table, 生产务必禁掉。这个默认值是 false 理论上做为默认配置是不科学的。
 clean-disabled: true
    # SQL 脚本的目录,多个路径使用逗号分隔 默认值 classpath:db/migration
 locations: classpath:db/migration
    # metadata 版本控制信息表 默认 flyway_schema_history
 table: flyway_schema_history
    # 若是没有 flyway_schema_history 这个 metadata 表, 在执行 flyway migrate 命令以前, 必须先执行 flyway baseline 命令
    # 设置为 true 后 flyway 将在须要 baseline 的时候, 自动执行一次 baseline。
 baseline-on-migrate: true
    # 指定 baseline 的版本号,默认值为 1, 低于该版本号的 SQL 文件, migrate 时会被忽略
 baseline-version: 1
    # 字符编码 默认 UTF-8
 encoding: UTF-8
    # 是否容许不按顺序迁移 开发建议 true 生产建议 false
 out-of-order: false
    # 须要 flyway 管控的 schema list,这里咱们配置为flyway 缺省的话, 使用spring.datasource.url 配置的那个 schema,
    # 能够指定多个schema, 但仅会在第一个schema下创建 metadata 表, 也仅在第一个schema应用migration sql 脚本.
    # 但flyway Clean 命令会依次在这些schema下都执行一遍. 因此 确保生产 spring.flyway.clean-disabled 为 true
 schemas: flyway
    # 执行迁移时是否自动调用验证 当你的 版本不符合逻辑 好比 你先执行了 DML 而没有 对应的DDL 会抛出异常
 validate-on-migrate: true
复制代码

请务必仔细阅读 Flyway 相关配置的说明。

4.3 编写 SQL 初始化脚本

咱们先编写一个初始化 SQL 文件,向 H2 数据库已经自动初始化的 schema flyway 添加一张 sys_user 表。请注意命名规则。脚本名称为 V1.0.1__Add_table_user.sqlSQL 脚本的位置在配置的 spring.flyway.locations 下。内容为:

use `flyway`;
CREATE TABLE `sys_user`
(
    `user_id`         int(10) unsigned NOT NULL AUTO_INCREMENT,
    `username`        varchar(1024)    NOT NULL unique ,
    `encode_password` varchar(1024)       NOT NULL,
    `age`             int(3)           NOT NULL,
    PRIMARY KEY (`user_id`)
) ENGINE = InnoDB
  DEFAULT CHARSET = utf8mb4;

insert into  flyway.sys_user values (1,'Felordcn','{noop}12345',18);
复制代码

启动 Spring Boot 应用 。打开 H2 数据库控制台 http://localhost:8080/h2-console ,在 JDBC URL 一栏粘贴 jdbc:h2:file:D:/h2/flyway;MODE=MySQL;DATABASE_TO_LOWER=TRUE 并点击 Connect 按钮会进入如下界面:

这里 -1 是由于咱们缺省了 Flyway 须要的 flyway_schema_history 表 。0 是由于 H2 数据库自动初始化了 Schema flyway ,其它数据库可能须要你手动来创建。

4.4 编写 SQL 变动脚本

咱们编写一个 V1.0.0__Delete_sysuser_felordcn.sql 来删除 V1.0.1__Add_table_user.sql 中初始化的用户。你会发现启动报错了,由于咱们开启了校验,因此对于逻辑错误的版本会抛出异常。咱们将版本号更改成 V1.0.2__Delete_sysuser_felordcn.sql 再次启动。经过 H2 数据库控制台咱们会发现多了一条变动记录:

同时 sys_user 表的数据也没有了,符合预期。

5. Flyway 最佳实践

经过上面的介绍相信你很快就会使用 Flyway 进行数据库版本控制了。这里总结了一些在实际开发中的使用经验:

  1. 生产务必禁 spring.flyway.cleanDisabled=false
  2. 尽可能避免使用 Undo 模式。
  3. 开发版本号尽可能根据团队来进行多层次的命名避免混乱。好比 V1.0.1__ProjectName_{Feature|fix}_Developer_Description.sql ,这种命名同时也能够获取更多脚本的开发者和相关功能的信息。
  4. spring.flyway.outOfOrder 取值 生产上使用 true,开发中使用 false
  5. 多个系统公用一个 数据库 schema 时配置spring.flyway.table 为不一样的系统设置不一样的 metadata 表名而不使用缺省值 flyway_schema_history

6. 总结

今天咱们对 Flyway 数据库版本迁移管理工具进行了介绍并将之与 Spring Boot 相结合。这将大大规范咱们的数据库管理,提升生产效率。同时也分享了一些至关有用的生产实践经验。

** 相关的 DEMO 可经过关注公众号:Felordcn 回复 flyway 进行获取。**

关注公众号:Felordcn获取更多资讯

我的博客:https://felord.cn

相关文章
相关标签/搜索