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 8.4 kB

2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  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. "gitlink.org.cn/cloudream/common/pkgs/logger"
  13. cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
  14. "gitlink.org.cn/cloudream/common/utils/sort2"
  15. stgglb "gitlink.org.cn/cloudream/storage/common/globals"
  16. "gitlink.org.cn/cloudream/storage/common/pkgs/connectivity"
  17. "gitlink.org.cn/cloudream/storage/common/pkgs/distlock/reqbuilder"
  18. "gitlink.org.cn/cloudream/storage/common/pkgs/ioswitch2"
  19. "gitlink.org.cn/cloudream/storage/common/pkgs/ioswitch2/parser"
  20. "gitlink.org.cn/cloudream/storage/common/pkgs/iterator"
  21. agtmq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/agent"
  22. coormq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/coordinator"
  23. )
  24. type UploadObjects struct {
  25. userID cdssdk.UserID
  26. packageID cdssdk.PackageID
  27. objectIter iterator.UploadingObjectIterator
  28. nodeAffinity *cdssdk.NodeID
  29. }
  30. type UploadObjectsResult struct {
  31. Objects []ObjectUploadResult
  32. }
  33. type ObjectUploadResult struct {
  34. Info *iterator.IterUploadingObject
  35. Error error
  36. Object cdssdk.Object
  37. }
  38. type UploadNodeInfo struct {
  39. Node cdssdk.Node
  40. Delay time.Duration
  41. IsSameLocation bool
  42. }
  43. type UploadObjectsContext struct {
  44. Distlock *distlock.Service
  45. Connectivity *connectivity.Collector
  46. }
  47. func NewUploadObjects(userID cdssdk.UserID, packageID cdssdk.PackageID, objIter iterator.UploadingObjectIterator, nodeAffinity *cdssdk.NodeID) *UploadObjects {
  48. return &UploadObjects{
  49. userID: userID,
  50. packageID: packageID,
  51. objectIter: objIter,
  52. nodeAffinity: nodeAffinity,
  53. }
  54. }
  55. func (t *UploadObjects) Execute(ctx *UploadObjectsContext) (*UploadObjectsResult, error) {
  56. defer t.objectIter.Close()
  57. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  58. if err != nil {
  59. return nil, fmt.Errorf("new coordinator client: %w", err)
  60. }
  61. getUserNodesResp, err := coorCli.GetUserNodes(coormq.NewGetUserNodes(t.userID))
  62. if err != nil {
  63. return nil, fmt.Errorf("getting user nodes: %w", err)
  64. }
  65. cons := ctx.Connectivity.GetAll()
  66. userNodes := lo.Map(getUserNodesResp.Nodes, func(node cdssdk.Node, index int) UploadNodeInfo {
  67. delay := time.Duration(math.MaxInt64)
  68. con, ok := cons[node.NodeID]
  69. if ok && con.Delay != nil {
  70. delay = *con.Delay
  71. }
  72. return UploadNodeInfo{
  73. Node: node,
  74. Delay: delay,
  75. IsSameLocation: node.LocationID == stgglb.Local.LocationID,
  76. }
  77. })
  78. if len(userNodes) == 0 {
  79. return nil, fmt.Errorf("user no available nodes")
  80. }
  81. // 给上传节点的IPFS加锁
  82. ipfsReqBlder := reqbuilder.NewBuilder()
  83. // 如果本地的IPFS也是存储系统的一个节点,那么从本地上传时,需要加锁
  84. if stgglb.Local.NodeID != nil {
  85. ipfsReqBlder.IPFS().Buzy(*stgglb.Local.NodeID)
  86. }
  87. for _, node := range userNodes {
  88. if stgglb.Local.NodeID != nil && node.Node.NodeID == *stgglb.Local.NodeID {
  89. continue
  90. }
  91. ipfsReqBlder.IPFS().Buzy(node.Node.NodeID)
  92. }
  93. // TODO 考虑加Object的Create锁
  94. // 防止上传的副本被清除
  95. ipfsMutex, err := ipfsReqBlder.MutexLock(ctx.Distlock)
  96. if err != nil {
  97. return nil, fmt.Errorf("acquire locks failed, err: %w", err)
  98. }
  99. defer ipfsMutex.Unlock()
  100. rets, err := uploadAndUpdatePackage(t.packageID, t.objectIter, userNodes, t.nodeAffinity)
  101. if err != nil {
  102. return nil, err
  103. }
  104. return &UploadObjectsResult{
  105. Objects: rets,
  106. }, nil
  107. }
  108. // chooseUploadNode 选择一个上传文件的节点
  109. // 1. 选择设置了亲和性的节点
  110. // 2. 从与当前客户端相同地域的节点中随机选一个
  111. // 3. 没有的话从所有节点选择延迟最低的节点
  112. func chooseUploadNode(nodes []UploadNodeInfo, nodeAffinity *cdssdk.NodeID) UploadNodeInfo {
  113. if nodeAffinity != nil {
  114. aff, ok := lo.Find(nodes, func(node UploadNodeInfo) bool { return node.Node.NodeID == *nodeAffinity })
  115. if ok {
  116. return aff
  117. }
  118. }
  119. sameLocationNodes := lo.Filter(nodes, func(e UploadNodeInfo, i int) bool { return e.IsSameLocation })
  120. if len(sameLocationNodes) > 0 {
  121. return sameLocationNodes[rand.Intn(len(sameLocationNodes))]
  122. }
  123. // 选择延迟最低的节点
  124. nodes = sort2.Sort(nodes, func(e1, e2 UploadNodeInfo) int { return sort2.Cmp(e1.Delay, e2.Delay) })
  125. return nodes[0]
  126. }
  127. func uploadAndUpdatePackage(packageID cdssdk.PackageID, objectIter iterator.UploadingObjectIterator, userNodes []UploadNodeInfo, nodeAffinity *cdssdk.NodeID) ([]ObjectUploadResult, error) {
  128. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  129. if err != nil {
  130. return nil, fmt.Errorf("new coordinator client: %w", err)
  131. }
  132. defer stgglb.CoordinatorMQPool.Release(coorCli)
  133. // 为所有文件选择相同的上传节点
  134. uploadNode := chooseUploadNode(userNodes, nodeAffinity)
  135. var uploadRets []ObjectUploadResult
  136. //上传文件夹
  137. var adds []coormq.AddObjectEntry
  138. for {
  139. objInfo, err := objectIter.MoveNext()
  140. if err == iterator.ErrNoMoreItem {
  141. break
  142. }
  143. if err != nil {
  144. return nil, fmt.Errorf("reading object: %w", err)
  145. }
  146. err = func() error {
  147. defer objInfo.File.Close()
  148. uploadTime := time.Now()
  149. fileHash, err := uploadFile(objInfo.File, uploadNode)
  150. if err != nil {
  151. return fmt.Errorf("uploading file: %w", err)
  152. }
  153. uploadRets = append(uploadRets, ObjectUploadResult{
  154. Info: objInfo,
  155. Error: err,
  156. })
  157. adds = append(adds, coormq.NewAddObjectEntry(objInfo.Path, objInfo.Size, fileHash, uploadTime, uploadNode.Node.NodeID))
  158. return nil
  159. }()
  160. if err != nil {
  161. return nil, err
  162. }
  163. }
  164. updateResp, err := coorCli.UpdatePackage(coormq.NewUpdatePackage(packageID, adds, nil))
  165. if err != nil {
  166. return nil, fmt.Errorf("updating package: %w", err)
  167. }
  168. updatedObjs := make(map[string]*cdssdk.Object)
  169. for _, obj := range updateResp.Added {
  170. o := obj
  171. updatedObjs[obj.Path] = &o
  172. }
  173. for i := range uploadRets {
  174. obj := updatedObjs[uploadRets[i].Info.Path]
  175. if obj == nil {
  176. uploadRets[i].Error = fmt.Errorf("object %s not found in package", uploadRets[i].Info.Path)
  177. continue
  178. }
  179. uploadRets[i].Object = *obj
  180. }
  181. return uploadRets, nil
  182. }
  183. func uploadFile(file io.Reader, uploadNode UploadNodeInfo) (string, error) {
  184. // 本地有IPFS,则直接从本地IPFS上传
  185. if stgglb.IPFSPool != nil {
  186. logger.Debug("try to use local IPFS to upload file")
  187. // 只有本地IPFS不是存储系统中的一个节点,才需要Pin文件
  188. fileHash, err := uploadToLocalIPFS(file, uploadNode.Node.NodeID, stgglb.Local.NodeID == nil)
  189. if err == nil {
  190. return fileHash, nil
  191. } else {
  192. logger.Warnf("upload to local IPFS failed, so try to upload to node %d, err: %s", uploadNode.Node.NodeID, err.Error())
  193. }
  194. }
  195. // 否则发送到agent上传
  196. fileHash, err := uploadToNode(file, uploadNode.Node)
  197. if err != nil {
  198. return "", fmt.Errorf("uploading to node %v: %w", uploadNode.Node.NodeID, err)
  199. }
  200. return fileHash, nil
  201. }
  202. func uploadToNode(file io.Reader, node cdssdk.Node) (string, error) {
  203. ft := ioswitch2.NewFromTo()
  204. fromExec, hd := ioswitch2.NewFromDriver(-1)
  205. ft.AddFrom(fromExec).AddTo(ioswitch2.NewToNode(node, -1, "fileHash"))
  206. parser := parser.NewParser(cdssdk.DefaultECRedundancy)
  207. plans := exec.NewPlanBuilder()
  208. err := parser.Parse(ft, plans)
  209. if err != nil {
  210. return "", fmt.Errorf("parsing plan: %w", err)
  211. }
  212. exec := plans.Execute()
  213. exec.BeginWrite(io.NopCloser(file), hd)
  214. ret, err := exec.Wait(context.TODO())
  215. if err != nil {
  216. return "", err
  217. }
  218. return ret["fileHash"].(string), nil
  219. }
  220. func uploadToLocalIPFS(file io.Reader, nodeID cdssdk.NodeID, shouldPin bool) (string, error) {
  221. ipfsCli, err := stgglb.IPFSPool.Acquire()
  222. if err != nil {
  223. return "", fmt.Errorf("new ipfs client: %w", err)
  224. }
  225. defer ipfsCli.Close()
  226. // 从本地IPFS上传文件
  227. fileHash, err := ipfsCli.CreateFile(file)
  228. if err != nil {
  229. return "", fmt.Errorf("creating ipfs file: %w", err)
  230. }
  231. if !shouldPin {
  232. return fileHash, nil
  233. }
  234. err = pinIPFSFile(nodeID, fileHash)
  235. if err != nil {
  236. return "", err
  237. }
  238. return fileHash, nil
  239. }
  240. func pinIPFSFile(nodeID cdssdk.NodeID, fileHash string) error {
  241. agtCli, err := stgglb.AgentMQPool.Acquire(nodeID)
  242. if err != nil {
  243. return fmt.Errorf("new agent client: %w", err)
  244. }
  245. defer stgglb.AgentMQPool.Release(agtCli)
  246. // 然后让最近节点pin本地上传的文件
  247. _, err = agtCli.PinObject(agtmq.ReqPinObject([]string{fileHash}, false))
  248. if err != nil {
  249. return fmt.Errorf("start pinning object: %w", err)
  250. }
  251. return nil
  252. }

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