使用 mvn liquibase:diff 时如何禁用删除表和列?

我正在设置一个发布流程,以尽可能轻松地升级我们的生产环境。

为了实现数据库方面的目标,我选择了 liquibase。每次环境升级时:

  1. 代码交付到服务器上,

  2. mvn liquibase:diff运行以生成先前版本的变更日志

  3. mvn spring-boot:run运行以启动应用程序并使用新的变更日志执行 liquibase,因此,从新交付的代码实体调整数据库。

问题是,例如,如果将java 中的字段移动到其他实体,liquibase将删除该列,而不将数据传输到新的字段位置。@Entity

所以我的问题是,我们可以配置 liquibase 来避免列和表删除吗?为了能够将数据复制到新位置,然后有效地永久删除列(或表)?

我们使用spring-boot 2.1.2liquibase maven 插件 3.4.1。我们的数据库是MySQL 5.7.27

我试图找到如何导出一个中的所有数据库创建/更改 以及另一个中的所有数据库创建/更改。就像我们可以在这两个之间执行,第三个专门用于将数据复制到新位置。Bu 在 liquibase 文档中没有找到解决方案:https://www.liquibase.org/documentation/diff.htmlchangeSetchangeLog.xml changeSetchangeLog.xmlchangeLogchangeLog

为了说明我的例子:

v1

@Entity

public class Person {

     private long id;

     private String name;

     private String phoneNumber;

}

成为


v2 : (创建实体Phone来替换实体Person的字段phoneNumber )


@Entity

public class Person {

     private long id;

     private String name;

     private Phone phone;

}


@Entity

public class Phone {

    private long id;

    private String phoneNumber;

    private String brand;

}

因此,当v2交付到服务器上并mvn liquibase:diff运行时,结果 ChangeLog 将删除phoneNumber列并在表person中创建列phone。在不从表person传输数据的情况下,将phoneNumber归档到新表phone字段phoneNumber中。


我想执行将Person.phoneNumber复制到新实体Phone.phoneNumber的 changeLog (手动编写) 。


那可能吗 ?或者有什么技巧可以正确地做到这一点?或者也许我以不好的方式使用 liquibase 来实现这一目标?


慕斯709654
浏览 84回答 1
1回答

森林海

也许我错过了一些东西,但据我了解,这diff并不是万能的工具,可以免除您的任何责任。在某些情况下,您必须自己编写变更集,我认为这就是其中之一。既然它diff确实是一个非常强大的命令,它也有它的缺点。理论上,差异工具还可以检查数据库之间的新数据、更新数据和丢失数据,但实际上这不起作用,原因有两个:表现。随着数据集的增长,要比较的信息量也会不断增加,直至无法管理。更改数据。在开发过程中,测试数据通常会添加到开发数据库中,而这些数据不应复制到其他数据库中。此外,新数据可能会添加到测试和生产数据库中,但不应仅仅因为开发数据库中不存在这些数据而将其删除。对于你的情况,我会做这样的事情:创建一个新表phone添加栏目person.phone_id将数据从 复制person.phone_number到phone.phone_number将数据从 复制phone.id到person.phone_idperson在和之间创建外键约束phone降低person.phone_number代码可能是这样的:<createTable tableName="phone">    <column name="id" autoIncrement="true" type="bigserial">        <constraints primaryKey="true" primaryKeyName="pk_phone"/>    </column>    <column name="phone_number" type="varchar(30)"/>    <column name="brand" type="varchar(255)"/></createTable><addColumn tableName="person">    <column name="phone_id" type="bigserial"/></addColumn><update tableName="phone">    <column name="phone_number" valueComputed="(select p.phone_nubmer from person p)"/></update><comment>Assuming that person.phone_number is unique</comment><update tableName="person">    <column name="phone_id" valueComputed="(select p.id from person p where p.phone_number = phone_number)"/></update><addForeignKeyConstraint baseTableName="person" baseColumnNames="phone_id"    constraintName="person_phone_id_phone_id_fk"    referencedTableName="phone" referencedColumnNames="id"/><dropColumn tableName="person" columnName="phone_number"/>
打开App,查看更多内容
随时随地看视频慕课网APP

相关分类

Java