mirror of
https://github.com/minio/minio.git
synced 2025-01-11 23:13:23 -05:00
fs/xl: Simplify bucket metadata reading. (#3486)
ObjectLayer GetObject() now returns the entire object if starting offset is 0 and length is negative. This also allows to simplify handler layer code where we always had to use GetObjectInfo() before proceeding to read bucket metadata files examples `policy.json`. This also reduces one additional call overhead.
This commit is contained in:
parent
f57f773189
commit
15b4c49621
@ -150,16 +150,8 @@ func readBucketPolicyJSON(bucket string, objAPI ObjectLayer) (bucketPolicyReader
|
|||||||
objLock.RLock()
|
objLock.RLock()
|
||||||
defer objLock.RUnlock()
|
defer objLock.RUnlock()
|
||||||
|
|
||||||
objInfo, err := objAPI.GetObjectInfo(minioMetaBucket, policyPath)
|
|
||||||
if err != nil {
|
|
||||||
if isErrObjectNotFound(err) || isErrIncompleteBody(err) {
|
|
||||||
return nil, BucketPolicyNotFound{Bucket: bucket}
|
|
||||||
}
|
|
||||||
errorIf(err, "Unable to load policy for the bucket %s.", bucket)
|
|
||||||
return nil, errorCause(err)
|
|
||||||
}
|
|
||||||
var buffer bytes.Buffer
|
var buffer bytes.Buffer
|
||||||
err = objAPI.GetObject(minioMetaBucket, policyPath, 0, objInfo.Size, &buffer)
|
err = objAPI.GetObject(minioMetaBucket, policyPath, 0, -1, &buffer)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if isErrObjectNotFound(err) || isErrIncompleteBody(err) {
|
if isErrObjectNotFound(err) || isErrIncompleteBody(err) {
|
||||||
return nil, BucketPolicyNotFound{Bucket: bucket}
|
return nil, BucketPolicyNotFound{Bucket: bucket}
|
||||||
|
@ -308,20 +308,8 @@ func loadNotificationConfig(bucket string, objAPI ObjectLayer) (*notificationCon
|
|||||||
objLock.RLock()
|
objLock.RLock()
|
||||||
defer objLock.RUnlock()
|
defer objLock.RUnlock()
|
||||||
|
|
||||||
objInfo, err := objAPI.GetObjectInfo(minioMetaBucket, ncPath)
|
|
||||||
if err != nil {
|
|
||||||
// 'notification.xml' not found return
|
|
||||||
// 'errNoSuchNotifications'. This is default when no
|
|
||||||
// bucket notifications are found on the bucket.
|
|
||||||
if isErrObjectNotFound(err) || isErrIncompleteBody(err) {
|
|
||||||
return nil, errNoSuchNotifications
|
|
||||||
}
|
|
||||||
errorIf(err, "Unable to load bucket-notification for bucket %s", bucket)
|
|
||||||
// Returns error for other errors.
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var buffer bytes.Buffer
|
var buffer bytes.Buffer
|
||||||
err = objAPI.GetObject(minioMetaBucket, ncPath, 0, objInfo.Size, &buffer)
|
err := objAPI.GetObject(minioMetaBucket, ncPath, 0, -1, &buffer) // Read everything.
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// 'notification.xml' not found return
|
// 'notification.xml' not found return
|
||||||
// 'errNoSuchNotifications'. This is default when no
|
// 'errNoSuchNotifications'. This is default when no
|
||||||
@ -363,20 +351,8 @@ func loadListenerConfig(bucket string, objAPI ObjectLayer) ([]listenerConfig, er
|
|||||||
objLock.RLock()
|
objLock.RLock()
|
||||||
defer objLock.RUnlock()
|
defer objLock.RUnlock()
|
||||||
|
|
||||||
objInfo, err := objAPI.GetObjectInfo(minioMetaBucket, lcPath)
|
|
||||||
if err != nil {
|
|
||||||
// 'listener.json' not found return
|
|
||||||
// 'errNoSuchNotifications'. This is default when no
|
|
||||||
// bucket notifications are found on the bucket.
|
|
||||||
if isErrObjectNotFound(err) {
|
|
||||||
return nil, errNoSuchNotifications
|
|
||||||
}
|
|
||||||
errorIf(err, "Unable to load bucket-listeners for bucket %s", bucket)
|
|
||||||
// Returns error for other errors.
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var buffer bytes.Buffer
|
var buffer bytes.Buffer
|
||||||
err = objAPI.GetObject(minioMetaBucket, lcPath, 0, objInfo.Size, &buffer)
|
err := objAPI.GetObject(minioMetaBucket, lcPath, 0, -1, &buffer)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// 'notification.xml' not found return
|
// 'notification.xml' not found return
|
||||||
// 'errNoSuchNotifications'. This is default when no
|
// 'errNoSuchNotifications'. This is default when no
|
||||||
|
@ -69,7 +69,7 @@ func TestInitEventNotifierFaultyDisks(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Test initEventNotifier() with faulty disks
|
// Test initEventNotifier() with faulty disks
|
||||||
for i := 1; i <= 5; i++ {
|
for i := 1; i <= 3; i++ {
|
||||||
fs.storage = newNaughtyDisk(fsstorage, map[int]error{i: errFaultyDisk}, nil)
|
fs.storage = newNaughtyDisk(fsstorage, map[int]error{i: errFaultyDisk}, nil)
|
||||||
if err := initEventNotifier(fs); errorCause(err) != errFaultyDisk {
|
if err := initEventNotifier(fs); errorCause(err) != errFaultyDisk {
|
||||||
t.Fatal("Unexpected error:", err)
|
t.Fatal("Unexpected error:", err)
|
||||||
|
16
cmd/fs-v1.go
16
cmd/fs-v1.go
@ -222,8 +222,8 @@ func (fs fsObjects) GetObject(bucket, object string, offset int64, length int64,
|
|||||||
if err = checkGetObjArgs(bucket, object); err != nil {
|
if err = checkGetObjArgs(bucket, object); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Offset and length cannot be negative.
|
// Offset cannot be negative.
|
||||||
if offset < 0 || length < 0 {
|
if offset < 0 {
|
||||||
return toObjectErr(traceError(errUnexpected), bucket, object)
|
return toObjectErr(traceError(errUnexpected), bucket, object)
|
||||||
}
|
}
|
||||||
// Writer cannot be nil.
|
// Writer cannot be nil.
|
||||||
@ -237,12 +237,13 @@ func (fs fsObjects) GetObject(bucket, object string, offset int64, length int64,
|
|||||||
return toObjectErr(traceError(err), bucket, object)
|
return toObjectErr(traceError(err), bucket, object)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reply back invalid range if the input offset and length fall out of range.
|
// For negative length we read everything.
|
||||||
if offset > fi.Size || length > fi.Size {
|
if length < 0 {
|
||||||
return traceError(InvalidRange{offset, length, fi.Size})
|
length = fi.Size - offset
|
||||||
}
|
}
|
||||||
// Reply if we have inputs with offset and length falling out of file size range.
|
|
||||||
if offset+length > fi.Size {
|
// Reply back invalid range if the input offset and length fall out of range.
|
||||||
|
if offset > fi.Size || offset+length > fi.Size {
|
||||||
return traceError(InvalidRange{offset, length, fi.Size})
|
return traceError(InvalidRange{offset, length, fi.Size})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -291,6 +292,7 @@ func (fs fsObjects) GetObject(bucket, object string, offset int64, length int64,
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns any error.
|
// Returns any error.
|
||||||
return toObjectErr(err, bucket, object)
|
return toObjectErr(err, bucket, object)
|
||||||
}
|
}
|
||||||
|
@ -121,28 +121,28 @@ func testGetObject(obj ObjectLayer, instanceType string, t TestErrHandler) {
|
|||||||
// Fetching the entire object.
|
// Fetching the entire object.
|
||||||
// Test case - 8.
|
// Test case - 8.
|
||||||
{bucketName, objectName, 0, int64(len(bytesData[0].byteData)), buffers[1], buffers[1], true, bytesData[0].byteData, nil},
|
{bucketName, objectName, 0, int64(len(bytesData[0].byteData)), buffers[1], buffers[1], true, bytesData[0].byteData, nil},
|
||||||
// Test case with content-range 1 to objectSize .
|
// Test case with `length` parameter set to a negative value.
|
||||||
// Test case - 9.
|
// Test case - 9.
|
||||||
|
{bucketName, objectName, 0, int64(-1), buffers[1], buffers[1], true, bytesData[0].byteData, nil},
|
||||||
|
// Test case with content-range 1 to objectSize .
|
||||||
|
// Test case - 10.
|
||||||
{bucketName, objectName, 1, int64(len(bytesData[0].byteData) - 1), buffers[1], buffers[1], true, bytesData[0].byteData[1:], nil},
|
{bucketName, objectName, 1, int64(len(bytesData[0].byteData) - 1), buffers[1], buffers[1], true, bytesData[0].byteData[1:], nil},
|
||||||
// Test case with content-range 100 to objectSize - 100.
|
// Test case with content-range 100 to objectSize - 100.
|
||||||
// Test case - 10.
|
// Test case - 11.
|
||||||
{bucketName, objectName, 100, int64(len(bytesData[0].byteData) - 200), buffers[1], buffers[1], true,
|
{bucketName, objectName, 100, int64(len(bytesData[0].byteData) - 200), buffers[1], buffers[1], true,
|
||||||
bytesData[0].byteData[100 : len(bytesData[0].byteData)-100], nil},
|
bytesData[0].byteData[100 : len(bytesData[0].byteData)-100], nil},
|
||||||
// Test case with offset greater than the size of the object
|
// Test case with offset greater than the size of the object
|
||||||
// Test case - 11.
|
// Test case - 12.
|
||||||
{bucketName, objectName, int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData)), buffers[0],
|
{bucketName, objectName, int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData)), buffers[0],
|
||||||
NewEOFWriter(buffers[0], 100), false, []byte{},
|
NewEOFWriter(buffers[0], 100), false, []byte{},
|
||||||
InvalidRange{int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData)), int64(len(bytesData[0].byteData))}},
|
InvalidRange{int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData)), int64(len(bytesData[0].byteData))}},
|
||||||
// Test case with offset greater than the size of the object.
|
// Test case with offset greater than the size of the object.
|
||||||
// Test case - 12.
|
// Test case - 13.
|
||||||
{bucketName, objectName, -1, int64(len(bytesData[0].byteData)), buffers[0], new(bytes.Buffer), false, []byte{}, errUnexpected},
|
{bucketName, objectName, -1, int64(len(bytesData[0].byteData)), buffers[0], new(bytes.Buffer), false, []byte{}, errUnexpected},
|
||||||
// Test case length parameter is more than the object size.
|
// Test case length parameter is more than the object size.
|
||||||
// Test case - 13.
|
// Test case - 14.
|
||||||
{bucketName, objectName, 0, int64(len(bytesData[0].byteData) + 1), buffers[1], buffers[1], false, bytesData[0].byteData,
|
{bucketName, objectName, 0, int64(len(bytesData[0].byteData) + 1), buffers[1], buffers[1], false, bytesData[0].byteData,
|
||||||
InvalidRange{0, int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData))}},
|
InvalidRange{0, int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData))}},
|
||||||
// Test case with `length` parameter set to a negative value.
|
|
||||||
// Test case - 14.
|
|
||||||
{bucketName, objectName, 0, int64(-1), buffers[1], buffers[1], false, bytesData[0].byteData, errUnexpected},
|
|
||||||
// Test case with offset + length > objectSize parameter set to a negative value.
|
// Test case with offset + length > objectSize parameter set to a negative value.
|
||||||
// Test case - 15.
|
// Test case - 15.
|
||||||
{bucketName, objectName, 2, int64(len(bytesData[0].byteData)), buffers[1], buffers[1], false, bytesData[0].byteData,
|
{bucketName, objectName, 2, int64(len(bytesData[0].byteData)), buffers[1], buffers[1], false, bytesData[0].byteData,
|
||||||
@ -391,34 +391,37 @@ func testGetObjectDiskNotFound(obj ObjectLayer, instanceType string, disks []str
|
|||||||
// Fetching the entire object.
|
// Fetching the entire object.
|
||||||
// Test case - 8.
|
// Test case - 8.
|
||||||
{bucketName, objectName, 0, int64(len(bytesData[0].byteData)), buffers[1], buffers[1], true, bytesData[0].byteData, nil},
|
{bucketName, objectName, 0, int64(len(bytesData[0].byteData)), buffers[1], buffers[1], true, bytesData[0].byteData, nil},
|
||||||
// Test case with content-range 1 to objectSize .
|
// Test case with `length` parameter set to a negative value.
|
||||||
// Test case - 9.
|
// Test case - 9.
|
||||||
|
{bucketName, objectName, 0, int64(-1), buffers[1], buffers[1], true, bytesData[0].byteData, nil},
|
||||||
|
// Test case with `length` parameter set to a negative value and offset is positive.
|
||||||
|
// Test case - 10.
|
||||||
|
{bucketName, objectName, 1, int64(-1), buffers[1], buffers[1], true, bytesData[0].byteData[1:], nil},
|
||||||
|
// Test case with content-range 1 to objectSize .
|
||||||
|
// Test case - 11.
|
||||||
{bucketName, objectName, 1, int64(len(bytesData[0].byteData) - 1), buffers[1], buffers[1], true, bytesData[0].byteData[1:], nil},
|
{bucketName, objectName, 1, int64(len(bytesData[0].byteData) - 1), buffers[1], buffers[1], true, bytesData[0].byteData[1:], nil},
|
||||||
// Test case with content-range 100 to objectSize - 100.
|
// Test case with content-range 100 to objectSize - 100.
|
||||||
// Test case - 10.
|
// Test case - 12.
|
||||||
{bucketName, objectName, 100, int64(len(bytesData[0].byteData) - 200), buffers[1], buffers[1], true,
|
{bucketName, objectName, 100, int64(len(bytesData[0].byteData) - 200), buffers[1], buffers[1], true,
|
||||||
bytesData[0].byteData[100 : len(bytesData[0].byteData)-100], nil},
|
bytesData[0].byteData[100 : len(bytesData[0].byteData)-100], nil},
|
||||||
// Test case with offset greater than the size of the object
|
// Test case with offset greater than the size of the object
|
||||||
// Test case - 11.
|
// Test case - 13.
|
||||||
{bucketName, objectName, int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData)), buffers[0],
|
{bucketName, objectName, int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData)), buffers[0],
|
||||||
NewEOFWriter(buffers[0], 100), false, []byte{},
|
NewEOFWriter(buffers[0], 100), false, []byte{},
|
||||||
InvalidRange{int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData)), int64(len(bytesData[0].byteData))}},
|
InvalidRange{int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData)), int64(len(bytesData[0].byteData))}},
|
||||||
// Test case with offset greater than the size of the object.
|
// Test case with offset greater than the size of the object.
|
||||||
// Test case - 12.
|
// Test case - 14.
|
||||||
{bucketName, objectName, -1, int64(len(bytesData[0].byteData)), buffers[0], new(bytes.Buffer), false, []byte{}, errUnexpected},
|
{bucketName, objectName, -1, int64(len(bytesData[0].byteData)), buffers[0], new(bytes.Buffer), false, []byte{}, errUnexpected},
|
||||||
// Test case length parameter is more than the object size.
|
// Test case length parameter is more than the object size.
|
||||||
// Test case - 13.
|
// Test case - 15.
|
||||||
{bucketName, objectName, 0, int64(len(bytesData[0].byteData) + 1), buffers[1], buffers[1], false, bytesData[0].byteData,
|
{bucketName, objectName, 0, int64(len(bytesData[0].byteData) + 1), buffers[1], buffers[1], false, bytesData[0].byteData,
|
||||||
InvalidRange{0, int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData))}},
|
InvalidRange{0, int64(len(bytesData[0].byteData) + 1), int64(len(bytesData[0].byteData))}},
|
||||||
// Test case with `length` parameter set to a negative value.
|
|
||||||
// Test case - 14.
|
|
||||||
{bucketName, objectName, 0, int64(-1), buffers[1], buffers[1], false, bytesData[0].byteData, errUnexpected},
|
|
||||||
// Test case with offset + length > objectSize parameter set to a negative value.
|
// Test case with offset + length > objectSize parameter set to a negative value.
|
||||||
// Test case - 15.
|
// Test case - 16.
|
||||||
{bucketName, objectName, 2, int64(len(bytesData[0].byteData)), buffers[1], buffers[1], false, bytesData[0].byteData,
|
{bucketName, objectName, 2, int64(len(bytesData[0].byteData)), buffers[1], buffers[1], false, bytesData[0].byteData,
|
||||||
InvalidRange{2, int64(len(bytesData[0].byteData)), int64(len(bytesData[0].byteData))}},
|
InvalidRange{2, int64(len(bytesData[0].byteData)), int64(len(bytesData[0].byteData))}},
|
||||||
// Test case with the writer set to nil.
|
// Test case with the writer set to nil.
|
||||||
// Test case - 16.
|
// Test case - 17.
|
||||||
{bucketName, objectName, 0, int64(len(bytesData[0].byteData)), buffers[1], nil, false, bytesData[0].byteData, errUnexpected},
|
{bucketName, objectName, 0, int64(len(bytesData[0].byteData)), buffers[1], nil, false, bytesData[0].byteData, errUnexpected},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -53,10 +53,12 @@ func (xl xlObjects) GetObject(bucket, object string, startOffset int64, length i
|
|||||||
if err := checkGetObjArgs(bucket, object); err != nil {
|
if err := checkGetObjArgs(bucket, object); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
// Start offset and length cannot be negative.
|
|
||||||
if startOffset < 0 || length < 0 {
|
// Start offset cannot be negative.
|
||||||
|
if startOffset < 0 {
|
||||||
return traceError(errUnexpected)
|
return traceError(errUnexpected)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Writer cannot be nil.
|
// Writer cannot be nil.
|
||||||
if writer == nil {
|
if writer == nil {
|
||||||
return traceError(errUnexpected)
|
return traceError(errUnexpected)
|
||||||
@ -88,13 +90,13 @@ func (xl xlObjects) GetObject(bucket, object string, startOffset int64, length i
|
|||||||
// Reorder parts metadata based on erasure distribution order.
|
// Reorder parts metadata based on erasure distribution order.
|
||||||
metaArr = getOrderedPartsMetadata(xlMeta.Erasure.Distribution, metaArr)
|
metaArr = getOrderedPartsMetadata(xlMeta.Erasure.Distribution, metaArr)
|
||||||
|
|
||||||
// Reply back invalid range if the input offset and length fall out of range.
|
// For negative length read everything.
|
||||||
if startOffset > xlMeta.Stat.Size || length > xlMeta.Stat.Size {
|
if length < 0 {
|
||||||
return traceError(InvalidRange{startOffset, length, xlMeta.Stat.Size})
|
length = xlMeta.Stat.Size - startOffset
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reply if we have inputs with offset and length.
|
// Reply back invalid range if the input offset and length fall out of range.
|
||||||
if startOffset+length > xlMeta.Stat.Size {
|
if startOffset > xlMeta.Stat.Size || startOffset+length > xlMeta.Stat.Size {
|
||||||
return traceError(InvalidRange{startOffset, length, xlMeta.Stat.Size})
|
return traceError(InvalidRange{startOffset, length, xlMeta.Stat.Size})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user