2022-01-14 10:32:35 -08:00
|
|
|
// Copyright (c) 2015-2022 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/>.
|
|
|
|
|
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
func prepareErasurePools() (ObjectLayer, []string, error) {
|
|
|
|
nDisks := 32
|
|
|
|
fsDirs, err := getRandomDisks(nDisks)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
pools := mustGetPoolEndpoints(fsDirs[:16]...)
|
|
|
|
pools = append(pools, mustGetPoolEndpoints(fsDirs[16:]...)...)
|
|
|
|
|
|
|
|
// Everything is fine, should return nil
|
|
|
|
objLayer, err := newErasureServerPools(context.Background(), pools)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
return objLayer, fsDirs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPoolMetaValidate(t *testing.T) {
|
|
|
|
objLayer1, fsDirs, err := prepareErasurePools()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer removeRoots(fsDirs)
|
|
|
|
|
|
|
|
meta := objLayer1.(*erasureServerPools).poolMeta
|
|
|
|
pools := objLayer1.(*erasureServerPools).serverPools
|
|
|
|
|
|
|
|
objLayer2, fsDirs, err := prepareErasurePools()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Initialization of object layer failed for Erasure setup: %s", err)
|
|
|
|
}
|
|
|
|
defer removeRoots(fsDirs)
|
|
|
|
|
|
|
|
newPools := objLayer2.(*erasureServerPools).serverPools
|
|
|
|
reducedPools := pools[1:]
|
|
|
|
orderChangePools := []*erasureSets{
|
|
|
|
pools[1],
|
|
|
|
pools[0],
|
|
|
|
}
|
|
|
|
|
|
|
|
var nmeta1 poolMeta
|
|
|
|
nmeta1.Version = poolMetaVersion
|
|
|
|
nmeta1.Pools = append(nmeta1.Pools, meta.Pools...)
|
|
|
|
for i, pool := range nmeta1.Pools {
|
|
|
|
if i == 0 {
|
|
|
|
nmeta1.Pools[i] = PoolStatus{
|
|
|
|
CmdLine: pool.CmdLine,
|
|
|
|
ID: i,
|
|
|
|
LastUpdate: UTCNow(),
|
|
|
|
Decommission: &PoolDecommissionInfo{
|
|
|
|
Complete: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var nmeta2 poolMeta
|
|
|
|
nmeta2.Version = poolMetaVersion
|
|
|
|
nmeta2.Pools = append(nmeta2.Pools, meta.Pools...)
|
|
|
|
for i, pool := range nmeta2.Pools {
|
|
|
|
if i == 0 {
|
|
|
|
nmeta2.Pools[i] = PoolStatus{
|
|
|
|
CmdLine: pool.CmdLine,
|
|
|
|
ID: i,
|
|
|
|
LastUpdate: UTCNow(),
|
|
|
|
Decommission: &PoolDecommissionInfo{
|
|
|
|
Complete: false,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
testCases := []struct {
|
|
|
|
meta poolMeta
|
|
|
|
pools []*erasureSets
|
|
|
|
expectedUpdate bool
|
|
|
|
expectedErr bool
|
|
|
|
name string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
meta: meta,
|
|
|
|
pools: pools,
|
|
|
|
name: "Correct",
|
|
|
|
expectedErr: false,
|
|
|
|
expectedUpdate: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
meta: meta,
|
|
|
|
pools: newPools,
|
2022-11-16 07:59:10 -08:00
|
|
|
name: "Correct-Update",
|
|
|
|
expectedErr: false,
|
|
|
|
expectedUpdate: true,
|
2022-01-14 10:32:35 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
meta: meta,
|
|
|
|
pools: reducedPools,
|
2022-11-16 07:59:10 -08:00
|
|
|
name: "Correct-Update",
|
|
|
|
expectedErr: false,
|
|
|
|
expectedUpdate: true,
|
2022-01-14 10:32:35 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
meta: meta,
|
|
|
|
pools: orderChangePools,
|
|
|
|
name: "Invalid-Orderchange",
|
2023-05-16 16:00:57 -07:00
|
|
|
expectedErr: false,
|
|
|
|
expectedUpdate: true,
|
2022-01-14 10:32:35 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
meta: nmeta1,
|
|
|
|
pools: pools,
|
|
|
|
name: "Invalid-Completed-Pool-Not-Removed",
|
|
|
|
expectedErr: true,
|
|
|
|
expectedUpdate: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
meta: nmeta2,
|
|
|
|
pools: pools,
|
|
|
|
name: "Correct-Decom-Pending",
|
|
|
|
expectedErr: false,
|
|
|
|
expectedUpdate: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
meta: nmeta2,
|
|
|
|
pools: reducedPools,
|
|
|
|
name: "Invalid-Decom-Pending-Pool-Removal",
|
2023-05-16 16:00:57 -07:00
|
|
|
expectedErr: false,
|
|
|
|
expectedUpdate: true,
|
2022-01-14 10:32:35 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
meta: nmeta1,
|
|
|
|
pools: reducedPools,
|
|
|
|
name: "Correct-Decom-Pool-Removed",
|
|
|
|
expectedErr: false,
|
|
|
|
expectedUpdate: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
meta: poolMeta{}, // no-pool info available fresh setup.
|
|
|
|
pools: pools,
|
|
|
|
name: "Correct-Fresh-Setup",
|
|
|
|
expectedErr: false,
|
|
|
|
expectedUpdate: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
meta: nmeta2,
|
|
|
|
pools: orderChangePools,
|
|
|
|
name: "Invalid-Orderchange-Decom",
|
2023-05-16 16:00:57 -07:00
|
|
|
expectedErr: false,
|
|
|
|
expectedUpdate: true,
|
2022-01-14 10:32:35 -08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
t.Parallel()
|
|
|
|
for _, testCase := range testCases {
|
|
|
|
testCase := testCase
|
|
|
|
t.Run(testCase.name, func(t *testing.T) {
|
|
|
|
update, err := testCase.meta.validate(testCase.pools)
|
|
|
|
if testCase.expectedErr {
|
|
|
|
t.Log(err)
|
|
|
|
}
|
|
|
|
if err != nil && !testCase.expectedErr {
|
|
|
|
t.Errorf("Expected success, but found %s", err)
|
|
|
|
}
|
|
|
|
if err == nil && testCase.expectedErr {
|
|
|
|
t.Error("Expected error, but got `nil`")
|
|
|
|
}
|
|
|
|
if update != testCase.expectedUpdate {
|
|
|
|
t.Errorf("Expected %t, got %t", testCase.expectedUpdate, update)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|