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_ec_package.go 7.0 kB

2 years ago
2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. package cmd
  2. import (
  3. "fmt"
  4. "io"
  5. "math/rand"
  6. "sync"
  7. "github.com/samber/lo"
  8. cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
  9. myio "gitlink.org.cn/cloudream/common/utils/io"
  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/ec"
  14. "gitlink.org.cn/cloudream/storage/common/pkgs/iterator"
  15. coormq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/coordinator"
  16. )
  17. type CreateECPackage struct {
  18. userID int64
  19. bucketID int64
  20. name string
  21. objectIter iterator.UploadingObjectIterator
  22. redundancy cdssdk.ECRedundancyInfo
  23. nodeAffinity *int64
  24. }
  25. type CreateECPackageResult struct {
  26. PackageID int64
  27. ObjectResults []ECObjectUploadResult
  28. }
  29. type ECObjectUploadResult struct {
  30. Info *iterator.IterUploadingObject
  31. Error error
  32. ObjectID int64
  33. }
  34. func NewCreateECPackage(userID int64, bucketID int64, name string, objIter iterator.UploadingObjectIterator, redundancy cdssdk.ECRedundancyInfo, nodeAffinity *int64) *CreateECPackage {
  35. return &CreateECPackage{
  36. userID: userID,
  37. bucketID: bucketID,
  38. name: name,
  39. objectIter: objIter,
  40. redundancy: redundancy,
  41. nodeAffinity: nodeAffinity,
  42. }
  43. }
  44. func (t *CreateECPackage) Execute(ctx *UpdatePackageContext) (*CreateECPackageResult, error) {
  45. defer t.objectIter.Close()
  46. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  47. if err != nil {
  48. return nil, fmt.Errorf("new coordinator client: %w", err)
  49. }
  50. mutex, err := reqbuilder.NewBuilder().
  51. Metadata().
  52. // 用于判断用户是否有桶的权限
  53. UserBucket().ReadOne(t.userID, t.bucketID).
  54. // 用于查询可用的上传节点
  55. Node().ReadAny().
  56. // 用于创建包信息
  57. Package().CreateOne(t.bucketID, t.name).
  58. // 用于创建包中的文件的信息
  59. Object().CreateAny().
  60. // 用于设置EC配置
  61. ObjectBlock().CreateAny().
  62. // 用于创建Cache记录
  63. Cache().CreateAny().
  64. MutexLock(ctx.Distlock)
  65. if err != nil {
  66. return nil, fmt.Errorf("acquire locks failed, err: %w", err)
  67. }
  68. defer mutex.Unlock()
  69. createPkgResp, err := coorCli.CreatePackage(coormq.NewCreatePackage(t.userID, t.bucketID, t.name,
  70. cdssdk.NewTypedRedundancyInfo(t.redundancy)))
  71. if err != nil {
  72. return nil, fmt.Errorf("creating package: %w", err)
  73. }
  74. getUserNodesResp, err := coorCli.GetUserNodes(coormq.NewGetUserNodes(t.userID))
  75. if err != nil {
  76. return nil, fmt.Errorf("getting user nodes: %w", err)
  77. }
  78. findCliLocResp, err := coorCli.FindClientLocation(coormq.NewFindClientLocation(stgglb.Local.ExternalIP))
  79. if err != nil {
  80. return nil, fmt.Errorf("finding client location: %w", err)
  81. }
  82. uploadNodeInfos := lo.Map(getUserNodesResp.Nodes, func(node model.Node, index int) UploadNodeInfo {
  83. return UploadNodeInfo{
  84. Node: node,
  85. IsSameLocation: node.LocationID == findCliLocResp.Location.LocationID,
  86. }
  87. })
  88. getECResp, err := coorCli.GetECConfig(coormq.NewGetECConfig(t.redundancy.ECName))
  89. if err != nil {
  90. return nil, fmt.Errorf("getting ec: %w", err)
  91. }
  92. // 给上传节点的IPFS加锁
  93. ipfsReqBlder := reqbuilder.NewBuilder()
  94. // 如果本地的IPFS也是存储系统的一个节点,那么从本地上传时,需要加锁
  95. if stgglb.Local.NodeID != nil {
  96. ipfsReqBlder.IPFS().CreateAnyRep(*stgglb.Local.NodeID)
  97. }
  98. for _, node := range uploadNodeInfos {
  99. if stgglb.Local.NodeID != nil && node.Node.NodeID == *stgglb.Local.NodeID {
  100. continue
  101. }
  102. ipfsReqBlder.IPFS().CreateAnyRep(node.Node.NodeID)
  103. }
  104. // 防止上传的副本被清除
  105. ipfsMutex, err := ipfsReqBlder.MutexLock(ctx.Distlock)
  106. if err != nil {
  107. return nil, fmt.Errorf("acquire locks failed, err: %w", err)
  108. }
  109. defer ipfsMutex.Unlock()
  110. // TODO 需要支持设置节点亲和性
  111. rets, err := uploadAndUpdateECPackage(createPkgResp.PackageID, t.objectIter, uploadNodeInfos, t.redundancy, getECResp.Config)
  112. if err != nil {
  113. return nil, err
  114. }
  115. return &CreateECPackageResult{
  116. PackageID: createPkgResp.PackageID,
  117. ObjectResults: rets,
  118. }, nil
  119. }
  120. func uploadAndUpdateECPackage(packageID int64, objectIter iterator.UploadingObjectIterator, uploadNodes []UploadNodeInfo, ecInfo cdssdk.ECRedundancyInfo, ec model.Ec) ([]ECObjectUploadResult, 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 []ECObjectUploadResult
  126. //上传文件夹
  127. var adds []coormq.AddECObjectInfo
  128. for {
  129. objInfo, err := objectIter.MoveNext()
  130. if err == iterator.ErrNoMoreItem {
  131. break
  132. }
  133. if err != nil {
  134. return nil, fmt.Errorf("reading object: %w", err)
  135. }
  136. err = func() error {
  137. defer objInfo.File.Close()
  138. fileHashes, uploadedNodeIDs, err := uploadECObject(objInfo, uploadNodes, ecInfo, ec)
  139. uploadRets = append(uploadRets, ECObjectUploadResult{
  140. Info: objInfo,
  141. Error: err,
  142. })
  143. if err != nil {
  144. return fmt.Errorf("uploading object: %w", err)
  145. }
  146. adds = append(adds, coormq.NewAddECObjectInfo(objInfo.Path, objInfo.Size, fileHashes, uploadedNodeIDs))
  147. return nil
  148. }()
  149. if err != nil {
  150. return nil, err
  151. }
  152. }
  153. _, err = coorCli.UpdateECPackage(coormq.NewUpdateECPackage(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 uploadECObject(obj *iterator.IterUploadingObject, uploadNodes []UploadNodeInfo, ecInfo cdssdk.ECRedundancyInfo, ecMod model.Ec) ([]string, []int64, error) {
  161. uploadNodes = shuffleNodes(uploadNodes, ecMod.EcN)
  162. rs, err := ec.NewRs(ecMod.EcK, ecMod.EcN, ecInfo.ChunkSize)
  163. if err != nil {
  164. return nil, nil, err
  165. }
  166. outputs := myio.ChunkedSplit(obj.File, ecInfo.ChunkSize, ecMod.EcK, myio.ChunkedSplitOption{
  167. PaddingZeros: true,
  168. })
  169. var readers []io.Reader
  170. for _, o := range outputs {
  171. readers = append(readers, o)
  172. }
  173. defer func() {
  174. for _, o := range outputs {
  175. o.Close()
  176. }
  177. }()
  178. encStrs := rs.EncodeAll(readers)
  179. wg := sync.WaitGroup{}
  180. nodeIDs := make([]int64, ecMod.EcN)
  181. fileHashes := make([]string, ecMod.EcN)
  182. anyErrs := make([]error, ecMod.EcN)
  183. for i := range encStrs {
  184. idx := i
  185. wg.Add(1)
  186. nodeIDs[idx] = uploadNodes[idx].Node.NodeID
  187. go func() {
  188. defer wg.Done()
  189. fileHashes[idx], anyErrs[idx] = uploadFile(encStrs[idx], uploadNodes[idx])
  190. }()
  191. }
  192. wg.Wait()
  193. for i, e := range anyErrs {
  194. if e != nil {
  195. return nil, nil, fmt.Errorf("uploading file to node %d: %w", uploadNodes[i].Node.NodeID, e)
  196. }
  197. }
  198. return fileHashes, nodeIDs, nil
  199. }
  200. func shuffleNodes(uploadNodes []UploadNodeInfo, extendTo int) []UploadNodeInfo {
  201. for i := len(uploadNodes); i < extendTo; i++ {
  202. uploadNodes = append(uploadNodes, uploadNodes[rand.Intn(len(uploadNodes))])
  203. }
  204. // 随机排列上传节点
  205. rand.Shuffle(len(uploadNodes), func(i, j int) {
  206. uploadNodes[i], uploadNodes[j] = uploadNodes[j], uploadNodes[i]
  207. })
  208. return uploadNodes
  209. }

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