You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
239 lines
6.8 KiB
239 lines
6.8 KiB
package s3api
|
|
|
|
import (
|
|
"net/http"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
|
|
"github.com/seaweedfs/seaweedfs/weed/s3api/s3_constants"
|
|
)
|
|
|
|
// CopySizeCalculator handles size calculations for different copy scenarios
|
|
type CopySizeCalculator struct {
|
|
srcSize int64
|
|
srcEncrypted bool
|
|
dstEncrypted bool
|
|
srcType EncryptionType
|
|
dstType EncryptionType
|
|
isCompressed bool
|
|
}
|
|
|
|
// EncryptionType represents different encryption types
|
|
type EncryptionType int
|
|
|
|
const (
|
|
EncryptionTypeNone EncryptionType = iota
|
|
EncryptionTypeSSEC
|
|
EncryptionTypeSSEKMS
|
|
EncryptionTypeSSES3
|
|
)
|
|
|
|
// NewCopySizeCalculator creates a new size calculator for copy operations
|
|
func NewCopySizeCalculator(entry *filer_pb.Entry, r *http.Request) *CopySizeCalculator {
|
|
calc := &CopySizeCalculator{
|
|
srcSize: int64(entry.Attributes.FileSize),
|
|
isCompressed: isCompressedEntry(entry),
|
|
}
|
|
|
|
// Determine source encryption type
|
|
calc.srcType, calc.srcEncrypted = getSourceEncryptionType(entry.Extended)
|
|
|
|
// Determine destination encryption type
|
|
calc.dstType, calc.dstEncrypted = getDestinationEncryptionType(r)
|
|
|
|
return calc
|
|
}
|
|
|
|
// CalculateTargetSize calculates the expected size of the target object
|
|
func (calc *CopySizeCalculator) CalculateTargetSize() int64 {
|
|
// For compressed objects, size calculation is complex
|
|
if calc.isCompressed {
|
|
return -1 // Indicates unknown size
|
|
}
|
|
|
|
switch {
|
|
case !calc.srcEncrypted && !calc.dstEncrypted:
|
|
// Plain → Plain: no size change
|
|
return calc.srcSize
|
|
|
|
case !calc.srcEncrypted && calc.dstEncrypted:
|
|
// Plain → Encrypted: no overhead since IV is in metadata
|
|
return calc.srcSize
|
|
|
|
case calc.srcEncrypted && !calc.dstEncrypted:
|
|
// Encrypted → Plain: no overhead since IV is in metadata
|
|
return calc.srcSize
|
|
|
|
case calc.srcEncrypted && calc.dstEncrypted:
|
|
// Encrypted → Encrypted: no overhead since IV is in metadata
|
|
return calc.srcSize
|
|
|
|
default:
|
|
return calc.srcSize
|
|
}
|
|
}
|
|
|
|
// CalculateActualSize calculates the actual unencrypted size of the content
|
|
func (calc *CopySizeCalculator) CalculateActualSize() int64 {
|
|
// With IV in metadata, encrypted and unencrypted sizes are the same
|
|
return calc.srcSize
|
|
}
|
|
|
|
// CalculateEncryptedSize calculates the encrypted size for the given encryption type
|
|
func (calc *CopySizeCalculator) CalculateEncryptedSize(encType EncryptionType) int64 {
|
|
// With IV in metadata, encrypted size equals actual size
|
|
return calc.CalculateActualSize()
|
|
}
|
|
|
|
// getSourceEncryptionType determines the encryption type of the source object
|
|
func getSourceEncryptionType(metadata map[string][]byte) (EncryptionType, bool) {
|
|
if IsSSECEncrypted(metadata) {
|
|
return EncryptionTypeSSEC, true
|
|
}
|
|
if IsSSEKMSEncrypted(metadata) {
|
|
return EncryptionTypeSSEKMS, true
|
|
}
|
|
if IsSSES3EncryptedInternal(metadata) {
|
|
return EncryptionTypeSSES3, true
|
|
}
|
|
return EncryptionTypeNone, false
|
|
}
|
|
|
|
// getDestinationEncryptionType determines the encryption type for the destination
|
|
func getDestinationEncryptionType(r *http.Request) (EncryptionType, bool) {
|
|
if IsSSECRequest(r) {
|
|
return EncryptionTypeSSEC, true
|
|
}
|
|
if IsSSEKMSRequest(r) {
|
|
return EncryptionTypeSSEKMS, true
|
|
}
|
|
if IsSSES3RequestInternal(r) {
|
|
return EncryptionTypeSSES3, true
|
|
}
|
|
return EncryptionTypeNone, false
|
|
}
|
|
|
|
// isCompressedEntry checks if the entry represents a compressed object
|
|
func isCompressedEntry(entry *filer_pb.Entry) bool {
|
|
// Check for compression indicators in metadata
|
|
if compressionType, exists := entry.Extended["compression"]; exists {
|
|
return string(compressionType) != ""
|
|
}
|
|
|
|
// Check MIME type for compressed formats
|
|
mimeType := entry.Attributes.Mime
|
|
compressedMimeTypes := []string{
|
|
"application/gzip",
|
|
"application/x-gzip",
|
|
"application/zip",
|
|
"application/x-compress",
|
|
"application/x-compressed",
|
|
}
|
|
|
|
for _, compressedType := range compressedMimeTypes {
|
|
if mimeType == compressedType {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// SizeTransitionInfo provides detailed information about size changes during copy
|
|
type SizeTransitionInfo struct {
|
|
SourceSize int64
|
|
TargetSize int64
|
|
ActualSize int64
|
|
SizeChange int64
|
|
SourceType EncryptionType
|
|
TargetType EncryptionType
|
|
IsCompressed bool
|
|
RequiresResize bool
|
|
}
|
|
|
|
// GetSizeTransitionInfo returns detailed size transition information
|
|
func (calc *CopySizeCalculator) GetSizeTransitionInfo() *SizeTransitionInfo {
|
|
targetSize := calc.CalculateTargetSize()
|
|
actualSize := calc.CalculateActualSize()
|
|
|
|
info := &SizeTransitionInfo{
|
|
SourceSize: calc.srcSize,
|
|
TargetSize: targetSize,
|
|
ActualSize: actualSize,
|
|
SizeChange: targetSize - calc.srcSize,
|
|
SourceType: calc.srcType,
|
|
TargetType: calc.dstType,
|
|
IsCompressed: calc.isCompressed,
|
|
RequiresResize: targetSize != calc.srcSize,
|
|
}
|
|
|
|
return info
|
|
}
|
|
|
|
// String returns a string representation of the encryption type
|
|
func (e EncryptionType) String() string {
|
|
switch e {
|
|
case EncryptionTypeNone:
|
|
return "None"
|
|
case EncryptionTypeSSEC:
|
|
return s3_constants.SSETypeC
|
|
case EncryptionTypeSSEKMS:
|
|
return s3_constants.SSETypeKMS
|
|
case EncryptionTypeSSES3:
|
|
return s3_constants.SSETypeS3
|
|
default:
|
|
return "Unknown"
|
|
}
|
|
}
|
|
|
|
// OptimizedSizeCalculation provides size calculations optimized for different scenarios
|
|
type OptimizedSizeCalculation struct {
|
|
Strategy UnifiedCopyStrategy
|
|
SourceSize int64
|
|
TargetSize int64
|
|
ActualContentSize int64
|
|
EncryptionOverhead int64
|
|
CanPreallocate bool
|
|
RequiresStreaming bool
|
|
}
|
|
|
|
// CalculateOptimizedSizes calculates sizes optimized for the copy strategy
|
|
func CalculateOptimizedSizes(entry *filer_pb.Entry, r *http.Request, strategy UnifiedCopyStrategy) *OptimizedSizeCalculation {
|
|
calc := NewCopySizeCalculator(entry, r)
|
|
info := calc.GetSizeTransitionInfo()
|
|
|
|
result := &OptimizedSizeCalculation{
|
|
Strategy: strategy,
|
|
SourceSize: info.SourceSize,
|
|
TargetSize: info.TargetSize,
|
|
ActualContentSize: info.ActualSize,
|
|
CanPreallocate: !info.IsCompressed && info.TargetSize > 0,
|
|
RequiresStreaming: info.IsCompressed || info.TargetSize < 0,
|
|
}
|
|
|
|
// Calculate encryption overhead for the target
|
|
// With IV in metadata, all encryption overhead is 0
|
|
result.EncryptionOverhead = 0
|
|
|
|
// Adjust based on strategy
|
|
switch strategy {
|
|
case CopyStrategyDirect:
|
|
// Direct copy: no size change
|
|
result.TargetSize = result.SourceSize
|
|
result.CanPreallocate = true
|
|
|
|
case CopyStrategyKeyRotation:
|
|
// Key rotation: size might change slightly due to different IVs
|
|
if info.SourceType == EncryptionTypeSSEC && info.TargetType == EncryptionTypeSSEC {
|
|
// SSE-C key rotation: same overhead
|
|
result.TargetSize = result.SourceSize
|
|
}
|
|
result.CanPreallocate = true
|
|
|
|
case CopyStrategyEncrypt, CopyStrategyDecrypt, CopyStrategyReencrypt:
|
|
// Size changes based on encryption transition
|
|
result.TargetSize = info.TargetSize
|
|
result.CanPreallocate = !info.IsCompressed
|
|
}
|
|
|
|
return result
|
|
}
|