mirror of
https://github.com/minio/minio.git
synced 2025-04-04 11:50:36 -04:00
api: Fix verification of checkLeafDirectory. (#1347)
This fixes a problem where leaf directory has more than 1000 entries, also resulting in listing issues, leading to an infinite loop. Fixes #1334
This commit is contained in:
parent
cb1116725b
commit
1284ecc6f2
@ -30,6 +30,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
// Minio meta volume.
|
||||||
minioMetaVolume = ".minio"
|
minioMetaVolume = ".minio"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -37,8 +38,9 @@ const (
|
|||||||
// yes returns all the files inside it.
|
// yes returns all the files inside it.
|
||||||
func (o objectAPI) checkLeafDirectory(prefixPath string) (isLeaf bool, fis []FileInfo) {
|
func (o objectAPI) checkLeafDirectory(prefixPath string) (isLeaf bool, fis []FileInfo) {
|
||||||
var allFileInfos []FileInfo
|
var allFileInfos []FileInfo
|
||||||
|
var markerPath string
|
||||||
for {
|
for {
|
||||||
fileInfos, eof, e := o.storage.ListFiles(minioMetaVolume, prefixPath, "", false, 1000)
|
fileInfos, eof, e := o.storage.ListFiles(minioMetaVolume, prefixPath, markerPath, false, 1000)
|
||||||
if e != nil {
|
if e != nil {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@ -46,6 +48,7 @@ func (o objectAPI) checkLeafDirectory(prefixPath string) (isLeaf bool, fis []Fil
|
|||||||
if eof {
|
if eof {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
markerPath = allFileInfos[len(allFileInfos)-1].Name
|
||||||
}
|
}
|
||||||
for _, fileInfo := range allFileInfos {
|
for _, fileInfo := range allFileInfos {
|
||||||
if fileInfo.Mode.IsDir() {
|
if fileInfo.Mode.IsDir() {
|
||||||
@ -80,7 +83,7 @@ func (o objectAPI) ListMultipartUploads(bucket, prefix, keyMarker, uploadIDMarke
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Verify if delimiter is anything other than '/', which we do not support.
|
// Verify if delimiter is anything other than '/', which we do not support.
|
||||||
if delimiter != "" && delimiter != slashPathSeparator {
|
if delimiter != "" && delimiter != slashSeparator {
|
||||||
return ListMultipartsInfo{}, probe.NewError(UnsupportedDelimiter{
|
return ListMultipartsInfo{}, probe.NewError(UnsupportedDelimiter{
|
||||||
Delimiter: delimiter,
|
Delimiter: delimiter,
|
||||||
})
|
})
|
||||||
@ -93,7 +96,7 @@ func (o objectAPI) ListMultipartUploads(bucket, prefix, keyMarker, uploadIDMarke
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if uploadIDMarker != "" {
|
if uploadIDMarker != "" {
|
||||||
if strings.HasSuffix(keyMarker, slashPathSeparator) {
|
if strings.HasSuffix(keyMarker, slashSeparator) {
|
||||||
return result, probe.NewError(InvalidUploadIDKeyCombination{
|
return result, probe.NewError(InvalidUploadIDKeyCombination{
|
||||||
UploadIDMarker: uploadIDMarker,
|
UploadIDMarker: uploadIDMarker,
|
||||||
KeyMarker: keyMarker,
|
KeyMarker: keyMarker,
|
||||||
@ -111,13 +114,13 @@ func (o objectAPI) ListMultipartUploads(bucket, prefix, keyMarker, uploadIDMarke
|
|||||||
}
|
}
|
||||||
|
|
||||||
recursive := true
|
recursive := true
|
||||||
if delimiter == slashPathSeparator {
|
if delimiter == slashSeparator {
|
||||||
recursive = false
|
recursive = false
|
||||||
}
|
}
|
||||||
result.IsTruncated = true
|
result.IsTruncated = true
|
||||||
result.MaxUploads = maxUploads
|
result.MaxUploads = maxUploads
|
||||||
newMaxUploads := 0
|
newMaxUploads := 0
|
||||||
// not using path.Join() as it strips off the trailing '/'.
|
// Not using path.Join() as it strips off the trailing '/'.
|
||||||
// Also bucket should always be followed by '/' even if prefix is empty.
|
// Also bucket should always be followed by '/' even if prefix is empty.
|
||||||
prefixPath := pathJoin(bucket, prefix)
|
prefixPath := pathJoin(bucket, prefix)
|
||||||
if recursive {
|
if recursive {
|
||||||
@ -135,9 +138,9 @@ func (o objectAPI) ListMultipartUploads(bucket, prefix, keyMarker, uploadIDMarke
|
|||||||
keyMarkerPath = fi.Name
|
keyMarkerPath = fi.Name
|
||||||
// fi.Name will look like bucket/object/uploadID, extract object and uploadID.
|
// fi.Name will look like bucket/object/uploadID, extract object and uploadID.
|
||||||
uploadID := path.Base(fi.Name)
|
uploadID := path.Base(fi.Name)
|
||||||
objectName := strings.TrimPrefix(path.Dir(fi.Name), bucket+slashPathSeparator)
|
objectName := strings.TrimPrefix(path.Dir(fi.Name), retainSlash(bucket))
|
||||||
if strings.Contains(uploadID, ".") {
|
if strings.Contains(uploadID, ".") {
|
||||||
// contains partnumber and md5sum info, skip this.
|
// Contains partnumber and md5sum info, skip this.
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
result.Uploads = append(result.Uploads, uploadMetadata{
|
result.Uploads = append(result.Uploads, uploadMetadata{
|
||||||
@ -189,14 +192,14 @@ func (o objectAPI) ListMultipartUploads(bucket, prefix, keyMarker, uploadIDMarke
|
|||||||
var uploads []uploadMetadata
|
var uploads []uploadMetadata
|
||||||
for _, fi := range fileInfos {
|
for _, fi := range fileInfos {
|
||||||
leaf, entries := o.checkLeafDirectory(fi.Name)
|
leaf, entries := o.checkLeafDirectory(fi.Name)
|
||||||
objectName := strings.TrimPrefix(fi.Name, bucket+slashPathSeparator)
|
objectName := strings.TrimPrefix(fi.Name, retainSlash(bucket))
|
||||||
if leaf {
|
if leaf {
|
||||||
for _, entry := range entries {
|
for _, entry := range entries {
|
||||||
if strings.Contains(entry.Name, ".") {
|
if strings.Contains(path.Base(entry.Name), ".") {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
uploads = append(uploads, uploadMetadata{
|
uploads = append(uploads, uploadMetadata{
|
||||||
Object: strings.TrimSuffix(objectName, slashPathSeparator),
|
Object: path.Dir(objectName),
|
||||||
UploadID: path.Base(entry.Name),
|
UploadID: path.Base(entry.Name),
|
||||||
Initiated: entry.ModTime,
|
Initiated: entry.ModTime,
|
||||||
})
|
})
|
||||||
@ -222,7 +225,7 @@ func (o objectAPI) ListMultipartUploads(bucket, prefix, keyMarker, uploadIDMarke
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
result.NextKeyMarker = uploads[index].Object
|
result.NextKeyMarker = uploads[index].Object
|
||||||
if strings.HasSuffix(uploads[index].Object, slashPathSeparator) {
|
if strings.HasSuffix(uploads[index].Object, slashSeparator) {
|
||||||
// for a directory entry
|
// for a directory entry
|
||||||
result.CommonPrefixes = append(result.CommonPrefixes, uploads[index].Object)
|
result.CommonPrefixes = append(result.CommonPrefixes, uploads[index].Object)
|
||||||
continue
|
continue
|
||||||
@ -393,22 +396,24 @@ func (o objectAPI) ListObjectParts(bucket, object, uploadID string, partNumberMa
|
|||||||
return ListPartsInfo{}, probe.NewError(InvalidUploadID{UploadID: uploadID})
|
return ListPartsInfo{}, probe.NewError(InvalidUploadID{UploadID: uploadID})
|
||||||
}
|
}
|
||||||
result := ListPartsInfo{}
|
result := ListPartsInfo{}
|
||||||
marker := ""
|
var markerPath string
|
||||||
nextPartNumberMarker := 0
|
nextPartNumberMarker := 0
|
||||||
uploadIDPath := path.Join(bucket, object, uploadID)
|
uploadIDPath := path.Join(bucket, object, uploadID)
|
||||||
// Figure out the marker for the next subsequent calls, if the
|
// Figure out the marker for the next subsequent calls, if the
|
||||||
// partNumberMarker is already set.
|
// partNumberMarker is already set.
|
||||||
if partNumberMarker > 0 {
|
if partNumberMarker > 0 {
|
||||||
fileInfos, _, e := o.storage.ListFiles(minioMetaVolume, uploadIDPath+"."+strconv.Itoa(partNumberMarker)+".", "", false, 1)
|
uploadIDPartPrefix := uploadIDPath + "." + strconv.Itoa(partNumberMarker) + "."
|
||||||
|
fileInfos, _, e := o.storage.ListFiles(minioMetaVolume, uploadIDPartPrefix, "", false, 1)
|
||||||
if e != nil {
|
if e != nil {
|
||||||
return result, probe.NewError(e)
|
return result, probe.NewError(e)
|
||||||
}
|
}
|
||||||
if len(fileInfos) == 0 {
|
if len(fileInfos) == 0 {
|
||||||
return result, probe.NewError(InvalidPart{})
|
return result, probe.NewError(InvalidPart{})
|
||||||
}
|
}
|
||||||
marker = fileInfos[0].Name
|
markerPath = fileInfos[0].Name
|
||||||
}
|
}
|
||||||
fileInfos, eof, e := o.storage.ListFiles(minioMetaVolume, uploadIDPath+".", marker, false, maxParts)
|
uploadIDPrefix := uploadIDPath + "."
|
||||||
|
fileInfos, eof, e := o.storage.ListFiles(minioMetaVolume, uploadIDPrefix, markerPath, false, maxParts)
|
||||||
if e != nil {
|
if e != nil {
|
||||||
return result, probe.NewError(InvalidPart{})
|
return result, probe.NewError(InvalidPart{})
|
||||||
}
|
}
|
||||||
|
@ -294,7 +294,7 @@ func (o objectAPI) ListObjects(bucket, prefix, marker, delimiter string, maxKeys
|
|||||||
return ListObjectsInfo{}, probe.NewError(ObjectNameInvalid{Bucket: bucket, Object: prefix})
|
return ListObjectsInfo{}, probe.NewError(ObjectNameInvalid{Bucket: bucket, Object: prefix})
|
||||||
}
|
}
|
||||||
// Verify if delimiter is anything other than '/', which we do not support.
|
// Verify if delimiter is anything other than '/', which we do not support.
|
||||||
if delimiter != "" && delimiter != slashPathSeparator {
|
if delimiter != "" && delimiter != slashSeparator {
|
||||||
return ListObjectsInfo{}, probe.NewError(UnsupportedDelimiter{
|
return ListObjectsInfo{}, probe.NewError(UnsupportedDelimiter{
|
||||||
Delimiter: delimiter,
|
Delimiter: delimiter,
|
||||||
})
|
})
|
||||||
@ -309,7 +309,7 @@ func (o objectAPI) ListObjects(bucket, prefix, marker, delimiter string, maxKeys
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
recursive := true
|
recursive := true
|
||||||
if delimiter == slashPathSeparator {
|
if delimiter == slashSeparator {
|
||||||
recursive = false
|
recursive = false
|
||||||
}
|
}
|
||||||
fileInfos, eof, e := o.storage.ListFiles(bucket, prefix, marker, recursive, maxKeys)
|
fileInfos, eof, e := o.storage.ListFiles(bucket, prefix, marker, recursive, maxKeys)
|
||||||
@ -325,7 +325,7 @@ func (o objectAPI) ListObjects(bucket, prefix, marker, delimiter string, maxKeys
|
|||||||
result := ListObjectsInfo{IsTruncated: !eof}
|
result := ListObjectsInfo{IsTruncated: !eof}
|
||||||
for _, fileInfo := range fileInfos {
|
for _, fileInfo := range fileInfos {
|
||||||
// With delimiter set we fill in NextMarker and Prefixes.
|
// With delimiter set we fill in NextMarker and Prefixes.
|
||||||
if delimiter == slashPathSeparator {
|
if delimiter == slashSeparator {
|
||||||
result.NextMarker = fileInfo.Name
|
result.NextMarker = fileInfo.Name
|
||||||
if fileInfo.Mode.IsDir() {
|
if fileInfo.Mode.IsDir() {
|
||||||
result.Prefixes = append(result.Prefixes, fileInfo.Name)
|
result.Prefixes = append(result.Prefixes, fileInfo.Name)
|
||||||
|
@ -23,10 +23,6 @@ import (
|
|||||||
"unicode/utf8"
|
"unicode/utf8"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
|
||||||
slashPathSeparator = "/"
|
|
||||||
)
|
|
||||||
|
|
||||||
// validBucket regexp.
|
// validBucket regexp.
|
||||||
var validBucket = regexp.MustCompile(`^[a-z0-9][a-z0-9\.\-]{1,61}[a-z0-9]$`)
|
var validBucket = regexp.MustCompile(`^[a-z0-9][a-z0-9\.\-]{1,61}[a-z0-9]$`)
|
||||||
|
|
||||||
@ -91,8 +87,17 @@ func IsValidObjectPrefix(object string) bool {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func pathJoin(path1 string, path2 string) string {
|
// Slash separator.
|
||||||
return strings.TrimSuffix(path1, slashPathSeparator) + slashPathSeparator + path2
|
const slashSeparator = "/"
|
||||||
|
|
||||||
|
// retainSlash - retains slash from a path.
|
||||||
|
func retainSlash(s string) string {
|
||||||
|
return strings.TrimSuffix(s, slashSeparator) + slashSeparator
|
||||||
|
}
|
||||||
|
|
||||||
|
// pathJoin - path join.
|
||||||
|
func pathJoin(s1 string, s2 string) string {
|
||||||
|
return retainSlash(s1) + s2
|
||||||
}
|
}
|
||||||
|
|
||||||
// validates location constraint from the request body.
|
// validates location constraint from the request body.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user