2021-04-19 03:41:13 +08:00
|
|
|
// Copyright (c) 2015-2021 MinIO, Inc.
|
|
|
|
//
|
|
|
|
// This file is part of MinIO Object Storage stack
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2021-03-10 06:43:16 +08:00
|
|
|
|
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
2025-09-29 04:59:21 +08:00
|
|
|
"sync"
|
2021-03-10 06:43:16 +08:00
|
|
|
"testing"
|
2025-09-29 04:59:21 +08:00
|
|
|
"time"
|
|
|
|
|
|
|
|
xhttp "github.com/minio/minio/internal/http"
|
2021-03-10 06:43:16 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
// Test redactLDAPPwd()
|
|
|
|
func TestRedactLDAPPwd(t *testing.T) {
|
|
|
|
testCases := []struct {
|
|
|
|
query string
|
|
|
|
expectedQuery string
|
|
|
|
}{
|
|
|
|
{"", ""},
|
2022-01-03 01:15:06 +08:00
|
|
|
{
|
|
|
|
"?Action=AssumeRoleWithLDAPIdentity&LDAPUsername=myusername&LDAPPassword=can+youreadthis%3F&Version=2011-06-15",
|
2021-03-10 06:43:16 +08:00
|
|
|
"?Action=AssumeRoleWithLDAPIdentity&LDAPUsername=myusername&LDAPPassword=*REDACTED*&Version=2011-06-15",
|
|
|
|
},
|
2022-01-03 01:15:06 +08:00
|
|
|
{
|
|
|
|
"LDAPPassword=can+youreadthis%3F&Version=2011-06-15&?Action=AssumeRoleWithLDAPIdentity&LDAPUsername=myusername",
|
2021-03-10 06:43:16 +08:00
|
|
|
"LDAPPassword=*REDACTED*&Version=2011-06-15&?Action=AssumeRoleWithLDAPIdentity&LDAPUsername=myusername",
|
|
|
|
},
|
2022-01-03 01:15:06 +08:00
|
|
|
{
|
|
|
|
"?Action=AssumeRoleWithLDAPIdentity&LDAPUsername=myusername&Version=2011-06-15&LDAPPassword=can+youreadthis%3F",
|
2021-03-10 06:43:16 +08:00
|
|
|
"?Action=AssumeRoleWithLDAPIdentity&LDAPUsername=myusername&Version=2011-06-15&LDAPPassword=*REDACTED*",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"?x=y&a=b",
|
|
|
|
"?x=y&a=b",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for i, test := range testCases {
|
|
|
|
gotQuery := redactLDAPPwd(test.query)
|
|
|
|
if gotQuery != test.expectedQuery {
|
|
|
|
t.Fatalf("test %d: expected %s got %s", i+1, test.expectedQuery, gotQuery)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2025-09-29 04:59:21 +08:00
|
|
|
|
|
|
|
// TestHTTPStatsRaceCondition tests the race condition fix for HTTPStats.
|
|
|
|
// This test specifically addresses the race between:
|
|
|
|
// - Write operations via updateStats.
|
|
|
|
// - Read operations via toServerHTTPStats(false).
|
|
|
|
func TestRaulStatsRaceCondition(t *testing.T) {
|
|
|
|
httpStats := newHTTPStats()
|
|
|
|
// Simulate the concurrent scenario from the original race condition:
|
|
|
|
// Multiple HTTP request handlers updating stats concurrently,
|
|
|
|
// while background processes are reading the stats for persistence.
|
|
|
|
const numWriters = 100 // Simulate many HTTP request handlers.
|
|
|
|
const numReaders = 50 // Simulate background stats readers.
|
|
|
|
const opsPerGoroutine = 100
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
for i := range numWriters {
|
|
|
|
wg.Add(1)
|
|
|
|
go func(writerID int) {
|
|
|
|
defer wg.Done()
|
|
|
|
for j := 0; j < opsPerGoroutine; j++ {
|
|
|
|
switch j % 4 {
|
|
|
|
case 0:
|
|
|
|
httpStats.updateStats("GetObject", &xhttp.ResponseRecorder{})
|
|
|
|
case 1:
|
|
|
|
httpStats.totalS3Requests.Inc("PutObject")
|
|
|
|
case 2:
|
|
|
|
httpStats.totalS3Errors.Inc("DeleteObject")
|
|
|
|
case 3:
|
|
|
|
httpStats.currentS3Requests.Inc("ListObjects")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range numReaders {
|
|
|
|
wg.Add(1)
|
|
|
|
go func(readerID int) {
|
|
|
|
defer wg.Done()
|
|
|
|
for range opsPerGoroutine {
|
|
|
|
_ = httpStats.toServerHTTPStats(false)
|
|
|
|
_ = httpStats.totalS3Requests.Load(false)
|
|
|
|
_ = httpStats.currentS3Requests.Load(false)
|
|
|
|
time.Sleep(1 * time.Microsecond)
|
|
|
|
}
|
|
|
|
}(i)
|
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
finalStats := httpStats.toServerHTTPStats(false)
|
|
|
|
totalRequests := 0
|
|
|
|
for _, v := range finalStats.TotalS3Requests.APIStats {
|
|
|
|
totalRequests += v
|
|
|
|
}
|
|
|
|
if totalRequests == 0 {
|
|
|
|
t.Error("Expected some total requests to be recorded, but got zero")
|
|
|
|
}
|
|
|
|
t.Logf("Total requests recorded: %d", totalRequests)
|
|
|
|
t.Logf("Race condition test passed - no races detected")
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestHTTPAPIStatsRaceCondition tests concurrent access to HTTPAPIStats specifically.
|
|
|
|
func TestRaulHTTPAPIStatsRaceCondition(t *testing.T) {
|
|
|
|
stats := &HTTPAPIStats{}
|
|
|
|
const numGoroutines = 50
|
|
|
|
const opsPerGoroutine = 1000
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
for i := range numGoroutines {
|
|
|
|
wg.Add(1)
|
|
|
|
go func(id int) {
|
|
|
|
defer wg.Done()
|
|
|
|
for j := 0; j < opsPerGoroutine; j++ {
|
|
|
|
stats.Inc("TestAPI")
|
|
|
|
}
|
|
|
|
}(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range numGoroutines / 2 {
|
|
|
|
wg.Add(1)
|
|
|
|
go func(id int) {
|
|
|
|
defer wg.Done()
|
|
|
|
for range opsPerGoroutine / 2 {
|
|
|
|
_ = stats.Load(false)
|
|
|
|
}
|
|
|
|
}(i)
|
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
finalStats := stats.Load(false)
|
|
|
|
expected := numGoroutines * opsPerGoroutine
|
|
|
|
actual := finalStats["TestAPI"]
|
|
|
|
if actual != expected {
|
|
|
|
t.Errorf("Race condition detected: expected %d, got %d (lost %d increments)",
|
|
|
|
expected, actual, expected-actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestBucketHTTPStatsRaceCondition tests concurrent access to bucket-level HTTP stats.
|
|
|
|
func TestRaulBucketHTTPStatsRaceCondition(t *testing.T) {
|
|
|
|
bucketStats := newBucketHTTPStats()
|
|
|
|
const numGoroutines = 50
|
|
|
|
const opsPerGoroutine = 100
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
for i := range numGoroutines {
|
|
|
|
wg.Add(1)
|
|
|
|
go func(id int) {
|
|
|
|
defer wg.Done()
|
|
|
|
bucketName := "test-bucket"
|
|
|
|
|
|
|
|
for range opsPerGoroutine {
|
|
|
|
bucketStats.updateHTTPStats(bucketName, "GetObject", nil)
|
|
|
|
recorder := &xhttp.ResponseRecorder{}
|
|
|
|
bucketStats.updateHTTPStats(bucketName, "GetObject", recorder)
|
|
|
|
_ = bucketStats.load(bucketName)
|
|
|
|
}
|
|
|
|
}(i)
|
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
stats := bucketStats.load("test-bucket")
|
|
|
|
if stats.totalS3Requests == nil {
|
|
|
|
t.Error("Expected bucket stats to be initialized")
|
|
|
|
}
|
|
|
|
t.Logf("Bucket HTTP stats race test passed")
|
|
|
|
}
|