mirror of
https://github.com/ceph/ceph-csi.git
synced 2025-01-09 21:39:29 +00:00
134 lines
4.0 KiB
Go
134 lines
4.0 KiB
Go
|
/*
|
||
|
*
|
||
|
* Copyright 2017 gRPC authors.
|
||
|
*
|
||
|
* 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.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
To format the benchmark result:
|
||
|
go run benchmark/benchresult/main.go resultfile
|
||
|
|
||
|
To see the performance change based on a old result:
|
||
|
go run benchmark/benchresult/main.go resultfile_old resultfile
|
||
|
It will print the comparison result of intersection benchmarks between two files.
|
||
|
|
||
|
*/
|
||
|
package main
|
||
|
|
||
|
import (
|
||
|
"encoding/gob"
|
||
|
"fmt"
|
||
|
"log"
|
||
|
"os"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"time"
|
||
|
|
||
|
"google.golang.org/grpc/benchmark/stats"
|
||
|
)
|
||
|
|
||
|
func createMap(fileName string, m map[string]stats.BenchResults) {
|
||
|
f, err := os.Open(fileName)
|
||
|
if err != nil {
|
||
|
log.Fatalf("Read file %s error: %s\n", fileName, err)
|
||
|
}
|
||
|
defer f.Close()
|
||
|
var data []stats.BenchResults
|
||
|
decoder := gob.NewDecoder(f)
|
||
|
if err = decoder.Decode(&data); err != nil {
|
||
|
log.Fatalf("Decode file %s error: %s\n", fileName, err)
|
||
|
}
|
||
|
for _, d := range data {
|
||
|
m[d.RunMode+"-"+d.Features.String()] = d
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func intChange(title string, val1, val2 int64) string {
|
||
|
return fmt.Sprintf("%10s %12s %12s %8.2f%%\n", title, strconv.FormatInt(val1, 10),
|
||
|
strconv.FormatInt(val2, 10), float64(val2-val1)*100/float64(val1))
|
||
|
}
|
||
|
|
||
|
func timeChange(title int, val1, val2 time.Duration) string {
|
||
|
return fmt.Sprintf("%10s %12s %12s %8.2f%%\n", strconv.Itoa(title)+" latency", val1.String(),
|
||
|
val2.String(), float64(val2-val1)*100/float64(val1))
|
||
|
}
|
||
|
|
||
|
func compareTwoMap(m1, m2 map[string]stats.BenchResults) {
|
||
|
for k2, v2 := range m2 {
|
||
|
if v1, ok := m1[k2]; ok {
|
||
|
changes := k2 + "\n"
|
||
|
changes += fmt.Sprintf("%10s %12s %12s %8s\n", "Title", "Before", "After", "Percentage")
|
||
|
changes += intChange("Bytes/op", v1.AllocedBytesPerOp, v2.AllocedBytesPerOp)
|
||
|
changes += intChange("Allocs/op", v1.AllocsPerOp, v2.AllocsPerOp)
|
||
|
changes += timeChange(v1.Latency[1].Percent, v1.Latency[1].Value, v2.Latency[1].Value)
|
||
|
changes += timeChange(v1.Latency[2].Percent, v1.Latency[2].Value, v2.Latency[2].Value)
|
||
|
fmt.Printf("%s\n", changes)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func compareBenchmark(file1, file2 string) {
|
||
|
var BenchValueFile1 map[string]stats.BenchResults
|
||
|
var BenchValueFile2 map[string]stats.BenchResults
|
||
|
BenchValueFile1 = make(map[string]stats.BenchResults)
|
||
|
BenchValueFile2 = make(map[string]stats.BenchResults)
|
||
|
|
||
|
createMap(file1, BenchValueFile1)
|
||
|
createMap(file2, BenchValueFile2)
|
||
|
|
||
|
compareTwoMap(BenchValueFile1, BenchValueFile2)
|
||
|
}
|
||
|
|
||
|
func printline(benchName, ltc50, ltc90, allocByte, allocsOp interface{}) {
|
||
|
fmt.Printf("%-80v%12v%12v%12v%12v\n", benchName, ltc50, ltc90, allocByte, allocsOp)
|
||
|
}
|
||
|
|
||
|
func formatBenchmark(fileName string) {
|
||
|
f, err := os.Open(fileName)
|
||
|
if err != nil {
|
||
|
log.Fatalf("Read file %s error: %s\n", fileName, err)
|
||
|
}
|
||
|
defer f.Close()
|
||
|
var data []stats.BenchResults
|
||
|
decoder := gob.NewDecoder(f)
|
||
|
if err = decoder.Decode(&data); err != nil {
|
||
|
log.Fatalf("Decode file %s error: %s\n", fileName, err)
|
||
|
}
|
||
|
if len(data) == 0 {
|
||
|
log.Fatalf("No data in file %s\n", fileName)
|
||
|
}
|
||
|
printPos := data[0].SharedPosion
|
||
|
fmt.Println("\nShared features:\n" + strings.Repeat("-", 20))
|
||
|
fmt.Print(stats.PartialPrintString(printPos, data[0].Features, true))
|
||
|
fmt.Println(strings.Repeat("-", 35))
|
||
|
for i := 0; i < len(data[0].SharedPosion); i++ {
|
||
|
printPos[i] = !printPos[i]
|
||
|
}
|
||
|
printline("Name", "latency-50", "latency-90", "Alloc (B)", "Alloc (#)")
|
||
|
for _, d := range data {
|
||
|
name := d.RunMode + stats.PartialPrintString(printPos, d.Features, false)
|
||
|
printline(name, d.Latency[1].Value.String(), d.Latency[2].Value.String(),
|
||
|
d.AllocedBytesPerOp, d.AllocsPerOp)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func main() {
|
||
|
if len(os.Args) == 2 {
|
||
|
formatBenchmark(os.Args[1])
|
||
|
} else {
|
||
|
compareBenchmark(os.Args[1], os.Args[2])
|
||
|
}
|
||
|
}
|