2016-07-10 01:31:32 +05:30
/ *
2020-06-12 20:04:01 -07:00
* MinIO Cloud Storage , ( C ) 2020 MinIO , Inc .
2016-07-10 01:31:32 +05:30
*
* 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 .
* /
2016-08-18 16:23:42 -07:00
package cmd
2016-07-10 01:31:32 +05:30
2016-07-21 19:07:00 -07:00
import (
2017-08-14 18:08:42 -07:00
"bytes"
"encoding/hex"
2016-09-09 11:08:18 +05:30
"encoding/json"
2016-07-21 19:07:00 -07:00
"testing"
2017-01-31 00:44:42 +01:00
2020-06-12 20:04:01 -07:00
"github.com/dustin/go-humanize"
jsoniter "github.com/json-iterator/go"
2016-07-21 19:07:00 -07:00
)
2016-07-10 01:31:32 +05:30
2020-06-12 20:04:01 -07:00
func TestIsXLMetaFormatValid ( t * testing . T ) {
tests := [ ] struct {
name int
version string
format string
want bool
2016-07-10 01:31:32 +05:30
} {
2020-06-12 20:04:01 -07:00
{ 1 , "123" , "fs" , false } ,
{ 2 , "123" , xlMetaFormat , false } ,
{ 3 , xlMetaVersion100 , "test" , false } ,
{ 4 , xlMetaVersion101 , "hello" , false } ,
{ 5 , xlMetaVersion100 , xlMetaFormat , true } ,
{ 6 , xlMetaVersion101 , xlMetaFormat , true } ,
}
for _ , tt := range tests {
if got := isXLMetaFormatValid ( tt . version , tt . format ) ; got != tt . want {
t . Errorf ( "Test %d: Expected %v but received %v" , tt . name , got , tt . want )
2016-11-21 01:47:26 -08:00
}
2016-07-10 01:31:32 +05:30
}
}
2016-07-12 18:23:40 -07:00
2020-06-12 20:04:01 -07:00
func TestIsXLMetaErasureInfoValid ( t * testing . T ) {
tests := [ ] struct {
name int
data int
parity int
want bool
2016-07-21 19:07:00 -07:00
} {
2020-06-12 20:04:01 -07:00
{ 1 , 5 , 6 , false } ,
{ 2 , 5 , 5 , true } ,
{ 3 , 0 , 5 , false } ,
{ 4 , 5 , 0 , false } ,
{ 5 , 5 , 0 , false } ,
{ 6 , 5 , 4 , true } ,
}
for _ , tt := range tests {
if got := isXLMetaErasureInfoValid ( tt . data , tt . parity ) ; got != tt . want {
t . Errorf ( "Test %d: Expected %v but received %v" , tt . name , got , tt . want )
2016-07-12 18:23:40 -07:00
}
}
}
2016-09-09 11:08:18 +05:30
2020-06-12 20:04:01 -07:00
// newTestXLMetaV1 - initializes new xlMetaV1Object, adds version, allocates a fresh erasure info and metadata.
func newTestXLMetaV1 ( ) xlMetaV1Object {
xlMeta := xlMetaV1Object { }
xlMeta . Version = xlMetaVersion101
2017-01-18 12:24:34 -08:00
xlMeta . Format = xlMetaFormat
xlMeta . Minio . Release = "test"
2017-08-14 18:08:42 -07:00
xlMeta . Erasure = ErasureInfo {
2016-09-09 11:08:18 +05:30
Algorithm : "klauspost/reedsolomon/vandermonde" ,
DataBlocks : 5 ,
ParityBlocks : 5 ,
BlockSize : 10485760 ,
Index : 10 ,
Distribution : [ ] int { 9 , 10 , 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 } ,
}
2020-06-12 20:04:01 -07:00
xlMeta . Stat = StatInfo {
2016-09-09 11:08:18 +05:30
Size : int64 ( 20 ) ,
2017-03-18 23:58:41 +05:30
ModTime : UTCNow ( ) ,
2016-09-09 11:08:18 +05:30
}
// Set meta data.
xlMeta . Meta = make ( map [ string ] string )
xlMeta . Meta [ "testKey1" ] = "val1"
xlMeta . Meta [ "testKey2" ] = "val2"
return xlMeta
}
2020-06-12 20:04:01 -07:00
func ( m * xlMetaV1Object ) AddTestObjectCheckSum ( partNumber int , algorithm BitrotAlgorithm , hash string ) {
2017-08-14 18:08:42 -07:00
checksum , err := hex . DecodeString ( hash )
if err != nil {
panic ( err )
2016-09-09 11:08:18 +05:30
}
2020-03-03 03:29:30 +03:00
m . Erasure . Checksums [ partNumber - 1 ] = ChecksumInfo { partNumber , algorithm , checksum }
2016-09-09 11:08:18 +05:30
}
// AddTestObjectPart - add a new object part in order.
2020-06-12 20:04:01 -07:00
func ( m * xlMetaV1Object ) AddTestObjectPart ( partNumber int , partSize int64 ) {
2019-01-05 14:16:43 -08:00
partInfo := ObjectPartInfo {
2016-09-09 11:08:18 +05:30
Number : partNumber ,
Size : partSize ,
}
// Proceed to include new part info.
2020-03-03 03:29:30 +03:00
m . Parts [ partNumber - 1 ] = partInfo
2016-09-09 11:08:18 +05:30
}
2020-06-12 20:04:01 -07:00
// Constructs xlMetaV1Object{} for given number of parts and converts it into bytes.
2016-09-09 11:08:18 +05:30
func getXLMetaBytes ( totalParts int ) [ ] byte {
xlSampleMeta := getSampleXLMeta ( totalParts )
xlMetaBytes , err := json . Marshal ( xlSampleMeta )
if err != nil {
panic ( err )
}
return xlMetaBytes
}
2020-06-12 20:04:01 -07:00
// Returns sample xlMetaV1Object{} for number of parts.
func getSampleXLMeta ( totalParts int ) xlMetaV1Object {
2016-09-09 11:08:18 +05:30
xlMeta := newTestXLMetaV1 ( )
// Number of checksum info == total parts.
2017-08-14 18:08:42 -07:00
xlMeta . Erasure . Checksums = make ( [ ] ChecksumInfo , totalParts )
2016-09-09 11:08:18 +05:30
// total number of parts.
2019-01-05 14:16:43 -08:00
xlMeta . Parts = make ( [ ] ObjectPartInfo , totalParts )
2016-09-09 11:08:18 +05:30
for i := 0 ; i < totalParts ; i ++ {
2020-06-12 20:04:01 -07:00
// hard coding hash and algo value for the checksum, Since we are benchmarking the parsing of xl.meta the magnitude doesn't affect the test,
2016-09-09 11:08:18 +05:30
// The magnitude doesn't make a difference, only the size does.
2020-03-03 03:29:30 +03:00
xlMeta . AddTestObjectCheckSum ( i + 1 , BLAKE2b512 , "a23f5eff248c4372badd9f3b2455a285cd4ca86c3d9a570b091d3fc5cd7ca6d9484bbea3f8c5d8d4f84daae96874419eda578fd736455334afbac2c924b3915a" )
xlMeta . AddTestObjectPart ( i + 1 , 67108864 )
2016-09-09 11:08:18 +05:30
}
return xlMeta
}
2019-09-05 15:51:27 -07:00
// Compare the unmarshaled XLMetaV1 with the one obtained from jsoniter parsing.
2020-06-12 20:04:01 -07:00
func compareXLMetaV1 ( t * testing . T , unMarshalXLMeta , jsoniterXLMeta xlMetaV1Object ) {
// Start comparing the fields of xlMetaV1Object obtained from jsoniter parsing with one parsed using json unmarshaling.
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Version != jsoniterXLMeta . Version {
t . Errorf ( "Expected the Version to be \"%s\", but got \"%s\"." , unMarshalXLMeta . Version , jsoniterXLMeta . Version )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Format != jsoniterXLMeta . Format {
t . Errorf ( "Expected the format to be \"%s\", but got \"%s\"." , unMarshalXLMeta . Format , jsoniterXLMeta . Format )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Stat . Size != jsoniterXLMeta . Stat . Size {
t . Errorf ( "Expected the stat size to be %v, but got %v." , unMarshalXLMeta . Stat . Size , jsoniterXLMeta . Stat . Size )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if ! unMarshalXLMeta . Stat . ModTime . Equal ( jsoniterXLMeta . Stat . ModTime ) {
t . Errorf ( "Expected the modTime to be \"%v\", but got \"%v\"." , unMarshalXLMeta . Stat . ModTime , jsoniterXLMeta . Stat . ModTime )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Erasure . Algorithm != jsoniterXLMeta . Erasure . Algorithm {
t . Errorf ( "Expected the erasure algorithm to be \"%v\", but got \"%v\"." , unMarshalXLMeta . Erasure . Algorithm , jsoniterXLMeta . Erasure . Algorithm )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Erasure . DataBlocks != jsoniterXLMeta . Erasure . DataBlocks {
t . Errorf ( "Expected the erasure data blocks to be %v, but got %v." , unMarshalXLMeta . Erasure . DataBlocks , jsoniterXLMeta . Erasure . DataBlocks )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Erasure . ParityBlocks != jsoniterXLMeta . Erasure . ParityBlocks {
t . Errorf ( "Expected the erasure parity blocks to be %v, but got %v." , unMarshalXLMeta . Erasure . ParityBlocks , jsoniterXLMeta . Erasure . ParityBlocks )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Erasure . BlockSize != jsoniterXLMeta . Erasure . BlockSize {
t . Errorf ( "Expected the erasure block size to be %v, but got %v." , unMarshalXLMeta . Erasure . BlockSize , jsoniterXLMeta . Erasure . BlockSize )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Erasure . Index != jsoniterXLMeta . Erasure . Index {
t . Errorf ( "Expected the erasure index to be %v, but got %v." , unMarshalXLMeta . Erasure . Index , jsoniterXLMeta . Erasure . Index )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if len ( unMarshalXLMeta . Erasure . Distribution ) != len ( jsoniterXLMeta . Erasure . Distribution ) {
t . Errorf ( "Expected the size of Erasure Distribution to be %d, but got %d." , len ( unMarshalXLMeta . Erasure . Distribution ) , len ( jsoniterXLMeta . Erasure . Distribution ) )
2016-09-09 11:08:18 +05:30
} else {
for i := 0 ; i < len ( unMarshalXLMeta . Erasure . Distribution ) ; i ++ {
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Erasure . Distribution [ i ] != jsoniterXLMeta . Erasure . Distribution [ i ] {
t . Errorf ( "Expected the Erasure Distribution to be %d, got %d." , unMarshalXLMeta . Erasure . Distribution [ i ] , jsoniterXLMeta . Erasure . Distribution [ i ] )
2016-09-09 11:08:18 +05:30
}
}
}
2019-09-05 15:51:27 -07:00
if len ( unMarshalXLMeta . Erasure . Checksums ) != len ( jsoniterXLMeta . Erasure . Checksums ) {
t . Errorf ( "Expected the size of Erasure Checksums to be %d, but got %d." , len ( unMarshalXLMeta . Erasure . Checksums ) , len ( jsoniterXLMeta . Erasure . Checksums ) )
2016-09-09 11:08:18 +05:30
} else {
2017-08-14 18:08:42 -07:00
for i := 0 ; i < len ( unMarshalXLMeta . Erasure . Checksums ) ; i ++ {
2020-03-03 03:29:30 +03:00
if unMarshalXLMeta . Erasure . Checksums [ i ] . PartNumber != jsoniterXLMeta . Erasure . Checksums [ i ] . PartNumber {
t . Errorf ( "Expected the Erasure Checksum PartNumber to be \"%d\", got \"%d\"." , unMarshalXLMeta . Erasure . Checksums [ i ] . PartNumber , jsoniterXLMeta . Erasure . Checksums [ i ] . PartNumber )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Erasure . Checksums [ i ] . Algorithm != jsoniterXLMeta . Erasure . Checksums [ i ] . Algorithm {
t . Errorf ( "Expected the Erasure Checksum Algorithm to be \"%s\", got \"%s\"." , unMarshalXLMeta . Erasure . Checksums [ i ] . Algorithm , jsoniterXLMeta . Erasure . Checksums [ i ] . Algorithm )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if ! bytes . Equal ( unMarshalXLMeta . Erasure . Checksums [ i ] . Hash , jsoniterXLMeta . Erasure . Checksums [ i ] . Hash ) {
t . Errorf ( "Expected the Erasure Checksum Hash to be \"%s\", got \"%s\"." , unMarshalXLMeta . Erasure . Checksums [ i ] . Hash , jsoniterXLMeta . Erasure . Checksums [ i ] . Hash )
2016-09-09 11:08:18 +05:30
}
}
}
2018-08-06 15:14:08 -07:00
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Minio . Release != jsoniterXLMeta . Minio . Release {
t . Errorf ( "Expected the Release string to be \"%s\", but got \"%s\"." , unMarshalXLMeta . Minio . Release , jsoniterXLMeta . Minio . Release )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if len ( unMarshalXLMeta . Parts ) != len ( jsoniterXLMeta . Parts ) {
t . Errorf ( "Expected info of %d parts to be present, but got %d instead." , len ( unMarshalXLMeta . Parts ) , len ( jsoniterXLMeta . Parts ) )
2016-09-09 11:08:18 +05:30
} else {
for i := 0 ; i < len ( unMarshalXLMeta . Parts ) ; i ++ {
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Parts [ i ] . Number != jsoniterXLMeta . Parts [ i ] . Number {
t . Errorf ( "Expected the number of part %d to be \"%d\", got \"%d\"." , i + 1 , unMarshalXLMeta . Parts [ i ] . Number , jsoniterXLMeta . Parts [ i ] . Number )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
if unMarshalXLMeta . Parts [ i ] . Size != jsoniterXLMeta . Parts [ i ] . Size {
t . Errorf ( "Expected the size of part %d to be %v, got %v." , i + 1 , unMarshalXLMeta . Parts [ i ] . Size , jsoniterXLMeta . Parts [ i ] . Size )
2016-09-09 11:08:18 +05:30
}
}
}
for key , val := range unMarshalXLMeta . Meta {
2019-09-05 15:51:27 -07:00
jsoniterVal , exists := jsoniterXLMeta . Meta [ key ]
2016-09-09 11:08:18 +05:30
if ! exists {
t . Errorf ( "No meta data entry for Key \"%s\" exists." , key )
}
2019-09-05 15:51:27 -07:00
if val != jsoniterVal {
t . Errorf ( "Expected the value for Meta data key \"%s\" to be \"%s\", but got \"%s\"." , key , val , jsoniterVal )
2016-09-09 11:08:18 +05:30
}
}
}
2019-09-05 15:51:27 -07:00
// Tests the correctness of constructing XLMetaV1 using jsoniter lib.
2016-09-09 11:08:18 +05:30
// The result will be compared with the result obtained from json.unMarshal of the byte data.
2019-09-05 15:51:27 -07:00
func TestGetXLMetaV1Jsoniter1 ( t * testing . T ) {
2016-09-09 11:08:18 +05:30
xlMetaJSON := getXLMetaBytes ( 1 )
2020-06-12 20:04:01 -07:00
var unMarshalXLMeta xlMetaV1Object
2016-09-09 11:08:18 +05:30
if err := json . Unmarshal ( xlMetaJSON , & unMarshalXLMeta ) ; err != nil {
2017-08-14 18:08:42 -07:00
t . Errorf ( "Unmarshalling failed: %v" , err )
2016-09-09 11:08:18 +05:30
}
2020-06-12 20:04:01 -07:00
var jsoniterXLMeta xlMetaV1Object
var json = jsoniter . ConfigCompatibleWithStandardLibrary
if err := json . Unmarshal ( xlMetaJSON , & jsoniterXLMeta ) ; err != nil {
2019-09-05 15:51:27 -07:00
t . Errorf ( "jsoniter parsing of XLMeta failed: %v" , err )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
compareXLMetaV1 ( t , unMarshalXLMeta , jsoniterXLMeta )
2016-09-09 11:08:18 +05:30
}
2019-09-05 15:51:27 -07:00
// Tests the correctness of constructing XLMetaV1 using jsoniter lib for XLMetaV1 of size 10 parts.
2016-09-09 11:08:18 +05:30
// The result will be compared with the result obtained from json.unMarshal of the byte data.
2019-09-05 15:51:27 -07:00
func TestGetXLMetaV1Jsoniter10 ( t * testing . T ) {
2016-09-09 11:08:18 +05:30
xlMetaJSON := getXLMetaBytes ( 10 )
2020-06-12 20:04:01 -07:00
var unMarshalXLMeta xlMetaV1Object
2016-09-09 11:08:18 +05:30
if err := json . Unmarshal ( xlMetaJSON , & unMarshalXLMeta ) ; err != nil {
2017-08-14 18:08:42 -07:00
t . Errorf ( "Unmarshalling failed: %v" , err )
2016-09-09 11:08:18 +05:30
}
2020-06-12 20:04:01 -07:00
var jsoniterXLMeta xlMetaV1Object
var json = jsoniter . ConfigCompatibleWithStandardLibrary
if err := json . Unmarshal ( xlMetaJSON , & jsoniterXLMeta ) ; err != nil {
2019-09-05 15:51:27 -07:00
t . Errorf ( "jsoniter parsing of XLMeta failed: %v" , err )
2016-09-09 11:08:18 +05:30
}
2020-06-12 20:04:01 -07:00
2019-09-05 15:51:27 -07:00
compareXLMetaV1 ( t , unMarshalXLMeta , jsoniterXLMeta )
2016-09-09 11:08:18 +05:30
}
2017-01-31 00:44:42 +01:00
// Test the predicted part size from the part index
func TestGetPartSizeFromIdx ( t * testing . T ) {
// Create test cases
testCases := [ ] struct {
totalSize int64
partSize int64
partIndex int
expectedSize int64
} {
// Total size is zero
{ 0 , 10 , 1 , 0 } ,
// part size 2MiB, total size 4MiB
{ 4 * humanize . MiByte , 2 * humanize . MiByte , 1 , 2 * humanize . MiByte } ,
{ 4 * humanize . MiByte , 2 * humanize . MiByte , 2 , 2 * humanize . MiByte } ,
{ 4 * humanize . MiByte , 2 * humanize . MiByte , 3 , 0 } ,
// part size 2MiB, total size 5MiB
{ 5 * humanize . MiByte , 2 * humanize . MiByte , 1 , 2 * humanize . MiByte } ,
{ 5 * humanize . MiByte , 2 * humanize . MiByte , 2 , 2 * humanize . MiByte } ,
{ 5 * humanize . MiByte , 2 * humanize . MiByte , 3 , 1 * humanize . MiByte } ,
{ 5 * humanize . MiByte , 2 * humanize . MiByte , 4 , 0 } ,
}
for i , testCase := range testCases {
2020-04-09 09:30:02 -07:00
s , err := calculatePartSizeFromIdx ( GlobalContext , testCase . totalSize , testCase . partSize , testCase . partIndex )
2017-01-31 15:34:49 -08:00
if err != nil {
t . Errorf ( "Test %d: Expected to pass but failed. %s" , i + 1 , err )
}
if err == nil && s != testCase . expectedSize {
2017-01-31 00:44:42 +01:00
t . Errorf ( "Test %d: The calculated part size is incorrect: expected = %d, found = %d\n" , i + 1 , testCase . expectedSize , s )
}
}
2017-01-31 15:34:49 -08:00
testCasesFailure := [ ] struct {
totalSize int64
partSize int64
partIndex int
err error
} {
2017-10-06 09:38:01 -07:00
// partSize is 0, returns error.
2017-01-31 15:34:49 -08:00
{ 10 , 0 , 1 , errPartSizeZero } ,
2017-10-06 09:38:01 -07:00
// partIndex is 0, returns error.
2017-01-31 15:34:49 -08:00
{ 10 , 1 , 0 , errPartSizeIndex } ,
2017-10-06 09:38:01 -07:00
// Total size is -1, returns error.
2018-09-28 09:06:17 +05:30
{ - 2 , 10 , 1 , errInvalidArgument } ,
2017-01-31 15:34:49 -08:00
}
for i , testCaseFailure := range testCasesFailure {
2020-04-09 09:30:02 -07:00
_ , err := calculatePartSizeFromIdx ( GlobalContext , testCaseFailure . totalSize , testCaseFailure . partSize , testCaseFailure . partIndex )
2017-01-31 15:34:49 -08:00
if err == nil {
t . Errorf ( "Test %d: Expected to failed but passed. %s" , i + 1 , err )
}
2018-04-10 09:36:37 -07:00
if err != nil && err != testCaseFailure . err {
t . Errorf ( "Test %d: Expected err %s, but got %s" , i + 1 , testCaseFailure . err , err )
2017-01-31 15:34:49 -08:00
}
}
2017-01-31 00:44:42 +01:00
}