| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | /* | 
					
						
							| 
									
										
										
										
											2017-01-19 04:24:34 +08:00
										 |  |  |  * Minio Cloud Storage, (C) 2016, 2017 Minio, Inc. | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-19 07:23:42 +08:00
										 |  |  | package cmd | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2016-07-09 05:28:06 +08:00
										 |  |  | 	"math" | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 	"math/rand" | 
					
						
							|  |  |  | 	"strconv" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							| 
									
										
										
										
											2016-11-23 10:18:22 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	humanize "github.com/dustin/go-humanize" | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-01 10:05:52 +08:00
										 |  |  | // Prepare XL/FS backend for benchmark.
 | 
					
						
							| 
									
										
										
										
											2016-08-31 10:22:27 +08:00
										 |  |  | func prepareBenchmarkBackend(instanceType string) (ObjectLayer, []string, error) { | 
					
						
							| 
									
										
										
										
											2017-03-01 10:05:52 +08:00
										 |  |  | 	return prepareTestBackend(instanceType) | 
					
						
							| 
									
										
										
										
											2016-08-31 10:22:27 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | // Benchmark utility functions for ObjectLayer.PutObject().
 | 
					
						
							|  |  |  | // Creates Object layer setup ( MakeBucket ) and then runs the PutObject benchmark.
 | 
					
						
							|  |  |  | func runPutObjectBenchmark(b *testing.B, obj ObjectLayer, objSize int) { | 
					
						
							|  |  |  | 	var err error | 
					
						
							|  |  |  | 	// obtains random bucket name.
 | 
					
						
							|  |  |  | 	bucket := getRandomBucketName() | 
					
						
							|  |  |  | 	// create bucket.
 | 
					
						
							|  |  |  | 	err = obj.MakeBucket(bucket) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// get text data generated for number of bytes equal to object size.
 | 
					
						
							|  |  |  | 	textData := generateBytesData(objSize) | 
					
						
							|  |  |  | 	// generate md5sum for the generated data.
 | 
					
						
							|  |  |  | 	// md5sum of the data to written is required as input for PutObject.
 | 
					
						
							|  |  |  | 	metadata := make(map[string]string) | 
					
						
							| 
									
										
										
										
											2016-11-22 05:51:05 +08:00
										 |  |  | 	metadata["md5Sum"] = getMD5Hash(textData) | 
					
						
							| 
									
										
										
										
											2016-10-03 06:51:49 +08:00
										 |  |  | 	sha256sum := "" | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | 	// benchmark utility which helps obtain number of allocations and bytes allocated per ops.
 | 
					
						
							|  |  |  | 	b.ReportAllocs() | 
					
						
							|  |  |  | 	// the actual benchmark for PutObject starts here. Reset the benchmark timer.
 | 
					
						
							|  |  |  | 	b.ResetTimer() | 
					
						
							|  |  |  | 	for i := 0; i < b.N; i++ { | 
					
						
							|  |  |  | 		// insert the object.
 | 
					
						
							| 
									
										
										
										
											2016-10-03 06:51:49 +08:00
										 |  |  | 		objInfo, err := obj.PutObject(bucket, "object"+strconv.Itoa(i), int64(len(textData)), bytes.NewBuffer(textData), metadata, sha256sum) | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			b.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-09-03 03:18:35 +08:00
										 |  |  | 		if objInfo.MD5Sum != metadata["md5Sum"] { | 
					
						
							|  |  |  | 			b.Fatalf("Write no: %d: Md5Sum mismatch during object write into the bucket: Expected %s, got %s", i+1, objInfo.MD5Sum, metadata["md5Sum"]) | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Benchmark ends here. Stop timer.
 | 
					
						
							|  |  |  | 	b.StopTimer() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-09 05:28:06 +08:00
										 |  |  | // Benchmark utility functions for ObjectLayer.PutObjectPart().
 | 
					
						
							|  |  |  | // Creates Object layer setup ( MakeBucket ) and then runs the PutObjectPart benchmark.
 | 
					
						
							|  |  |  | func runPutObjectPartBenchmark(b *testing.B, obj ObjectLayer, partSize int) { | 
					
						
							|  |  |  | 	var err error | 
					
						
							|  |  |  | 	// obtains random bucket name.
 | 
					
						
							|  |  |  | 	bucket := getRandomBucketName() | 
					
						
							|  |  |  | 	object := getRandomObjectName() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// create bucket.
 | 
					
						
							|  |  |  | 	err = obj.MakeBucket(bucket) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-23 10:18:22 +08:00
										 |  |  | 	objSize := 128 * humanize.MiByte | 
					
						
							| 
									
										
										
										
											2016-07-09 05:28:06 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// PutObjectPart returns md5Sum of the object inserted.
 | 
					
						
							|  |  |  | 	// md5Sum variable is assigned with that value.
 | 
					
						
							|  |  |  | 	var md5Sum, uploadID string | 
					
						
							|  |  |  | 	// get text data generated for number of bytes equal to object size.
 | 
					
						
							|  |  |  | 	textData := generateBytesData(objSize) | 
					
						
							|  |  |  | 	// generate md5sum for the generated data.
 | 
					
						
							|  |  |  | 	// md5sum of the data to written is required as input for NewMultipartUpload.
 | 
					
						
							|  |  |  | 	metadata := make(map[string]string) | 
					
						
							| 
									
										
										
										
											2016-11-22 05:51:05 +08:00
										 |  |  | 	metadata["md5Sum"] = getMD5Hash(textData) | 
					
						
							| 
									
										
										
										
											2016-10-03 06:51:49 +08:00
										 |  |  | 	sha256sum := "" | 
					
						
							| 
									
										
										
										
											2016-07-09 05:28:06 +08:00
										 |  |  | 	uploadID, err = obj.NewMultipartUpload(bucket, object, metadata) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var textPartData []byte | 
					
						
							|  |  |  | 	// benchmark utility which helps obtain number of allocations and bytes allocated per ops.
 | 
					
						
							|  |  |  | 	b.ReportAllocs() | 
					
						
							|  |  |  | 	// the actual benchmark for PutObjectPart starts here. Reset the benchmark timer.
 | 
					
						
							|  |  |  | 	b.ResetTimer() | 
					
						
							|  |  |  | 	for i := 0; i < b.N; i++ { | 
					
						
							|  |  |  | 		// insert the object.
 | 
					
						
							|  |  |  | 		totalPartsNR := int(math.Ceil(float64(objSize) / float64(partSize))) | 
					
						
							|  |  |  | 		for j := 0; j < totalPartsNR; j++ { | 
					
						
							|  |  |  | 			if j < totalPartsNR-1 { | 
					
						
							|  |  |  | 				textPartData = textData[j*partSize : (j+1)*partSize-1] | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				textPartData = textData[j*partSize:] | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			metadata := make(map[string]string) | 
					
						
							| 
									
										
										
										
											2016-11-22 05:51:05 +08:00
										 |  |  | 			metadata["md5Sum"] = getMD5Hash([]byte(textPartData)) | 
					
						
							| 
									
										
										
										
											2017-02-01 01:38:34 +08:00
										 |  |  | 			var partInfo PartInfo | 
					
						
							|  |  |  | 			partInfo, err = obj.PutObjectPart(bucket, object, uploadID, j, int64(len(textPartData)), bytes.NewBuffer(textPartData), metadata["md5Sum"], sha256sum) | 
					
						
							| 
									
										
										
										
											2016-07-09 05:28:06 +08:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				b.Fatal(err) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-02-01 01:38:34 +08:00
										 |  |  | 			if partInfo.ETag != metadata["md5Sum"] { | 
					
						
							| 
									
										
										
										
											2016-07-09 05:28:06 +08:00
										 |  |  | 				b.Fatalf("Write no: %d: Md5Sum mismatch during object write into the bucket: Expected %s, got %s", i+1, md5Sum, metadata["md5Sum"]) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Benchmark ends here. Stop timer.
 | 
					
						
							|  |  |  | 	b.StopTimer() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // creates XL/FS backend setup, obtains the object layer and calls the runPutObjectPartBenchmark function.
 | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | func benchmarkPutObjectPart(b *testing.B, instanceType string, objSize int) { | 
					
						
							| 
									
										
										
										
											2017-01-19 04:24:34 +08:00
										 |  |  | 	rootPath, err := newTestConfig(globalMinioDefaultRegion) | 
					
						
							| 
									
										
										
										
											2016-10-25 09:45:06 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Unable to initialize config. %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(rootPath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-09 05:28:06 +08:00
										 |  |  | 	// create a temp XL/FS backend.
 | 
					
						
							| 
									
										
										
										
											2016-08-31 10:22:27 +08:00
										 |  |  | 	objLayer, disks, err := prepareBenchmarkBackend(instanceType) | 
					
						
							| 
									
										
										
										
											2016-07-09 05:28:06 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Failed obtaining Temp Backend: <ERROR> %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// cleaning up the backend by removing all the directories and files created on function return.
 | 
					
						
							|  |  |  | 	defer removeRoots(disks) | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | 	// uses *testing.B and the object Layer to run the benchmark.
 | 
					
						
							|  |  |  | 	runPutObjectPartBenchmark(b, objLayer, objSize) | 
					
						
							| 
									
										
										
										
											2016-07-09 05:28:06 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | // creates XL/FS backend setup, obtains the object layer and calls the runPutObjectBenchmark function.
 | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | func benchmarkPutObject(b *testing.B, instanceType string, objSize int) { | 
					
						
							| 
									
										
										
										
											2017-01-19 04:24:34 +08:00
										 |  |  | 	rootPath, err := newTestConfig(globalMinioDefaultRegion) | 
					
						
							| 
									
										
										
										
											2016-10-25 09:45:06 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Unable to initialize config. %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(rootPath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | 	// create a temp XL/FS backend.
 | 
					
						
							| 
									
										
										
										
											2016-08-31 10:22:27 +08:00
										 |  |  | 	objLayer, disks, err := prepareBenchmarkBackend(instanceType) | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Failed obtaining Temp Backend: <ERROR> %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// cleaning up the backend by removing all the directories and files created on function return.
 | 
					
						
							|  |  |  | 	defer removeRoots(disks) | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | 	// uses *testing.B and the object Layer to run the benchmark.
 | 
					
						
							|  |  |  | 	runPutObjectBenchmark(b, objLayer, objSize) | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | // creates XL/FS backend setup, obtains the object layer and runs parallel benchmark for put object.
 | 
					
						
							|  |  |  | func benchmarkPutObjectParallel(b *testing.B, instanceType string, objSize int) { | 
					
						
							| 
									
										
										
										
											2017-01-19 04:24:34 +08:00
										 |  |  | 	rootPath, err := newTestConfig(globalMinioDefaultRegion) | 
					
						
							| 
									
										
										
										
											2016-10-25 09:45:06 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Unable to initialize config. %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(rootPath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | 	// create a temp XL/FS backend.
 | 
					
						
							| 
									
										
										
										
											2016-08-31 10:22:27 +08:00
										 |  |  | 	objLayer, disks, err := prepareBenchmarkBackend(instanceType) | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Failed obtaining Temp Backend: <ERROR> %s", err) | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | 	// cleaning up the backend by removing all the directories and files created on function return.
 | 
					
						
							|  |  |  | 	defer removeRoots(disks) | 
					
						
							|  |  |  | 	// uses *testing.B and the object Layer to run the benchmark.
 | 
					
						
							|  |  |  | 	runPutObjectBenchmarkParallel(b, objLayer, objSize) | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Benchmark utility functions for ObjectLayer.GetObject().
 | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | // Creates Object layer setup ( MakeBucket, PutObject) and then runs the benchmark.
 | 
					
						
							|  |  |  | func runGetObjectBenchmark(b *testing.B, obj ObjectLayer, objSize int) { | 
					
						
							| 
									
										
										
										
											2017-01-19 04:24:34 +08:00
										 |  |  | 	rootPath, err := newTestConfig(globalMinioDefaultRegion) | 
					
						
							| 
									
										
										
										
											2016-10-25 09:45:06 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Unable to initialize config. %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(rootPath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 	// obtains random bucket name.
 | 
					
						
							|  |  |  | 	bucket := getRandomBucketName() | 
					
						
							|  |  |  | 	// create bucket.
 | 
					
						
							|  |  |  | 	err = obj.MakeBucket(bucket) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-03 06:51:49 +08:00
										 |  |  | 	sha256sum := "" | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 	for i := 0; i < 10; i++ { | 
					
						
							|  |  |  | 		// get text data generated for number of bytes equal to object size.
 | 
					
						
							|  |  |  | 		textData := generateBytesData(objSize) | 
					
						
							|  |  |  | 		// generate md5sum for the generated data.
 | 
					
						
							|  |  |  | 		// md5sum of the data to written is required as input for PutObject.
 | 
					
						
							|  |  |  | 		// PutObject is the functions which writes the data onto the FS/XL backend.
 | 
					
						
							|  |  |  | 		metadata := make(map[string]string) | 
					
						
							| 
									
										
										
										
											2016-11-22 05:51:05 +08:00
										 |  |  | 		metadata["md5Sum"] = getMD5Hash(textData) | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 		// insert the object.
 | 
					
						
							| 
									
										
										
										
											2016-09-03 03:18:35 +08:00
										 |  |  | 		var objInfo ObjectInfo | 
					
						
							| 
									
										
										
										
											2016-10-03 06:51:49 +08:00
										 |  |  | 		objInfo, err = obj.PutObject(bucket, "object"+strconv.Itoa(i), int64(len(textData)), bytes.NewBuffer(textData), metadata, sha256sum) | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			b.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-09-03 03:18:35 +08:00
										 |  |  | 		if objInfo.MD5Sum != metadata["md5Sum"] { | 
					
						
							|  |  |  | 			b.Fatalf("Write no: %d: Md5Sum mismatch during object write into the bucket: Expected %s, got %s", i+1, objInfo.MD5Sum, metadata["md5Sum"]) | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// benchmark utility which helps obtain number of allocations and bytes allocated per ops.
 | 
					
						
							|  |  |  | 	b.ReportAllocs() | 
					
						
							|  |  |  | 	// the actual benchmark for GetObject starts here. Reset the benchmark timer.
 | 
					
						
							|  |  |  | 	b.ResetTimer() | 
					
						
							|  |  |  | 	for i := 0; i < b.N; i++ { | 
					
						
							|  |  |  | 		var buffer = new(bytes.Buffer) | 
					
						
							|  |  |  | 		err = obj.GetObject(bucket, "object"+strconv.Itoa(i%10), 0, int64(objSize), buffer) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			b.Error(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Benchmark ends here. Stop timer.
 | 
					
						
							|  |  |  | 	b.StopTimer() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // randomly picks a character and returns its equivalent byte array.
 | 
					
						
							|  |  |  | func getRandomByte() []byte { | 
					
						
							|  |  |  | 	const letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ" | 
					
						
							|  |  |  | 	// seeding the random number generator.
 | 
					
						
							| 
									
										
										
										
											2017-03-19 02:28:41 +08:00
										 |  |  | 	rand.Seed(UTCNow().UnixNano()) | 
					
						
							| 
									
										
										
										
											2016-06-29 13:12:36 +08:00
										 |  |  | 	var b byte | 
					
						
							|  |  |  | 	// pick a character randomly.
 | 
					
						
							|  |  |  | 	b = letterBytes[rand.Intn(len(letterBytes))] | 
					
						
							|  |  |  | 	return []byte{b} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // picks a random byte and repeats it to size bytes.
 | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | func generateBytesData(size int) []byte { | 
					
						
							| 
									
										
										
										
											2016-06-28 01:01:09 +08:00
										 |  |  | 	// repeat the random character chosen size
 | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 	return bytes.Repeat(getRandomByte(), size) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // creates XL/FS backend setup, obtains the object layer and calls the runGetObjectBenchmark function.
 | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | func benchmarkGetObject(b *testing.B, instanceType string, objSize int) { | 
					
						
							| 
									
										
										
										
											2017-01-19 04:24:34 +08:00
										 |  |  | 	rootPath, err := newTestConfig(globalMinioDefaultRegion) | 
					
						
							| 
									
										
										
										
											2016-10-25 09:45:06 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Unable to initialize config. %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(rootPath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 	// create a temp XL/FS backend.
 | 
					
						
							| 
									
										
										
										
											2016-08-31 10:22:27 +08:00
										 |  |  | 	objLayer, disks, err := prepareBenchmarkBackend(instanceType) | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Failed obtaining Temp Backend: <ERROR> %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// cleaning up the backend by removing all the directories and files created.
 | 
					
						
							|  |  |  | 	defer removeRoots(disks) | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | 	//  uses *testing.B and the object Layer to run the benchmark.
 | 
					
						
							|  |  |  | 	runGetObjectBenchmark(b, objLayer, objSize) | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | // creates XL/FS backend setup, obtains the object layer and runs parallel benchmark for ObjectLayer.GetObject() .
 | 
					
						
							|  |  |  | func benchmarkGetObjectParallel(b *testing.B, instanceType string, objSize int) { | 
					
						
							| 
									
										
										
										
											2017-01-19 04:24:34 +08:00
										 |  |  | 	rootPath, err := newTestConfig(globalMinioDefaultRegion) | 
					
						
							| 
									
										
										
										
											2016-10-25 09:45:06 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Unable to initialize config. %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(rootPath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | 	// create a temp XL/FS backend.
 | 
					
						
							| 
									
										
										
										
											2016-08-31 10:22:27 +08:00
										 |  |  | 	objLayer, disks, err := prepareBenchmarkBackend(instanceType) | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Failed obtaining Temp Backend: <ERROR> %s", err) | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-07-22 02:17:28 +08:00
										 |  |  | 	// cleaning up the backend by removing all the directories and files created.
 | 
					
						
							|  |  |  | 	defer removeRoots(disks) | 
					
						
							|  |  |  | 	//  uses *testing.B and the object Layer to run the benchmark.
 | 
					
						
							|  |  |  | 	runGetObjectBenchmarkParallel(b, objLayer, objSize) | 
					
						
							| 
									
										
										
										
											2016-06-26 11:22:04 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Parallel benchmark utility functions for ObjectLayer.PutObject().
 | 
					
						
							|  |  |  | // Creates Object layer setup ( MakeBucket ) and then runs the PutObject benchmark.
 | 
					
						
							|  |  |  | func runPutObjectBenchmarkParallel(b *testing.B, obj ObjectLayer, objSize int) { | 
					
						
							| 
									
										
										
										
											2017-01-19 04:24:34 +08:00
										 |  |  | 	rootPath, err := newTestConfig(globalMinioDefaultRegion) | 
					
						
							| 
									
										
										
										
											2016-10-25 09:45:06 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Unable to initialize config. %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(rootPath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 	// obtains random bucket name.
 | 
					
						
							|  |  |  | 	bucket := getRandomBucketName() | 
					
						
							|  |  |  | 	// create bucket.
 | 
					
						
							|  |  |  | 	err = obj.MakeBucket(bucket) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// get text data generated for number of bytes equal to object size.
 | 
					
						
							|  |  |  | 	textData := generateBytesData(objSize) | 
					
						
							|  |  |  | 	// generate md5sum for the generated data.
 | 
					
						
							|  |  |  | 	// md5sum of the data to written is required as input for PutObject.
 | 
					
						
							|  |  |  | 	metadata := make(map[string]string) | 
					
						
							| 
									
										
										
										
											2016-11-22 05:51:05 +08:00
										 |  |  | 	metadata["md5Sum"] = getMD5Hash([]byte(textData)) | 
					
						
							| 
									
										
										
										
											2016-10-03 06:51:49 +08:00
										 |  |  | 	sha256sum := "" | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 	// benchmark utility which helps obtain number of allocations and bytes allocated per ops.
 | 
					
						
							|  |  |  | 	b.ReportAllocs() | 
					
						
							|  |  |  | 	// the actual benchmark for PutObject starts here. Reset the benchmark timer.
 | 
					
						
							|  |  |  | 	b.ResetTimer() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	b.RunParallel(func(pb *testing.PB) { | 
					
						
							|  |  |  | 		i := 0 | 
					
						
							|  |  |  | 		for pb.Next() { | 
					
						
							|  |  |  | 			// insert the object.
 | 
					
						
							| 
									
										
										
										
											2016-10-03 06:51:49 +08:00
										 |  |  | 			objInfo, err := obj.PutObject(bucket, "object"+strconv.Itoa(i), int64(len(textData)), bytes.NewBuffer(textData), metadata, sha256sum) | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				b.Fatal(err) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-09-03 03:18:35 +08:00
										 |  |  | 			if objInfo.MD5Sum != metadata["md5Sum"] { | 
					
						
							|  |  |  | 				b.Fatalf("Write no: Md5Sum mismatch during object write into the bucket: Expected %s, got %s", objInfo.MD5Sum, metadata["md5Sum"]) | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			i++ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Benchmark ends here. Stop timer.
 | 
					
						
							|  |  |  | 	b.StopTimer() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Parallel benchmark utility functions for ObjectLayer.GetObject().
 | 
					
						
							|  |  |  | // Creates Object layer setup ( MakeBucket, PutObject) and then runs the benchmark.
 | 
					
						
							|  |  |  | func runGetObjectBenchmarkParallel(b *testing.B, obj ObjectLayer, objSize int) { | 
					
						
							| 
									
										
										
										
											2017-01-19 04:24:34 +08:00
										 |  |  | 	rootPath, err := newTestConfig(globalMinioDefaultRegion) | 
					
						
							| 
									
										
										
										
											2016-10-25 09:45:06 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatalf("Unable to initialize config. %s", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer removeAll(rootPath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 	// obtains random bucket name.
 | 
					
						
							|  |  |  | 	bucket := getRandomBucketName() | 
					
						
							|  |  |  | 	// create bucket.
 | 
					
						
							|  |  |  | 	err = obj.MakeBucket(bucket) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		b.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i := 0; i < 10; i++ { | 
					
						
							|  |  |  | 		// get text data generated for number of bytes equal to object size.
 | 
					
						
							|  |  |  | 		textData := generateBytesData(objSize) | 
					
						
							|  |  |  | 		// generate md5sum for the generated data.
 | 
					
						
							|  |  |  | 		// md5sum of the data to written is required as input for PutObject.
 | 
					
						
							|  |  |  | 		// PutObject is the functions which writes the data onto the FS/XL backend.
 | 
					
						
							|  |  |  | 		metadata := make(map[string]string) | 
					
						
							| 
									
										
										
										
											2016-11-22 05:51:05 +08:00
										 |  |  | 		metadata["md5Sum"] = getMD5Hash([]byte(textData)) | 
					
						
							| 
									
										
										
										
											2016-10-03 06:51:49 +08:00
										 |  |  | 		sha256sum := "" | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 		// insert the object.
 | 
					
						
							| 
									
										
										
										
											2016-09-03 03:18:35 +08:00
										 |  |  | 		var objInfo ObjectInfo | 
					
						
							| 
									
										
										
										
											2016-10-03 06:51:49 +08:00
										 |  |  | 		objInfo, err = obj.PutObject(bucket, "object"+strconv.Itoa(i), int64(len(textData)), bytes.NewBuffer(textData), metadata, sha256sum) | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			b.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-09-03 03:18:35 +08:00
										 |  |  | 		if objInfo.MD5Sum != metadata["md5Sum"] { | 
					
						
							|  |  |  | 			b.Fatalf("Write no: %d: Md5Sum mismatch during object write into the bucket: Expected %s, got %s", i+1, objInfo.MD5Sum, metadata["md5Sum"]) | 
					
						
							| 
									
										
										
										
											2016-07-11 02:08:45 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// benchmark utility which helps obtain number of allocations and bytes allocated per ops.
 | 
					
						
							|  |  |  | 	b.ReportAllocs() | 
					
						
							|  |  |  | 	// the actual benchmark for GetObject starts here. Reset the benchmark timer.
 | 
					
						
							|  |  |  | 	b.ResetTimer() | 
					
						
							|  |  |  | 	b.RunParallel(func(pb *testing.PB) { | 
					
						
							|  |  |  | 		i := 0 | 
					
						
							|  |  |  | 		for pb.Next() { | 
					
						
							|  |  |  | 			err = obj.GetObject(bucket, "object"+strconv.Itoa(i), 0, int64(objSize), ioutil.Discard) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				b.Error(err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			i++ | 
					
						
							|  |  |  | 			if i == 10 { | 
					
						
							|  |  |  | 				i = 0 | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	// Benchmark ends here. Stop timer.
 | 
					
						
							|  |  |  | 	b.StopTimer() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |