File size: 4,163 Bytes
7107f0b |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 |
package db
import (
"fmt"
"github.com/alist-org/alist/v3/internal/model"
"github.com/pkg/errors"
"gorm.io/gorm"
)
// why don't need `cache` for storage?
// because all storage store in `op.storagesMap`
// the most of the read operation is from `op.storagesMap`
// just for persistence in database
// CreateStorage just insert storage to database
func CreateStorage(storage *model.Storage) error {
return errors.WithStack(db.Create(storage).Error)
}
// UpdateStorage just update storage in database
func UpdateStorage(storage *model.Storage) error {
return errors.WithStack(db.Save(storage).Error)
}
// DeleteStorageById just delete storage from database by id
func DeleteStorageById(id uint) error {
return errors.WithStack(db.Delete(&model.Storage{}, id).Error)
}
// GetStorages Get all storages from database order by index
func GetStorages(pageIndex, pageSize int) ([]model.Storage, int64, error) {
storageDB := db.Model(&model.Storage{})
var count int64
if err := storageDB.Count(&count).Error; err != nil {
return nil, 0, errors.Wrapf(err, "failed get storages count")
}
var storages []model.Storage
if err := storageDB.Order(columnName("order")).Offset((pageIndex - 1) * pageSize).Limit(pageSize).Find(&storages).Error; err != nil {
return nil, 0, errors.WithStack(err)
}
return storages, count, nil
}
// GetStorageById Get Storage by id, used to update storage usually
func GetStorageById(id uint) (*model.Storage, error) {
var storage model.Storage
storage.ID = id
if err := db.First(&storage).Error; err != nil {
return nil, errors.WithStack(err)
}
return &storage, nil
}
// GetStorageByMountPath Get Storage by mountPath, used to update storage usually
func GetStorageByMountPath(mountPath string) (*model.Storage, error) {
var storage model.Storage
if err := db.Where("mount_path = ?", mountPath).First(&storage).Error; err != nil {
return nil, errors.WithStack(err)
}
return &storage, nil
}
func GetEnabledStorages() ([]model.Storage, error) {
var storages []model.Storage
if err := db.Where(fmt.Sprintf("%s = ?", columnName("disabled")), false).Find(&storages).Error; err != nil {
return nil, errors.WithStack(err)
}
return storages, nil
}
func GetGroupStorages(groupName string) ([]model.Storage, error) {
var storages []model.Storage
if err := db.Where(fmt.Sprintf("%s = ?", columnName("group")), groupName).Find(&storages).Error; err != nil {
return nil, errors.WithStack(err)
}
return storages, nil
}
func UpdateGroupStorages(groupName string, changedAdditions map[string]interface{}) error {
var storages []model.Storage
if err := db.Where(fmt.Sprintf("%s = ?", columnName("group")), groupName).Find(&storages).Error; err != nil {
return errors.WithStack(err)
}
// 动态构建 SQL 表达式
ids := extractField(storages, func(u model.Storage) int { return int(u.ID) }) //提取同组存储的id组成数组
//方案一:更新数据为字段名:新值类型
expr := "addition"
var args []interface{}
for key, val := range changedAdditions {
expr = fmt.Sprintf("JSON_SET(%s, ?, ?)", expr)
args = append(args, "$."+key, val)
}
updates := map[string]interface{}{
"addition": gorm.Expr(expr, args...),
}
// 执行更新
if updateErr := db.Model(&model.Storage{}).Where("id IN ?", ids).Updates(updates).Error; updateErr != nil {
return errors.WithStack(updateErr)
}
// 方案二 更新数据为旧数据:新数据类型
// var keys []string
// for oldStr := range changedAdditions {
// keys = append(keys, oldStr)
// }
// sort.Strings(keys) // 按字典序排序
// // 按排序后的键遍历
// expr := "addition"
// for _, oldStr := range keys {
// newStr := changedAdditions[oldStr]
// expr = fmt.Sprintf("REPLACE(%s, '%s', '%s')", expr, oldStr, newStr)
// }
// if updateErr := db.Model(&model.Storage{}).Where("id IN ?", ids).Update("addition", gorm.Expr(expr)).Error; updateErr != nil {
// return errors.WithStack(updateErr)
// }
return nil
}
func extractField[T any, F any](slice []T, getter func(T) F) []F {
result := make([]F, 0, len(slice))
for _, item := range slice {
result = append(result, getter(item))
}
return result
}
|