Liquibase-數(shù)據(jù)庫(kù)腳本版本管理控制

簡(jiǎn)介

Liquibase是一個(gè)用于跟蹤荷憋、管理和應(yīng)用數(shù)據(jù)庫(kù)變化的開源的數(shù)據(jù)庫(kù)重構(gòu)工具衔峰。它將所有數(shù)據(jù)庫(kù)的變化(包括結(jié)構(gòu)和數(shù)據(jù))都保存在XML文件中,便于版本控制。

Liquibase使參與應(yīng)用程序發(fā)布過(guò)程的任何人都可以輕松地:
  • 不依賴于特定的數(shù)據(jù)庫(kù),Liquibase會(huì)自動(dòng)適配目標(biāo)數(shù)據(jù)庫(kù)進(jìn)行腳本初始化默责,目前支持至少30種主流數(shù)據(jù)庫(kù)。
  • 提供數(shù)據(jù)庫(kù)比較功能么库,比較結(jié)果保存在XML中傻丝,基于該XML可以用Liquibase輕松部署或升級(jí)數(shù)據(jù)庫(kù)。
  • XML記錄/存儲(chǔ)數(shù)據(jù)庫(kù)變化诉儒,其中以authorid唯一標(biāo)識(shí)一個(gè)變化(ChangSet)葡缰,支持?jǐn)?shù)據(jù)庫(kù)變化的合并,因此支持多開發(fā)人員同時(shí)工作。
  • 在數(shù)據(jù)庫(kù)中保存數(shù)據(jù)庫(kù)修改歷史(DatabaseChangeHistory)泛释,在數(shù)據(jù)庫(kù)升級(jí)時(shí)自動(dòng)跳過(guò)已應(yīng)用的變化(ChangSet)滤愕。
  • 提供變化應(yīng)用的回滾功能,可按時(shí)間怜校、數(shù)量或標(biāo)簽(tag)回滾已應(yīng)用的變化间影。通過(guò)這種方式,開發(fā)人員可輕易的還原數(shù)據(jù)庫(kù)在任何時(shí)間點(diǎn)的狀態(tài)茄茁。
  • 可生成數(shù)據(jù)庫(kù)修改文檔(HTML格式)
  • 將所有變化(包括結(jié)構(gòu)和數(shù)據(jù))存在XML文件中魂贬,便于版本控制的工具
  • 支持Springboot,只需要導(dǎo)入依賴裙顽。
  • application.yml配置(可選)
  • 不配置付燥,默認(rèn)去resource/db/changelog下找db.changelog-mastert.yaml文件

Quick Start

  • step1:通過(guò)PowerDesigner等數(shù)據(jù)庫(kù)建模工具生成數(shù)據(jù)庫(kù)建表語(yǔ)句
  • step2:通過(guò)命令行對(duì)數(shù)據(jù)庫(kù)生成(全量)變更集(change set)
  • step3:通過(guò)命令行對(duì)數(shù)據(jù)庫(kù)生成(增量)變更集(change set)
  • step4:校驗(yàn)數(shù)據(jù)庫(kù)中的變更
圖一

SpringBoot 、Maven 集成

添加maven依賴
<parent>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-parent</artifactId>
      <version>2.2.2.RELEASE</version>
      <relativePath/> <!-- lookup parent from repository -->
</parent>

<properties>
        <liquibase.version>4.8.0</liquibase.version>
</properties>

<dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-jdbc</artifactId>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.13</version>
        </dependency>
        <dependency>
            <groupId>org.liquibase</groupId>
            <artifactId>liquibase-core</artifactId>
            <version>${liquibase.version}</version>
        </dependency>
    </dependencies>

