You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
|
|
|
package utils
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
|
|
|
|
|
|
|
"gorm.io/gorm"
|
|
|
|
"gorm.io/gorm/schema"
|
|
|
|
)
|
|
|
|
|
|
|
|
func FindAndInsert[V2 any, V3 schema.Tabler](v2db *gorm.DB, v3db *gorm.DB, fn func(V2) (V3, bool)) (string, error) {
|
|
|
|
const batchSize = 100
|
|
|
|
var t V3
|
|
|
|
name := t.TableName()
|
|
|
|
if err := v3db.AutoMigrate(&t); err != nil {
|
|
|
|
return name, fmt.Errorf("auto migrate v3 %s failed %w", name, err)
|
|
|
|
}
|
|
|
|
for i := 0; ; i++ {
|
|
|
|
var v2s []V2
|
|
|
|
if err := v2db.Offset(i * batchSize).Limit(batchSize).Find(&v2s).Error; err != nil {
|
|
|
|
return name, fmt.Errorf("find v2 %s failed %w", name, err)
|
|
|
|
}
|
|
|
|
if len(v2s) == 0 {
|
|
|
|
return name, nil
|
|
|
|
}
|
|
|
|
v3s := make([]V3, 0, len(v2s))
|
|
|
|
for _, v := range v2s {
|
|
|
|
res, ok := fn(v)
|
|
|
|
if ok {
|
|
|
|
v3s = append(v3s, res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(v3s) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err := v3db.Create(&v3s).Error; err != nil {
|
|
|
|
return name, fmt.Errorf("insert v3 %s failed %w", name, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type TakeList []Task
|
|
|
|
|
|
|
|
func (l *TakeList) Append(fn ...Task) {
|
|
|
|
*l = append(*l, fn...)
|
|
|
|
}
|
|
|
|
|
|
|
|
type Task func() (string, error)
|
|
|
|
|
|
|
|
func RunTask(concurrency int, tasks TakeList) []string {
|
|
|
|
if len(tasks) == 0 {
|
|
|
|
return []string{}
|
|
|
|
}
|
|
|
|
if concurrency < 1 {
|
|
|
|
concurrency = 1
|
|
|
|
}
|
|
|
|
if concurrency > len(tasks) {
|
|
|
|
concurrency = len(tasks)
|
|
|
|
}
|
|
|
|
|
|
|
|
taskCh := make(chan func() (string, error), 4)
|
|
|
|
go func() {
|
|
|
|
defer close(taskCh)
|
|
|
|
for i := range tasks {
|
|
|
|
taskCh <- tasks[i]
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
var lock sync.Mutex
|
|
|
|
var failedTables []string
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(concurrency)
|
|
|
|
var count int64
|
|
|
|
|
|
|
|
for i := 0; i < concurrency; i++ {
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
for task := range taskCh {
|
|
|
|
name, err := task()
|
|
|
|
index := atomic.AddInt64(&count, 1)
|
|
|
|
if err == nil {
|
|
|
|
log.Printf("[%d/%d] %s success\n", index, len(tasks), name)
|
|
|
|
} else {
|
|
|
|
lock.Lock()
|
|
|
|
failedTables = append(failedTables, name)
|
|
|
|
lock.Unlock()
|
|
|
|
log.Printf("[%d/%d] %s failed %s\n", index, len(tasks), name, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
if len(failedTables) == 0 {
|
|
|
|
log.Println("all tables success")
|
|
|
|
} else {
|
|
|
|
log.Printf("failed tables %d: %+v\n", len(failedTables), failedTables)
|
|
|
|
}
|
|
|
|
|
|
|
|
return failedTables
|
|
|
|
}
|