1

Golang migrate 做数据库变更管理

 2 years ago
source link: https://jiajunhuang.com/articles/2022_04_28-golang_migrate_iofs.md.html
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.
neoserver,ios ssh client

Golang migrate 做数据库变更管理

最近在使用 golang-migrate 做数据库变更管理,按照官方的教程,需要本地 先下载一个二进制,命令行生成变更文件:

$ migrate create -ext sql -dir db/migrations -seq create_article_table
...项目路径/db/migrations/000011_create_article_table.up.sql
...项目路径/db/migrations/000011_create_article_table.down.sql

这样,就会在 ./db/migrations 下有一堆的 .sql 文件,其中 000011 是编号,从 000001 开始递增, create_article_table 是传入的文件名,.up.sql 代表向前变更时执行的SQL,.down.sql 代表回滚时执行的SQL。

我们在编辑完之后,可以这样在本地变更数据库:

$ export POSTGRESQL_URL='postgres://postgres:密码@localhost:5432/dbname?sslmode=disable'
$ migrate -database ${POSTGRESQL_URL} -path db/migrations up
no change

但是在实际使用中,我们在发布项目之后,如果每次都还要去线上变更一下,那就比较麻烦了,能不能让代码自己来运行呢? 当然可以。

代码自动执行变更

migrate 既可以使用命令行,又可以以库的方式调用。migrate中,migration文件存放的地方,叫做 source,migrate支持多种 source例如 iofs, github, gitlab, s3 等等。

我想使用的就是官方的 embed,migrate封装为 iofs

package main

import (
    "context"
    "embed"
    "time"

    _ "github.com/golang-migrate/migrate/v4/database/postgres"
    "github.com/sirupsen/logrus"
)

var (
    //go:embed db/migrations/*.sql
    fs embed.FS
)

func initDB(config *Config) {
    var err error
    uri := fmt.Sprintf(
        "postgres://%s:%s@%s:%s/%s?sslmode=disable",
        config.DBUser, config.DBPassword, config.DBHost, config.DBPort, config.DBName,
    )
    ctx := context.Background()
    db, err = pgxpool.Connect(ctx, uri)
    if err != nil {
        logrus.Fatalf("connect to db failed: %v", err)
    }

    if err = db.Ping(ctx); err != nil {
        logrus.Fatalf("could not connect to database: %v", err)
    }

    d, err := iofs.New(fs, "db/migrations")
    if err != nil {
        logrus.Fatalf("could not open migrations: %v", err)
    }
    m, err := migrate.NewWithSourceInstance("iofs", d, uri)
    if err != nil {
        logrus.Fatalf("could not init migrate: %v", err)
    }
    err = m.Up()
    if err != nil {
        logrus.Errorf("migrate up error: %v", err)
    }
}

func main() {
    config := GetConfig()
    initDB(config)
}

也就是在应用启动以后,立刻初始化数据库,连接到数据库之后立刻开始执行数据库变更,看最上面的 var fs embed.FS 以及上一行的注释,这是Go官方提供的将文件打包到二进制的方式,本质是编译时将文件内容读取,编译后,赋值到 fs 变量里。

这样就可以在每次启动之后,自动先做数据库变更然后才开始执行代码了。但是有一点值得注意,跑数据库变更的程序,最好只部署 一份,否则容易出现竞争问题。

这篇文章记录了我使用migrate的方式,这种方式比所有操作都在命令行执行更加方便,也不再需要将sql文件同步到服务器,当然 缺点就是二进制文件会变的更大一些。有利有弊,不过我更倾向于这种方式。


微信公众号
关注公众号,获得及时更新

使用Tornado和rst来写博客

Haskell do notation

foldl 和 foldr 的变换

Haskell TypeClass 笔记

重新捡起你那吃灰的树莓派

Tornado 源码阅读

JavaScript权威指南笔记

Python零碎知识汇总

C语言的位操作

分治

关于python的decorator和descriptor

程序设计实践笔记

Thinking Recursively

Block I/O

如何解读c的声明




About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK