You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

upload_objects.go 6.7 kB

2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. package cmd
  2. import (
  3. "context"
  4. "fmt"
  5. "io"
  6. "math"
  7. "math/rand"
  8. "time"
  9. "github.com/samber/lo"
  10. "gitlink.org.cn/cloudream/common/pkgs/distlock"
  11. "gitlink.org.cn/cloudream/common/pkgs/ioswitch/exec"
  12. cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
  13. "gitlink.org.cn/cloudream/common/utils/sort2"
  14. stgglb "gitlink.org.cn/cloudream/storage/common/globals"
  15. "gitlink.org.cn/cloudream/storage/common/pkgs/connectivity"
  16. "gitlink.org.cn/cloudream/storage/common/pkgs/distlock/reqbuilder"
  17. "gitlink.org.cn/cloudream/storage/common/pkgs/ioswitch2"
  18. "gitlink.org.cn/cloudream/storage/common/pkgs/ioswitch2/parser"
  19. "gitlink.org.cn/cloudream/storage/common/pkgs/iterator"
  20. coormq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/coordinator"
  21. )
  22. type UploadObjects struct {
  23. userID cdssdk.UserID
  24. packageID cdssdk.PackageID
  25. objectIter iterator.UploadingObjectIterator
  26. nodeAffinity *cdssdk.NodeID
  27. }
  28. type UploadObjectsResult struct {
  29. Objects []ObjectUploadResult
  30. }
  31. type ObjectUploadResult struct {
  32. Info *iterator.IterUploadingObject
  33. Error error
  34. Object cdssdk.Object
  35. }
  36. type UploadNodeInfo struct {
  37. Node cdssdk.Node
  38. Delay time.Duration
  39. IsSameLocation bool
  40. }
  41. type UploadObjectsContext struct {
  42. Distlock *distlock.Service
  43. Connectivity *connectivity.Collector
  44. }
  45. func NewUploadObjects(userID cdssdk.UserID, packageID cdssdk.PackageID, objIter iterator.UploadingObjectIterator, nodeAffinity *cdssdk.NodeID) *UploadObjects {
  46. return &UploadObjects{
  47. userID: userID,
  48. packageID: packageID,
  49. objectIter: objIter,
  50. nodeAffinity: nodeAffinity,
  51. }
  52. }
  53. func (t *UploadObjects) Execute(ctx *UploadObjectsContext) (*UploadObjectsResult, error) {
  54. defer t.objectIter.Close()
  55. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  56. if err != nil {
  57. return nil, fmt.Errorf("new coordinator client: %w", err)
  58. }
  59. getUserNodesResp, err := coorCli.GetUserNodes(coormq.NewGetUserNodes(t.userID))
  60. if err != nil {
  61. return nil, fmt.Errorf("getting user nodes: %w", err)
  62. }
  63. cons := ctx.Connectivity.GetAll()
  64. userNodes := lo.Map(getUserNodesResp.Nodes, func(node cdssdk.Node, index int) UploadNodeInfo {
  65. delay := time.Duration(math.MaxInt64)
  66. con, ok := cons[node.NodeID]
  67. if ok && con.Delay != nil {
  68. delay = *con.Delay
  69. }
  70. return UploadNodeInfo{
  71. Node: node,
  72. Delay: delay,
  73. IsSameLocation: node.LocationID == stgglb.Local.LocationID,
  74. }
  75. })
  76. if len(userNodes) == 0 {
  77. return nil, fmt.Errorf("user no available nodes")
  78. }
  79. // 给上传节点的IPFS加锁
  80. ipfsReqBlder := reqbuilder.NewBuilder()
  81. // 如果本地的IPFS也是存储系统的一个节点,那么从本地上传时,需要加锁
  82. if stgglb.Local.NodeID != nil {
  83. ipfsReqBlder.IPFS().Buzy(*stgglb.Local.NodeID)
  84. }
  85. for _, node := range userNodes {
  86. if stgglb.Local.NodeID != nil && node.Node.NodeID == *stgglb.Local.NodeID {
  87. continue
  88. }
  89. ipfsReqBlder.IPFS().Buzy(node.Node.NodeID)
  90. }
  91. // TODO 考虑加Object的Create锁
  92. // 防止上传的副本被清除
  93. ipfsMutex, err := ipfsReqBlder.MutexLock(ctx.Distlock)
  94. if err != nil {
  95. return nil, fmt.Errorf("acquire locks failed, err: %w", err)
  96. }
  97. defer ipfsMutex.Unlock()
  98. rets, err := uploadAndUpdatePackage(t.packageID, t.objectIter, userNodes, t.nodeAffinity)
  99. if err != nil {
  100. return nil, err
  101. }
  102. return &UploadObjectsResult{
  103. Objects: rets,
  104. }, nil
  105. }
  106. // chooseUploadNode 选择一个上传文件的节点
  107. // 1. 选择设置了亲和性的节点
  108. // 2. 从与当前客户端相同地域的节点中随机选一个
  109. // 3. 没有的话从所有节点选择延迟最低的节点
  110. func chooseUploadNode(nodes []UploadNodeInfo, nodeAffinity *cdssdk.NodeID) UploadNodeInfo {
  111. if nodeAffinity != nil {
  112. aff, ok := lo.Find(nodes, func(node UploadNodeInfo) bool { return node.Node.NodeID == *nodeAffinity })
  113. if ok {
  114. return aff
  115. }
  116. }
  117. sameLocationNodes := lo.Filter(nodes, func(e UploadNodeInfo, i int) bool { return e.IsSameLocation })
  118. if len(sameLocationNodes) > 0 {
  119. return sameLocationNodes[rand.Intn(len(sameLocationNodes))]
  120. }
  121. // 选择延迟最低的节点
  122. nodes = sort2.Sort(nodes, func(e1, e2 UploadNodeInfo) int { return sort2.Cmp(e1.Delay, e2.Delay) })
  123. return nodes[0]
  124. }
  125. func uploadAndUpdatePackage(packageID cdssdk.PackageID, objectIter iterator.UploadingObjectIterator, userNodes []UploadNodeInfo, nodeAffinity *cdssdk.NodeID) ([]ObjectUploadResult, error) {
  126. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  127. if err != nil {
  128. return nil, fmt.Errorf("new coordinator client: %w", err)
  129. }
  130. defer stgglb.CoordinatorMQPool.Release(coorCli)
  131. // 为所有文件选择相同的上传节点
  132. uploadNode := chooseUploadNode(userNodes, nodeAffinity)
  133. var uploadRets []ObjectUploadResult
  134. //上传文件夹
  135. var adds []coormq.AddObjectEntry
  136. for {
  137. objInfo, err := objectIter.MoveNext()
  138. if err == iterator.ErrNoMoreItem {
  139. break
  140. }
  141. if err != nil {
  142. return nil, fmt.Errorf("reading object: %w", err)
  143. }
  144. err = func() error {
  145. defer objInfo.File.Close()
  146. uploadTime := time.Now()
  147. fileHash, err := uploadFile(objInfo.File, uploadNode)
  148. if err != nil {
  149. return fmt.Errorf("uploading file: %w", err)
  150. }
  151. uploadRets = append(uploadRets, ObjectUploadResult{
  152. Info: objInfo,
  153. Error: err,
  154. })
  155. adds = append(adds, coormq.NewAddObjectEntry(objInfo.Path, objInfo.Size, fileHash, uploadTime, uploadNode.Node.NodeID))
  156. return nil
  157. }()
  158. if err != nil {
  159. return nil, err
  160. }
  161. }
  162. updateResp, err := coorCli.UpdatePackage(coormq.NewUpdatePackage(packageID, adds, nil))
  163. if err != nil {
  164. return nil, fmt.Errorf("updating package: %w", err)
  165. }
  166. updatedObjs := make(map[string]*cdssdk.Object)
  167. for _, obj := range updateResp.Added {
  168. o := obj
  169. updatedObjs[obj.Path] = &o
  170. }
  171. for i := range uploadRets {
  172. obj := updatedObjs[uploadRets[i].Info.Path]
  173. if obj == nil {
  174. uploadRets[i].Error = fmt.Errorf("object %s not found in package", uploadRets[i].Info.Path)
  175. continue
  176. }
  177. uploadRets[i].Object = *obj
  178. }
  179. return uploadRets, nil
  180. }
  181. func uploadFile(file io.Reader, uploadNode UploadNodeInfo) (string, error) {
  182. ft := ioswitch2.NewFromTo()
  183. fromExec, hd := ioswitch2.NewFromDriver(-1)
  184. ft.AddFrom(fromExec).AddTo(ioswitch2.NewToNode(uploadNode.Node, -1, "fileHash"))
  185. parser := parser.NewParser(cdssdk.DefaultECRedundancy)
  186. plans := exec.NewPlanBuilder()
  187. err := parser.Parse(ft, plans)
  188. if err != nil {
  189. return "", fmt.Errorf("parsing plan: %w", err)
  190. }
  191. // TODO2 注入依赖
  192. exeCtx := exec.NewExecContext()
  193. exec := plans.Execute(exeCtx)
  194. exec.BeginWrite(io.NopCloser(file), hd)
  195. ret, err := exec.Wait(context.TODO())
  196. if err != nil {
  197. return "", err
  198. }
  199. return ret["fileHash"].(string), nil
  200. }

本项目旨在将云际存储公共基础设施化,使个人及企业可低门槛使用高效的云际存储服务(安装开箱即用云际存储客户端即可,无需关注其他组件的部署),同时支持用户灵活便捷定制云际存储的功能细节。