mirror of
https://github.com/minio/minio.git
synced 2025-01-12 07:23:23 -05:00
33cd910d3a
backend/fs: More cleanup and start using checkBucketArg.
315 lines
8.3 KiB
Go
315 lines
8.3 KiB
Go
/*
|
|
* Minio Cloud Storage, (C) 2016 Minio, Inc.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package main
|
|
|
|
import (
|
|
"errors"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
func scanMultipartDir(bucketDir, prefixPath, markerPath, uploadIDMarker string, recursive bool) multipartObjectInfoChannel {
|
|
objectInfoCh := make(chan multipartObjectInfo, listObjectsLimit)
|
|
timeoutCh := make(chan struct{}, 1)
|
|
|
|
// TODO: check if bucketDir is absolute path
|
|
scanDir := bucketDir
|
|
dirDepth := bucketDir
|
|
|
|
if prefixPath != "" {
|
|
if !filepath.IsAbs(prefixPath) {
|
|
tmpPrefixPath := filepath.Join(bucketDir, prefixPath)
|
|
if strings.HasSuffix(prefixPath, string(os.PathSeparator)) {
|
|
tmpPrefixPath += string(os.PathSeparator)
|
|
}
|
|
prefixPath = tmpPrefixPath
|
|
}
|
|
|
|
// TODO: check if prefixPath starts with bucketDir
|
|
|
|
// Case #1: if prefixPath is /mnt/mys3/mybucket/2012/photos/paris, then
|
|
// dirDepth is /mnt/mys3/mybucket/2012/photos
|
|
// Case #2: if prefixPath is /mnt/mys3/mybucket/2012/photos/, then
|
|
// dirDepth is /mnt/mys3/mybucket/2012/photos
|
|
dirDepth = filepath.Dir(prefixPath)
|
|
scanDir = dirDepth
|
|
} else {
|
|
prefixPath = bucketDir
|
|
}
|
|
|
|
if markerPath != "" {
|
|
if !filepath.IsAbs(markerPath) {
|
|
tmpMarkerPath := filepath.Join(bucketDir, markerPath)
|
|
if strings.HasSuffix(markerPath, string(os.PathSeparator)) {
|
|
tmpMarkerPath += string(os.PathSeparator)
|
|
}
|
|
|
|
markerPath = tmpMarkerPath
|
|
}
|
|
|
|
// TODO: check markerPath must be a file
|
|
if uploadIDMarker != "" {
|
|
markerPath = filepath.Join(markerPath, uploadIDMarker+multipartUploadIDSuffix)
|
|
}
|
|
|
|
// TODO: check if markerPath starts with bucketDir
|
|
// TODO: check if markerPath starts with prefixPath
|
|
|
|
// Case #1: if markerPath is /mnt/mys3/mybucket/2012/photos/gophercon.png, then
|
|
// scanDir is /mnt/mys3/mybucket/2012/photos
|
|
// Case #2: if markerPath is /mnt/mys3/mybucket/2012/photos/gophercon.png/1fbd117a-268a-4ed0-85c9-8cc3888cbf20.uploadid, then
|
|
// scanDir is /mnt/mys3/mybucket/2012/photos/gophercon.png
|
|
// Case #3: if markerPath is /mnt/mys3/mybucket/2012/photos/, then
|
|
// scanDir is /mnt/mys3/mybucket/2012/photos
|
|
|
|
scanDir = filepath.Dir(markerPath)
|
|
} else {
|
|
markerPath = bucketDir
|
|
}
|
|
|
|
// Have bucketDir ends with os.PathSeparator
|
|
if !strings.HasSuffix(bucketDir, string(os.PathSeparator)) {
|
|
bucketDir += string(os.PathSeparator)
|
|
}
|
|
|
|
// Remove os.PathSeparator if scanDir ends with
|
|
if strings.HasSuffix(scanDir, string(os.PathSeparator)) {
|
|
scanDir = filepath.Dir(scanDir)
|
|
}
|
|
|
|
// goroutine - retrieves directory entries, makes ObjectInfo and sends into the channel.
|
|
go func() {
|
|
defer close(objectInfoCh)
|
|
defer close(timeoutCh)
|
|
|
|
// send function - returns true if ObjectInfo is sent
|
|
// within (time.Second * 15) else false on timeout.
|
|
send := func(oi multipartObjectInfo) bool {
|
|
timer := time.After(time.Second * 15)
|
|
select {
|
|
case objectInfoCh <- oi:
|
|
return true
|
|
case <-timer:
|
|
timeoutCh <- struct{}{}
|
|
return false
|
|
}
|
|
}
|
|
|
|
for {
|
|
// Filters scandir entries. This filter function is
|
|
// specific for multipart listing.
|
|
multipartFilterFn := func(dirent fsDirent) bool {
|
|
// Verify if dirent is a directory a regular file
|
|
// with match uploadID suffix.
|
|
if dirent.IsDir() || (dirent.IsRegular() && strings.HasSuffix(dirent.name, multipartUploadIDSuffix)) {
|
|
// Return if dirent matches prefix and
|
|
// lexically higher than marker.
|
|
return strings.HasPrefix(dirent.name, prefixPath) && dirent.name > markerPath
|
|
}
|
|
return false
|
|
}
|
|
dirents, err := scandir(scanDir, multipartFilterFn, false)
|
|
if err != nil {
|
|
send(multipartObjectInfo{Err: err})
|
|
return
|
|
}
|
|
|
|
var dirent fsDirent
|
|
for len(dirents) > 0 {
|
|
dirent, dirents = dirents[0], dirents[1:]
|
|
if dirent.IsRegular() {
|
|
// Handle uploadid file
|
|
name := strings.Replace(filepath.Dir(dirent.name), bucketDir, "", 1)
|
|
if name == "" {
|
|
// This should not happen ie uploadid file should not be in bucket directory
|
|
send(multipartObjectInfo{Err: errors.New("Corrupted metadata")})
|
|
return
|
|
}
|
|
|
|
uploadID := strings.Split(filepath.Base(dirent.name), multipartUploadIDSuffix)[0]
|
|
|
|
// Solaris and older unixes have modTime to be
|
|
// empty, fall back to os.Stat() to fill missing values.
|
|
if dirent.modTime.IsZero() {
|
|
if fi, e := os.Stat(dirent.name); e == nil {
|
|
dirent.modTime = fi.ModTime()
|
|
} else {
|
|
send(multipartObjectInfo{Err: e})
|
|
return
|
|
}
|
|
}
|
|
|
|
objInfo := multipartObjectInfo{
|
|
Name: name,
|
|
UploadID: uploadID,
|
|
ModifiedTime: dirent.modTime,
|
|
}
|
|
|
|
if !send(objInfo) {
|
|
return
|
|
}
|
|
|
|
continue
|
|
}
|
|
|
|
multipartSubDirentFilterFn := func(dirent fsDirent) bool {
|
|
return dirent.IsDir() || (dirent.IsRegular() && strings.HasSuffix(dirent.name, multipartUploadIDSuffix))
|
|
}
|
|
// Fetch sub dirents.
|
|
subDirents, err := scandir(dirent.name, multipartSubDirentFilterFn, false)
|
|
if err != nil {
|
|
send(multipartObjectInfo{Err: err})
|
|
return
|
|
}
|
|
|
|
subDirFound := false
|
|
uploadIDDirents := []fsDirent{}
|
|
// If subDirents has a directory, then current dirent needs to be sent
|
|
for _, subdirent := range subDirents {
|
|
if subdirent.IsDir() {
|
|
subDirFound = true
|
|
|
|
if recursive {
|
|
break
|
|
}
|
|
}
|
|
|
|
if !recursive && subdirent.IsRegular() {
|
|
uploadIDDirents = append(uploadIDDirents, subdirent)
|
|
}
|
|
}
|
|
|
|
// Send directory only for non-recursive listing
|
|
if !recursive && (subDirFound || len(subDirents) == 0) {
|
|
// Solaris and older unixes have modTime to be
|
|
// empty, fall back to os.Stat() to fill missing values.
|
|
if dirent.modTime.IsZero() {
|
|
if fi, e := os.Stat(dirent.name); e == nil {
|
|
dirent.modTime = fi.ModTime()
|
|
} else {
|
|
send(multipartObjectInfo{Err: e})
|
|
return
|
|
}
|
|
}
|
|
|
|
objInfo := multipartObjectInfo{
|
|
Name: strings.Replace(dirent.name, bucketDir, "", 1),
|
|
ModifiedTime: dirent.modTime,
|
|
IsDir: true,
|
|
}
|
|
|
|
if !send(objInfo) {
|
|
return
|
|
}
|
|
}
|
|
|
|
if recursive {
|
|
dirents = append(subDirents, dirents...)
|
|
} else {
|
|
dirents = append(uploadIDDirents, dirents...)
|
|
}
|
|
}
|
|
|
|
if !recursive {
|
|
break
|
|
}
|
|
|
|
markerPath = scanDir + string(os.PathSeparator)
|
|
if scanDir = filepath.Dir(scanDir); scanDir < dirDepth {
|
|
break
|
|
}
|
|
}
|
|
}()
|
|
|
|
// Return multipart info.
|
|
return multipartObjectInfoChannel{ch: objectInfoCh, timeoutCh: timeoutCh}
|
|
}
|
|
|
|
// multipartObjectInfo - Multipart object info
|
|
type multipartObjectInfo struct {
|
|
Name string
|
|
UploadID string
|
|
ModifiedTime time.Time
|
|
IsDir bool
|
|
Err error
|
|
}
|
|
|
|
// multipartObjectInfoChannel - multipart object info channel
|
|
type multipartObjectInfoChannel struct {
|
|
ch <-chan multipartObjectInfo
|
|
objInfo *multipartObjectInfo
|
|
closed bool
|
|
timeoutCh <-chan struct{}
|
|
timedOut bool
|
|
}
|
|
|
|
func (oic *multipartObjectInfoChannel) Read() (multipartObjectInfo, bool) {
|
|
if oic.closed {
|
|
return multipartObjectInfo{}, false
|
|
}
|
|
if oic.objInfo == nil {
|
|
// First read.
|
|
if oi, ok := <-oic.ch; ok {
|
|
oic.objInfo = &oi
|
|
} else {
|
|
oic.closed = true
|
|
return multipartObjectInfo{}, false
|
|
}
|
|
}
|
|
|
|
retObjInfo := *oic.objInfo
|
|
status := true
|
|
oic.objInfo = nil
|
|
|
|
// Read once more to know whether it was last read.
|
|
if oi, ok := <-oic.ch; ok {
|
|
oic.objInfo = &oi
|
|
} else {
|
|
oic.closed = true
|
|
}
|
|
|
|
return retObjInfo, status
|
|
}
|
|
|
|
// IsClosed - return whether channel is closed or not.
|
|
func (oic multipartObjectInfoChannel) IsClosed() bool {
|
|
if oic.objInfo != nil {
|
|
return false
|
|
}
|
|
return oic.closed
|
|
}
|
|
|
|
// IsTimedOut - return whether channel is closed due to timeout.
|
|
func (oic multipartObjectInfoChannel) IsTimedOut() bool {
|
|
if oic.timedOut {
|
|
return true
|
|
}
|
|
|
|
select {
|
|
case _, ok := <-oic.timeoutCh:
|
|
if ok {
|
|
oic.timedOut = true
|
|
return true
|
|
}
|
|
return false
|
|
default:
|
|
return false
|
|
}
|
|
}
|