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.0 kB

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

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