<build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <configuration>
                    <fork>true</fork>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.liquibase</groupId>
                <artifactId>liquibase-maven-plugin</artifactId>
                <version>${liquibase.version}</version>
                <configuration>
                    <propertyFile>src/main/resources/liquibase.properties</propertyFile>
                    <diffChangeLogFile>${basedir}/src/main/resources/config/liquibase/changelog/${project.version}_changelog.xml</diffChangeLogFile>
                    <outputChangeLogFile>${basedir}/src/main/resources/config/liquibase/changelog/generate_changelog.xml</outputChangeLogFile>
                    <!-- 是否需要彈出確認(rèn)框 -->
                    <promptOnNonLocalDatabase>false</promptOnNonLocalDatabase>
                    <!--輸出文件的編碼 -->
                    <outputFileEncoding>UTF-8</outputFileEncoding>
                </configuration>
                <executions>
                    <execution>
                        <phase>process-resources</phase>
                        <goals>
                            <goal>diff</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>
application.properties
spring.datasource.username=root
spring.datasource.password=123456
spring.datasource.url=jdbc:mysql://localhost:3306/world1?serverTimezone=UTC
spring.datasource.driver-class-name=com.mysql.jdbc.Driver

# 是否啟用
spring.liquibase.enabled=true
 # 指定配置文件路徑
spring.liquibase.change-log=classpath:/config/liquibase/master.xml
liquibase.properties
changeLogFile=src/main/resources/config/liquibase/changelog/0.0.1-SNAPSHOT_changelog.xml
zone=UTC
verbose=true
 # 覆蓋本地 ddl dml
dropFirst=false

#目標(biāo)數(shù)據(jù)庫(kù)
driver=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/world1?serverTimezone=UTC
username=root
password=123456

#引用數(shù)據(jù)庫(kù)
referenceDriver=com.mysql.cj.jdbc.Driver
referenceUrl=jdbc:mysql://localhost:3306/world?serverTimezone=UTC
referenceUsername=root
referencePassword=123456
添加/生成 liquibase xml

master.xml

liquibase 配置文件入口愈犹,主要用來(lái)引用其他的changelog.xml键科,如下配置文件中的include,當(dāng)然也可以使用includeAll來(lái)引用整個(gè)目錄

<?xml version="1.0" encoding="utf-8"?>
<databaseChangeLog
        xmlns="http://www.liquibase.org/xml/ns/dbchangelog"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-3.5.xsd">

    <!-- 項(xiàng)目自身的sql腳本 -->
    <include file="config/liquibase/changelog/0.0.1-SNAPSHOT_changelog.xml" relativeToChangelogFile="false"/>
<!--    <includeAll path="config/liquibase/changelog" relativeToChangelogFile="false"/>-->
</databaseChangeLog>

generate_changelog.xml

使用mvn 插件逆向生成xml漩怎,主要記錄了ddl的變化信息勋颖,比如 如下配置文件中需要?jiǎng)?chuàng)建的表

圖二
啟動(dòng)測(cè)試
圖三
測(cè)試changeset版本控制

我們?cè)趕pms_tenant_user_attr_config表中修改一個(gè)type字段的長(zhǎng)度,便直接在之前的changeset中修改了字段勋锤,如下圖所示饭玲,然后啟動(dòng)項(xiàng)目看結(jié)果


圖四

報(bào)錯(cuò)信息是我們直接修改了changeset后導(dǎo)致md5值與之前的不匹配(直接在之前的changeset中做了修改)

