2018-02-15 17:45:57 -08:00
/ *
2019-04-09 11:39:42 -07:00
* MinIO Cloud Storage , ( C ) 2018 MinIO , Inc .
2018-02-15 17:45:57 -08:00
*
* 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 (
"fmt"
2018-08-15 16:35:21 -07:00
"strconv"
2018-02-15 17:45:57 -08:00
"strings"
2019-05-29 16:35:12 -07:00
"github.com/minio/minio-go/v6/pkg/set"
2019-10-04 10:35:33 -07:00
"github.com/minio/minio/cmd/config"
2018-02-15 17:45:57 -08:00
"github.com/minio/minio/pkg/ellipses"
2019-10-04 10:35:33 -07:00
"github.com/minio/minio/pkg/env"
2018-02-15 17:45:57 -08:00
)
// This file implements and supports ellipses pattern for
// `minio server` command line arguments.
// Endpoint set represents parsed ellipses values, also provides
// methods to get the sets of endpoints.
type endpointSet struct {
argPatterns [ ] ellipses . ArgPattern
endpoints [ ] string // Endpoints saved from previous GetEndpoints().
setIndexes [ ] [ ] uint64 // All the sets.
}
// Supported set sizes this is used to find the optimal
// single set size.
var setSizes = [ ] uint64 { 4 , 6 , 8 , 10 , 12 , 14 , 16 }
// getDivisibleSize - returns a greatest common divisor of
// all the ellipses sizes.
func getDivisibleSize ( totalSizes [ ] uint64 ) ( result uint64 ) {
gcd := func ( x , y uint64 ) uint64 {
for y != 0 {
x , y = y , x % y
}
return x
}
result = totalSizes [ 0 ]
for i := 1 ; i < len ( totalSizes ) ; i ++ {
result = gcd ( result , totalSizes [ i ] )
}
return result
}
2019-11-19 17:42:27 -08:00
// isValidSetSize - checks whether given count is a valid set size for erasure coding.
var isValidSetSize = func ( count uint64 ) bool {
return ( count >= setSizes [ 0 ] && count <= setSizes [ len ( setSizes ) - 1 ] && count % 2 == 0 )
}
2018-02-15 17:45:57 -08:00
// getSetIndexes returns list of indexes which provides the set size
// on each index, this function also determines the final set size
// The final set size has the affinity towards choosing smaller
// indexes (total sets)
2019-11-19 17:42:27 -08:00
func getSetIndexes ( args [ ] string , totalSizes [ ] uint64 , customSetDriveCount uint64 ) ( setIndexes [ ] [ ] uint64 , err error ) {
2018-02-15 17:45:57 -08:00
if len ( totalSizes ) == 0 || len ( args ) == 0 {
return nil , errInvalidArgument
}
setIndexes = make ( [ ] [ ] uint64 , len ( totalSizes ) )
2018-05-08 19:04:36 -07:00
for _ , totalSize := range totalSizes {
2018-02-15 17:45:57 -08:00
// Check if totalSize has minimum range upto setSize
2018-08-15 16:35:21 -07:00
if totalSize < setSizes [ 0 ] || totalSize < customSetDriveCount {
2020-01-07 09:13:44 -08:00
msg := fmt . Sprintf ( "Incorrect number of endpoints provided %s" , args )
return nil , config . ErrInvalidNumberOfErasureEndpoints ( nil ) . Msg ( msg )
2018-02-15 17:45:57 -08:00
}
}
var setSize uint64
commonSize := getDivisibleSize ( totalSizes )
if commonSize > setSizes [ len ( setSizes ) - 1 ] {
prevD := commonSize / setSizes [ 0 ]
for _ , i := range setSizes {
if commonSize % i == 0 {
d := commonSize / i
if d <= prevD {
prevD = d
setSize = i
}
}
}
} else {
setSize = commonSize
}
2018-08-15 16:35:21 -07:00
possibleSetCounts := func ( setSize uint64 ) ( ss [ ] uint64 ) {
for _ , s := range setSizes {
if setSize % s == 0 {
ss = append ( ss , s )
}
}
return ss
}
if customSetDriveCount > 0 {
msg := fmt . Sprintf ( "Invalid set drive count, leads to non-uniform distribution for the given number of disks. Possible values for custom set count are %d" , possibleSetCounts ( setSize ) )
if customSetDriveCount > setSize {
2019-10-04 10:35:33 -07:00
return nil , config . ErrInvalidErasureSetSize ( nil ) . Msg ( msg )
2018-08-15 16:35:21 -07:00
}
if setSize % customSetDriveCount != 0 {
2019-10-04 10:35:33 -07:00
return nil , config . ErrInvalidErasureSetSize ( nil ) . Msg ( msg )
2018-08-15 16:35:21 -07:00
}
setSize = customSetDriveCount
2018-02-15 17:45:57 -08:00
}
// Check whether setSize is with the supported range.
if ! isValidSetSize ( setSize ) {
2020-01-07 09:13:44 -08:00
msg := fmt . Sprintf ( "Incorrect number of endpoints provided %s" , args )
return nil , config . ErrInvalidNumberOfErasureEndpoints ( nil ) . Msg ( msg )
2018-02-15 17:45:57 -08:00
}
for i := range totalSizes {
for j := uint64 ( 0 ) ; j < totalSizes [ i ] / setSize ; j ++ {
setIndexes [ i ] = append ( setIndexes [ i ] , setSize )
}
}
return setIndexes , nil
}
// Returns all the expanded endpoints, each argument is expanded separately.
func ( s endpointSet ) getEndpoints ( ) ( endpoints [ ] string ) {
if len ( s . endpoints ) != 0 {
return s . endpoints
}
for _ , argPattern := range s . argPatterns {
for _ , lbls := range argPattern . Expand ( ) {
endpoints = append ( endpoints , strings . Join ( lbls , "" ) )
}
}
s . endpoints = endpoints
return endpoints
}
// Get returns the sets representation of the endpoints
// this function also intelligently decides on what will
// be the right set size etc.
func ( s endpointSet ) Get ( ) ( sets [ ] [ ] string ) {
var k = uint64 ( 0 )
endpoints := s . getEndpoints ( )
for i := range s . setIndexes {
for j := range s . setIndexes [ i ] {
sets = append ( sets , endpoints [ k : s . setIndexes [ i ] [ j ] + k ] )
k = s . setIndexes [ i ] [ j ] + k
}
}
return sets
}
// Return the total size for each argument patterns.
func getTotalSizes ( argPatterns [ ] ellipses . ArgPattern ) [ ] uint64 {
var totalSizes [ ] uint64
for _ , argPattern := range argPatterns {
var totalSize uint64 = 1
for _ , p := range argPattern {
totalSize = totalSize * uint64 ( len ( p . Seq ) )
}
totalSizes = append ( totalSizes , totalSize )
}
return totalSizes
}
// Parses all arguments and returns an endpointSet which is a collection
// of endpoints following the ellipses pattern, this is what is used
// by the object layer for initializing itself.
2019-11-19 17:42:27 -08:00
func parseEndpointSet ( customSetDriveCount uint64 , args ... string ) ( ep endpointSet , err error ) {
2018-02-15 17:45:57 -08:00
var argPatterns = make ( [ ] ellipses . ArgPattern , len ( args ) )
for i , arg := range args {
2018-02-28 14:30:00 -08:00
patterns , perr := ellipses . FindEllipsesPatterns ( arg )
if perr != nil {
2019-10-04 10:35:33 -07:00
return endpointSet { } , config . ErrInvalidErasureEndpoints ( nil ) . Msg ( perr . Error ( ) )
2018-02-15 17:45:57 -08:00
}
argPatterns [ i ] = patterns
}
2019-11-19 17:42:27 -08:00
ep . setIndexes , err = getSetIndexes ( args , getTotalSizes ( argPatterns ) , customSetDriveCount )
2018-02-15 17:45:57 -08:00
if err != nil {
2019-10-04 10:35:33 -07:00
return endpointSet { } , config . ErrInvalidErasureEndpoints ( nil ) . Msg ( err . Error ( ) )
2018-02-15 17:45:57 -08:00
}
ep . argPatterns = argPatterns
return ep , nil
}
2019-06-10 07:57:42 -07:00
// GetAllSets - parses all ellipses input arguments, expands them into
// corresponding list of endpoints chunked evenly in accordance with a
// specific set size.
2018-02-15 17:45:57 -08:00
// For example: {1...64} is divided into 4 sets each of size 16.
// This applies to even distributed setup syntax as well.
2020-01-07 09:13:44 -08:00
func GetAllSets ( customSetDriveCount uint64 , args ... string ) ( [ ] [ ] string , error ) {
2018-02-15 17:45:57 -08:00
var setArgs [ ] [ ] string
if ! ellipses . HasEllipses ( args ... ) {
var setIndexes [ ] [ ] uint64
// Check if we have more one args.
if len ( args ) > 1 {
var err error
2019-11-19 17:42:27 -08:00
setIndexes , err = getSetIndexes ( args , [ ] uint64 { uint64 ( len ( args ) ) } , customSetDriveCount )
2018-02-15 17:45:57 -08:00
if err != nil {
return nil , err
}
} else {
// We are in FS setup, proceed forward.
2018-02-26 10:22:03 -08:00
setIndexes = [ ] [ ] uint64 { { uint64 ( len ( args ) ) } }
2018-02-15 17:45:57 -08:00
}
s := endpointSet {
endpoints : args ,
setIndexes : setIndexes ,
}
setArgs = s . Get ( )
} else {
2019-11-19 17:42:27 -08:00
s , err := parseEndpointSet ( customSetDriveCount , args ... )
2018-02-15 17:45:57 -08:00
if err != nil {
return nil , err
}
setArgs = s . Get ( )
}
uniqueArgs := set . NewStringSet ( )
for _ , sargs := range setArgs {
for _ , arg := range sargs {
if uniqueArgs . Contains ( arg ) {
2019-10-04 10:35:33 -07:00
return nil , config . ErrInvalidErasureEndpoints ( nil ) . Msg ( fmt . Sprintf ( "Input args (%s) has duplicate ellipses" , args ) )
2018-02-15 17:45:57 -08:00
}
uniqueArgs . Add ( arg )
}
}
return setArgs , nil
}
// CreateServerEndpoints - validates and creates new endpoints from input args, supports
// both ellipses and without ellipses transparently.
2020-01-07 09:13:44 -08:00
func createServerEndpoints ( serverAddr string , args ... string ) (
endpointZones EndpointZones , setDriveCount int ,
setupType SetupType , err error ) {
2019-11-19 17:42:27 -08:00
if len ( args ) == 0 {
2019-11-21 04:24:51 -08:00
return nil , - 1 , - 1 , errInvalidArgument
2018-02-15 17:45:57 -08:00
}
2020-01-07 09:13:44 -08:00
if v := env . Get ( "MINIO_ERASURE_SET_DRIVE_COUNT" , "" ) ; v != "" {
setDriveCount , err = strconv . Atoi ( v )
if err != nil {
return nil , - 1 , - 1 , config . ErrInvalidErasureSetSize ( err )
}
if ! isValidSetSize ( uint64 ( setDriveCount ) ) {
return nil , - 1 , - 1 , config . ErrInvalidErasureSetSize ( nil )
}
}
2019-11-19 17:42:27 -08:00
if ! ellipses . HasEllipses ( args ... ) {
2020-01-07 09:13:44 -08:00
setArgs , err := GetAllSets ( uint64 ( setDriveCount ) , args ... )
2019-11-19 17:42:27 -08:00
if err != nil {
2019-11-21 04:24:51 -08:00
return nil , - 1 , - 1 , err
2019-11-19 17:42:27 -08:00
}
2019-12-19 13:45:56 -08:00
endpointList , newSetupType , err := CreateEndpoints ( serverAddr , false , setArgs ... )
2019-11-19 17:42:27 -08:00
if err != nil {
2019-11-21 04:24:51 -08:00
return nil , - 1 , - 1 , err
2019-11-19 17:42:27 -08:00
}
endpointZones = append ( endpointZones , ZoneEndpoints {
SetCount : len ( setArgs ) ,
DrivesPerSet : len ( setArgs [ 0 ] ) ,
Endpoints : endpointList ,
} )
setupType = newSetupType
2019-11-21 04:24:51 -08:00
return endpointZones , len ( setArgs [ 0 ] ) , setupType , nil
2018-02-15 17:45:57 -08:00
}
2020-01-07 09:13:44 -08:00
var prevSetupType SetupType
2019-12-19 13:45:56 -08:00
var foundPrevLocal bool
2019-11-19 17:42:27 -08:00
for _ , arg := range args {
2020-01-07 09:13:44 -08:00
setArgs , err := GetAllSets ( uint64 ( setDriveCount ) , arg )
2019-11-19 17:42:27 -08:00
if err != nil {
2019-11-21 04:24:51 -08:00
return nil , - 1 , - 1 , err
2019-11-19 17:42:27 -08:00
}
2020-01-07 09:13:44 -08:00
var endpointList Endpoints
endpointList , setupType , err = CreateEndpoints ( serverAddr , foundPrevLocal , setArgs ... )
2019-11-19 17:42:27 -08:00
if err != nil {
2019-11-21 04:24:51 -08:00
return nil , - 1 , - 1 , err
2019-11-19 17:42:27 -08:00
}
2020-01-07 09:13:44 -08:00
if setDriveCount != 0 && setDriveCount != len ( setArgs [ 0 ] ) {
return nil , - 1 , - 1 , fmt . Errorf ( "All zones should have same drive per set ratio - expected %d, got %d" , setDriveCount , len ( setArgs [ 0 ] ) )
2019-11-19 17:42:27 -08:00
}
2020-01-07 09:13:44 -08:00
if prevSetupType != UnknownSetupType && prevSetupType != setupType {
return nil , - 1 , - 1 , fmt . Errorf ( "All zones should be of the same setup-type to maintain the original SLA expectations - expected %s, got %s" , prevSetupType , setupType )
}
if err = endpointZones . Add ( ZoneEndpoints {
2019-11-19 17:42:27 -08:00
SetCount : len ( setArgs ) ,
DrivesPerSet : len ( setArgs [ 0 ] ) ,
Endpoints : endpointList ,
2020-01-07 09:13:44 -08:00
} ) ; err != nil {
return nil , - 1 , - 1 , err
}
foundPrevLocal = endpointList . atleastOneEndpointLocal ( )
if setDriveCount == 0 {
setDriveCount = len ( setArgs [ 0 ] )
}
prevSetupType = setupType
2019-11-19 17:42:27 -08:00
}
2019-11-26 11:42:10 -08:00
2020-01-07 09:13:44 -08:00
return endpointZones , setDriveCount , setupType , nil
2018-02-15 17:45:57 -08:00
}