mirror of
https://github.com/minio/minio.git
synced 2025-01-23 12:43:16 -05:00
24d9d7e5fa
This fix removes logrus package dependency and refactors the console logging as the only logging mechanism by removing file logging support. It rearranges the log message format and adds stack trace information whenever trace information is not available in the error structure. It also adds `--json` flag support for server logging. When minio server is started with `--json` flag, all log messages are displayed in json format, with no start-up and informational log messages. Fixes #5265 #5220 #5197
236 lines
7.3 KiB
Go
236 lines
7.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 cmd
|
|
|
|
import (
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
// WARNING:
|
|
//
|
|
// Expected source line number is hard coded, 32, in the
|
|
// following test. Adding new code before this test or changing its
|
|
// position will cause the line number to change and the test to FAIL
|
|
// Tests getSource().
|
|
func TestGetSource(t *testing.T) {
|
|
currentSource := func() string { return getSource() }
|
|
gotSource := currentSource()
|
|
// Hard coded line number, 32, in the "expectedSource" value
|
|
expectedSource := "[namespace-lock_test.go:32:TestGetSource()]"
|
|
if gotSource != expectedSource {
|
|
t.Errorf("expected : %s, got : %s", expectedSource, gotSource)
|
|
}
|
|
}
|
|
|
|
// Tests functionality provided by namespace lock.
|
|
func TestNamespaceLockTest(t *testing.T) {
|
|
// List of test cases.
|
|
testCases := []struct {
|
|
lk func(s1, s2, s3 string, t time.Duration) bool
|
|
unlk func(s1, s2, s3 string)
|
|
rlk func(s1, s2, s3 string, t time.Duration) bool
|
|
runlk func(s1, s2, s3 string)
|
|
lkCount int
|
|
lockedRefCount uint
|
|
unlockedRefCount uint
|
|
shouldPass bool
|
|
}{
|
|
{
|
|
lk: globalNSMutex.Lock,
|
|
unlk: globalNSMutex.Unlock,
|
|
lockedRefCount: 1,
|
|
unlockedRefCount: 0,
|
|
shouldPass: true,
|
|
},
|
|
{
|
|
rlk: globalNSMutex.RLock,
|
|
runlk: globalNSMutex.RUnlock,
|
|
lockedRefCount: 4,
|
|
unlockedRefCount: 2,
|
|
shouldPass: true,
|
|
},
|
|
{
|
|
rlk: globalNSMutex.RLock,
|
|
runlk: globalNSMutex.RUnlock,
|
|
lockedRefCount: 1,
|
|
unlockedRefCount: 0,
|
|
shouldPass: true,
|
|
},
|
|
}
|
|
|
|
// Run all test cases.
|
|
|
|
// Write lock tests.
|
|
testCase := testCases[0]
|
|
if !testCase.lk("a", "b", "c", 60*time.Second) { // lock once.
|
|
t.Fatalf("Failed to acquire lock")
|
|
}
|
|
nsLk, ok := globalNSMutex.lockMap[nsParam{"a", "b"}]
|
|
if !ok && testCase.shouldPass {
|
|
t.Errorf("Lock in map missing.")
|
|
}
|
|
// Validate locked ref count.
|
|
if testCase.lockedRefCount != nsLk.ref && testCase.shouldPass {
|
|
t.Errorf("Test %d fails, expected to pass. Wanted ref count is %d, got %d", 1, testCase.lockedRefCount, nsLk.ref)
|
|
}
|
|
testCase.unlk("a", "b", "c") // unlock once.
|
|
if testCase.unlockedRefCount != nsLk.ref && testCase.shouldPass {
|
|
t.Errorf("Test %d fails, expected to pass. Wanted ref count is %d, got %d", 1, testCase.unlockedRefCount, nsLk.ref)
|
|
}
|
|
_, ok = globalNSMutex.lockMap[nsParam{"a", "b"}]
|
|
if ok && !testCase.shouldPass {
|
|
t.Errorf("Lock map found after unlock.")
|
|
}
|
|
|
|
// Read lock tests.
|
|
testCase = testCases[1]
|
|
if !testCase.rlk("a", "b", "c", 60*time.Second) { // lock once.
|
|
t.Fatalf("Failed to acquire first read lock")
|
|
}
|
|
if !testCase.rlk("a", "b", "c", 60*time.Second) { // lock second time.
|
|
t.Fatalf("Failed to acquire second read lock")
|
|
}
|
|
if !testCase.rlk("a", "b", "c", 60*time.Second) { // lock third time.
|
|
t.Fatalf("Failed to acquire third read lock")
|
|
}
|
|
if !testCase.rlk("a", "b", "c", 60*time.Second) { // lock fourth time.
|
|
t.Fatalf("Failed to acquire fourth read lock")
|
|
}
|
|
nsLk, ok = globalNSMutex.lockMap[nsParam{"a", "b"}]
|
|
if !ok && testCase.shouldPass {
|
|
t.Errorf("Lock in map missing.")
|
|
}
|
|
// Validate locked ref count.
|
|
if testCase.lockedRefCount != nsLk.ref && testCase.shouldPass {
|
|
t.Errorf("Test %d fails, expected to pass. Wanted ref count is %d, got %d", 1, testCase.lockedRefCount, nsLk.ref)
|
|
}
|
|
|
|
testCase.runlk("a", "b", "c") // unlock once.
|
|
testCase.runlk("a", "b", "c") // unlock second time.
|
|
if testCase.unlockedRefCount != nsLk.ref && testCase.shouldPass {
|
|
t.Errorf("Test %d fails, expected to pass. Wanted ref count is %d, got %d", 2, testCase.unlockedRefCount, nsLk.ref)
|
|
}
|
|
_, ok = globalNSMutex.lockMap[nsParam{"a", "b"}]
|
|
if !ok && testCase.shouldPass {
|
|
t.Errorf("Lock map not found.")
|
|
}
|
|
|
|
// Read lock 0 ref count.
|
|
testCase = testCases[2]
|
|
if !testCase.rlk("a", "c", "d", 60*time.Second) { // lock once.
|
|
t.Fatalf("Failed to acquire read lock")
|
|
}
|
|
|
|
nsLk, ok = globalNSMutex.lockMap[nsParam{"a", "c"}]
|
|
if !ok && testCase.shouldPass {
|
|
t.Errorf("Lock in map missing.")
|
|
}
|
|
// Validate locked ref count.
|
|
if testCase.lockedRefCount != nsLk.ref && testCase.shouldPass {
|
|
t.Errorf("Test %d fails, expected to pass. Wanted ref count is %d, got %d", 3, testCase.lockedRefCount, nsLk.ref)
|
|
}
|
|
testCase.runlk("a", "c", "d") // unlock once.
|
|
if testCase.unlockedRefCount != nsLk.ref && testCase.shouldPass {
|
|
t.Errorf("Test %d fails, expected to pass. Wanted ref count is %d, got %d", 3, testCase.unlockedRefCount, nsLk.ref)
|
|
}
|
|
_, ok = globalNSMutex.lockMap[nsParam{"a", "c"}]
|
|
if ok && !testCase.shouldPass {
|
|
t.Errorf("Lock map not found.")
|
|
}
|
|
}
|
|
|
|
func TestNamespaceLockTimedOut(t *testing.T) {
|
|
|
|
// Get write lock
|
|
if !globalNSMutex.Lock("my-bucket", "my-object", "abc", 60*time.Second) {
|
|
t.Fatalf("Failed to acquire lock")
|
|
}
|
|
|
|
// Second attempt for write lock on same resource should time out
|
|
locked := globalNSMutex.Lock("my-bucket", "my-object", "def", 1*time.Second)
|
|
if locked {
|
|
t.Fatalf("Should not have acquired lock")
|
|
}
|
|
|
|
// Read lock on same resource should also time out
|
|
locked = globalNSMutex.RLock("my-bucket", "my-object", "def", 1*time.Second)
|
|
if locked {
|
|
t.Fatalf("Should not have acquired read lock while write lock is active")
|
|
}
|
|
|
|
// Release write lock
|
|
globalNSMutex.Unlock("my-bucket", "my-object", "abc")
|
|
|
|
// Get read lock
|
|
if !globalNSMutex.RLock("my-bucket", "my-object", "ghi", 60*time.Second) {
|
|
t.Fatalf("Failed to acquire read lock")
|
|
}
|
|
|
|
// Write lock on same resource should time out
|
|
locked = globalNSMutex.Lock("my-bucket", "my-object", "klm", 1*time.Second)
|
|
if locked {
|
|
t.Fatalf("Should not have acquired lock")
|
|
}
|
|
|
|
// 2nd read lock should be just fine
|
|
if !globalNSMutex.RLock("my-bucket", "my-object", "nop", 60*time.Second) {
|
|
t.Fatalf("Failed to acquire second read lock")
|
|
}
|
|
|
|
// Release both read locks
|
|
globalNSMutex.RUnlock("my-bucket", "my-object", "ghi")
|
|
globalNSMutex.RUnlock("my-bucket", "my-object", "nop")
|
|
}
|
|
|
|
// Tests functionality to forcefully unlock locks.
|
|
func TestNamespaceForceUnlockTest(t *testing.T) {
|
|
|
|
// Create lock.
|
|
lock := globalNSMutex.NewNSLock("bucket", "object")
|
|
if lock.GetLock(newDynamicTimeout(60*time.Second, time.Second)) != nil {
|
|
t.Fatalf("Failed to get lock")
|
|
}
|
|
// Forcefully unlock lock.
|
|
globalNSMutex.ForceUnlock("bucket", "object")
|
|
|
|
ch := make(chan struct{}, 1)
|
|
|
|
go func() {
|
|
// Try to claim lock again.
|
|
anotherLock := globalNSMutex.NewNSLock("bucket", "object")
|
|
if anotherLock.GetLock(newDynamicTimeout(60*time.Second, time.Second)) != nil {
|
|
t.Fatalf("Failed to get lock")
|
|
}
|
|
// And signal success.
|
|
ch <- struct{}{}
|
|
}()
|
|
|
|
select {
|
|
case <-ch:
|
|
// Signalled so all is fine.
|
|
break
|
|
|
|
case <-time.After(100 * time.Millisecond):
|
|
// In case we hit the time out, the lock has not been cleared.
|
|
t.Errorf("Lock not cleared.")
|
|
}
|
|
|
|
// Clean up lock.
|
|
globalNSMutex.ForceUnlock("bucket", "object")
|
|
}
|