forked from TrueCloudLab/frostfs-s3-gw
[#583] Fix list-buckets vhs routing
The problem is that with VHS requests, the list-buckets operation does not work because the request is filtered on list-objects-v1. Since list-buckets can also have query parameters, in the end it is necessary to distinguish list-buckets from list-objects-v1 only by the presence of the bucket name in the URL (provided that the request is in VHS style). Signed-off-by: Roman Loginov <r.loginov@yadro.com>
This commit is contained in:
parent
f2274b2786
commit
09412d8f20
4 changed files with 126 additions and 35 deletions
|
@ -30,6 +30,9 @@ const (
|
||||||
QueryMaxKeys = "max-keys"
|
QueryMaxKeys = "max-keys"
|
||||||
QueryMarker = "marker"
|
QueryMarker = "marker"
|
||||||
QueryEncodingType = "encoding-type"
|
QueryEncodingType = "encoding-type"
|
||||||
|
QueryMaxBuckets = "max-buckets"
|
||||||
|
QueryContinuationToken = "continuation-token"
|
||||||
|
QueryBucketRegion = "bucket-region"
|
||||||
amzTagging = "x-amz-tagging"
|
amzTagging = "x-amz-tagging"
|
||||||
|
|
||||||
unmatchedBucketOperation = "UnmatchedBucketOperation"
|
unmatchedBucketOperation = "UnmatchedBucketOperation"
|
||||||
|
|
|
@ -159,7 +159,7 @@ func NewRouter(cfg Config) *chi.Mux {
|
||||||
defaultRouter.Get("/", named(s3middleware.ListBucketsOperation, cfg.Handler.ListBucketsHandler))
|
defaultRouter.Get("/", named(s3middleware.ListBucketsOperation, cfg.Handler.ListBucketsHandler))
|
||||||
attachErrorHandler(defaultRouter)
|
attachErrorHandler(defaultRouter)
|
||||||
|
|
||||||
vhsRouter := bucketRouter(cfg.Handler)
|
vhsRouter := newDomainRouter(cfg.Handler)
|
||||||
router := newGlobalRouter(defaultRouter, vhsRouter)
|
router := newGlobalRouter(defaultRouter, vhsRouter)
|
||||||
|
|
||||||
api.Mount("/", router)
|
api.Mount("/", router)
|
||||||
|
@ -169,12 +169,43 @@ func NewRouter(cfg Config) *chi.Mux {
|
||||||
return api
|
return api
|
||||||
}
|
}
|
||||||
|
|
||||||
type globalRouter struct {
|
type domainRouter struct {
|
||||||
pathStyleRouter chi.Router
|
bucketRouter chi.Router
|
||||||
vhsRouter chi.Router
|
defaultRouter chi.Router
|
||||||
}
|
}
|
||||||
|
|
||||||
func newGlobalRouter(pathStyleRouter, vhsRouter chi.Router) *globalRouter {
|
func newDomainRouter(handler Handler) *domainRouter {
|
||||||
|
defaultRouter := chi.NewRouter()
|
||||||
|
defaultRouter.Group(func(r chi.Router) {
|
||||||
|
r.Method(http.MethodGet, "/", NewHandlerFilter().
|
||||||
|
Add(NewFilter().
|
||||||
|
AllowedQueries(s3middleware.QueryMaxBuckets, s3middleware.QueryPrefix,
|
||||||
|
s3middleware.QueryContinuationToken, s3middleware.QueryBucketRegion).
|
||||||
|
Handler(named(s3middleware.ListBucketsOperation, handler.ListBucketsHandler))).
|
||||||
|
DefaultHandler(notSupportedHandler()))
|
||||||
|
})
|
||||||
|
attachErrorHandler(defaultRouter)
|
||||||
|
|
||||||
|
return &domainRouter{
|
||||||
|
bucketRouter: bucketRouter(handler),
|
||||||
|
defaultRouter: defaultRouter,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *domainRouter) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if reqInfo := s3middleware.GetReqInfo(r.Context()); reqInfo.BucketName != "" {
|
||||||
|
g.bucketRouter.ServeHTTP(w, r)
|
||||||
|
} else {
|
||||||
|
g.defaultRouter.ServeHTTP(w, r)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type globalRouter struct {
|
||||||
|
pathStyleRouter chi.Router
|
||||||
|
vhsRouter *domainRouter
|
||||||
|
}
|
||||||
|
|
||||||
|
func newGlobalRouter(pathStyleRouter chi.Router, vhsRouter *domainRouter) *globalRouter {
|
||||||
return &globalRouter{
|
return &globalRouter{
|
||||||
pathStyleRouter: pathStyleRouter,
|
pathStyleRouter: pathStyleRouter,
|
||||||
vhsRouter: vhsRouter,
|
vhsRouter: vhsRouter,
|
||||||
|
@ -182,12 +213,11 @@ func newGlobalRouter(pathStyleRouter, vhsRouter chi.Router) *globalRouter {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *globalRouter) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
func (g *globalRouter) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
router := g.pathStyleRouter
|
|
||||||
if reqInfo := s3middleware.GetReqInfo(r.Context()); reqInfo.RequestVHSEnabled {
|
if reqInfo := s3middleware.GetReqInfo(r.Context()); reqInfo.RequestVHSEnabled {
|
||||||
router = g.vhsRouter
|
g.vhsRouter.ServeHTTP(w, r)
|
||||||
|
} else {
|
||||||
|
g.pathStyleRouter.ServeHTTP(w, r)
|
||||||
}
|
}
|
||||||
|
|
||||||
router.ServeHTTP(w, r)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func named(name string, handlerFunc http.HandlerFunc) http.HandlerFunc {
|
func named(name string, handlerFunc http.HandlerFunc) http.HandlerFunc {
|
||||||
|
@ -338,9 +368,6 @@ func bucketRouter(h Handler) chi.Router {
|
||||||
AllowedQueries(s3middleware.QueryDelimiter, s3middleware.QueryMaxKeys, s3middleware.QueryPrefix,
|
AllowedQueries(s3middleware.QueryDelimiter, s3middleware.QueryMaxKeys, s3middleware.QueryPrefix,
|
||||||
s3middleware.QueryMarker, s3middleware.QueryEncodingType).
|
s3middleware.QueryMarker, s3middleware.QueryEncodingType).
|
||||||
Handler(named(s3middleware.ListObjectsV1Operation, h.ListObjectsV1Handler))).
|
Handler(named(s3middleware.ListObjectsV1Operation, h.ListObjectsV1Handler))).
|
||||||
Add(NewFilter().
|
|
||||||
NoQueries().
|
|
||||||
Handler(listWrapper(h))).
|
|
||||||
DefaultHandler(notSupportedHandler()))
|
DefaultHandler(notSupportedHandler()))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -422,18 +449,6 @@ func bucketRouter(h Handler) chi.Router {
|
||||||
return bktRouter
|
return bktRouter
|
||||||
}
|
}
|
||||||
|
|
||||||
func listWrapper(h Handler) http.HandlerFunc {
|
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
if reqInfo := s3middleware.GetReqInfo(r.Context()); reqInfo.BucketName == "" {
|
|
||||||
reqInfo.API = s3middleware.ListBucketsOperation
|
|
||||||
h.ListBucketsHandler(w, r)
|
|
||||||
} else {
|
|
||||||
reqInfo.API = s3middleware.ListObjectsV1Operation
|
|
||||||
h.ListObjectsV1Handler(w, r)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func objectRouter(h Handler) chi.Router {
|
func objectRouter(h Handler) chi.Router {
|
||||||
objRouter := chi.NewRouter()
|
objRouter := chi.NewRouter()
|
||||||
|
|
||||||
|
|
|
@ -138,13 +138,14 @@ func (hf *HandlerFilters) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (hf *HandlerFilters) match(r *http.Request) http.Handler {
|
func (hf *HandlerFilters) match(r *http.Request) http.Handler {
|
||||||
|
queries := r.URL.Query()
|
||||||
|
|
||||||
LOOP:
|
LOOP:
|
||||||
for _, filter := range hf.filters {
|
for _, filter := range hf.filters {
|
||||||
if filter.noQueries && len(r.URL.Query()) > 0 {
|
if filter.noQueries && len(queries) > 0 {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if len(filter.allowedQueries) > 0 {
|
if len(filter.allowedQueries) > 0 {
|
||||||
queries := r.URL.Query()
|
|
||||||
for key := range queries {
|
for key := range queries {
|
||||||
if _, ok := filter.allowedQueries[key]; !ok {
|
if _, ok := filter.allowedQueries[key]; !ok {
|
||||||
continue LOOP
|
continue LOOP
|
||||||
|
@ -158,8 +159,8 @@ LOOP:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for _, query := range filter.queries {
|
for _, query := range filter.queries {
|
||||||
queryVal := r.URL.Query().Get(query.Key)
|
queryVal := queries.Get(query.Key)
|
||||||
if !r.URL.Query().Has(query.Key) || query.Value != "" && query.Value != queryVal {
|
if !queries.Has(query.Key) || query.Value != "" && query.Value != queryVal {
|
||||||
continue LOOP
|
continue LOOP
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -903,6 +903,78 @@ func TestRouterListObjectsV2Domains(t *testing.T) {
|
||||||
require.Equal(t, s3middleware.ListObjectsV2Operation, resp.Method)
|
require.Equal(t, s3middleware.ListObjectsV2Operation, resp.Method)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestRouterListingVHS(t *testing.T) {
|
||||||
|
baseDomain := "domain.com"
|
||||||
|
baseDomainWithBkt := "bucket.domain.com"
|
||||||
|
chiRouter := prepareRouter(t, enableVHSDomains(baseDomain))
|
||||||
|
chiRouter.handler.buckets["bucket"] = &data.BucketInfo{}
|
||||||
|
|
||||||
|
for _, tc := range []struct {
|
||||||
|
name string
|
||||||
|
host string
|
||||||
|
queries string
|
||||||
|
expectedOperation string
|
||||||
|
notSupported bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "list-object-v1 without query params",
|
||||||
|
host: baseDomainWithBkt,
|
||||||
|
expectedOperation: s3middleware.ListObjectsV1Operation,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "list-buckets without query params",
|
||||||
|
host: baseDomain,
|
||||||
|
expectedOperation: s3middleware.ListBucketsOperation,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "list-objects-v1 with prefix param",
|
||||||
|
host: baseDomainWithBkt,
|
||||||
|
queries: func() string {
|
||||||
|
query := make(url.Values)
|
||||||
|
query.Set(s3middleware.QueryPrefix, "prefix")
|
||||||
|
return query.Encode()
|
||||||
|
}(),
|
||||||
|
expectedOperation: s3middleware.ListObjectsV1Operation,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "list-buckets with prefix param",
|
||||||
|
host: baseDomain,
|
||||||
|
queries: func() string {
|
||||||
|
query := make(url.Values)
|
||||||
|
query.Set(s3middleware.QueryPrefix, "prefix")
|
||||||
|
return query.Encode()
|
||||||
|
}(),
|
||||||
|
expectedOperation: s3middleware.ListBucketsOperation,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "not supported operation",
|
||||||
|
host: baseDomain,
|
||||||
|
queries: func() string {
|
||||||
|
query := make(url.Values)
|
||||||
|
query.Set("invalid", "invalid")
|
||||||
|
return query.Encode()
|
||||||
|
}(),
|
||||||
|
notSupported: true,
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
r := httptest.NewRequest(http.MethodGet, "/", nil)
|
||||||
|
r.URL.RawQuery = tc.queries
|
||||||
|
r.Host = tc.host
|
||||||
|
chiRouter.ServeHTTP(w, r)
|
||||||
|
|
||||||
|
if tc.notSupported {
|
||||||
|
assertAPIError(t, w, apierr.ErrNotSupported)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := readResponse(t, w)
|
||||||
|
require.Equal(t, tc.expectedOperation, resp.Method)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func enableVHSDomains(domains ...string) option {
|
func enableVHSDomains(domains ...string) option {
|
||||||
return func(cfg *Config) {
|
return func(cfg *Config) {
|
||||||
setting := cfg.MiddlewareSettings.(*middlewareSettingsMock)
|
setting := cfg.MiddlewareSettings.(*middlewareSettingsMock)
|
||||||
|
|
Loading…
Reference in a new issue