mirror of
https://github.com/minio/minio.git
synced 2025-01-12 07:23:23 -05:00
40852801ea
guessIsRPCReq() considers all POST requests as RPC but doesn't check if this is an object operation API or not, which is actually confusing bucket forwarder handler when it receives a new multipart upload API which is a POST http request. Due to this bug, users having a federated setup are not able to upload a multipart object using an endpoint which doesn't actually contain the specified bucket that will store the object. Hence this commit will fix the described issue.
231 lines
6.4 KiB
Go
231 lines
6.4 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 cmd
|
|
|
|
import (
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"net/url"
|
|
"strconv"
|
|
"testing"
|
|
|
|
"github.com/minio/minio/cmd/crypto"
|
|
)
|
|
|
|
// Tests getRedirectLocation function for all its criteria.
|
|
func TestRedirectLocation(t *testing.T) {
|
|
testCases := []struct {
|
|
urlPath string
|
|
location string
|
|
}{
|
|
{
|
|
// 1. When urlPath is '/minio'
|
|
urlPath: minioReservedBucketPath,
|
|
location: minioReservedBucketPath + "/",
|
|
},
|
|
{
|
|
// 2. When urlPath is '/'
|
|
urlPath: "/",
|
|
location: minioReservedBucketPath + "/",
|
|
},
|
|
{
|
|
// 3. When urlPath is '/webrpc'
|
|
urlPath: "/webrpc",
|
|
location: minioReservedBucketPath + "/webrpc",
|
|
},
|
|
{
|
|
// 4. When urlPath is '/login'
|
|
urlPath: "/login",
|
|
location: minioReservedBucketPath + "/login",
|
|
},
|
|
{
|
|
// 5. When urlPath is '/favicon.ico'
|
|
urlPath: "/favicon.ico",
|
|
location: minioReservedBucketPath + "/favicon.ico",
|
|
},
|
|
{
|
|
// 6. When urlPath is '/unknown'
|
|
urlPath: "/unknown",
|
|
location: "",
|
|
},
|
|
}
|
|
|
|
// Validate all conditions.
|
|
for i, testCase := range testCases {
|
|
loc := getRedirectLocation(testCase.urlPath)
|
|
if testCase.location != loc {
|
|
t.Errorf("Test %d: Unexpected location expected %s, got %s", i+1, testCase.location, loc)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Tests request guess function for net/rpc requests.
|
|
func TestGuessIsRPC(t *testing.T) {
|
|
if guessIsRPCReq(nil) {
|
|
t.Fatal("Unexpected return for nil request")
|
|
}
|
|
|
|
u, err := url.Parse("http://localhost:9000/minio/lock")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
r := &http.Request{
|
|
Proto: "HTTP/1.0",
|
|
Method: http.MethodPost,
|
|
URL: u,
|
|
}
|
|
if !guessIsRPCReq(r) {
|
|
t.Fatal("Test shouldn't fail for a possible net/rpc request.")
|
|
}
|
|
r = &http.Request{
|
|
Proto: "HTTP/1.1",
|
|
Method: http.MethodGet,
|
|
}
|
|
if guessIsRPCReq(r) {
|
|
t.Fatal("Test shouldn't report as net/rpc for a non net/rpc request.")
|
|
}
|
|
}
|
|
|
|
// Tests browser request guess function.
|
|
func TestGuessIsBrowser(t *testing.T) {
|
|
if guessIsBrowserReq(nil) {
|
|
t.Fatal("Unexpected return for nil request")
|
|
}
|
|
r := &http.Request{
|
|
Header: http.Header{},
|
|
}
|
|
r.Header.Set("User-Agent", "Mozilla")
|
|
if !guessIsBrowserReq(r) {
|
|
t.Fatal("Test shouldn't fail for a possible browser request.")
|
|
}
|
|
r = &http.Request{
|
|
Header: http.Header{},
|
|
}
|
|
r.Header.Set("User-Agent", "mc")
|
|
if guessIsBrowserReq(r) {
|
|
t.Fatal("Test shouldn't report as browser for a non browser request.")
|
|
}
|
|
}
|
|
|
|
var isHTTPHeaderSizeTooLargeTests = []struct {
|
|
header http.Header
|
|
shouldFail bool
|
|
}{
|
|
{header: generateHeader(0, 0), shouldFail: false},
|
|
{header: generateHeader(1024, 0), shouldFail: false},
|
|
{header: generateHeader(2048, 0), shouldFail: false},
|
|
{header: generateHeader(8*1024+1, 0), shouldFail: true},
|
|
{header: generateHeader(0, 1024), shouldFail: false},
|
|
{header: generateHeader(0, 2048), shouldFail: true},
|
|
{header: generateHeader(0, 2048+1), shouldFail: true},
|
|
}
|
|
|
|
func generateHeader(size, usersize int) http.Header {
|
|
header := http.Header{}
|
|
for i := 0; i < size; i++ {
|
|
header.Add(strconv.Itoa(i), "")
|
|
}
|
|
userlength := 0
|
|
for i := 0; userlength < usersize; i++ {
|
|
userlength += len(userMetadataKeyPrefixes[0] + strconv.Itoa(i))
|
|
header.Add(userMetadataKeyPrefixes[0]+strconv.Itoa(i), "")
|
|
}
|
|
return header
|
|
}
|
|
|
|
func TestIsHTTPHeaderSizeTooLarge(t *testing.T) {
|
|
for i, test := range isHTTPHeaderSizeTooLargeTests {
|
|
if res := isHTTPHeaderSizeTooLarge(test.header); res != test.shouldFail {
|
|
t.Errorf("Test %d: Expected %v got %v", i, res, test.shouldFail)
|
|
}
|
|
}
|
|
}
|
|
|
|
var containsReservedMetadataTests = []struct {
|
|
header http.Header
|
|
shouldFail bool
|
|
}{
|
|
{
|
|
header: http.Header{"X-Minio-Key": []string{"value"}},
|
|
},
|
|
{
|
|
header: http.Header{crypto.SSEIV: []string{"iv"}},
|
|
shouldFail: true,
|
|
},
|
|
{
|
|
header: http.Header{crypto.SSESealAlgorithm: []string{SSESealAlgorithmDareSha256}},
|
|
shouldFail: true,
|
|
},
|
|
{
|
|
header: http.Header{crypto.SSECSealedKey: []string{"mac"}},
|
|
shouldFail: true,
|
|
},
|
|
{
|
|
header: http.Header{ReservedMetadataPrefix + "Key": []string{"value"}},
|
|
shouldFail: true,
|
|
},
|
|
}
|
|
|
|
func TestContainsReservedMetadata(t *testing.T) {
|
|
for i, test := range containsReservedMetadataTests {
|
|
if contains := containsReservedMetadata(test.header); contains && !test.shouldFail {
|
|
t.Errorf("Test %d: contains reserved header but should not fail", i)
|
|
} else if !contains && test.shouldFail {
|
|
t.Errorf("Test %d: does not contain reserved header but failed", i)
|
|
}
|
|
}
|
|
}
|
|
|
|
var sseTLSHandlerTests = []struct {
|
|
URL *url.URL
|
|
Header http.Header
|
|
IsTLS, ShouldFail bool
|
|
}{
|
|
{URL: &url.URL{}, Header: http.Header{}, IsTLS: false, ShouldFail: false}, // 0
|
|
{URL: &url.URL{}, Header: http.Header{crypto.SSECAlgorithm: []string{"AES256"}}, IsTLS: false, ShouldFail: true}, // 1
|
|
{URL: &url.URL{}, Header: http.Header{crypto.SSECAlgorithm: []string{"AES256"}}, IsTLS: true, ShouldFail: false}, // 2
|
|
{URL: &url.URL{}, Header: http.Header{crypto.SSECKey: []string{""}}, IsTLS: true, ShouldFail: false}, // 3
|
|
{URL: &url.URL{}, Header: http.Header{crypto.SSECopyAlgorithm: []string{""}}, IsTLS: false, ShouldFail: true}, // 4
|
|
}
|
|
|
|
func TestSSETLSHandler(t *testing.T) {
|
|
defer func(isSSL bool) { globalIsSSL = isSSL }(globalIsSSL) // reset globalIsSSL after test
|
|
|
|
var okHandler http.HandlerFunc = func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusOK)
|
|
}
|
|
for i, test := range sseTLSHandlerTests {
|
|
globalIsSSL = test.IsTLS
|
|
|
|
w := httptest.NewRecorder()
|
|
r := new(http.Request)
|
|
r.Header = test.Header
|
|
r.URL = test.URL
|
|
|
|
h := setSSETLSHandler(okHandler)
|
|
h.ServeHTTP(w, r)
|
|
|
|
switch {
|
|
case test.ShouldFail && w.Code == http.StatusOK:
|
|
t.Errorf("Test %d: should fail but status code is HTTP %d", i, w.Code)
|
|
case !test.ShouldFail && w.Code != http.StatusOK:
|
|
t.Errorf("Test %d: should not fail but status code is HTTP %d and not 200 OK", i, w.Code)
|
|
}
|
|
}
|
|
}
|