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.

create_package.go 7.7 kB

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

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