mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-31 08:29:33 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			120 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			120 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright © 2023 OpenIM. All rights reserved.
 | |
| //
 | |
| // Licensed under the Apache License, Version 2.0 (the "License");
 | |
| // you may not use this file except in compliance with the License.
 | |
| // You may obtain a copy of the License at
 | |
| //
 | |
| //     http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, software
 | |
| // distributed under the License is distributed on an "AS IS" BASIS,
 | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| // See the License for the specific language governing permissions and
 | |
| // limitations under the License.
 | |
| 
 | |
| 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
 | |
| }
 |