@ -4,33 +4,44 @@ import (
"context"
"context"
"encoding/xml"
"encoding/xml"
"fmt"
"fmt"
"github.com/aws/aws-sdk-go/service/s3"
"github.com/seaweedfs/seaweedfs/weed/glog"
"github.com/seaweedfs/seaweedfs/weed/glog"
"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
"github.com/seaweedfs/seaweedfs/weed/s3api/s3_constants"
"github.com/seaweedfs/seaweedfs/weed/s3api/s3_constants"
"github.com/seaweedfs/seaweedfs/weed/s3api/s3err"
"io"
"io"
"net/http"
"net/http"
"net/url"
"net/url"
"strconv"
"strconv"
"strings"
"strings"
"time"
"github.com/seaweedfs/seaweedfs/weed/filer"
"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
"github.com/seaweedfs/seaweedfs/weed/s3api/s3err"
)
)
type OptionalString struct {
string
set bool
}
func ( o OptionalString ) MarshalXML ( e * xml . Encoder , startElement xml . StartElement ) error {
if ! o . set {
return nil
}
return e . EncodeElement ( o . string , startElement )
}
type ListBucketResultV2 struct {
type ListBucketResultV2 struct {
XMLName xml . Name ` xml:"http://s3.amazonaws.com/doc/2006-03-01/ ListBucketResult" `
Name string ` xml:"Name" `
Prefix string ` xml:"Prefix" `
MaxKeys int ` xml:"MaxKeys" `
Delimiter string ` xml:"Delimiter,omitempty" `
IsTruncated bool ` xml:"IsTruncated" `
Contents [ ] ListEntry ` xml:"Contents,omitempty" `
CommonPrefixes [ ] PrefixEntry ` xml:"CommonPrefixes,omitempty" `
ContinuationToken string ` xml:"ContinuationToken,omitempty" `
NextContinuationToken string ` xml:"NextContinuationToken,omitempty" `
KeyCount int ` xml:"KeyCount" `
StartAfter string ` xml:"StartAfter,omitempty" `
XMLName xml . Name ` xml:"http://s3.amazonaws.com/doc/2006-03-01/ ListBucketResult" `
Name string ` xml:"Name" `
Prefix string ` xml:"Prefix" `
MaxKeys int ` xml:"MaxKeys" `
Delimiter string ` xml:"Delimiter,omitempty" `
IsTruncated bool ` xml:"IsTruncated" `
Contents [ ] ListEntry ` xml:"Contents,omitempty" `
CommonPrefixes [ ] PrefixEntry ` xml:"CommonPrefixes,omitempty" `
ContinuationToken OptionalString ` xml:"ContinuationToken,omitempty" `
NextContinuationToken string ` xml:"NextContinuationToken,omitempty" `
EncodingType string ` xml:"EncodingType,omitempty" `
KeyCount int ` xml:"KeyCount" `
StartAfter string ` xml:"StartAfter,omitempty" `
}
}
func ( s3a * S3ApiServer ) ListObjectsV2Handler ( w http . ResponseWriter , r * http . Request ) {
func ( s3a * S3ApiServer ) ListObjectsV2Handler ( w http . ResponseWriter , r * http . Request ) {
@ -41,19 +52,19 @@ func (s3a *S3ApiServer) ListObjectsV2Handler(w http.ResponseWriter, r *http.Requ
bucket , _ := s3_constants . GetBucketAndObject ( r )
bucket , _ := s3_constants . GetBucketAndObject ( r )
glog . V ( 3 ) . Infof ( "ListObjectsV2Handler %s" , bucket )
glog . V ( 3 ) . Infof ( "ListObjectsV2Handler %s" , bucket )
originalPrefix , continuationToken , startAfter , delimiter , _ , maxKeys := getListObjectsV2Args ( r . URL . Query ( ) )
originalPrefix , startAfter , delimiter , continuationToken , encodingTypeUrl , fetchOwner , maxKeys := getListObjectsV2Args ( r . URL . Query ( ) )
if maxKeys < 0 {
if maxKeys < 0 {
s3err . WriteErrorResponse ( w , r , s3err . ErrInvalidMaxKeys )
s3err . WriteErrorResponse ( w , r , s3err . ErrInvalidMaxKeys )
return
return
}
}
marker := continuationToken
if continuationToken == "" {
marker := continuationToken . string
if ! continuationToken . set {
marker = startAfter
marker = startAfter
}
}
response , err := s3a . listFilerEntries ( bucket , originalPrefix , maxKeys , marker , delimiter )
response , err := s3a . listFilerEntries ( bucket , originalPrefix , maxKeys , marker , delimiter , encodingTypeUrl , fetchOwner )
if err != nil {
if err != nil {
s3err . WriteErrorResponse ( w , r , s3err . ErrInternalError )
s3err . WriteErrorResponse ( w , r , s3err . ErrInternalError )
@ -81,6 +92,9 @@ func (s3a *S3ApiServer) ListObjectsV2Handler(w http.ResponseWriter, r *http.Requ
Prefix : response . Prefix ,
Prefix : response . Prefix ,
StartAfter : startAfter ,
StartAfter : startAfter ,
}
}
if encodingTypeUrl {
responseV2 . EncodingType = s3 . EncodingTypeUrl
}
writeSuccessResponseXML ( w , r , responseV2 )
writeSuccessResponseXML ( w , r , responseV2 )
}
}
@ -93,14 +107,14 @@ func (s3a *S3ApiServer) ListObjectsV1Handler(w http.ResponseWriter, r *http.Requ
bucket , _ := s3_constants . GetBucketAndObject ( r )
bucket , _ := s3_constants . GetBucketAndObject ( r )
glog . V ( 3 ) . Infof ( "ListObjectsV1Handler %s" , bucket )
glog . V ( 3 ) . Infof ( "ListObjectsV1Handler %s" , bucket )
originalPrefix , marker , delimiter , maxKeys := getListObjectsV1Args ( r . URL . Query ( ) )
originalPrefix , marker , delimiter , encodingTypeUrl , maxKeys := getListObjectsV1Args ( r . URL . Query ( ) )
if maxKeys < 0 {
if maxKeys < 0 {
s3err . WriteErrorResponse ( w , r , s3err . ErrInvalidMaxKeys )
s3err . WriteErrorResponse ( w , r , s3err . ErrInvalidMaxKeys )
return
return
}
}
response , err := s3a . listFilerEntries ( bucket , originalPrefix , maxKeys , marker , delimiter )
response , err := s3a . listFilerEntries ( bucket , originalPrefix , maxKeys , marker , delimiter , encodingTypeUrl , true )
if err != nil {
if err != nil {
s3err . WriteErrorResponse ( w , r , s3err . ErrInternalError )
s3err . WriteErrorResponse ( w , r , s3err . ErrInternalError )
@ -117,7 +131,7 @@ func (s3a *S3ApiServer) ListObjectsV1Handler(w http.ResponseWriter, r *http.Requ
writeSuccessResponseXML ( w , r , response )
writeSuccessResponseXML ( w , r , response )
}
}
func ( s3a * S3ApiServer ) listFilerEntries ( bucket string , originalPrefix string , maxKeys int , originalMarker string , delimiter string ) ( response ListBucketResult , err error ) {
func ( s3a * S3ApiServer ) listFilerEntries ( bucket string , originalPrefix string , maxKeys int , originalMarker string , delimiter string , encodingTypeUrl bool , fetchOwner bool ) ( response ListBucketResult , err error ) {
// convert full path prefix into directory name and prefix for entry name
// convert full path prefix into directory name and prefix for entry name
requestDir , prefix , marker := normalizePrefixMarker ( originalPrefix , originalMarker )
requestDir , prefix , marker := normalizePrefixMarker ( originalPrefix , originalMarker )
bucketPrefix := fmt . Sprintf ( "%s/%s/" , s3a . option . BucketsPath , bucket )
bucketPrefix := fmt . Sprintf ( "%s/%s/" , s3a . option . BucketsPath , bucket )
@ -134,36 +148,22 @@ func (s3a *S3ApiServer) listFilerEntries(bucket string, originalPrefix string, m
maxKeys : maxKeys ,
maxKeys : maxKeys ,
prefixEndsOnDelimiter : strings . HasSuffix ( originalPrefix , "/" ) && len ( originalMarker ) == 0 ,
prefixEndsOnDelimiter : strings . HasSuffix ( originalPrefix , "/" ) && len ( originalMarker ) == 0 ,
}
}
if s3a . option . AllowListRecursive {
// Todo remove force disable
if s3a . option . AllowListRecursive && prefix == "force_disable" {
err = s3a . WithFilerClient ( false , func ( client filer_pb . SeaweedFilerClient ) error {
err = s3a . WithFilerClient ( false , func ( client filer_pb . SeaweedFilerClient ) error {
glog . V ( 0 ) . Infof ( "doListFilerRecursiveEntries reqDir: %s, prefix: %s, delimiter: %s, cursor: %+v" , reqDir , prefix , delimiter , cursor )
glog . V ( 0 ) . Infof ( "doListFilerRecursiveEntries reqDir: %s, prefix: %s, delimiter: %s, cursor: %+v" , reqDir , prefix , delimiter , cursor )
nextMarker , doErr = s3a . doListFilerRecursiveEntries ( client , reqDir , prefix , cursor , marker , delimiter , false ,
nextMarker , doErr = s3a . doListFilerRecursiveEntries ( client , reqDir , prefix , cursor , marker , delimiter , false ,
func ( dir string , entry * filer_pb . Entry ) {
func ( dir string , entry * filer_pb . Entry ) {
dirName , entryName , prefixName := entryUrlEncode ( dir , entry . Name , encodingTypeUrl )
if entry . IsDirectory {
if entry . IsDirectory {
if delimiter == "/" { // A response can contain CommonPrefixes only if you specify a delimiter.
if delimiter == "/" { // A response can contain CommonPrefixes only if you specify a delimiter.
commonPrefixes = append ( commonPrefixes , PrefixEntry {
commonPrefixes = append ( commonPrefixes , PrefixEntry {
Prefix : fmt . Sprintf ( "%s/%s/" , dir , entry . Name ) [ len ( bucketPrefix ) : ] ,
Prefix : fmt . Sprintf ( "%s/%s/" , dirName , prefix Name ) [ len ( bucketPrefix ) : ] ,
} )
} )
}
}
return
return
}
}
storageClass := "STANDARD"
if v , ok := entry . Extended [ s3_constants . AmzStorageClass ] ; ok {
storageClass = string ( v )
}
contents = append ( contents , ListEntry {
Key : fmt . Sprintf ( "%s/%s" , dir , entry . Name ) [ len ( bucketPrefix ) : ] ,
LastModified : time . Unix ( entry . Attributes . Mtime , 0 ) . UTC ( ) ,
ETag : "\"" + filer . ETag ( entry ) + "\"" ,
Size : int64 ( filer . FileSize ( entry ) ) ,
Owner : CanonicalUser {
ID : fmt . Sprintf ( "%x" , entry . Attributes . Uid ) ,
DisplayName : entry . Attributes . UserName ,
} ,
StorageClass : StorageClass ( storageClass ) ,
} )
contents = append ( contents , newListEntry ( entry , dirName , entryName , bucketPrefix , fetchOwner , entry . IsDirectoryKeyObject ( ) ) )
cursor . maxKeys --
cursor . maxKeys --
} ,
} ,
)
)
@ -180,6 +180,9 @@ func (s3a *S3ApiServer) listFilerEntries(bucket string, originalPrefix string, m
Contents : contents ,
Contents : contents ,
CommonPrefixes : commonPrefixes ,
CommonPrefixes : commonPrefixes ,
}
}
if encodingTypeUrl {
response . EncodingType = s3 . EncodingTypeUrl
}
return
return
}
}
@ -189,23 +192,16 @@ func (s3a *S3ApiServer) listFilerEntries(bucket string, originalPrefix string, m
empty := true
empty := true
nextMarker , doErr = s3a . doListFilerEntries ( client , reqDir , prefix , cursor , marker , delimiter , false , func ( dir string , entry * filer_pb . Entry ) {
nextMarker , doErr = s3a . doListFilerEntries ( client , reqDir , prefix , cursor , marker , delimiter , false , func ( dir string , entry * filer_pb . Entry ) {
empty = false
empty = false
glog . V ( 5 ) . Infof ( "doListFilerEntries dir: %s entry: %+v" , dir , entry )
dirName , entryName , prefixName := entryUrlEncode ( dir , entry . Name , encodingTypeUrl )
if entry . IsDirectory {
if entry . IsDirectory {
if entry . IsDirectoryKeyObject ( ) {
if entry . IsDirectoryKeyObject ( ) {
contents = append ( contents , ListEntry {
Key : fmt . Sprintf ( "%s/%s/" , dir , entry . Name ) [ len ( bucketPrefix ) : ] ,
LastModified : time . Unix ( entry . Attributes . Mtime , 0 ) . UTC ( ) ,
ETag : "\"" + filer . ETag ( entry ) + "\"" ,
Owner : CanonicalUser {
ID : fmt . Sprintf ( "%x" , entry . Attributes . Uid ) ,
DisplayName : entry . Attributes . UserName ,
} ,
StorageClass : "STANDARD" ,
} )
contents = append ( contents , newListEntry ( entry , dirName , entryName , bucketPrefix , fetchOwner , true ) )
cursor . maxKeys --
cursor . maxKeys --
// https://docs.aws.amazon.com/AmazonS3/latest/API/API_ListObjectsV2.html
// https://docs.aws.amazon.com/AmazonS3/latest/API/API_ListObjectsV2.html
} else if delimiter == "/" { // A response can contain CommonPrefixes only if you specify a delimiter.
} else if delimiter == "/" { // A response can contain CommonPrefixes only if you specify a delimiter.
commonPrefixes = append ( commonPrefixes , PrefixEntry {
commonPrefixes = append ( commonPrefixes , PrefixEntry {
Prefix : fmt . Sprintf ( "%s/%s/" , dir , entry . Name ) [ len ( bucketPrefix ) : ] ,
Prefix : fmt . Sprintf ( "%s/%s/" , dirName , prefixName ) [ len ( bucketPrefix ) : ] ,
} )
} )
//All of the keys (up to 1,000) rolled up into a common prefix count as a single return when calculating the number of returns.
//All of the keys (up to 1,000) rolled up into a common prefix count as a single return when calculating the number of returns.
cursor . maxKeys --
cursor . maxKeys --
@ -243,21 +239,7 @@ func (s3a *S3ApiServer) listFilerEntries(bucket string, originalPrefix string, m
}
}
}
}
if ! delimiterFound {
if ! delimiterFound {
storageClass := "STANDARD"
if v , ok := entry . Extended [ s3_constants . AmzStorageClass ] ; ok {
storageClass = string ( v )
}
contents = append ( contents , ListEntry {
Key : fmt . Sprintf ( "%s/%s" , dir , entry . Name ) [ len ( bucketPrefix ) : ] ,
LastModified : time . Unix ( entry . Attributes . Mtime , 0 ) . UTC ( ) ,
ETag : "\"" + filer . ETag ( entry ) + "\"" ,
Size : int64 ( filer . FileSize ( entry ) ) ,
Owner : CanonicalUser {
ID : fmt . Sprintf ( "%x" , entry . Attributes . Uid ) ,
DisplayName : entry . Attributes . UserName ,
} ,
StorageClass : StorageClass ( storageClass ) ,
} )
contents = append ( contents , newListEntry ( entry , dirName , entryName , bucketPrefix , fetchOwner , false ) )
cursor . maxKeys --
cursor . maxKeys --
}
}
}
}
@ -291,7 +273,9 @@ func (s3a *S3ApiServer) listFilerEntries(bucket string, originalPrefix string, m
Contents : contents ,
Contents : contents ,
CommonPrefixes : commonPrefixes ,
CommonPrefixes : commonPrefixes ,
}
}
if encodingTypeUrl {
response . EncodingType = s3 . EncodingTypeUrl
}
return nil
return nil
} )
} )
@ -514,11 +498,12 @@ func (s3a *S3ApiServer) doListFilerEntries(client filer_pb.SeaweedFilerClient, d
return
return
}
}
func getListObjectsV2Args ( values url . Values ) ( prefix , token , startAfter , delimiter string , fetchOwner bool , maxkeys int ) {
func getListObjectsV2Args ( values url . Values ) ( prefix , startAfter , delimiter string , token OptionalString , encodingTypeUrl bool , fetchOwner bool , maxkeys int ) {
prefix = values . Get ( "prefix" )
prefix = values . Get ( "prefix" )
token = values . Get ( "continuation-token" )
token = OptionalString { set : values . Has ( "continuation-token" ) , string : values . Get ( "continuation-token" ) }
startAfter = values . Get ( "start-after" )
startAfter = values . Get ( "start-after" )
delimiter = values . Get ( "delimiter" )
delimiter = values . Get ( "delimiter" )
encodingTypeUrl = values . Get ( "encoding-type" ) == s3 . EncodingTypeUrl
if values . Get ( "max-keys" ) != "" {
if values . Get ( "max-keys" ) != "" {
maxkeys , _ = strconv . Atoi ( values . Get ( "max-keys" ) )
maxkeys , _ = strconv . Atoi ( values . Get ( "max-keys" ) )
} else {
} else {
@ -528,10 +513,11 @@ func getListObjectsV2Args(values url.Values) (prefix, token, startAfter, delimit
return
return
}
}
func getListObjectsV1Args ( values url . Values ) ( prefix , marker , delimiter string , maxkeys int ) {
func getListObjectsV1Args ( values url . Values ) ( prefix , marker , delimiter string , encodingTypeUrl bool , maxkeys int ) {
prefix = values . Get ( "prefix" )
prefix = values . Get ( "prefix" )
marker = values . Get ( "marker" )
marker = values . Get ( "marker" )
delimiter = values . Get ( "delimiter" )
delimiter = values . Get ( "delimiter" )
encodingTypeUrl = values . Get ( "encoding-type" ) == "url"
if values . Get ( "max-keys" ) != "" {
if values . Get ( "max-keys" ) != "" {
maxkeys , _ = strconv . Atoi ( values . Get ( "max-keys" ) )
maxkeys , _ = strconv . Atoi ( values . Get ( "max-keys" ) )
} else {
} else {