XL/fs: GetObject should validate all its inputs. (#2142)

Fixes #2141
Fixes #2139
This commit is contained in:
Harshavardhana 2016-07-08 07:46:49 -07:00 committed by GitHub
parent ca1b1921c4
commit ec35330ebb
6 changed files with 79 additions and 51 deletions

View File

@ -74,8 +74,8 @@ func setObjectHeaders(w http.ResponseWriter, objInfo ObjectInfo, contentRange *h
w.Header().Set("Content-Length", strconv.FormatInt(objInfo.Size, 10)) w.Header().Set("Content-Length", strconv.FormatInt(objInfo.Size, 10))
// for providing ranged content // for providing ranged content
if contentRange != nil && contentRange.firstBytePos > -1 { if contentRange != nil && contentRange.offsetBegin > -1 {
// override content-length // Override content-length
w.Header().Set("Content-Length", strconv.FormatInt(contentRange.getLength(), 10)) w.Header().Set("Content-Length", strconv.FormatInt(contentRange.getLength(), 10))
w.Header().Set("Content-Range", contentRange.String()) w.Header().Set("Content-Range", contentRange.String())
w.WriteHeader(http.StatusPartialContent) w.WriteHeader(http.StatusPartialContent)

View File

@ -221,14 +221,24 @@ func (fs fsObjects) GetObject(bucket, object string, offset int64, length int64,
if offset < 0 || length < 0 { if offset < 0 || length < 0 {
return toObjectErr(errUnexpected, bucket, object) return toObjectErr(errUnexpected, bucket, object)
} }
// Writer cannot be nil.
if writer == nil {
return toObjectErr(errUnexpected, bucket, object)
}
// Stat the file to get file size.
fi, err := fs.storage.StatFile(bucket, object) fi, err := fs.storage.StatFile(bucket, object)
if err != nil { if err != nil {
return toObjectErr(err, bucket, object) return toObjectErr(err, bucket, object)
} }
if offset > fi.Size { // Reply back invalid range if the input offset and length fall out of range.
return InvalidRange{} if offset > fi.Size || length > fi.Size {
return InvalidRange{offset, length, fi.Size}
}
// Reply if we have inputs with offset and length falling out of file size range.
if offset+length > fi.Size {
return InvalidRange{offset, length, fi.Size}
} }
var totalLeft = length var totalLeft = length

View File

@ -32,10 +32,14 @@ const (
var errInvalidRange = errors.New("Invalid range") var errInvalidRange = errors.New("Invalid range")
// InvalidRange - invalid range typed error. // InvalidRange - invalid range typed error.
type InvalidRange struct{} type InvalidRange struct {
offsetBegin int64
offsetEnd int64
resourceSize int64
}
func (e InvalidRange) Error() string { func (e InvalidRange) Error() string {
return "The requested range is not satisfiable" return fmt.Sprintf("The requested range \"bytes %d-%d/%d\" is not satisfiable.", e.offsetBegin, e.offsetEnd, e.resourceSize)
} }
// Valid byte position regexp // Valid byte position regexp
@ -43,22 +47,22 @@ var validBytePos = regexp.MustCompile(`^[0-9]+$`)
// HttpRange specifies the byte range to be sent to the client. // HttpRange specifies the byte range to be sent to the client.
type httpRange struct { type httpRange struct {
firstBytePos int64 offsetBegin int64
lastBytePos int64 offsetEnd int64
size int64 resourceSize int64
} }
// String populate range stringer interface // String populate range stringer interface
func (hrange httpRange) String() string { func (hrange httpRange) String() string {
return fmt.Sprintf("bytes %d-%d/%d", hrange.firstBytePos, hrange.lastBytePos, hrange.size) return fmt.Sprintf("bytes %d-%d/%d", hrange.offsetBegin, hrange.offsetEnd, hrange.resourceSize)
} }
// getLength - get length from the range. // getlength - get length from the range.
func (hrange httpRange) getLength() int64 { func (hrange httpRange) getLength() int64 {
return 1 + hrange.lastBytePos - hrange.firstBytePos return 1 + hrange.offsetEnd - hrange.offsetBegin
} }
func parseRequestRange(rangeString string, size int64) (hrange *httpRange, err error) { func parseRequestRange(rangeString string, resourceSize int64) (hrange *httpRange, err error) {
// Return error if given range string doesn't start with byte range prefix. // Return error if given range string doesn't start with byte range prefix.
if !strings.HasPrefix(rangeString, byteRangePrefix) { if !strings.HasPrefix(rangeString, byteRangePrefix) {
return nil, fmt.Errorf("'%s' does not start with '%s'", rangeString, byteRangePrefix) return nil, fmt.Errorf("'%s' does not start with '%s'", rangeString, byteRangePrefix)
@ -73,73 +77,73 @@ func parseRequestRange(rangeString string, size int64) (hrange *httpRange, err e
return nil, fmt.Errorf("'%s' does not have a valid range value", rangeString) return nil, fmt.Errorf("'%s' does not have a valid range value", rangeString)
} }
firstBytePosString := byteRangeString[:sepIndex] offsetBeginString := byteRangeString[:sepIndex]
firstBytePos := int64(-1) offsetBegin := int64(-1)
// Convert firstBytePosString only if its not empty. // Convert offsetBeginString only if its not empty.
if len(firstBytePosString) > 0 { if len(offsetBeginString) > 0 {
if !validBytePos.MatchString(firstBytePosString) { if !validBytePos.MatchString(offsetBeginString) {
return nil, fmt.Errorf("'%s' does not have a valid first byte position value", rangeString) return nil, fmt.Errorf("'%s' does not have a valid first byte position value", rangeString)
} }
if firstBytePos, err = strconv.ParseInt(firstBytePosString, 10, 64); err != nil { if offsetBegin, err = strconv.ParseInt(offsetBeginString, 10, 64); err != nil {
return nil, fmt.Errorf("'%s' does not have a valid first byte position value", rangeString) return nil, fmt.Errorf("'%s' does not have a valid first byte position value", rangeString)
} }
} }
lastBytePosString := byteRangeString[sepIndex+1:] offsetEndString := byteRangeString[sepIndex+1:]
lastBytePos := int64(-1) offsetEnd := int64(-1)
// Convert lastBytePosString only if its not empty. // Convert offsetEndString only if its not empty.
if len(lastBytePosString) > 0 { if len(offsetEndString) > 0 {
if !validBytePos.MatchString(lastBytePosString) { if !validBytePos.MatchString(offsetEndString) {
return nil, fmt.Errorf("'%s' does not have a valid last byte position value", rangeString) return nil, fmt.Errorf("'%s' does not have a valid last byte position value", rangeString)
} }
if lastBytePos, err = strconv.ParseInt(lastBytePosString, 10, 64); err != nil { if offsetEnd, err = strconv.ParseInt(offsetEndString, 10, 64); err != nil {
return nil, fmt.Errorf("'%s' does not have a valid last byte position value", rangeString) return nil, fmt.Errorf("'%s' does not have a valid last byte position value", rangeString)
} }
} }
// rangeString contains first and last byte positions. eg. "bytes=2-5" // rangeString contains first and last byte positions. eg. "bytes=2-5"
if firstBytePos > -1 && lastBytePos > -1 { if offsetBegin > -1 && offsetEnd > -1 {
if firstBytePos > lastBytePos { if offsetBegin > offsetEnd {
// Last byte position is not greater than first byte position. eg. "bytes=5-2" // Last byte position is not greater than first byte position. eg. "bytes=5-2"
return nil, fmt.Errorf("'%s' does not have valid range value", rangeString) return nil, fmt.Errorf("'%s' does not have valid range value", rangeString)
} }
// First and last byte positions should not be >= size. // First and last byte positions should not be >= resourceSize.
if firstBytePos >= size { if offsetBegin >= resourceSize {
return nil, errInvalidRange return nil, errInvalidRange
} }
if lastBytePos >= size { if offsetEnd >= resourceSize {
lastBytePos = size - 1 offsetEnd = resourceSize - 1
} }
} else if firstBytePos > -1 { } else if offsetBegin > -1 {
// rangeString contains only first byte position. eg. "bytes=8-" // rangeString contains only first byte position. eg. "bytes=8-"
if firstBytePos >= size { if offsetBegin >= resourceSize {
// First byte position should not be >= size. // First byte position should not be >= resourceSize.
return nil, errInvalidRange return nil, errInvalidRange
} }
lastBytePos = size - 1 offsetEnd = resourceSize - 1
} else if lastBytePos > -1 { } else if offsetEnd > -1 {
// rangeString contains only last byte position. eg. "bytes=-3" // rangeString contains only last byte position. eg. "bytes=-3"
if lastBytePos == 0 { if offsetEnd == 0 {
// Last byte position should not be zero eg. "bytes=-0" // Last byte position should not be zero eg. "bytes=-0"
return nil, errInvalidRange return nil, errInvalidRange
} }
if lastBytePos >= size { if offsetEnd >= resourceSize {
firstBytePos = 0 offsetBegin = 0
} else { } else {
firstBytePos = size - lastBytePos offsetBegin = resourceSize - offsetEnd
} }
lastBytePos = size - 1 offsetEnd = resourceSize - 1
} else { } else {
// rangeString contains first and last byte positions missing. eg. "bytes=-" // rangeString contains first and last byte positions missing. eg. "bytes=-"
return nil, fmt.Errorf("'%s' does not have valid range value", rangeString) return nil, fmt.Errorf("'%s' does not have valid range value", rangeString)
} }
return &httpRange{firstBytePos, lastBytePos, size}, nil return &httpRange{offsetBegin, offsetEnd, resourceSize}, nil
} }

View File

@ -22,10 +22,10 @@ import "testing"
func TestParseRequestRange(t *testing.T) { func TestParseRequestRange(t *testing.T) {
// Test success cases. // Test success cases.
successCases := []struct { successCases := []struct {
rangeString string rangeString string
firstBytePos int64 offsetBegin int64
lastBytePos int64 offsetEnd int64
length int64 length int64
}{ }{
{"bytes=2-5", 2, 5, 4}, {"bytes=2-5", 2, 5, 4},
{"bytes=2-20", 2, 9, 8}, {"bytes=2-20", 2, 9, 8},
@ -42,12 +42,12 @@ func TestParseRequestRange(t *testing.T) {
t.Fatalf("expected: <nil>, got: %s", err) t.Fatalf("expected: <nil>, got: %s", err)
} }
if hrange.firstBytePos != successCase.firstBytePos { if hrange.offsetBegin != successCase.offsetBegin {
t.Fatalf("expected: %d, got: %d", successCase.firstBytePos, hrange.firstBytePos) t.Fatalf("expected: %d, got: %d", successCase.offsetBegin, hrange.offsetBegin)
} }
if hrange.lastBytePos != successCase.lastBytePos { if hrange.offsetEnd != successCase.offsetEnd {
t.Fatalf("expected: %d, got: %d", successCase.lastBytePos, hrange.lastBytePos) t.Fatalf("expected: %d, got: %d", successCase.offsetEnd, hrange.offsetEnd)
} }
if hrange.getLength() != successCase.length { if hrange.getLength() != successCase.length {
t.Fatalf("expected: %d, got: %d", successCase.length, hrange.getLength()) t.Fatalf("expected: %d, got: %d", successCase.length, hrange.getLength())

View File

@ -138,7 +138,7 @@ func (api objectAPIHandlers) GetObjectHandler(w http.ResponseWriter, r *http.Req
startOffset := int64(0) startOffset := int64(0)
length := objInfo.Size length := objInfo.Size
if hrange != nil { if hrange != nil {
startOffset = hrange.firstBytePos startOffset = hrange.offsetBegin
length = hrange.getLength() length = hrange.getLength()
} }
// Reads the object at startOffset and writes to mw. // Reads the object at startOffset and writes to mw.

View File

@ -52,6 +52,10 @@ func (xl xlObjects) GetObject(bucket, object string, startOffset int64, length i
if startOffset < 0 || length < 0 { if startOffset < 0 || length < 0 {
return toObjectErr(errUnexpected, bucket, object) return toObjectErr(errUnexpected, bucket, object)
} }
// Writer cannot be nil.
if writer == nil {
return toObjectErr(errUnexpected, bucket, object)
}
// Lock the object before reading. // Lock the object before reading.
nsMutex.RLock(bucket, object) nsMutex.RLock(bucket, object)
defer nsMutex.RUnlock(bucket, object) defer nsMutex.RUnlock(bucket, object)
@ -78,6 +82,16 @@ func (xl xlObjects) GetObject(bucket, object string, startOffset int64, length i
} }
} }
// Reply back invalid range if the input offset and length fall out of range.
if startOffset > xlMeta.Stat.Size || length > xlMeta.Stat.Size {
return InvalidRange{startOffset, length, xlMeta.Stat.Size}
}
// Reply if we have inputs with offset and length.
if startOffset+length > xlMeta.Stat.Size {
return InvalidRange{startOffset, length, xlMeta.Stat.Size}
}
// Get start part index and offset. // Get start part index and offset.
partIndex, partOffset, err := xlMeta.ObjectToPartOffset(startOffset) partIndex, partOffset, err := xlMeta.ObjectToPartOffset(startOffset)
if err != nil { if err != nil {