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.

iterator.go 5.5 kB

7 months ago
7 months ago
7 months ago
7 months ago
6 months ago
7 months ago
7 months ago
7 months ago
7 months ago
7 months ago
7 months ago
7 months ago
7 months ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. package downloader
  2. import (
  3. "context"
  4. "fmt"
  5. "io"
  6. "reflect"
  7. "gitlink.org.cn/cloudream/common/pkgs/ioswitch/exec"
  8. "gitlink.org.cn/cloudream/common/pkgs/logger"
  9. "gitlink.org.cn/cloudream/common/utils/io2"
  10. "gitlink.org.cn/cloudream/common/utils/math2"
  11. "gitlink.org.cn/cloudream/jcs-pub/client/internal/downloader/strategy"
  12. "gitlink.org.cn/cloudream/jcs-pub/client/types"
  13. stgglb "gitlink.org.cn/cloudream/jcs-pub/common/globals"
  14. "gitlink.org.cn/cloudream/jcs-pub/common/pkgs/distlock"
  15. "gitlink.org.cn/cloudream/jcs-pub/common/pkgs/ioswitch2"
  16. "gitlink.org.cn/cloudream/jcs-pub/common/pkgs/ioswitch2/parser"
  17. "gitlink.org.cn/cloudream/jcs-pub/common/pkgs/iterator"
  18. )
  19. type downloadSpaceInfo struct {
  20. Space types.UserSpaceDetail
  21. ObjectPinned bool
  22. Blocks []types.ObjectBlock
  23. Distance float64
  24. }
  25. type DownloadContext struct {
  26. PubLock *distlock.Service
  27. }
  28. type DownloadObjectIterator struct {
  29. OnClosing func()
  30. downloader *Downloader
  31. reqs []downloadReqeust2
  32. currentIndex int
  33. }
  34. func NewDownloadObjectIterator(downloader *Downloader, downloadObjs []downloadReqeust2) *DownloadObjectIterator {
  35. return &DownloadObjectIterator{
  36. downloader: downloader,
  37. reqs: downloadObjs,
  38. }
  39. }
  40. func (i *DownloadObjectIterator) MoveNext() (*Downloading, error) {
  41. if i.currentIndex >= len(i.reqs) {
  42. return nil, iterator.ErrNoMoreItem
  43. }
  44. req := i.reqs[i.currentIndex]
  45. if req.Detail == nil {
  46. return &Downloading{
  47. Object: nil,
  48. File: nil,
  49. Request: req.Raw,
  50. }, nil
  51. }
  52. strg, err := i.downloader.selector.Select(strategy.Request{
  53. Detail: *req.Detail,
  54. Range: math2.NewRange(req.Raw.Offset, req.Raw.Length),
  55. DestLocation: stgglb.Local.Location,
  56. })
  57. if err != nil {
  58. return nil, fmt.Errorf("selecting download strategy: %w", err)
  59. }
  60. var reader io.ReadCloser
  61. switch strg := strg.(type) {
  62. case *strategy.DirectStrategy:
  63. reader, err = i.downloadDirect(req, *strg)
  64. if err != nil {
  65. return nil, fmt.Errorf("downloading object %v: %w", req.Raw.ObjectID, err)
  66. }
  67. case *strategy.ECReconstructStrategy:
  68. reader, err = i.downloadECReconstruct(req, *strg)
  69. if err != nil {
  70. return nil, fmt.Errorf("downloading ec object %v: %w", req.Raw.ObjectID, err)
  71. }
  72. case *strategy.LRCReconstructStrategy:
  73. reader, err = i.downloadLRCReconstruct(req, *strg)
  74. if err != nil {
  75. return nil, fmt.Errorf("downloading lrc object %v: %w", req.Raw.ObjectID, err)
  76. }
  77. default:
  78. return nil, fmt.Errorf("unsupported strategy type: %v", reflect.TypeOf(strg))
  79. }
  80. i.currentIndex++
  81. return &Downloading{
  82. Object: &req.Detail.Object,
  83. File: reader,
  84. Request: req.Raw,
  85. }, nil
  86. }
  87. func (i *DownloadObjectIterator) Close() {
  88. if i.OnClosing != nil {
  89. i.OnClosing()
  90. }
  91. }
  92. func (i *DownloadObjectIterator) downloadDirect(req downloadReqeust2, strg strategy.DirectStrategy) (io.ReadCloser, error) {
  93. logger.Debugf("downloading object %v from storage %v", req.Raw.ObjectID, strg.UserSpace.UserSpace.Storage.String())
  94. var strHandle *exec.DriverReadStream
  95. ft := ioswitch2.NewFromTo()
  96. toExec, handle := ioswitch2.NewToDriver(ioswitch2.RawStream())
  97. toExec.Range = math2.Range{
  98. Offset: req.Raw.Offset,
  99. }
  100. if req.Raw.Length != -1 {
  101. len := req.Raw.Length
  102. toExec.Range.Length = &len
  103. }
  104. ft.AddFrom(ioswitch2.NewFromShardstore(req.Detail.Object.FileHash, strg.UserSpace, ioswitch2.RawStream())).AddTo(toExec)
  105. strHandle = handle
  106. plans := exec.NewPlanBuilder()
  107. if err := parser.Parse(ft, plans); err != nil {
  108. return nil, fmt.Errorf("parsing plan: %w", err)
  109. }
  110. exeCtx := exec.NewExecContext()
  111. exec.SetValueByType(exeCtx, i.downloader.stgPool)
  112. exec := plans.Execute(exeCtx)
  113. go func() {
  114. _, err := exec.Wait(context.TODO())
  115. if err != nil {
  116. logger.Warnf("downloading object %v: %v", req.Raw.ObjectID, err)
  117. }
  118. }()
  119. return exec.BeginRead(strHandle)
  120. }
  121. func (i *DownloadObjectIterator) downloadECReconstruct(req downloadReqeust2, strg strategy.ECReconstructStrategy) (io.ReadCloser, error) {
  122. var logStrs []any = []any{fmt.Sprintf("downloading ec object %v from: ", req.Raw.ObjectID)}
  123. for i, b := range strg.Blocks {
  124. if i > 0 {
  125. logStrs = append(logStrs, ", ")
  126. }
  127. logStrs = append(logStrs, fmt.Sprintf("%v@%v", b.Index, strg.UserSpaces[i].UserSpace.Storage.String()))
  128. }
  129. logger.Debug(logStrs...)
  130. downloadBlks := make([]downloadBlock, len(strg.Blocks))
  131. for i, b := range strg.Blocks {
  132. downloadBlks[i] = downloadBlock{
  133. Block: b,
  134. Space: strg.UserSpaces[i],
  135. }
  136. }
  137. pr, pw := io.Pipe()
  138. go func() {
  139. readPos := req.Raw.Offset
  140. totalReadLen := req.Detail.Object.Size - req.Raw.Offset
  141. if req.Raw.Length >= 0 {
  142. totalReadLen = math2.Min(req.Raw.Length, totalReadLen)
  143. }
  144. firstStripIndex := readPos / strg.Redundancy.StripSize()
  145. stripIter := NewStripIterator(i.downloader, req.Detail.Object, downloadBlks, strg.Redundancy, firstStripIndex, i.downloader.strips, i.downloader.cfg.ECStripPrefetchCount)
  146. defer stripIter.Close()
  147. for totalReadLen > 0 {
  148. strip, err := stripIter.MoveNext()
  149. if err == iterator.ErrNoMoreItem {
  150. pw.CloseWithError(io.ErrUnexpectedEOF)
  151. return
  152. }
  153. if err != nil {
  154. pw.CloseWithError(err)
  155. return
  156. }
  157. readRelativePos := readPos - strip.Position
  158. curReadLen := math2.Min(totalReadLen, strg.Redundancy.StripSize()-readRelativePos)
  159. err = io2.WriteAll(pw, strip.Data[readRelativePos:readRelativePos+curReadLen])
  160. if err != nil {
  161. pw.CloseWithError(err)
  162. return
  163. }
  164. totalReadLen -= curReadLen
  165. readPos += curReadLen
  166. }
  167. pw.Close()
  168. }()
  169. return pr, nil
  170. }

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