2016-09-01 00:09:08 +05:30
/ *
* Minio Cloud Storage , ( C ) 2015 , 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 (
"path"
"strconv"
2016-09-21 19:58:50 -07:00
"strings"
2016-09-01 00:09:08 +05:30
"sync"
2016-09-21 19:58:50 -07:00
"testing"
2016-09-01 00:09:08 +05:30
"time"
)
// API suite container common to both FS and XL.
2016-10-11 00:50:27 -07:00
type TestRPCControlSuite struct {
2016-09-21 19:58:50 -07:00
serverType string
testServer TestServer
testAuthConf * authConfig
2016-09-01 00:09:08 +05:30
}
// Setting up the test suite.
// Starting the Test server with temporary FS backend.
2016-10-11 00:50:27 -07:00
func ( s * TestRPCControlSuite ) SetUpSuite ( c * testing . T ) {
2016-10-06 02:30:54 -07:00
s . testServer = StartTestControlRPCServer ( c , s . serverType )
2016-09-21 19:58:50 -07:00
s . testAuthConf = & authConfig {
address : s . testServer . Server . Listener . Addr ( ) . String ( ) ,
accessKey : s . testServer . AccessKey ,
secretKey : s . testServer . SecretKey ,
path : path . Join ( reservedBucket , controlPath ) ,
2016-10-11 00:50:27 -07:00
loginMethod : "Control.LoginHandler" ,
2016-09-21 19:58:50 -07:00
}
2016-09-01 00:09:08 +05:30
}
2016-09-21 19:58:50 -07:00
// No longer used with gocheck, but used in explicit teardown code in
// each test function. // Called implicitly by "gopkg.in/check.v1"
// after all tests are run.
2016-10-11 00:50:27 -07:00
func ( s * TestRPCControlSuite ) TearDownSuite ( c * testing . T ) {
2016-09-01 00:09:08 +05:30
s . testServer . Stop ( )
}
2016-09-21 19:58:50 -07:00
func TestRPCControlLock ( t * testing . T ) {
//setup code
2016-10-11 00:50:27 -07:00
s := & TestRPCControlSuite { serverType : "XL" }
2016-09-21 19:58:50 -07:00
s . SetUpSuite ( t )
//run test
s . testRPCControlLock ( t )
//teardown code
s . TearDownSuite ( t )
}
2016-09-01 00:09:08 +05:30
// Tests to validate the correctness of lock instrumentation control RPC end point.
2016-10-11 00:50:27 -07:00
func ( s * TestRPCControlSuite ) testRPCControlLock ( c * testing . T ) {
2016-09-01 00:09:08 +05:30
expectedResult := [ ] lockStateCase {
// Test case - 1.
// Case where 10 read locks are held.
// Entry for any of the 10 reads locks has to be found.
// Since they held in a loop, Lock origin for first 10 read locks (opsID 0-9) should be the same.
{
volume : "my-bucket" ,
path : "my-object" ,
opsID : "0" ,
readLock : true ,
lockOrigin : "[lock held] in github.com/minio/minio/cmd.TestLockStats[/Users/hackintoshrao/mycode/go/src/github.com/minio/minio/cmd/namespace-lock_test.go:298]" ,
// expected metrics.
expectedErr : nil ,
expectedLockStatus : "Running" ,
expectedGlobalLockCount : 10 ,
expectedRunningLockCount : 10 ,
expectedBlockedLockCount : 0 ,
expectedVolPathLockCount : 10 ,
expectedVolPathRunningCount : 10 ,
expectedVolPathBlockCount : 0 ,
} ,
// Test case 2.
2016-09-01 02:54:37 -07:00
// Testing the existence of entry for the last read lock (read lock with opsID "9").
2016-09-01 00:09:08 +05:30
{
volume : "my-bucket" ,
path : "my-object" ,
opsID : "9" ,
readLock : true ,
lockOrigin : "[lock held] in github.com/minio/minio/cmd.TestLockStats[/Users/hackintoshrao/mycode/go/src/github.com/minio/minio/cmd/namespace-lock_test.go:298]" ,
// expected metrics.
expectedErr : nil ,
expectedLockStatus : "Running" ,
expectedGlobalLockCount : 10 ,
expectedRunningLockCount : 10 ,
expectedBlockedLockCount : 0 ,
expectedVolPathLockCount : 10 ,
expectedVolPathRunningCount : 10 ,
expectedVolPathBlockCount : 0 ,
} ,
// Test case 3.
// Hold a write lock, and it should block since 10 read locks
// on <"my-bucket", "my-object"> are still held.
{
volume : "my-bucket" ,
path : "my-object" ,
opsID : "10" ,
readLock : false ,
lockOrigin : "[lock held] in github.com/minio/minio/cmd.TestLockStats[/Users/hackintoshrao/mycode/go/src/github.com/minio/minio/cmd/namespace-lock_test.go:298]" ,
// expected metrics.
expectedErr : nil ,
expectedLockStatus : "Blocked" ,
expectedGlobalLockCount : 11 ,
expectedRunningLockCount : 10 ,
expectedBlockedLockCount : 1 ,
expectedVolPathLockCount : 11 ,
expectedVolPathRunningCount : 10 ,
expectedVolPathBlockCount : 1 ,
} ,
// Test case 4.
// Expected result when all the read locks are released and the blocked write lock acquires the lock.
{
volume : "my-bucket" ,
path : "my-object" ,
opsID : "10" ,
readLock : false ,
lockOrigin : "[lock held] in github.com/minio/minio/cmd.TestLockStats[/Users/hackintoshrao/mycode/go/src/github.com/minio/minio/cmd/namespace-lock_test.go:298]" ,
// expected metrics.
expectedErr : nil ,
expectedLockStatus : "Running" ,
expectedGlobalLockCount : 1 ,
expectedRunningLockCount : 1 ,
expectedBlockedLockCount : 0 ,
expectedVolPathLockCount : 1 ,
expectedVolPathRunningCount : 1 ,
expectedVolPathBlockCount : 0 ,
} ,
// Test case - 5.
// At the end after locks are released, its verified whether the counters are set to 0.
{
volume : "my-bucket" ,
path : "my-object" ,
// expected metrics.
expectedErr : nil ,
expectedLockStatus : "Blocked" ,
expectedGlobalLockCount : 0 ,
expectedRunningLockCount : 0 ,
expectedBlockedLockCount : 0 ,
} ,
}
// used to make sure that the tests don't end till locks held in other go routines are released.
var wg sync . WaitGroup
// Hold 5 read locks. We should find the info about these in the RPC response.
// hold 10 read locks.
// Then call the RPC control end point for obtaining lock instrumentation info.
for i := 0 ; i < 10 ; i ++ {
nsMutex . RLock ( "my-bucket" , "my-object" , strconv . Itoa ( i ) )
}
2016-09-21 19:58:50 -07:00
client := newAuthClient ( s . testAuthConf )
2016-09-01 00:09:08 +05:30
defer client . Close ( )
args := & GenericArgs { }
2016-10-11 00:50:27 -07:00
reply := make ( map [ string ] * SystemLockState )
2016-09-01 00:09:08 +05:30
// Call the lock instrumentation RPC end point.
2016-10-11 00:50:27 -07:00
err := client . Call ( "Control.LockInfo" , args , & reply )
2016-09-01 00:09:08 +05:30
if err != nil {
c . Errorf ( "Add: expected no error but got string %q" , err . Error ( ) )
}
// expected lock info.
expectedLockStats := expectedResult [ 0 ]
// verify the actual lock info with the expected one.
2016-09-01 02:54:37 -07:00
// verify the existence entry for first read lock (read lock with opsID "0").
2016-10-11 00:50:27 -07:00
verifyRPCLockInfoResponse ( expectedLockStats , reply , c , 1 )
2016-09-01 00:09:08 +05:30
expectedLockStats = expectedResult [ 1 ]
// verify the actual lock info with the expected one.
2016-09-01 02:54:37 -07:00
// verify the existence entry for last read lock (read lock with opsID "9").
2016-10-11 00:50:27 -07:00
verifyRPCLockInfoResponse ( expectedLockStats , reply , c , 2 )
2016-09-01 00:09:08 +05:30
// now hold a write lock in a different go routine and it should block since 10 read locks are
// still held.
wg . Add ( 1 )
go func ( ) {
defer wg . Done ( )
// blocks till all read locks are released.
nsMutex . Lock ( "my-bucket" , "my-object" , strconv . Itoa ( 10 ) )
// Once the above attempt to lock is unblocked/acquired, we verify the stats and release the lock.
expectedWLockStats := expectedResult [ 3 ]
// Since the write lock acquired here, the number of blocked locks should reduce by 1 and
// count of running locks should increase by 1.
// Call the RPC control handle to fetch the lock instrumentation info.
2016-10-11 00:50:27 -07:00
reply = make ( map [ string ] * SystemLockState )
2016-09-01 00:09:08 +05:30
// Call the lock instrumentation RPC end point.
2016-10-11 00:50:27 -07:00
err = client . Call ( "Control.LockInfo" , args , & reply )
2016-09-01 00:09:08 +05:30
if err != nil {
c . Errorf ( "Add: expected no error but got string %q" , err . Error ( ) )
}
2016-10-11 00:50:27 -07:00
verifyRPCLockInfoResponse ( expectedWLockStats , reply , c , 4 )
2016-09-01 00:09:08 +05:30
// release the write lock.
nsMutex . Unlock ( "my-bucket" , "my-object" , strconv . Itoa ( 10 ) )
} ( )
// waiting for a second so that the attempt to acquire the write lock in
// the above go routines gets blocked.
time . Sleep ( 1 * time . Second )
// The write lock should have got blocked by now,
// check whether the entry for one blocked lock exists.
expectedLockStats = expectedResult [ 2 ]
// Call the RPC control handle to fetch the lock instrumentation info.
2016-10-11 00:50:27 -07:00
reply = make ( map [ string ] * SystemLockState )
2016-09-01 00:09:08 +05:30
// Call the lock instrumentation RPC end point.
2016-10-11 00:50:27 -07:00
err = client . Call ( "Control.LockInfo" , args , & reply )
2016-09-01 00:09:08 +05:30
if err != nil {
c . Errorf ( "Add: expected no error but got string %q" , err . Error ( ) )
}
2016-10-11 00:50:27 -07:00
verifyRPCLockInfoResponse ( expectedLockStats , reply , c , 3 )
2016-09-01 00:09:08 +05:30
// Release all the read locks held.
// the blocked write lock in the above go routines should get unblocked.
for i := 0 ; i < 10 ; i ++ {
nsMutex . RUnlock ( "my-bucket" , "my-object" , strconv . Itoa ( i ) )
}
wg . Wait ( )
2016-09-01 02:54:37 -07:00
// Since all the locks are released. There should not be any entry in the lock info.
2016-09-01 00:09:08 +05:30
// and all the counters should be set to 0.
2016-10-11 00:50:27 -07:00
reply = make ( map [ string ] * SystemLockState )
2016-09-01 00:09:08 +05:30
// Call the lock instrumentation RPC end point.
2016-10-11 00:50:27 -07:00
err = client . Call ( "Control.LockInfo" , args , & reply )
2016-09-01 00:09:08 +05:30
if err != nil {
c . Errorf ( "Add: expected no error but got string %q" , err . Error ( ) )
}
2016-10-11 00:50:27 -07:00
for _ , rpcLockInfo := range reply {
if rpcLockInfo . TotalAcquiredLocks != 0 && rpcLockInfo . TotalLocks != 0 && rpcLockInfo . TotalBlockedLocks != 0 {
c . Fatalf ( "The counters are not reset properly after all locks are released" )
}
if len ( rpcLockInfo . LocksInfoPerObject ) != 0 {
c . Fatalf ( "Since all locks are released there shouldn't have been any lock info entry, but found %d" , len ( rpcLockInfo . LocksInfoPerObject ) )
}
2016-09-01 00:09:08 +05:30
}
}
2016-10-11 00:50:27 -07:00
func TestControlHealDiskMetadataH ( t * testing . T ) {
2016-10-14 19:57:40 -07:00
// Setup code
2016-10-11 00:50:27 -07:00
s := & TestRPCControlSuite { serverType : "XL" }
2016-09-21 19:58:50 -07:00
s . SetUpSuite ( t )
2016-10-14 19:57:40 -07:00
// Run test
s . testControlHealFormatH ( t )
2016-09-21 19:58:50 -07:00
2016-10-14 19:57:40 -07:00
// Teardown code
2016-09-21 19:58:50 -07:00
s . TearDownSuite ( t )
}
2016-10-14 19:57:40 -07:00
// TestControlHandlerHealFormat - Registers and call the `HealFormatHandler`, asserts to validate the success.
func ( s * TestRPCControlSuite ) testControlHealFormatH ( c * testing . T ) {
2016-09-01 00:09:08 +05:30
// The suite has already started the test RPC server, just send RPC calls.
2016-09-21 19:58:50 -07:00
client := newAuthClient ( s . testAuthConf )
2016-09-01 00:09:08 +05:30
defer client . Close ( )
args := & GenericArgs { }
reply := & GenericReply { }
2016-10-14 19:57:40 -07:00
err := client . Call ( "Control.HealFormatHandler" , args , reply )
2016-09-01 00:09:08 +05:30
if err != nil {
2016-10-14 19:57:40 -07:00
c . Errorf ( "Test failed with <ERROR> %s" , err )
2016-09-21 19:58:50 -07:00
}
}
2016-10-11 00:50:27 -07:00
func TestControlHealObjectH ( t * testing . T ) {
2016-10-14 19:57:40 -07:00
// Setup code
2016-10-11 00:50:27 -07:00
s := & TestRPCControlSuite { serverType : "XL" }
2016-09-21 19:58:50 -07:00
s . SetUpSuite ( t )
2016-10-14 19:57:40 -07:00
// Run test
s . testControlHealObjectsH ( t )
2016-09-21 19:58:50 -07:00
2016-10-14 19:57:40 -07:00
// Teardown code
2016-09-21 19:58:50 -07:00
s . TearDownSuite ( t )
}
2016-10-14 19:57:40 -07:00
func ( s * TestRPCControlSuite ) testControlHealObjectsH ( t * testing . T ) {
2016-09-21 19:58:50 -07:00
client := newAuthClient ( s . testAuthConf )
defer client . Close ( )
2016-10-14 19:57:40 -07:00
objAPI := newObjectLayerFn ( )
err := objAPI . MakeBucket ( "testbucket" )
2016-09-21 19:58:50 -07:00
if err != nil {
2016-10-14 19:57:40 -07:00
t . Fatalf ( "Create bucket failed with <ERROR> %s" , err )
2016-09-21 19:58:50 -07:00
}
datum := strings . NewReader ( "a" )
2016-10-14 19:57:40 -07:00
_ , err = objAPI . PutObject ( "testbucket" , "testobject1" , 1 , datum , nil , "" )
2016-09-21 19:58:50 -07:00
if err != nil {
2016-10-14 19:57:40 -07:00
t . Fatalf ( "Put object failed with <ERROR> %s" , err )
}
datum = strings . NewReader ( "a" )
_ , err = objAPI . PutObject ( "testbucket" , "testobject2" , 1 , datum , nil , "" )
if err != nil {
t . Fatalf ( "Put object failed with <ERROR> %s" , err )
2016-09-21 19:58:50 -07:00
}
2016-10-14 19:57:40 -07:00
args := & HealObjectArgs {
Bucket : "testbucket" ,
Objects : [ ] ObjectInfo {
{
Name : "testobject1" ,
} , {
Name : "testobject2" ,
} ,
} ,
}
reply := & HealObjectReply { }
err = client . Call ( "Control.HealObjectsHandler" , args , reply )
2016-09-21 19:58:50 -07:00
if err != nil {
2016-10-14 19:57:40 -07:00
t . Errorf ( "Test failed with <ERROR> %s" , err )
2016-09-21 19:58:50 -07:00
}
}
2016-10-11 00:50:27 -07:00
func TestControlListObjectsHealH ( t * testing . T ) {
2016-10-14 19:57:40 -07:00
// Setup code
2016-10-11 00:50:27 -07:00
s := & TestRPCControlSuite { serverType : "XL" }
2016-09-21 19:58:50 -07:00
s . SetUpSuite ( t )
2016-10-14 19:57:40 -07:00
// Run test
2016-10-11 00:50:27 -07:00
s . testControlListObjectsHealH ( t )
2016-09-21 19:58:50 -07:00
2016-10-14 19:57:40 -07:00
// Teardown code
2016-09-21 19:58:50 -07:00
s . TearDownSuite ( t )
}
2016-10-11 00:50:27 -07:00
func ( s * TestRPCControlSuite ) testControlListObjectsHealH ( t * testing . T ) {
2016-09-21 19:58:50 -07:00
client := newAuthClient ( s . testAuthConf )
defer client . Close ( )
2016-10-14 19:57:40 -07:00
objAPI := newObjectLayerFn ( )
// Create a bucket
err := objAPI . MakeBucket ( "testbucket" )
2016-09-21 19:58:50 -07:00
if err != nil {
2016-10-14 19:57:40 -07:00
t . Fatalf ( "Create bucket failed - %s" , err )
2016-09-21 19:58:50 -07:00
}
r := strings . NewReader ( "0" )
2016-10-14 19:57:40 -07:00
_ , err = objAPI . PutObject ( "testbucket" , "testObj-0" , 1 , r , nil , "" )
2016-09-21 19:58:50 -07:00
if err != nil {
2016-10-14 19:57:40 -07:00
t . Fatalf ( "Object creation failed - %s" , err )
2016-09-21 19:58:50 -07:00
}
args := & HealListArgs {
GenericArgs { } , "testbucket" , "testObj-" ,
"" , "" , 100 ,
}
reply := & GenericReply { }
2016-10-11 00:50:27 -07:00
err = client . Call ( "Control.ListObjectsHealHandler" , args , reply )
2016-09-21 19:58:50 -07:00
if err != nil {
2016-10-14 19:57:40 -07:00
t . Errorf ( "Test failed - %s" , err )
2016-09-01 00:09:08 +05:30
}
}