答疑
  • 問:liquibase如何判斷 是同一changeset的?
  • 答:authorid唯一標(biāo)識(shí)一個(gè)變化(ChangSet

  • 問: liquibase是如何進(jìn)行changeset版本控制的叁执?
  • 答:Liquibase會(huì)對(duì)已經(jīng)執(zhí)行的changelog的每一個(gè)changeSet的內(nèi)容進(jìn)行md5計(jì)算咱枉,生成的值是databasechanglog表的MD5SUM字段。
    當(dāng)重新啟動(dòng)Liquibase時(shí)徒恋,會(huì)對(duì)每個(gè)changeSet進(jìn)行md5值計(jì)算,與databasechanglog表中的MD5SUM字段進(jìn)行對(duì)比欢伏,如果不一致入挣,說(shuō)明changeSet值已經(jīng)被修改,無(wú)法啟動(dòng)成功硝拧。
plugin-比較數(shù)據(jù)庫(kù)差異

首先使用liquibase diff 功能前径筏,我們?cè)?code>liquibase.properties中加入?yún)⒖嫉臄?shù)據(jù)庫(kù) world配置信息,用于差異比較的數(shù)據(jù)庫(kù)障陶,以此數(shù)據(jù)庫(kù)為準(zhǔn)滋恬,生成diff xml;然后再用liquibase插件看下生成的 diff xml信息抱究;執(zhí)行完畢后恢氯,查看diff xml 內(nèi)容如下:

圖五

plugin-生成增量的changelog SQL

讀取diff生成的差異changeset,輸出增加更新的SQL文件

圖六

圖七

總結(jié)

  • 首次使用liquibase,可以直接根據(jù)數(shù)據(jù)庫(kù)逆向生成changeSet xml
  • 增量使用liquibase勋拟,使用diff命令生成需要增量更新的changeSet xml(也可用 updateSQL 生成可執(zhí)行的SQL語(yǔ)句)
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末勋磕,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子敢靡,更是在濱河造成了極大的恐慌挂滓,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,509評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件啸胧,死亡現(xiàn)場(chǎng)離奇詭異赶站,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)纺念,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門贝椿,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人柠辞,你說(shuō)我怎么就攤上這事团秽。” “怎么了叭首?”我有些...
    開封第一講書人閱讀 163,875評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵习勤,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我焙格,道長(zhǎng)图毕,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,441評(píng)論 1 293
  • 正文 為了忘掉前任眷唉,我火速辦了婚禮予颤,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘冬阳。我一直安慰自己蛤虐,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,488評(píng)論 6 392
  • 文/花漫 我一把揭開白布肝陪。 她就那樣靜靜地躺著驳庭,像睡著了一般。 火紅的嫁衣襯著肌膚如雪氯窍。 梳的紋絲不亂的頭發(fā)上饲常,一...
    開封第一講書人閱讀 51,365評(píng)論 1 302
  • 那天,我揣著相機(jī)與錄音狼讨,去河邊找鬼贝淤。 笑死,一個(gè)胖子當(dāng)著我的面吹牛政供,可吹牛的內(nèi)容都是我干的播聪。 我是一名探鬼主播朽基,決...
    沈念sama閱讀 40,190評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼犬耻!你這毒婦竟也來(lái)了踩晶?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,062評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤枕磁,失蹤者是張志新(化名)和其女友劉穎渡蜻,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體计济,經(jīng)...
    沈念sama閱讀 45,500評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡茸苇,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,706評(píng)論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了沦寂。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片学密。...
    茶點(diǎn)故事閱讀 39,834評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖传藏,靈堂內(nèi)的尸體忽然破棺而出腻暮,到底是詐尸還是另有隱情,我是刑警寧澤毯侦,帶...
    沈念sama閱讀 35,559評(píng)論 5 345
  • 正文 年R本政府宣布哭靖,位于F島的核電站,受9級(jí)特大地震影響侈离,放射性物質(zhì)發(fā)生泄漏试幽。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,167評(píng)論 3 328
  • 文/蒙蒙 一卦碾、第九天 我趴在偏房一處隱蔽的房頂上張望铺坞。 院中可真熱鬧,春花似錦洲胖、人聲如沸济榨。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,779評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)腿短。三九已至,卻和暖如春绘梦,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背赴魁。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工卸奉, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人颖御。 一個(gè)月前我還...
    沈念sama閱讀 47,958評(píng)論 2 370
  • 正文 我出身青樓榄棵,卻偏偏與公主長(zhǎng)得像凝颇,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子疹鳄,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,779評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容