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_rep_package.go 8.7 kB

2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. package cmd
  2. import (
  3. "fmt"
  4. "io"
  5. "math/rand"
  6. "time"
  7. "github.com/samber/lo"
  8. "gitlink.org.cn/cloudream/common/pkgs/distlock"
  9. "gitlink.org.cn/cloudream/common/pkgs/logger"
  10. cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
  11. stgglb "gitlink.org.cn/cloudream/storage/common/globals"
  12. "gitlink.org.cn/cloudream/storage/common/pkgs/db/model"
  13. "gitlink.org.cn/cloudream/storage/common/pkgs/distlock/reqbuilder"
  14. "gitlink.org.cn/cloudream/storage/common/pkgs/iterator"
  15. agtmq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/agent"
  16. coormq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/coordinator"
  17. )
  18. type UploadNodeInfo struct {
  19. Node model.Node
  20. IsSameLocation bool
  21. }
  22. type CreateRepPackage struct {
  23. userID int64
  24. bucketID int64
  25. name string
  26. objectIter iterator.UploadingObjectIterator
  27. redundancy cdssdk.RepRedundancyInfo
  28. nodeAffinity *int64
  29. }
  30. type UpdatePackageContext struct {
  31. Distlock *distlock.Service
  32. }
  33. type CreateRepPackageResult struct {
  34. PackageID int64
  35. ObjectResults []RepObjectUploadResult
  36. }
  37. type RepObjectUploadResult struct {
  38. Info *iterator.IterUploadingObject
  39. Error error
  40. FileHash string
  41. ObjectID int64
  42. }
  43. func NewCreateRepPackage(userID int64, bucketID int64, name string, objIter iterator.UploadingObjectIterator, redundancy cdssdk.RepRedundancyInfo, nodeAffinity *int64) *CreateRepPackage {
  44. return &CreateRepPackage{
  45. userID: userID,
  46. bucketID: bucketID,
  47. name: name,
  48. objectIter: objIter,
  49. redundancy: redundancy,
  50. nodeAffinity: nodeAffinity,
  51. }
  52. }
  53. func (t *CreateRepPackage) Execute(ctx *UpdatePackageContext) (*CreateRepPackageResult, 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. reqBlder := reqbuilder.NewBuilder()
  60. // 如果本地的IPFS也是存储系统的一个节点,那么从本地上传时,需要加锁
  61. if stgglb.Local.NodeID != nil {
  62. reqBlder.IPFS().CreateAnyRep(*stgglb.Local.NodeID)
  63. }
  64. mutex, err := reqBlder.
  65. Metadata().
  66. // 用于判断用户是否有桶的权限
  67. UserBucket().ReadOne(t.userID, t.bucketID).
  68. // 用于查询可用的上传节点
  69. Node().ReadAny().
  70. // 用于创建包信息
  71. Package().CreateOne(t.bucketID, t.name).
  72. // 用于创建包中的文件的信息
  73. Object().CreateAny().
  74. // 用于设置EC配置
  75. ObjectBlock().CreateAny().
  76. // 用于创建Cache记录
  77. Cache().CreateAny().
  78. MutexLock(ctx.Distlock)
  79. if err != nil {
  80. return nil, fmt.Errorf("acquire locks failed, err: %w", err)
  81. }
  82. defer mutex.Unlock()
  83. createPkgResp, err := coorCli.CreatePackage(coormq.NewCreatePackage(t.userID, t.bucketID, t.name,
  84. cdssdk.NewTypedRedundancyInfo(t.redundancy)))
  85. if err != nil {
  86. return nil, fmt.Errorf("creating package: %w", err)
  87. }
  88. getUserNodesResp, err := coorCli.GetUserNodes(coormq.NewGetUserNodes(t.userID))
  89. if err != nil {
  90. return nil, fmt.Errorf("getting user nodes: %w", err)
  91. }
  92. findCliLocResp, err := coorCli.FindClientLocation(coormq.NewFindClientLocation(stgglb.Local.ExternalIP))
  93. if err != nil {
  94. return nil, fmt.Errorf("finding client location: %w", err)
  95. }
  96. nodeInfos := lo.Map(getUserNodesResp.Nodes, func(node model.Node, index int) UploadNodeInfo {
  97. return UploadNodeInfo{
  98. Node: node,
  99. IsSameLocation: node.LocationID == findCliLocResp.Location.LocationID,
  100. }
  101. })
  102. uploadNode := t.chooseUploadNode(nodeInfos, t.nodeAffinity)
  103. // 防止上传的副本被清除
  104. ipfsMutex, err := reqbuilder.NewBuilder().
  105. IPFS().CreateAnyRep(uploadNode.Node.NodeID).
  106. MutexLock(ctx.Distlock)
  107. if err != nil {
  108. return nil, fmt.Errorf("acquire locks failed, err: %w", err)
  109. }
  110. defer ipfsMutex.Unlock()
  111. rets, err := uploadAndUpdateRepPackage(createPkgResp.PackageID, t.objectIter, uploadNode)
  112. if err != nil {
  113. return nil, err
  114. }
  115. return &CreateRepPackageResult{
  116. PackageID: createPkgResp.PackageID,
  117. ObjectResults: rets,
  118. }, nil
  119. }
  120. func uploadAndUpdateRepPackage(packageID int64, objectIter iterator.UploadingObjectIterator, uploadNode UploadNodeInfo) ([]RepObjectUploadResult, error) {
  121. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  122. if err != nil {
  123. return nil, fmt.Errorf("new coordinator client: %w", err)
  124. }
  125. var uploadRets []RepObjectUploadResult
  126. var adds []coormq.AddRepObjectInfo
  127. for {
  128. objInfo, err := objectIter.MoveNext()
  129. if err == iterator.ErrNoMoreItem {
  130. break
  131. }
  132. if err != nil {
  133. return nil, fmt.Errorf("reading object: %w", err)
  134. }
  135. err = func() error {
  136. defer objInfo.File.Close()
  137. fileHash, err := uploadFile(objInfo.File, uploadNode)
  138. uploadRets = append(uploadRets, RepObjectUploadResult{
  139. Info: objInfo,
  140. Error: err,
  141. FileHash: fileHash,
  142. })
  143. if err != nil {
  144. return fmt.Errorf("uploading object: %w", err)
  145. }
  146. adds = append(adds, coormq.NewAddRepObjectInfo(objInfo.Path, objInfo.Size, fileHash, []int64{uploadNode.Node.NodeID}))
  147. return nil
  148. }()
  149. if err != nil {
  150. return nil, err
  151. }
  152. }
  153. _, err = coorCli.UpdateRepPackage(coormq.NewUpdateRepPackage(packageID, adds, nil))
  154. if err != nil {
  155. return nil, fmt.Errorf("updating package: %w", err)
  156. }
  157. return uploadRets, nil
  158. }
  159. // 上传文件
  160. func uploadFile(file io.Reader, uploadNode UploadNodeInfo) (string, error) {
  161. // 本地有IPFS,则直接从本地IPFS上传
  162. if stgglb.IPFSPool != nil {
  163. logger.Infof("try to use local IPFS to upload file")
  164. // 只有本地IPFS不是存储系统中的一个节点,才需要Pin文件
  165. fileHash, err := uploadToLocalIPFS(file, uploadNode.Node.NodeID, stgglb.Local.NodeID == nil)
  166. if err == nil {
  167. return fileHash, nil
  168. } else {
  169. logger.Warnf("upload to local IPFS failed, so try to upload to node %d, err: %s", uploadNode.Node.NodeID, err.Error())
  170. }
  171. }
  172. // 否则发送到agent上传
  173. // 如果客户端与节点在同一个地域,则使用内网地址连接节点
  174. nodeIP := uploadNode.Node.ExternalIP
  175. grpcPort := uploadNode.Node.ExternalGRPCPort
  176. if uploadNode.IsSameLocation {
  177. nodeIP = uploadNode.Node.LocalIP
  178. grpcPort = uploadNode.Node.LocalGRPCPort
  179. logger.Infof("client and node %d are at the same location, use local ip", uploadNode.Node.NodeID)
  180. }
  181. fileHash, err := uploadToNode(file, nodeIP, grpcPort)
  182. if err != nil {
  183. return "", fmt.Errorf("upload to node %s failed, err: %w", nodeIP, err)
  184. }
  185. return fileHash, nil
  186. }
  187. // chooseUploadNode 选择一个上传文件的节点
  188. // 1. 选择设置了亲和性的节点
  189. // 2. 从与当前客户端相同地域的节点中随机选一个
  190. // 3. 没有用的话从所有节点中随机选一个
  191. func (t *CreateRepPackage) chooseUploadNode(nodes []UploadNodeInfo, nodeAffinity *int64) UploadNodeInfo {
  192. if nodeAffinity != nil {
  193. aff, ok := lo.Find(nodes, func(node UploadNodeInfo) bool { return node.Node.NodeID == *nodeAffinity })
  194. if ok {
  195. return aff
  196. }
  197. }
  198. sameLocationNodes := lo.Filter(nodes, func(e UploadNodeInfo, i int) bool { return e.IsSameLocation })
  199. if len(sameLocationNodes) > 0 {
  200. return sameLocationNodes[rand.Intn(len(sameLocationNodes))]
  201. }
  202. return nodes[rand.Intn(len(nodes))]
  203. }
  204. func uploadToNode(file io.Reader, nodeIP string, grpcPort int) (string, error) {
  205. rpcCli, err := stgglb.AgentRPCPool.Acquire(nodeIP, grpcPort)
  206. if err != nil {
  207. return "", fmt.Errorf("new agent rpc client: %w", err)
  208. }
  209. defer rpcCli.Close()
  210. return rpcCli.SendIPFSFile(file)
  211. }
  212. func uploadToLocalIPFS(file io.Reader, nodeID int64, shouldPin bool) (string, error) {
  213. ipfsCli, err := stgglb.IPFSPool.Acquire()
  214. if err != nil {
  215. return "", fmt.Errorf("new ipfs client: %w", err)
  216. }
  217. defer ipfsCli.Close()
  218. // 从本地IPFS上传文件
  219. fileHash, err := ipfsCli.CreateFile(file)
  220. if err != nil {
  221. return "", fmt.Errorf("creating ipfs file: %w", err)
  222. }
  223. if !shouldPin {
  224. return fileHash, nil
  225. }
  226. err = pinIPFSFile(nodeID, fileHash)
  227. if err != nil {
  228. return "", err
  229. }
  230. return fileHash, nil
  231. }
  232. func pinIPFSFile(nodeID int64, fileHash string) error {
  233. agtCli, err := stgglb.AgentMQPool.Acquire(nodeID)
  234. if err != nil {
  235. return fmt.Errorf("new agent client: %w", err)
  236. }
  237. defer stgglb.AgentMQPool.Release(agtCli)
  238. // 然后让最近节点pin本地上传的文件
  239. pinObjResp, err := agtCli.StartPinningObject(agtmq.NewStartPinningObject(fileHash))
  240. if err != nil {
  241. return fmt.Errorf("start pinning object: %w", err)
  242. }
  243. for {
  244. waitResp, err := agtCli.WaitPinningObject(agtmq.NewWaitPinningObject(pinObjResp.TaskID, int64(time.Second)*5))
  245. if err != nil {
  246. return fmt.Errorf("waitting pinning object: %w", err)
  247. }
  248. if waitResp.IsComplete {
  249. if waitResp.Error != "" {
  250. return fmt.Errorf("agent pinning object: %s", waitResp.Error)
  251. }
  252. break
  253. }
  254. }
  255. return nil
  256. }

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