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_object.go 8.6 kB

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

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