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.

multipart_upload.go 4.3 kB

10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
10 months ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. package s3
  2. import (
  3. "context"
  4. "crypto/sha256"
  5. "io"
  6. "github.com/aws/aws-sdk-go-v2/aws"
  7. "github.com/aws/aws-sdk-go-v2/service/s3"
  8. s3types "github.com/aws/aws-sdk-go-v2/service/s3/types"
  9. "gitlink.org.cn/cloudream/common/utils/io2"
  10. "gitlink.org.cn/cloudream/common/utils/os2"
  11. "gitlink.org.cn/cloudream/common/utils/sort2"
  12. stgtypes "gitlink.org.cn/cloudream/jcs-pub/common/pkgs/storage/types"
  13. jcstypes "gitlink.org.cn/cloudream/jcs-pub/common/types"
  14. )
  15. type Multiparter struct {
  16. detail *jcstypes.UserSpaceDetail
  17. feat *jcstypes.MultipartUploadFeature
  18. bucket string
  19. cli *s3.Client
  20. }
  21. func NewMultiparter(detail *jcstypes.UserSpaceDetail, feat *jcstypes.MultipartUploadFeature, bkt string, cli *s3.Client) *Multiparter {
  22. return &Multiparter{
  23. detail: detail,
  24. feat: feat,
  25. bucket: bkt,
  26. cli: cli,
  27. }
  28. }
  29. func (*Multiparter) MinPartSize() int64 {
  30. return 5 * 1024 * 1024 // 5MB
  31. }
  32. func (*Multiparter) MaxPartSize() int64 {
  33. return 5 * 1024 * 1024 * 1024 // 5GB
  34. }
  35. func (m *Multiparter) Initiate(ctx context.Context) (stgtypes.MultipartTask, error) {
  36. tempFileName := os2.GenerateRandomFileName(10)
  37. tempDir := m.detail.UserSpace.WorkingDir.Clone()
  38. tempDir.Push(stgtypes.TempWorkingDir)
  39. tempFilePath := tempDir.Clone()
  40. tempFilePath.Push(tempFileName)
  41. resp, err := m.cli.CreateMultipartUpload(ctx, &s3.CreateMultipartUploadInput{
  42. Bucket: aws.String(m.bucket),
  43. Key: aws.String(tempFilePath.String()),
  44. ChecksumAlgorithm: s3types.ChecksumAlgorithmSha256,
  45. })
  46. if err != nil {
  47. return nil, err
  48. }
  49. return &MultipartTask{
  50. cli: m.cli,
  51. bucket: m.bucket,
  52. tempDir: tempDir,
  53. tempFileName: tempFileName,
  54. tempFilePath: tempFilePath,
  55. uploadID: *resp.UploadId,
  56. }, nil
  57. }
  58. func (m *Multiparter) UploadPart(ctx context.Context, init stgtypes.MultipartInitState, partSize int64, partNumber int, stream io.Reader) (stgtypes.UploadedPartInfo, error) {
  59. hashStr := io2.NewReadHasher(sha256.New(), stream)
  60. resp, err := m.cli.UploadPart(ctx, &s3.UploadPartInput{
  61. Bucket: aws.String(init.Bucket),
  62. Key: aws.String(init.Key),
  63. UploadId: aws.String(init.UploadID),
  64. PartNumber: aws.Int32(int32(partNumber)),
  65. Body: hashStr,
  66. })
  67. if err != nil {
  68. return stgtypes.UploadedPartInfo{}, err
  69. }
  70. return stgtypes.UploadedPartInfo{
  71. ETag: *resp.ETag,
  72. PartNumber: partNumber,
  73. PartHash: hashStr.Sum(),
  74. }, nil
  75. }
  76. type MultipartTask struct {
  77. cli *s3.Client
  78. bucket string
  79. tempDir jcstypes.JPath
  80. tempFileName string
  81. tempFilePath jcstypes.JPath
  82. uploadID string
  83. }
  84. func (i *MultipartTask) InitState() stgtypes.MultipartInitState {
  85. return stgtypes.MultipartInitState{
  86. UploadID: i.uploadID,
  87. Bucket: i.bucket,
  88. Key: i.tempFilePath.String(),
  89. }
  90. }
  91. func (i *MultipartTask) JoinParts(ctx context.Context, parts []stgtypes.UploadedPartInfo) (stgtypes.FileInfo, error) {
  92. parts = sort2.Sort(parts, func(l, r stgtypes.UploadedPartInfo) int {
  93. return l.PartNumber - r.PartNumber
  94. })
  95. s3Parts := make([]s3types.CompletedPart, len(parts))
  96. for i, part := range parts {
  97. s3Parts[i] = s3types.CompletedPart{
  98. ETag: aws.String(part.ETag),
  99. PartNumber: aws.Int32(int32(part.PartNumber)),
  100. }
  101. }
  102. partHashes := make([][]byte, len(parts))
  103. for i, part := range parts {
  104. partHashes[i] = part.PartHash
  105. }
  106. _, err := i.cli.CompleteMultipartUpload(ctx, &s3.CompleteMultipartUploadInput{
  107. Bucket: aws.String(i.bucket),
  108. Key: aws.String(i.tempFilePath.String()),
  109. UploadId: aws.String(i.uploadID),
  110. MultipartUpload: &s3types.CompletedMultipartUpload{
  111. Parts: s3Parts,
  112. },
  113. })
  114. if err != nil {
  115. return stgtypes.FileInfo{}, err
  116. }
  117. headResp, err := i.cli.HeadObject(ctx, &s3.HeadObjectInput{
  118. Bucket: aws.String(i.bucket),
  119. Key: aws.String(i.tempFilePath.String()),
  120. })
  121. if err != nil {
  122. return stgtypes.FileInfo{}, err
  123. }
  124. hash := jcstypes.CalculateCompositeHash(partHashes)
  125. return stgtypes.FileInfo{
  126. Path: i.tempFilePath,
  127. Size: *headResp.ContentLength,
  128. Hash: hash,
  129. }, nil
  130. }
  131. func (i *MultipartTask) Close() {
  132. i.cli.AbortMultipartUpload(context.Background(), &s3.AbortMultipartUploadInput{
  133. Bucket: aws.String(i.bucket),
  134. Key: aws.String(i.tempFilePath.String()),
  135. UploadId: aws.String(i.uploadID),
  136. })
  137. }

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