2018-08-07 11:43:51 +02:00
|
|
|
package router
|
|
|
|
|
|
|
|
import (
|
2018-08-07 12:05:32 +02:00
|
|
|
"bytes"
|
2018-08-07 11:43:51 +02:00
|
|
|
"math/rand"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2019-10-25 00:27:02 +02:00
|
|
|
"github.com/kataras/iris/v12/context"
|
2020-08-16 06:07:36 +02:00
|
|
|
|
|
|
|
"github.com/kataras/golog"
|
2018-08-07 11:43:51 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// randStringBytesMaskImprSrc helps us to generate random paths for the test,
|
|
|
|
// the below piece of code is external, as an answer to a stackoverflow question.
|
|
|
|
//
|
|
|
|
// START.
|
|
|
|
const letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|
|
|
const (
|
|
|
|
letterIdxBits = 6 // 6 bits to represent a letter index
|
|
|
|
letterIdxMask = 1<<letterIdxBits - 1 // All 1-bits, as many as letterIdxBits
|
|
|
|
letterIdxMax = 63 / letterIdxBits // # of letter indices fitting in 63 bits
|
|
|
|
)
|
|
|
|
|
|
|
|
var src = rand.NewSource(time.Now().UnixNano())
|
|
|
|
|
|
|
|
func randStringBytesMaskImprSrc(n int) string {
|
|
|
|
b := make([]byte, n)
|
|
|
|
// A src.Int63() generates 63 random bits, enough for letterIdxMax characters!
|
|
|
|
for i, cache, remain := n-1, src.Int63(), letterIdxMax; i >= 0; {
|
|
|
|
if remain == 0 {
|
|
|
|
cache, remain = src.Int63(), letterIdxMax
|
|
|
|
}
|
|
|
|
if idx := int(cache & letterIdxMask); idx < len(letterBytes) {
|
|
|
|
b[i] = letterBytes[idx]
|
|
|
|
i--
|
|
|
|
}
|
|
|
|
cache >>= letterIdxBits
|
|
|
|
remain--
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.ToLower(string(b))
|
|
|
|
}
|
|
|
|
|
|
|
|
// END.
|
|
|
|
|
|
|
|
func genPaths(routesLength, minCharLength, maxCharLength int) []string {
|
2018-08-07 12:05:32 +02:00
|
|
|
// b := new(strings.Builder)
|
|
|
|
b := new(bytes.Buffer)
|
2018-08-07 11:43:51 +02:00
|
|
|
paths := make([]string, routesLength)
|
|
|
|
pathStart := '/'
|
|
|
|
for i := 0; i < routesLength; i++ {
|
|
|
|
pathSegmentCharsLength := rand.Intn(maxCharLength-minCharLength) + minCharLength
|
|
|
|
|
|
|
|
b.WriteRune(pathStart)
|
|
|
|
b.WriteString(randStringBytesMaskImprSrc(pathSegmentCharsLength))
|
|
|
|
b.WriteString("/{name:string}/") // sugar.
|
|
|
|
b.WriteString(randStringBytesMaskImprSrc(pathSegmentCharsLength))
|
|
|
|
b.WriteString("/{age:int}/end")
|
|
|
|
paths[i] = b.String()
|
|
|
|
|
|
|
|
b.Reset()
|
|
|
|
}
|
|
|
|
|
|
|
|
return paths
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build 1296(=144*9(the available http methods)) routes
|
|
|
|
// with up to 2*range(15-42)+ 2 named paths lowercase letters
|
|
|
|
// and 12 request handlers each.
|
|
|
|
//
|
|
|
|
// GOCACHE=off && go test -run=XXX -bench=BenchmarkAPIBuilder$ -benchtime=10s
|
|
|
|
func BenchmarkAPIBuilder(b *testing.B) {
|
2023-07-08 01:08:18 +02:00
|
|
|
rand.New(rand.NewSource(time.Now().Unix()))
|
2018-08-07 11:43:51 +02:00
|
|
|
|
2020-07-10 22:21:09 +02:00
|
|
|
noOpHandler := func(ctx *context.Context) {}
|
2018-08-07 11:43:51 +02:00
|
|
|
handlersPerRoute := make(context.Handlers, 12)
|
|
|
|
for i := 0; i < cap(handlersPerRoute); i++ {
|
|
|
|
handlersPerRoute[i] = noOpHandler
|
|
|
|
}
|
|
|
|
|
|
|
|
routesLength := 144
|
|
|
|
// i.e /gzhyweumidvelqewrvoyqmzopvuxli/{name:string}/bibrkratnrrhvsjwsxygfwmqwhcstc/{age:int}/end
|
|
|
|
paths := genPaths(routesLength, 15, 42)
|
|
|
|
|
2020-08-16 06:07:36 +02:00
|
|
|
api := NewAPIBuilder(golog.Default)
|
2020-04-30 15:16:43 +02:00
|
|
|
requestHandler := NewDefaultHandler(nil, nil)
|
2018-08-07 11:43:51 +02:00
|
|
|
|
|
|
|
b.ReportAllocs()
|
|
|
|
b.ResetTimer()
|
|
|
|
|
|
|
|
for i := 0; i < routesLength; i++ {
|
|
|
|
api.Any(paths[i], handlersPerRoute...)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := requestHandler.Build(api); err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
b.StopTimer()
|
|
|
|
|
|
|
|
b.Logf("%d routes have just builded\n", len(api.GetRoutes()))
|
|
|
|
}
|