From 71407f8a938f18b2c9d5a282173c47ba462e8a1c Mon Sep 17 00:00:00 2001 From: Mark Reed Date: Mon, 13 May 2024 03:04:14 +0000 Subject: [PATCH] Add go benches in examples --- examples/bench.go | 126 ++++++++++++++ examples/lua/json-echo.lua | 18 ++ examples/lua/json.lua | 18 ++ examples/test.go | 329 +++++++++++++++++++++++++++++++++++++ 4 files changed, 491 insertions(+) create mode 100644 examples/bench.go create mode 100755 examples/lua/json-echo.lua create mode 100755 examples/lua/json.lua create mode 100644 examples/test.go diff --git a/examples/bench.go b/examples/bench.go new file mode 100644 index 0000000..2e7157d --- /dev/null +++ b/examples/bench.go @@ -0,0 +1,126 @@ +package main + +import ( + "errors" + "fmt" + "io" + "math/rand" + "net" + "os" + "time" + "bytes" + "flag" + "strconv" +) + +var( + limitSeconds int + numConnections int + hostname string + port int + batch int + html bool + req string + buffer bytes.Buffer +) + +func load_buffer() { + for i := 0; i < batch; i++ { + a := rand.Intn(1000) + b := rand.Intn(1000) + if ( batch > 1 ) { buffer.WriteString(fmt.Sprintf(`[`)) } + if i < batch-1 { + buffer.WriteString(fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":%d,"session_id":%d},"id":%d},`, a, b, i)) + } else { + buffer.WriteString(fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":%d,"session_id":%d},"id":%d}`, a, b, i)) + } + } + if ( batch > 1 ) { buffer.WriteString(fmt.Sprintf(`]`)) } + //fmt.Printf("%s\n",buffer.String()) +} + +func client(c chan int, tcpAddr *net.TCPAddr, tid int ) { + reply := make([]byte, 4096) + transmits := 0 + conn, err := net.DialTCP("tcp", nil, tcpAddr) + if err != nil { + println("Dial failed:", err.Error()) + os.Exit(1) + } + + start := time.Now() + for { + _, err = conn.Write(buffer.Bytes()) + if err != nil { + fmt.Printf("Write Error: %v\n", err) + break + } + + _, err := conn.Read(reply) + //fmt.Printf("Reply\n%s",reply[:l]) + if err != nil && !errors.Is(err, io.EOF) { + break + } + if time.Since(start).Seconds() >= float64(limitSeconds) { + break + } + transmits++ + } + conn.Close() + c <- transmits +} + +func formatInt(number int64) string { + output := strconv.FormatInt(number, 10) + startOffset := 3 + if number < 0 { + startOffset++ + } + for outputIndex := len(output); outputIndex > startOffset; { + outputIndex -= 3 + output = output[:outputIndex] + "," + output[outputIndex:] + } + return output +} + + +func main() { + + flag.StringVar(&hostname, "h", "localhost", "hostname") + flag.IntVar(&port, "p", 8545, "port") + flag.IntVar(&numConnections, "c", 16, "Number of connections") + flag.IntVar(&limitSeconds, "s", 2, "Stop after n seconds") + flag.IntVar(&batch, "b", 1, "Batch n requests together") + flag.BoolVar(&html, "html", false, "Send an html request instead of jsonrpc") + flag.Parse() + + fmt.Printf("DELME before connecting") + servAddr := fmt.Sprintf(`%s:%d`,hostname,port) + tcpAddr, err := net.ResolveTCPAddr("tcp", servAddr) + if err != nil { + println("ResolveTCPAddr failed:", err.Error()) + os.Exit(1) + } + + load_buffer(); + + fmt.Printf("Benchmarking jsonrpc for %d seconds with %d connections and a batch size of %d \n", limitSeconds, numConnections, batch); + + start := time.Now() + c := make(chan int) + for i := 0; i < numConnections; i++ { + go client( c, tcpAddr, i ) + } + + // Wait for all connections to finish + transmits := 0 + for i := 0; i < numConnections; i++ { + transmits += <-c + } + + elapsed := time.Since(start) + latency := float64(elapsed.Microseconds()) / float64(transmits) + speed := int64((float64(transmits) / float64(elapsed.Seconds())) * float64(batch)) + fmt.Printf(" %s commands/second, mean latency %.1fu\n", formatInt(speed), latency) + +} diff --git a/examples/lua/json-echo.lua b/examples/lua/json-echo.lua new file mode 100755 index 0000000..b02a48a --- /dev/null +++ b/examples/lua/json-echo.lua @@ -0,0 +1,18 @@ +-- example script demonstrating HTTP pipelining + +init = function(args) + + depth = tonumber(args[1]) or 1 + wrk.headers["Content-Type"] = "application/json" + + local r = {} + for i=1,depth do + r[i] = wrk.format('POST','/', nil, '{"jsonrpc":"2.0","method":"echo","params":{"data":"echomesomedata"},"id":0}') + end + req = table.concat(r) + +end + +request = function() + return req +end diff --git a/examples/lua/json.lua b/examples/lua/json.lua new file mode 100755 index 0000000..055a070 --- /dev/null +++ b/examples/lua/json.lua @@ -0,0 +1,18 @@ +-- example script demonstrating HTTP pipelining + +init = function(args) + + depth = tonumber(args[1]) or 1 + wrk.headers["Content-Type"] = "application/json" + + local r = {} + for i=1,depth do + r[i] = wrk.format('POST','/json', nil, '{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":55,"session_id":21},"id":0}') + end + req = table.concat(r) + +end + +request = function() + return req +end diff --git a/examples/test.go b/examples/test.go new file mode 100644 index 0000000..e1499bd --- /dev/null +++ b/examples/test.go @@ -0,0 +1,329 @@ +package main + +import ( + "errors" + "fmt" + "io" + "math/rand" + "net" + "os" + "time" + "bytes" + "flag" + "strings" +) + +var( + limitSeconds int + numConnections int + hostname string + port int + batch int + html bool + req string + buffer bytes.Buffer +) + +func load_buffer( n int) { + for i := 0; i < n; i++ { + a := rand.Intn(1000) + b := rand.Intn(1000) + if ( n > 1 ) { buffer.WriteString(fmt.Sprintf(`[`)) } + if i < n-1 { + buffer.WriteString(fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":%d,"session_id":%d},"id":%d},`, a, b, i)) + } else { + buffer.WriteString(fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":%d,"session_id":%d},"id":%d}`, a, b, i)) + } + } + if ( n > 1 ) { buffer.WriteString(fmt.Sprintf(`]`)) } + //fmt.Printf("%s\n",buffer.String()) +} + +func test_rpc(tcpAddr *net.TCPAddr ) { + print(" Test rpc ... ") + conn, err := net.DialTCP("tcp", nil, tcpAddr) + if err != nil { + println("connection failed:", err.Error()) + return + } + reply := make([]byte, 4096) + + req = fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":46,"session_id":0},"id":0}`) + _, err = conn.Write([]byte(req)) + if err != nil { + println("write error: %v\n", err) + return + } + n, err := conn.Read(reply) + if err != nil && !errors.Is(err, io.EOF) { + println("read error: %v\n", err) + return + } + rep := fmt.Sprintf(`{"jsonrpc":"2.0","id":0,"result":true}`) + if rep != string(reply[:n]) { + println("unexpected reply") + println(" exp: ", rep) + println(" act: ", string(reply[:n])) + return + } + + println("successful") +} +func test_html(tcpAddr *net.TCPAddr ) { + print(" Test html ... ") + conn, err := net.DialTCP("tcp", nil, tcpAddr) + if err != nil { + println("connection failed:", err.Error()) + return + } + reply := make([]byte, 4096) + + jrpc := fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":46,"session_id":0},"id":0}`) + req := fmt.Sprintf("POST / HTTP/1.1\r\nHost: localhost:8558\r\nUser-Agent: python-requests/2.31.0\r\nAccept-Encoding: gzip, deflate\r\nAccept: */*\r\nConnection: keep-alive\r\nContent-Length: %d\r\nContent-Type: application/json\r\n\r\n%s", len(jrpc), jrpc) + + _, err = conn.Write([]byte(req)) + if err != nil { + println("write error: %v\n", err) + return + } + n, err := conn.Read(reply) + if err != nil && !errors.Is(err, io.EOF) { + println("read error: %v\n", err) + return + } + rep := fmt.Sprintf("HTTP/1.1 200 OK\r\nContent-Length: 38 \r\nContent-Type: application/json\r\n\r\n{\"jsonrpc\":\"2.0\",\"id\":0,\"result\":true}") + + if rep != string(reply[:n]) { + println("unexpected reply") + println(" exp: ", rep) + println(" act: ", string(reply[:n])) + return + } + + println("successful") +} +func test_big(tcpAddr *net.TCPAddr ) { + print(" Test 4097 byte json ... ") + conn, err := net.DialTCP("tcp", nil, tcpAddr) + if err != nil { + println("connection failed:", err.Error()) + return + } + reply := make([]byte, 4096) + pad := bytes.Repeat([]byte("a"), 3992) + req = fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":46,"session_id":0},"id":0, "padding":"%s"}`,pad) + _, err = conn.Write([]byte(req)) + if err != nil { + println("write error: %v\n", err) + return + } + n, err := conn.Read(reply) + if err != nil && !errors.Is(err, io.EOF) { + println("read error: %v\n", err) + return + } + rep := `{"jsonrpc":"2.0","id":0,"result":true}` + if strings.Compare( rep, string(reply[:n]) ) != 0 { + println("unexpected reply") + println(" exp: ", rep) + println(" act: ", string(reply[:n])) + return + } + + println("successful") +} +func test_partial(tcpAddr *net.TCPAddr ) { + print(" Test partial ... ") + conn, err := net.DialTCP("tcp", nil, tcpAddr) + if err != nil { + println("connection failed:", err.Error()) + return + } + reply := make([]byte, 4096) + + req = fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":46,"session_id"`) + _, err = conn.Write([]byte(req)) + if err != nil { + println("write error: %v\n", err) + return + } + time.Sleep(1000 * time.Millisecond) + req = fmt.Sprintf(`:0},"id":0}`) + _, err = conn.Write([]byte(req)) + if err != nil { + println("write second part error: %v\n", err) + return + } + n, err := conn.Read(reply) + if err != nil && !errors.Is(err, io.EOF) { + println("read error: %v\n", err) + return + } + rep := fmt.Sprintf(`{"jsonrpc":"2.0","id":0,"result":true}`) + if strings.Compare( rep, string(reply[:n]) ) != 0 { + println("unexpected reply") + println(" exp: ", rep) + println(" act: ", string(reply[:n])) + return + } + + println("successful") +} + + +func client(c chan int, tcpAddr *net.TCPAddr, tid int ) { + reply := make([]byte, 4096) + + start := time.Now() + transmits := 0 + conn, err := net.DialTCP("tcp", nil, tcpAddr) + if err != nil { + println("Dial failed:", err.Error()) + os.Exit(1) + } + + for { + _, err = conn.Write(buffer.Bytes()) + if err != nil { + fmt.Printf("Write Error: %v\n", err) + break + } + + //conn.SetReadDeadline(time.Now().Add(time.Second*5)) + _, err := conn.Read(reply) + //fmt.Printf("Reply\n%s",reply[:l]) + if err != nil && !errors.Is(err, io.EOF) { + break + } + if time.Since(start).Seconds() >= float64(limitSeconds) { + break + } + transmits++ + } + conn.Close() + c <- transmits +} +func test_echo(tcpAddr *net.TCPAddr ) { + print(" Test rpc ... ") + conn, err := net.DialTCP("tcp", nil, tcpAddr) + if err != nil { + println("connection failed:", err.Error()) + return + } + reply := make([]byte, 4096) + + req := fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":46,"session_id":0},"id":0}`) + _, err = conn.Write([]byte(req)) + if err != nil { + println("write error: %v\n", err) + return + } + n, err := conn.Read(reply) + if err != nil && !errors.Is(err, io.EOF) { + println("read error: %v\n", err) + return + } + rep := fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":46,"session_id":0},"id":0}`) + if rep != string(reply[:n]) { + println("unexpected reply") + println(" exp: ", rep) + println(" act: ", string(reply[:n])) + return + } + + println("successful") +} + +func test_generic(name string, tcpAddr *net.TCPAddr, req []byte, reply []byte ) { + print(" Test ",name," ... ") + conn, err := net.DialTCP("tcp", nil, tcpAddr) + if err != nil { + println("connection failed:", err.Error()) + return + } + rbuf := make([]byte, 4096) + + _, err = conn.Write([]byte(req)) + if err != nil { + println("write error: %v\n", err) + return + } + n, err := conn.Read(rbuf) + if err != nil && !errors.Is(err, io.EOF) { + println("read error: %v\n", err) + return + } + if string(reply) != string(rbuf[:n]) { + println("unexpected reply") + println(" exp: ", string(reply)) + println(" act: ", string(rbuf[:n])) + return + } + + println("successful") +} + + +func main() { + + flag.StringVar(&hostname, "h", "localhost", "hostname") + flag.IntVar(&port, "p", 8545, "port") + flag.IntVar(&numConnections, "c", 16, "Number of connections") + flag.IntVar(&limitSeconds, "s", 2, "Stop after n seconds") + flag.IntVar(&batch, "b", 1, "Batch n requests together") + flag.BoolVar(&html, "html", false, "Send an html request instead of jsonrpc") + flag.Parse() + + //var b [16]bytes.Buffer + //b[0].WriteString("foo") + //b[1].WriteString("bar") + //fmt.Println(b) + + servAddr := fmt.Sprintf(`%s:%d`,hostname,port) + tcpAddr, err := net.ResolveTCPAddr("tcp", servAddr) + if err != nil { + println("ResolveTCPAddr failed:", err.Error()) + os.Exit(1) + } + + //test_rpc(tcpAddr) + //test_html(tcpAddr) + //test_big(tcpAddr) + //test_partial(tcpAddr) + + + req := fmt.Sprintf(`{"jsonrpc":"2.0","method":"validate_session","params":{"user_id":46,"session_id":0},"id":0}`) + rep := fmt.Sprintf(`{"jsonrpc":"2.0","id":0,"result":true}`) + test_generic("validate_session", tcpAddr, []byte(req), []byte(rep) ) + + req = fmt.Sprintf(`{"jsonrpc":"2.0","method":"echo","params":{"data":"session_id"},"id":0}`) + rep = fmt.Sprintf(`{"jsonrpc":"2.0","id":0,"result":"session_id"}`) + test_generic("echo", tcpAddr, []byte(req), []byte(rep) ) + + req = fmt.Sprintf(`{"jsonrpc":"2.0","method":"create_user","params":{"age":46,"name":"My Name","bio":"My bio","avatar":"fdasfsadbfasdfasdwefdsahfsds"},"id":0}`) + rep = fmt.Sprintf(`{"jsonrpc":"2.0","method":"create_user","params":{"age":46,"name":"My Name","bio":"My bio","avatar":"fdasfsadbfasdfasdwefdsahfsds"},"id":0}`) + test_generic("create_user", tcpAddr, []byte(req), []byte(rep) ) + + req = fmt.Sprintf(`{"jsonrpc":"2.0","method":"set","params":{"key":"test","value":"val"},"id":0}`) + rep = fmt.Sprintf(`{"jsonrpc":"2.0","id":0,"result":"OK"}`) + test_generic("create_user", tcpAddr, []byte(req), []byte(rep) ) + req = fmt.Sprintf(`{"jsonrpc":"2.0","method":"get","params":{"key":"test","value":"val"},"id":0}`) + rep = fmt.Sprintf(`{"jsonrpc":"2.0","id":0,"result":"val"}`) + test_generic("create_user", tcpAddr, []byte(req), []byte(rep) ) + + //load_buffer(batch) + + + //start := time.Now() + //c := make(chan int) + ////for i := 0; i < numConnections; i++ { + //go client( c, tcpAddr, i ) + //} + + // Wait for all connections to finish + //transmits := 0 + //for i := 0; i < numConnections; i++ { + //transmits += <-c + //} + +}