|
|
|
@ -5,8 +5,8 @@ import ( |
|
|
|
|
"crypto/md5" |
|
|
|
|
"crypto/sha512" |
|
|
|
|
"encoding/base64" |
|
|
|
|
"encoding/gob" |
|
|
|
|
"encoding/hex" |
|
|
|
|
"encoding/json" |
|
|
|
|
"errors" |
|
|
|
|
"fmt" |
|
|
|
|
"io" |
|
|
|
@ -73,7 +73,7 @@ func (fs *fsDriver) ListMultipartUploads(bucket string, resources drivers.Bucket |
|
|
|
|
defer activeSessionFile.Close() |
|
|
|
|
|
|
|
|
|
var deserializedActiveSession map[string]*MultipartSession |
|
|
|
|
decoder := gob.NewDecoder(activeSessionFile) |
|
|
|
|
decoder := json.NewDecoder(activeSessionFile) |
|
|
|
|
err = decoder.Decode(&deserializedActiveSession) |
|
|
|
|
if err != nil { |
|
|
|
|
return drivers.BucketMultipartResourcesMetadata{}, iodine.New(err, nil) |
|
|
|
@ -142,21 +142,6 @@ func (fs *fsDriver) NewMultipartUpload(bucket, key, contentType string) (string, |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(drivers.InternalError{}, nil) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
var activeSessionFile *os.File |
|
|
|
|
if _, err := os.Stat(bucketPath + "$activeSession"); os.IsNotExist(err) { |
|
|
|
|
activeSessionFile, err = os.OpenFile(bucketPath+"$activeSession", os.O_WRONLY|os.O_CREATE, 0600) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
|
} |
|
|
|
|
} else { |
|
|
|
|
activeSessionFile, err = os.OpenFile(bucketPath+"$activeSession", os.O_WRONLY, 0600) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
defer activeSessionFile.Close() |
|
|
|
|
|
|
|
|
|
objectPath := path.Join(bucketPath, key) |
|
|
|
|
objectDir := path.Dir(objectPath) |
|
|
|
|
if _, err := os.Stat(objectDir); os.IsNotExist(err) { |
|
|
|
@ -174,6 +159,20 @@ func (fs *fsDriver) NewMultipartUpload(bucket, key, contentType string) (string, |
|
|
|
|
}, nil) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
var activeSessionFile *os.File |
|
|
|
|
if _, err := os.Stat(bucketPath + "$activeSession"); os.IsNotExist(err) { |
|
|
|
|
activeSessionFile, err = os.OpenFile(bucketPath+"$activeSession", os.O_WRONLY|os.O_CREATE, 0600) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
|
} |
|
|
|
|
} else { |
|
|
|
|
activeSessionFile, err = os.OpenFile(bucketPath+"$activeSession", os.O_WRONLY, 0600) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
defer activeSessionFile.Close() |
|
|
|
|
|
|
|
|
|
id := []byte(strconv.FormatInt(rand.Int63(), 10) + bucket + key + time.Now().String()) |
|
|
|
|
uploadIDSum := sha512.Sum512(id) |
|
|
|
|
uploadID := base64.URLEncoding.EncodeToString(uploadIDSum[:])[:47] |
|
|
|
@ -192,12 +191,12 @@ func (fs *fsDriver) NewMultipartUpload(bucket, key, contentType string) (string, |
|
|
|
|
mpartSession.Parts = parts |
|
|
|
|
fs.multiparts.ActiveSession[key] = mpartSession |
|
|
|
|
|
|
|
|
|
encoder := gob.NewEncoder(file) |
|
|
|
|
encoder := json.NewEncoder(file) |
|
|
|
|
err = encoder.Encode(mpartSession) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
|
} |
|
|
|
|
encoder = gob.NewEncoder(activeSessionFile) |
|
|
|
|
encoder = json.NewEncoder(activeSessionFile) |
|
|
|
|
err = encoder.Encode(fs.multiparts.ActiveSession) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
@ -329,17 +328,17 @@ func (fs *fsDriver) CreateObjectPart(bucket, key, uploadID string, partID int, c |
|
|
|
|
defer multiPartfile.Close() |
|
|
|
|
|
|
|
|
|
var deserializedMultipartSession MultipartSession |
|
|
|
|
decoder := gob.NewDecoder(multiPartfile) |
|
|
|
|
decoder := json.NewDecoder(multiPartfile) |
|
|
|
|
err = decoder.Decode(&deserializedMultipartSession) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
|
} |
|
|
|
|
deserializedMultipartSession.Parts = append(deserializedMultipartSession.Parts, &partMetadata) |
|
|
|
|
deserializedMultipartSession.TotalParts++ |
|
|
|
|
fs.multiparts.ActiveSession[uploadID] = &deserializedMultipartSession |
|
|
|
|
fs.multiparts.ActiveSession[key] = &deserializedMultipartSession |
|
|
|
|
|
|
|
|
|
sort.Sort(partNumber(deserializedMultipartSession.Parts)) |
|
|
|
|
encoder := gob.NewEncoder(multiPartfile) |
|
|
|
|
encoder := json.NewEncoder(multiPartfile) |
|
|
|
|
err = encoder.Encode(&deserializedMultipartSession) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
@ -425,7 +424,7 @@ func (fs *fsDriver) CompleteMultipartUpload(bucket, key, uploadID string, parts |
|
|
|
|
} |
|
|
|
|
md5sum := hex.EncodeToString(h.Sum(nil)) |
|
|
|
|
|
|
|
|
|
delete(fs.multiparts.ActiveSession, uploadID) |
|
|
|
|
delete(fs.multiparts.ActiveSession, key) |
|
|
|
|
for partNumber := range parts { |
|
|
|
|
err = os.Remove(objectPath + fmt.Sprintf("$%d", partNumber)) |
|
|
|
|
if err != nil { |
|
|
|
@ -447,19 +446,20 @@ func (fs *fsDriver) CompleteMultipartUpload(bucket, key, uploadID string, parts |
|
|
|
|
ContentType: "application/octet-stream", |
|
|
|
|
Md5sum: h.Sum(nil), |
|
|
|
|
} |
|
|
|
|
// serialize metadata to gob
|
|
|
|
|
encoder := gob.NewEncoder(file) |
|
|
|
|
// serialize metadata to json
|
|
|
|
|
encoder := json.NewEncoder(file) |
|
|
|
|
err = encoder.Encode(metadata) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
activeSessionFile, err := os.OpenFile(bucketPath+"$activeSession", os.O_WRONLY, 0600) |
|
|
|
|
activeSessionFile, err := os.OpenFile(bucketPath+"$activeSession", os.O_WRONLY|os.O_TRUNC, 0600) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
|
} |
|
|
|
|
defer activeSessionFile.Close() |
|
|
|
|
encoder = gob.NewEncoder(activeSessionFile) |
|
|
|
|
fmt.Println(fs.multiparts.ActiveSession) |
|
|
|
|
encoder = json.NewEncoder(activeSessionFile) |
|
|
|
|
err = encoder.Encode(fs.multiparts.ActiveSession) |
|
|
|
|
if err != nil { |
|
|
|
|
return "", iodine.New(err, nil) |
|
|
|
@ -514,7 +514,7 @@ func (fs *fsDriver) ListObjectParts(bucket, key string, resources drivers.Object |
|
|
|
|
defer multiPartfile.Close() |
|
|
|
|
|
|
|
|
|
var deserializedMultipartSession MultipartSession |
|
|
|
|
decoder := gob.NewDecoder(multiPartfile) |
|
|
|
|
decoder := json.NewDecoder(multiPartfile) |
|
|
|
|
err = decoder.Decode(&deserializedMultipartSession) |
|
|
|
|
if err != nil { |
|
|
|
|
return drivers.ObjectResourcesMetadata{}, iodine.New(err, nil) |
|
|
|
@ -570,14 +570,14 @@ func (fs *fsDriver) AbortMultipartUpload(bucket, key, uploadID string) error { |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
var deserializedMultipartSession MultipartSession |
|
|
|
|
decoder := gob.NewDecoder(multiPartfile) |
|
|
|
|
decoder := json.NewDecoder(multiPartfile) |
|
|
|
|
err = decoder.Decode(&deserializedMultipartSession) |
|
|
|
|
if err != nil { |
|
|
|
|
return iodine.New(err, nil) |
|
|
|
|
} |
|
|
|
|
multiPartfile.Close() // close it right here, since we will delete it subsequently
|
|
|
|
|
|
|
|
|
|
delete(fs.multiparts.ActiveSession, uploadID) |
|
|
|
|
delete(fs.multiparts.ActiveSession, key) |
|
|
|
|
for _, part := range deserializedMultipartSession.Parts { |
|
|
|
|
err = os.RemoveAll(objectPath + fmt.Sprintf("$%d", part.PartNumber)) |
|
|
|
|
if err != nil { |
|
|
|
|