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_rep_objects.go 8.8 kB

2 years ago
2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. package task
  2. import (
  3. "fmt"
  4. "io"
  5. "math/rand"
  6. "time"
  7. "github.com/samber/lo"
  8. "gitlink.org.cn/cloudream/common/pkgs/distlock/reqbuilder"
  9. "gitlink.org.cn/cloudream/common/pkgs/logger"
  10. "gitlink.org.cn/cloudream/common/utils/ipfs"
  11. "gitlink.org.cn/cloudream/storage-client/internal/config"
  12. "gitlink.org.cn/cloudream/storage-common/utils"
  13. mygrpc "gitlink.org.cn/cloudream/storage-common/utils/grpc"
  14. agtcli "gitlink.org.cn/cloudream/storage-common/pkgs/mq/client/agent"
  15. ramsg "gitlink.org.cn/cloudream/storage-common/pkgs/mq/message"
  16. agtmsg "gitlink.org.cn/cloudream/storage-common/pkgs/mq/message/agent"
  17. coormsg "gitlink.org.cn/cloudream/storage-common/pkgs/mq/message/coordinator"
  18. agentcaller "gitlink.org.cn/cloudream/storage-common/pkgs/proto"
  19. "google.golang.org/grpc"
  20. "google.golang.org/grpc/credentials/insecure"
  21. )
  22. // UploadObjects和UploadRepResults为一一对应关系
  23. type UploadRepObjects struct {
  24. userID int64
  25. bucketID int64
  26. repCount int
  27. Objects []UploadObject
  28. Results []UploadSingleRepObjectResult
  29. IsUploading bool
  30. }
  31. type UploadRepObjectsResult struct {
  32. Objects []UploadObject
  33. Results []UploadSingleRepObjectResult
  34. IsUploading bool
  35. }
  36. type UploadObject struct {
  37. ObjectName string
  38. File io.ReadCloser
  39. FileSize int64
  40. }
  41. type UploadSingleRepObjectResult struct {
  42. Error error
  43. FileHash string
  44. ObjectID int64
  45. }
  46. func NewUploadRepObjects(userID int64, bucketID int64, uploadObjects []UploadObject, repCount int) *UploadRepObjects {
  47. return &UploadRepObjects{
  48. userID: userID,
  49. bucketID: bucketID,
  50. Objects: uploadObjects,
  51. repCount: repCount,
  52. }
  53. }
  54. func (t *UploadRepObjects) Execute(ctx TaskContext, complete CompleteFn) {
  55. err := t.do(ctx)
  56. complete(err, CompleteOption{
  57. RemovingDelay: time.Minute,
  58. })
  59. }
  60. func (t *UploadRepObjects) do(ctx TaskContext) error {
  61. reqBlder := reqbuilder.NewBuilder()
  62. for _, uploadObject := range t.Objects {
  63. reqBlder.Metadata().
  64. // 用于防止创建了多个同名对象
  65. Object().CreateOne(t.bucketID, uploadObject.ObjectName)
  66. }
  67. mutex, err := reqBlder.
  68. Metadata().
  69. // 用于判断用户是否有桶的权限
  70. UserBucket().ReadOne(t.userID, t.bucketID).
  71. // 用于查询可用的上传节点
  72. Node().ReadAny().
  73. // 用于设置Rep配置
  74. ObjectRep().CreateAny().
  75. // 用于创建Cache记录
  76. Cache().CreateAny().
  77. MutexLock(ctx.DistLock)
  78. if err != nil {
  79. return fmt.Errorf("acquire locks failed, err: %w", err)
  80. }
  81. defer mutex.Unlock()
  82. var repWriteResps []*coormsg.PreUploadResp
  83. //判断是否所有文件都符合上传条件
  84. hasFailure := true
  85. for i := 0; i < len(t.Objects); i++ {
  86. repWriteResp, err := t.preUploadSingleObject(ctx, t.Objects[i])
  87. if err != nil {
  88. hasFailure = false
  89. t.Results = append(t.Results,
  90. UploadSingleRepObjectResult{
  91. Error: err,
  92. FileHash: "",
  93. ObjectID: 0,
  94. })
  95. continue
  96. }
  97. t.Results = append(t.Results, UploadSingleRepObjectResult{})
  98. repWriteResps = append(repWriteResps, repWriteResp)
  99. }
  100. // 不满足上传条件,返回各文件检查结果
  101. if !hasFailure {
  102. return nil
  103. }
  104. //上传文件夹
  105. t.IsUploading = true
  106. for i := 0; i < len(repWriteResps); i++ {
  107. objectID, fileHash, err := t.uploadSingleObject(ctx, t.Objects[i], repWriteResps[i])
  108. // 记录文件上传结果
  109. t.Results[i] = UploadSingleRepObjectResult{
  110. Error: err,
  111. FileHash: fileHash,
  112. ObjectID: objectID,
  113. }
  114. }
  115. return nil
  116. }
  117. // 检查单个文件是否能够上传
  118. func (t *UploadRepObjects) preUploadSingleObject(ctx TaskContext, uploadObject UploadObject) (*coormsg.PreUploadResp, error) {
  119. //发送写请求,请求Coor分配写入节点Ip
  120. repWriteResp, err := ctx.Coordinator.PreUploadRepObject(coormsg.NewPreUploadRepObjectBody(t.bucketID, uploadObject.ObjectName, uploadObject.FileSize, t.userID, config.Cfg().ExternalIP))
  121. if err != nil {
  122. return nil, fmt.Errorf("pre upload rep object: %w", err)
  123. }
  124. if len(repWriteResp.Nodes) == 0 {
  125. return nil, fmt.Errorf("no node to upload file")
  126. }
  127. return repWriteResp, nil
  128. }
  129. // 上传文件
  130. func (t *UploadRepObjects) uploadSingleObject(ctx TaskContext, uploadObject UploadObject, preResp *coormsg.PreUploadResp) (int64, string, error) {
  131. uploadNode := t.chooseUploadNode(preResp.Nodes)
  132. var fileHash string
  133. uploadedNodeIDs := []int64{}
  134. willUploadToNode := true
  135. // 本地有IPFS,则直接从本地IPFS上传
  136. if ctx.IPFS != nil {
  137. logger.Infof("try to use local IPFS to upload file")
  138. var err error
  139. fileHash, err = uploadToLocalIPFS(ctx.IPFS, uploadObject.File, uploadNode.ID)
  140. if err != nil {
  141. logger.Warnf("upload to local IPFS failed, so try to upload to node %d, err: %s", uploadNode.ID, err.Error())
  142. } else {
  143. willUploadToNode = false
  144. }
  145. }
  146. // 否则发送到agent上传
  147. if willUploadToNode {
  148. // 如果客户端与节点在同一个地域,则使用内网地址连接节点
  149. nodeIP := uploadNode.ExternalIP
  150. if uploadNode.IsSameLocation {
  151. nodeIP = uploadNode.LocalIP
  152. logger.Infof("client and node %d are at the same location, use local ip\n", uploadNode.ID)
  153. }
  154. mutex, err := reqbuilder.NewBuilder().
  155. // 防止上传的副本被清除
  156. IPFS().CreateAnyRep(uploadNode.ID).
  157. MutexLock(ctx.DistLock)
  158. if err != nil {
  159. return 0, "", fmt.Errorf("acquire locks failed, err: %w", err)
  160. }
  161. defer mutex.Unlock()
  162. fileHash, err = uploadToNode(uploadObject.File, nodeIP)
  163. if err != nil {
  164. return 0, "", fmt.Errorf("upload to node %s failed, err: %w", nodeIP, err)
  165. }
  166. uploadedNodeIDs = append(uploadedNodeIDs, uploadNode.ID)
  167. }
  168. dirName := utils.GetDirectoryName(uploadObject.ObjectName)
  169. // 记录写入的文件的Hash
  170. createResp, err := ctx.Coordinator.CreateRepObject(coormsg.NewCreateRepObject(t.bucketID, uploadObject.ObjectName, uploadObject.FileSize, t.repCount, t.userID, uploadedNodeIDs, fileHash, dirName))
  171. if err != nil {
  172. return 0, "", fmt.Errorf("creating rep object: %w", err)
  173. }
  174. return createResp.ObjectID, fileHash, nil
  175. }
  176. // chooseUploadNode 选择一个上传文件的节点
  177. // 1. 从与当前客户端相同地域的节点中随机选一个
  178. // 2. 没有用的话从所有节点中随机选一个
  179. func (t *UploadRepObjects) chooseUploadNode(nodes []ramsg.RespNode) ramsg.RespNode {
  180. sameLocationNodes := lo.Filter(nodes, func(e ramsg.RespNode, i int) bool { return e.IsSameLocation })
  181. if len(sameLocationNodes) > 0 {
  182. return sameLocationNodes[rand.Intn(len(sameLocationNodes))]
  183. }
  184. return nodes[rand.Intn(len(nodes))]
  185. }
  186. func uploadToNode(file io.ReadCloser, nodeIP string) (string, error) {
  187. // 建立grpc连接,发送请求
  188. grpcAddr := fmt.Sprintf("%s:%d", nodeIP, config.Cfg().GRPCPort)
  189. grpcCon, err := grpc.Dial(grpcAddr, grpc.WithTransportCredentials(insecure.NewCredentials()))
  190. if err != nil {
  191. return "", fmt.Errorf("connect to grpc server at %s failed, err: %w", grpcAddr, err)
  192. }
  193. defer grpcCon.Close()
  194. client := agentcaller.NewFileTransportClient(grpcCon)
  195. upload, err := mygrpc.SendFileAsStream(client)
  196. if err != nil {
  197. return "", fmt.Errorf("request to send file failed, err: %w", err)
  198. }
  199. // 发送文件数据
  200. _, err = io.Copy(upload, file)
  201. if err != nil {
  202. // 发生错误则关闭连接
  203. upload.Abort(io.ErrClosedPipe)
  204. return "", fmt.Errorf("copy file date to upload stream failed, err: %w", err)
  205. }
  206. // 发送EOF消息,并获得FileHash
  207. fileHash, err := upload.Finish()
  208. if err != nil {
  209. upload.Abort(io.ErrClosedPipe)
  210. return "", fmt.Errorf("send EOF failed, err: %w", err)
  211. }
  212. return fileHash, nil
  213. }
  214. func uploadToLocalIPFS(ipfs *ipfs.IPFS, file io.ReadCloser, nodeID int64) (string, error) {
  215. // 从本地IPFS上传文件
  216. writer, err := ipfs.CreateFile()
  217. if err != nil {
  218. return "", fmt.Errorf("create IPFS file failed, err: %w", err)
  219. }
  220. _, err = io.Copy(writer, file)
  221. if err != nil {
  222. return "", fmt.Errorf("copy file data to IPFS failed, err: %w", err)
  223. }
  224. fileHash, err := writer.Finish()
  225. if err != nil {
  226. return "", fmt.Errorf("finish writing IPFS failed, err: %w", err)
  227. }
  228. // 然后让最近节点pin本地上传的文件
  229. agentClient, err := agtcli.NewClient(nodeID, &config.Cfg().RabbitMQ)
  230. if err != nil {
  231. return "", fmt.Errorf("create agent client to %d failed, err: %w", nodeID, err)
  232. }
  233. defer agentClient.Close()
  234. pinObjResp, err := agentClient.StartPinningObject(agtmsg.NewStartPinningObject(fileHash))
  235. if err != nil {
  236. return "", fmt.Errorf("start pinning object: %w", err)
  237. }
  238. for {
  239. waitResp, err := agentClient.WaitPinningObject(agtmsg.NewWaitPinningObject(pinObjResp.TaskID, int64(time.Second)*5))
  240. if err != nil {
  241. return "", fmt.Errorf("waitting pinning object: %w", err)
  242. }
  243. if waitResp.IsComplete {
  244. if waitResp.Error != "" {
  245. return "", fmt.Errorf("agent pinning object: %s", waitResp.Error)
  246. }
  247. break
  248. }
  249. }
  250. return fileHash, nil
  251. }

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