|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240 |
- package cmd
-
- import (
- "context"
- "fmt"
- "io"
- "math"
- "math/rand"
- "time"
-
- "github.com/samber/lo"
-
- "gitlink.org.cn/cloudream/common/pkgs/distlock"
- "gitlink.org.cn/cloudream/common/pkgs/ioswitch/exec"
- cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
- "gitlink.org.cn/cloudream/common/utils/sort2"
-
- stgglb "gitlink.org.cn/cloudream/storage/common/globals"
- stgmod "gitlink.org.cn/cloudream/storage/common/models"
- "gitlink.org.cn/cloudream/storage/common/pkgs/connectivity"
- "gitlink.org.cn/cloudream/storage/common/pkgs/distlock/reqbuilder"
- "gitlink.org.cn/cloudream/storage/common/pkgs/ioswitch2"
- "gitlink.org.cn/cloudream/storage/common/pkgs/ioswitch2/ops2"
- "gitlink.org.cn/cloudream/storage/common/pkgs/ioswitch2/parser"
- "gitlink.org.cn/cloudream/storage/common/pkgs/iterator"
- coormq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/coordinator"
- "gitlink.org.cn/cloudream/storage/common/pkgs/storage/mgr"
- )
-
- type UploadObjects struct {
- userID cdssdk.UserID
- packageID cdssdk.PackageID
- objectIter iterator.UploadingObjectIterator
- stgAffinity cdssdk.StorageID
- }
-
- type UploadObjectsResult struct {
- Objects []ObjectUploadResult
- }
-
- type ObjectUploadResult struct {
- Info *iterator.IterUploadingObject
- Error error
- Object cdssdk.Object
- }
-
- type UploadStorageInfo struct {
- Storage stgmod.StorageDetail
- Delay time.Duration
- IsSameLocation bool
- }
-
- type UploadObjectsContext struct {
- Distlock *distlock.Service
- Connectivity *connectivity.Collector
- StgMgr *mgr.Manager
- }
-
- func NewUploadObjects(userID cdssdk.UserID, packageID cdssdk.PackageID, objIter iterator.UploadingObjectIterator, stgAffinity cdssdk.StorageID) *UploadObjects {
- return &UploadObjects{
- userID: userID,
- packageID: packageID,
- objectIter: objIter,
- stgAffinity: stgAffinity,
- }
- }
-
- func (t *UploadObjects) Execute(ctx *UploadObjectsContext) (*UploadObjectsResult, error) {
- defer t.objectIter.Close()
-
- coorCli, err := stgglb.CoordinatorMQPool.Acquire()
- if err != nil {
- return nil, fmt.Errorf("new coordinator client: %w", err)
- }
-
- getUserStgsResp, err := coorCli.GetUserStorageDetails(coormq.ReqGetUserStorageDetails(t.userID))
- if err != nil {
- return nil, fmt.Errorf("getting user storages: %w", err)
- }
-
- cons := ctx.Connectivity.GetAll()
- var userStgs []UploadStorageInfo
- for _, stg := range getUserStgsResp.Storages {
- if stg.MasterHub == nil {
- continue
- }
-
- delay := time.Duration(math.MaxInt64)
-
- con, ok := cons[stg.MasterHub.HubID]
- if ok && con.Delay != nil {
- delay = *con.Delay
- }
-
- userStgs = append(userStgs, UploadStorageInfo{
- Storage: stg,
- Delay: delay,
- IsSameLocation: stg.MasterHub.LocationID == stgglb.Local.LocationID,
- })
- }
-
- if len(userStgs) == 0 {
- return nil, fmt.Errorf("user no available storages")
- }
-
- // 给上传节点的IPFS加锁
- lockBlder := reqbuilder.NewBuilder()
- for _, us := range userStgs {
- lockBlder.Shard().Buzy(us.Storage.Storage.StorageID)
- }
- // TODO 考虑加Object的Create锁
- // 防止上传的副本被清除
- ipfsMutex, err := lockBlder.MutexLock(ctx.Distlock)
- if err != nil {
- return nil, fmt.Errorf("acquire locks failed, err: %w", err)
- }
- defer ipfsMutex.Unlock()
-
- rets, err := uploadAndUpdatePackage(ctx, t.packageID, t.objectIter, userStgs, t.stgAffinity)
- if err != nil {
- return nil, err
- }
-
- return &UploadObjectsResult{
- Objects: rets,
- }, nil
- }
-
- // chooseUploadStorage 选择一个上传文件的节点
- // 1. 选择设置了亲和性的节点
- // 2. 从与当前客户端相同地域的节点中随机选一个
- // 3. 没有的话从所有节点选择延迟最低的节点
- func chooseUploadStorage(storages []UploadStorageInfo, stgAffinity cdssdk.StorageID) UploadStorageInfo {
- if stgAffinity > 0 {
- aff, ok := lo.Find(storages, func(storage UploadStorageInfo) bool { return storage.Storage.Storage.StorageID == stgAffinity })
- if ok {
- return aff
- }
- }
-
- sameLocationStorages := lo.Filter(storages, func(e UploadStorageInfo, i int) bool { return e.IsSameLocation })
- if len(sameLocationStorages) > 0 {
- return sameLocationStorages[rand.Intn(len(sameLocationStorages))]
- }
-
- // 选择延迟最低的节点
- storages = sort2.Sort(storages, func(e1, e2 UploadStorageInfo) int { return sort2.Cmp(e1.Delay, e2.Delay) })
-
- return storages[0]
- }
-
- func uploadAndUpdatePackage(ctx *UploadObjectsContext, packageID cdssdk.PackageID, objectIter iterator.UploadingObjectIterator, userStorages []UploadStorageInfo, stgAffinity cdssdk.StorageID) ([]ObjectUploadResult, error) {
- coorCli, err := stgglb.CoordinatorMQPool.Acquire()
- if err != nil {
- return nil, fmt.Errorf("new coordinator client: %w", err)
- }
- defer stgglb.CoordinatorMQPool.Release(coorCli)
-
- // 为所有文件选择相同的上传节点
- uploadStorage := chooseUploadStorage(userStorages, stgAffinity)
-
- var uploadRets []ObjectUploadResult
- //上传文件夹
- var adds []coormq.AddObjectEntry
- for {
- objInfo, err := objectIter.MoveNext()
- if err == iterator.ErrNoMoreItem {
- break
- }
- if err != nil {
- return nil, fmt.Errorf("reading object: %w", err)
- }
- err = func() error {
- defer objInfo.File.Close()
-
- uploadTime := time.Now()
- fileHash, err := uploadFile(ctx, objInfo.File, uploadStorage)
- if err != nil {
- return fmt.Errorf("uploading file: %w", err)
- }
-
- uploadRets = append(uploadRets, ObjectUploadResult{
- Info: objInfo,
- Error: err,
- })
-
- adds = append(adds, coormq.NewAddObjectEntry(objInfo.Path, objInfo.Size, fileHash, uploadTime, uploadStorage.Storage.Storage.StorageID))
- return nil
- }()
- if err != nil {
- return nil, err
- }
- }
-
- updateResp, err := coorCli.UpdatePackage(coormq.NewUpdatePackage(packageID, adds, nil))
- if err != nil {
- return nil, fmt.Errorf("updating package: %w", err)
- }
-
- updatedObjs := make(map[string]*cdssdk.Object)
- for _, obj := range updateResp.Added {
- o := obj
- updatedObjs[obj.Path] = &o
- }
-
- for i := range uploadRets {
- obj := updatedObjs[uploadRets[i].Info.Path]
- if obj == nil {
- uploadRets[i].Error = fmt.Errorf("object %s not found in package", uploadRets[i].Info.Path)
- continue
- }
- uploadRets[i].Object = *obj
- }
-
- return uploadRets, nil
- }
-
- func uploadFile(ctx *UploadObjectsContext, file io.Reader, uploadStg UploadStorageInfo) (cdssdk.FileHash, error) {
- ft := ioswitch2.NewFromTo()
- fromExec, hd := ioswitch2.NewFromDriver(-1)
- ft.AddFrom(fromExec).AddTo(ioswitch2.NewToShardStore(*uploadStg.Storage.MasterHub, uploadStg.Storage.Storage, -1, "fileHash"))
-
- parser := parser.NewParser(cdssdk.DefaultECRedundancy)
- plans := exec.NewPlanBuilder()
- err := parser.Parse(ft, plans)
- if err != nil {
- return "", fmt.Errorf("parsing plan: %w", err)
- }
-
- exeCtx := exec.NewExecContext()
- exec.SetValueByType(exeCtx, ctx.StgMgr)
- exec := plans.Execute(exeCtx)
- exec.BeginWrite(io.NopCloser(file), hd)
- ret, err := exec.Wait(context.TODO())
- if err != nil {
- return "", err
- }
-
- return ret["fileHash"].(*ops2.FileHashValue).Hash, nil
- }
|