2021-04-18 12:41:13 -07:00
// Copyright (c) 2015-2021 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/>.
2016-07-10 01:31:32 +05:30
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"
2021-11-18 12:15:22 -08:00
"fmt"
"math/rand"
2016-07-21 19:07:00 -07:00
"testing"
2021-11-18 12:15:22 -08:00
"time"
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"
2021-11-18 12:15:22 -08:00
xhttp "github.com/minio/minio/internal/http"
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
2022-01-02 09:15:06 -08:00
json := jsoniter . ConfigCompatibleWithStandardLibrary
2020-06-12 20:04:01 -07:00
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
2022-01-02 09:15:06 -08:00
json := jsoniter . ConfigCompatibleWithStandardLibrary
2020-06-12 20:04:01 -07:00
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
}
2021-11-18 12:15:22 -08:00
func BenchmarkXlMetaV2Shallow ( b * testing . B ) {
fi := FileInfo {
Volume : "volume" ,
Name : "object-name" ,
VersionID : "756100c6-b393-4981-928a-d49bbc164741" ,
IsLatest : true ,
Deleted : false ,
TransitionStatus : "PENDING" ,
DataDir : "bffea160-ca7f-465f-98bc-9b4f1c3ba1ef" ,
XLV1 : false ,
ModTime : time . Now ( ) ,
Size : 1234456 ,
Mode : 0 ,
Metadata : map [ string ] string {
xhttp . AmzRestore : "FAILED" ,
xhttp . ContentMD5 : mustGetUUID ( ) ,
xhttp . AmzBucketReplicationStatus : "PENDING" ,
xhttp . ContentType : "application/json" ,
} ,
2022-01-02 09:15:06 -08:00
Parts : [ ] ObjectPartInfo {
{
Number : 1 ,
Size : 1234345 ,
ActualSize : 1234345 ,
} ,
2021-11-18 12:15:22 -08:00
{
Number : 2 ,
Size : 1234345 ,
ActualSize : 1234345 ,
} ,
} ,
Erasure : ErasureInfo {
Algorithm : ReedSolomon . String ( ) ,
DataBlocks : 4 ,
ParityBlocks : 2 ,
BlockSize : 10000 ,
Index : 1 ,
Distribution : [ ] int { 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 } ,
2022-01-02 09:15:06 -08:00
Checksums : [ ] ChecksumInfo {
{
PartNumber : 1 ,
Algorithm : HighwayHash256S ,
Hash : nil ,
} ,
2021-11-18 12:15:22 -08:00
{
PartNumber : 2 ,
Algorithm : HighwayHash256S ,
Hash : nil ,
} ,
} ,
} ,
}
for _ , size := range [ ] int { 1 , 10 , 1000 , 100_000 } {
b . Run ( fmt . Sprint ( size , "-versions" ) , func ( b * testing . B ) {
var xl xlMetaV2
ids := make ( [ ] string , size )
for i := 0 ; i < size ; i ++ {
fi . VersionID = mustGetUUID ( )
fi . DataDir = mustGetUUID ( )
ids [ i ] = fi . VersionID
fi . ModTime = fi . ModTime . Add ( - time . Second )
xl . AddVersion ( fi )
}
// Encode all. This is used for benchmarking.
enc , err := xl . AppendTo ( nil )
if err != nil {
b . Fatal ( err )
}
b . Logf ( "Serialized size: %d bytes" , len ( enc ) )
rng := rand . New ( rand . NewSource ( 0 ) )
2022-01-02 09:15:06 -08:00
dump := make ( [ ] byte , len ( enc ) )
2021-11-18 12:15:22 -08:00
b . Run ( "UpdateObjectVersion" , func ( b * testing . B ) {
b . SetBytes ( int64 ( size ) )
b . ResetTimer ( )
b . ReportAllocs ( )
for i := 0 ; i < b . N ; i ++ {
// Load...
xl = xlMetaV2 { }
err := xl . Load ( enc )
if err != nil {
b . Fatal ( err )
}
// Update modtime for resorting...
fi . ModTime = fi . ModTime . Add ( - time . Second )
// Update a random version.
fi . VersionID = ids [ rng . Intn ( size ) ]
// Update...
err = xl . UpdateObjectVersion ( fi )
if err != nil {
b . Fatal ( err )
}
// Save...
dump , err = xl . AppendTo ( dump [ : 0 ] )
if err != nil {
b . Fatal ( err )
}
}
} )
b . Run ( "DeleteVersion" , func ( b * testing . B ) {
b . SetBytes ( int64 ( size ) )
b . ResetTimer ( )
b . ReportAllocs ( )
for i := 0 ; i < b . N ; i ++ {
// Load...
xl = xlMetaV2 { }
err := xl . Load ( enc )
if err != nil {
b . Fatal ( err )
}
// Update a random version.
fi . VersionID = ids [ rng . Intn ( size ) ]
// Delete...
_ , _ , err = xl . DeleteVersion ( fi )
if err != nil {
b . Fatal ( err )
}
// Save...
dump , err = xl . AppendTo ( dump [ : 0 ] )
if err != nil {
b . Fatal ( err )
}
}
} )
b . Run ( "AddVersion" , func ( b * testing . B ) {
b . SetBytes ( int64 ( size ) )
b . ResetTimer ( )
b . ReportAllocs ( )
for i := 0 ; i < b . N ; i ++ {
// Load...
xl = xlMetaV2 { }
err := xl . Load ( enc )
if err != nil {
b . Fatal ( err )
}
// Update modtime for resorting...
fi . ModTime = fi . ModTime . Add ( - time . Second )
// Update a random version.
fi . VersionID = mustGetUUID ( )
// Add...
err = xl . AddVersion ( fi )
if err != nil {
b . Fatal ( err )
}
// Save...
dump , err = xl . AppendTo ( dump [ : 0 ] )
if err != nil {
b . Fatal ( err )
}
}
} )
b . Run ( "ToFileInfo" , func ( b * testing . B ) {
b . SetBytes ( int64 ( size ) )
b . ResetTimer ( )
b . ReportAllocs ( )
for i := 0 ; i < b . N ; i ++ {
// Load...
xl = xlMetaV2 { }
err := xl . Load ( enc )
if err != nil {
b . Fatal ( err )
}
// List...
_ , err = xl . ToFileInfo ( "volume" , "path" , ids [ rng . Intn ( size ) ] )
if err != nil {
b . Fatal ( err )
}
}
} )
b . Run ( "ListVersions" , func ( b * testing . B ) {
b . SetBytes ( int64 ( size ) )
b . ResetTimer ( )
b . ReportAllocs ( )
for i := 0 ; i < b . N ; i ++ {
// Load...
xl = xlMetaV2 { }
err := xl . Load ( enc )
if err != nil {
b . Fatal ( err )
}
// List...
_ , err = xl . ListVersions ( "volume" , "path" )
if err != nil {
b . Fatal ( err )
}
}
} )
b . Run ( "ToFileInfoNew" , func ( b * testing . B ) {
b . SetBytes ( int64 ( size ) )
b . ResetTimer ( )
b . ReportAllocs ( )
for i := 0 ; i < b . N ; i ++ {
buf , _ := isIndexedMetaV2 ( enc )
if buf == nil {
b . Fatal ( "buf == nil" )
}
_ , err = buf . ToFileInfo ( "volume" , "path" , ids [ rng . Intn ( size ) ] )
if err != nil {
b . Fatal ( err )
}
}
} )
b . Run ( "ListVersionsNew" , func ( b * testing . B ) {
b . SetBytes ( int64 ( size ) )
b . ResetTimer ( )
b . ReportAllocs ( )
for i := 0 ; i < b . N ; i ++ {
buf , _ := isIndexedMetaV2 ( enc )
if buf == nil {
b . Fatal ( "buf == nil" )
}
_ , err = buf . ListVersions ( "volume" , "path" )
if err != nil {
b . Fatal ( err )
}
}
} )
} )
}
}