migrent is a library to manage data migrations for applications using ent.
Ent is one of the most popular ORM libraries for Golang, and automatic schema migration is already supported.
However, with regard to data migration/incremental migration history management, there is some room for adding more functionalities.
When an application using ent needs master data (=data needed to be registered in advance), the users need to have the code to manage the data by themselves because Upsert
is not yet implemented, and it is not possible to "Up" and "Down" migration like
existing tools such as goose, sql-migrate,
and golang-migrate.
migrent enables the apply and roleback of data migrations by making an internal "migration" entity in a DB, while keeping the great part of ent, such as type-safety.
go get github.com/dakimura/migrent
package main
import (
"context"
"github.com/dakimura/migrent/example"
"github.com/dakimura/migrent/example/ent"
"github.com/dakimura/migrent"
_ "github.com/go-sql-driver/mysql"
)
// example master data definition
var masterData = []ent.User{
{Age: 12, Name: "Alice"},
{Age: 24, Name: "Bob"},
{Age: 36, Name: "Carol"},
{Age: 48, Name: "David"},
{Age: 60, Name: "Eve"},
}
func main() {
ctx := context.Background()
dialect := "mysql"
dsn := "user:password@tcp(IPAddress:Port)/dbname?parse=True"
// initialize ent client for your DB
userDBCli, _ := ent.Open(dialect, dsn)
// initialize migrent client (you can use the same dsn).
// because migrent creates an internal entity to manage the migration history
// in the DB, an ent client for it is necessary
cli, _ := migrent.Open(dialect, dsn)
// define your migration
migrations := map[migrent.MigrationName]migrent.Migration{
"user_data1": example.NewUserMasterDataMigration(masterData, userDBCli.User),
}
// --- execute migration(Up)
cli.Up(ctx, migrations)
// --- execute migration(Down)
cli.Down(ctx, migrations)
}
You need to define what migration you need when Up
or Down
is called.
type Migration interface {
Up(ctx context.Context) error
Down(ctx context.Context) error
}
It doesn't have to be idempotent because migrent manages whether each migration has already been applied. The following is the simple example of a migration that inserts master data to User entity.
package example
import (
"context"
"fmt"
"github.com/dakimura/migrent/example/ent/user"
ent "github.com/dakimura/migrent/example/ent"
)
type UserMasterDataMigration struct {
Data []ent.User
Client *ent.UserClient
}
func NewUserMasterDataMigration(data []ent.User, client *ent.UserClient,
) *UserMasterDataMigration {
return &UserMasterDataMigration{
Data: data,
Client: client,
}
}
// Up inserts data to User entity
func (m *UserMasterDataMigration) Up(ctx context.Context) error {
data := make([]*ent.UserCreate, len(m.Data))
for i, rec := range m.Data {
data[i] = m.Client.Create().SetAge(rec.Age).SetName(rec.Name)
}
_, err := m.Client.CreateBulk(data...).Save(ctx)
if err != nil {
return fmt.Errorf("create User entities: %w", err)
}
return nil
}
// Down deletes data from User entity
func (m *UserMasterDataMigration) Down(ctx context.Context) error {
users := make([]string, len(m.Data))
for i, u := range m.Data {
users[i] = u.Name
}
_, err := m.Client.
Delete().
Where(user.NameIn(users...)).Exec(ctx)
if err != nil {
return fmt.Errorf("delete User entities: %w", err)
}
return nil
}
migrent is licensed under MIT as found in the LICENSE file.