1 Star 0 Fork 0

chuang / gorm-shard

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
dialector.go 2.88 KB
一键复制 编辑 原始数据 按行查看 历史
chuang 提交于 2024-02-29 10:29 . 搬运工
package sharding
import (
"fmt"
"gorm.io/gorm/migrator"
"gorm.io/gorm/schema"
"gorm.io/gorm"
)
type ShardingDialector struct {
gorm.Dialector
sharding *Sharding
}
type ShardingMigrator struct {
gorm.Migrator
sharding *Sharding
dialector gorm.Dialector
}
func NewShardingDialector(d gorm.Dialector, s *Sharding) ShardingDialector {
return ShardingDialector{
Dialector: d,
sharding: s,
}
}
func (d ShardingDialector) Migrator(db *gorm.DB) gorm.Migrator {
m := d.Dialector.Migrator(db)
return ShardingMigrator{
Migrator: m,
sharding: d.sharding,
dialector: d.Dialector,
}
}
func (m ShardingMigrator) AutoMigrate(dst ...any) error {
shardingDsts, noShardingDsts, err := m.splitShardingDsts(dst...)
if err != nil {
return err
}
stmt := &gorm.Statement{DB: m.sharding.DB}
for _, sd := range shardingDsts {
tx := stmt.DB.Session(&gorm.Session{}).Table(sd.table)
if err := m.dialector.Migrator(tx).AutoMigrate(sd.dst); err != nil {
return err
}
}
if len(noShardingDsts) > 0 {
tx := stmt.DB.Session(&gorm.Session{})
tx.Statement.Settings.Store(ShardingIgnoreStoreKey, nil)
defer tx.Statement.Settings.Delete(ShardingIgnoreStoreKey)
if err := m.dialector.Migrator(tx).AutoMigrate(noShardingDsts...); err != nil {
return err
}
}
return nil
}
// BuildIndexOptions build index options
func (m ShardingMigrator) BuildIndexOptions(opts []schema.IndexOption, stmt *gorm.Statement) (results []interface{}) {
return m.Migrator.(migrator.BuildIndexOptionsInterface).BuildIndexOptions(opts, stmt)
}
func (m ShardingMigrator) DropTable(dst ...any) error {
shardingDsts, noShardingDsts, err := m.splitShardingDsts(dst...)
if err != nil {
return err
}
for _, sd := range shardingDsts {
if err := m.Migrator.DropTable(sd.table); err != nil {
return err
}
}
if len(noShardingDsts) > 0 {
if err := m.Migrator.DropTable(noShardingDsts...); err != nil {
return err
}
}
return nil
}
type shardingDst struct {
table string
dst any
}
// splite sharding or normal dsts
func (m ShardingMigrator) splitShardingDsts(dsts ...any) (shardingDsts []shardingDst,
noShardingDsts []any, err error) {
shardingDsts = make([]shardingDst, 0)
noShardingDsts = make([]any, 0)
for _, model := range dsts {
stmt := &gorm.Statement{DB: m.sharding.DB}
err = stmt.Parse(model)
if err != nil {
return
}
if cfg, ok := m.sharding.configs[stmt.Table]; ok {
// support sharding table
suffixs := cfg.ShardingSuffixs()
if len(suffixs) == 0 {
err = fmt.Errorf("sharding table:%s suffixs is empty", stmt.Table)
return
}
for _, suffix := range suffixs {
shardingTable := stmt.Table + suffix
shardingDsts = append(shardingDsts, shardingDst{
table: shardingTable,
dst: model,
})
}
if cfg.DoubleWrite {
noShardingDsts = append(noShardingDsts, model)
}
} else {
noShardingDsts = append(noShardingDsts, model)
}
}
return
}
Go
1
https://gitee.com/hellochuang/gorm-shard.git
git@gitee.com:hellochuang/gorm-shard.git
hellochuang
gorm-shard
gorm-shard
v1.0.0

搜索帮助

53164aa7 5694891 3bd8fe86 5694891