Skip to content

Commit e129211

Browse files
authored
Merge pull request #4 from tombokombo/master
rework
2 parents 8e2c9ba + 74e114b commit e129211

File tree

4 files changed

+138
-63
lines changed

4 files changed

+138
-63
lines changed

Dockerfile

+7-6
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,13 @@
1-
FROM golang:1.9 AS builder
1+
FROM golang:1.10-alpine AS builder
22

33
RUN mkdir /app
4-
ADD . /app/
4+
ADD ./main.go /app/
55
WORKDIR /app
6-
RUN go get github.com/go-sql-driver/mysql
6+
RUN apk update && apk add git && go get github.com/go-sql-driver/mysql && go get github.com/namsral/flag
77
RUN go build -o main .
88

99
FROM alpine:3.10
10-
11-
COPY --from=builder /app/main /usr/bin/mysql-load-generator
12-
CMD ["/usr/bin/mysql-load-generator"]
10+
RUN mkdir /app
11+
COPY --from=builder /app/main /app/mysql-loader
12+
COPY config.json /app/config.json
13+
CMD ["/app/mysql-loader"]

Readme.md

+8-7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
# MemSQL / MySQL Load Tester
22

33
This is a tool for generating load tests based on custom queries for MemSQL / MySQL databases.
4+
This fork brings substitution, query timeouts, wait group timeout, connection pool, periodic qps and latency logs, higher performance
45

56
## How does it work?
67

@@ -11,13 +12,13 @@ Json configuration guide:
1112

1213
* connectionString (string) - the connection string to the MySQL / MemSQL Database
1314
* requestsPerSecond (int) - how many statements per second to execute
14-
* printLogs (bool) - print queries executed including time taken for each query and a total average
15+
* printLogs (bool) - print queries executed
1516
* timeToRun (int) - how long to run the test in seconds
16-
* queries (string array) - the queries to run
17+
* queries (string array) - the queries to run, can use substitution placeholder
18+
* substitution ( map ) - specify placeholder for substitution with random number according specified min,max (int) rules
1719

1820
## Run as Docker
19-
20-
Simply build using the Dockerfile.
21-
Load tests are a great fit for [Azure Container Instances].
22-
23-
[Azure Container Instances]: https://azure.microsoft.com/en-us/services/container-instances/
21+
Prepare config.json and run
22+
```
23+
docker run --network=host -v $(pwd)/config.json:/app/config.json -it --rm tombokombo/mysql-loader:latest
24+
```

config.json

+8-4
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,13 @@
11
{
22
"connectionString": "USER:PASSWORD@tcp(HOST:PORT)/DB",
3-
"requestsPerSecond": 1,
3+
"requestsPerSecond": 4500,
44
"printLogs": true,
5-
"timeToRun": 3,
5+
"timeToRunInSeconds": 1200,
6+
"poolConnections": 1000,
7+
"dryRun" : false,
8+
"queryTimeout": 120,
69
"queries" : [
7-
"SELECT * from Table"
8-
]
10+
"SELECT * FROM test LIMIT WHERE id=:rand: LIMIT 1"
11+
],
12+
"substitution" : [ {"key": ":rand:", "min": 1 ,"max": 3182356 }, {"key": ":rand_tiny:", "min": 1 ,"max": 10 } ]
913
}

main.go

+115-46
Original file line numberDiff line numberDiff line change
@@ -1,42 +1,105 @@
11
package main
22

33
import (
4+
"context"
45
"database/sql"
56
"encoding/json"
67
"fmt"
78
"io/ioutil"
89
"math/rand"
10+
"os"
911
"strconv"
12+
"strings"
1013
"sync"
14+
"sync/atomic"
1115
"time"
1216

17+
"github.com/namsral/flag"
18+
1319
_ "github.com/go-sql-driver/mysql"
1420
)
1521

22+
type Substitution struct {
23+
Key string `json:"key"`
24+
Min int `json:"min"`
25+
Max int `json:"max"`
26+
}
27+
1628
type Config struct {
17-
RequestsPerSecond int `json:"requestsPerSecond"`
18-
Queries []string `json:"queries"`
19-
ConnectionString string `json:"connectionString"`
20-
PrintLogs bool `json:"printLogs"`
21-
TimeToRunInSeconds int `json:"timeToRunInSeconds"`
29+
RequestsPerSecond int `json:"requestsPerSecond"`
30+
Queries []string `json:"queries"`
31+
ConnectionString string `json:"connectionString"`
32+
PrintLogs bool `json:"printLogs"`
33+
TimeToRunInSeconds int `json:"timeToRunInSeconds"`
34+
PoolConnections int `json:"poolConnections"`
35+
DryRun bool `json:"dryRun"`
36+
QueryTimeout int `json:"queryTimeout"`
37+
ConnectionLifetime time.Duration `json:"connectionLifeTime"`
38+
Substitution []Substitution `json:"substitution"`
2239
}
2340

24-
var config Config
41+
var config = Config{
42+
RequestsPerSecond: 100,
43+
Queries: make([]string, 0),
44+
ConnectionString: "",
45+
PrintLogs: true,
46+
TimeToRunInSeconds: 10,
47+
PoolConnections: 10,
48+
DryRun: true,
49+
QueryTimeout: 10,
50+
ConnectionLifetime: 0,
51+
Substitution: []Substitution{},
52+
}
53+
var configFilePath string
2554
var db *sql.DB
2655
var queryTimesInMS []int
56+
var ops int64 = 0
57+
var latency int64 = 0
2758

2859
var wg sync.WaitGroup
2960

3061
func main() {
31-
loadConfig()
62+
flag.StringVar(&configFilePath, "configFile", "/app/config.json", "path to config file")
63+
flag.Parse()
64+
go printQps()
65+
loadConfig(configFilePath)
3266
openMemSQLConnection()
3367
dispatchQueries()
34-
displayAverageQueryTime()
68+
}
69+
70+
func printQps() {
71+
ticker := time.NewTicker(5 * time.Second)
72+
for range ticker.C {
73+
loadedOps := atomic.LoadInt64(&ops)
74+
atomic.StoreInt64(&ops, 0)
75+
loadedLatency := atomic.LoadInt64(&latency)
76+
atomic.StoreInt64(&latency, 0)
77+
fmt.Println("qps: ", loadedOps/5, ", average latency: ", time.Duration(loadedLatency/loadedOps))
78+
79+
}
80+
81+
}
82+
83+
func waitTimeout(wg *sync.WaitGroup, timeout time.Duration) bool {
84+
c := make(chan struct{})
85+
go func() {
86+
defer close(c)
87+
wg.Wait()
88+
}()
89+
select {
90+
case <-c:
91+
return false // completed normally
92+
case <-time.After(timeout):
93+
return true // timed out
94+
}
3595
}
3696

3797
func openMemSQLConnection() {
3898
var err error
3999
db, err = sql.Open("mysql", config.ConnectionString)
100+
db.SetConnMaxLifetime(config.ConnectionLifetime)
101+
db.SetMaxIdleConns(config.PoolConnections)
102+
db.SetMaxOpenConns(config.PoolConnections)
40103

41104
if err != nil {
42105
panic(err.Error())
@@ -48,45 +111,30 @@ func openMemSQLConnection() {
48111
panic(err.Error())
49112
}
50113

51-
fmt.Println("Connection succeeded")
52-
}
53-
54-
func trackFuncTime(start time.Time) {
55-
elapsed := time.Since(start)
56-
queryTimesInMS = append(queryTimesInMS, int(elapsed/time.Millisecond))
57-
58-
if config.PrintLogs {
59-
fmt.Println(elapsed)
60-
}
114+
fmt.Println("Connection succeeded with pool size", config.PoolConnections)
61115
}
62116

63117
func executeQuery(query string) {
64118
defer wg.Done()
65-
defer trackFuncTime(time.Now())
119+
ctx, cancel := context.WithTimeout(context.Background(), time.Duration(config.QueryTimeout)*time.Second)
120+
defer cancel()
66121

67-
_, err := db.Exec(query)
68-
69-
if err != nil {
70-
fmt.Println(err.Error())
122+
for _, subs := range config.Substitution {
123+
query = strings.Replace(query, subs.Key, strconv.Itoa(rand.Intn(subs.Max-subs.Min)+subs.Min), 1)
71124
}
72125

73126
if config.PrintLogs {
74-
fmt.Println("Executed " + query)
127+
fmt.Println("Executed ", query)
75128
}
76-
}
77-
78-
func displayAverageQueryTime() {
79-
if config.PrintLogs {
80-
var totalvalues int
81-
82-
for _, val := range queryTimesInMS {
83-
totalvalues += val
129+
if config.DryRun == false {
130+
start := time.Now()
131+
_, err := db.ExecContext(ctx, query)
132+
if err == nil {
133+
atomic.AddInt64(&ops, 1)
134+
atomic.AddInt64(&latency, int64(time.Since(start)))
135+
} else {
136+
fmt.Fprintf(os.Stderr, "query err: %s\n", err.Error())
84137
}
85-
86-
var count = len(queryTimesInMS)
87-
sum := totalvalues / count
88-
89-
fmt.Println("Average time: " + strconv.Itoa(sum) + "ms")
90138
}
91139
}
92140

@@ -97,23 +145,44 @@ func dispatchQueries() {
97145
shouldEnd := start.Add(time.Second * time.Duration(config.TimeToRunInSeconds))
98146

99147
for time.Now().Before(shouldEnd) {
100-
for i := 0; i < config.RequestsPerSecond; i++ {
101-
wg.Add(1)
102-
go executeQuery(config.Queries[rand.Intn(len(config.Queries))])
148+
l := len(config.Queries)
149+
for i := 0; i < config.RequestsPerSecond/l; i++ {
150+
wg.Add(l)
151+
for q := 0; q < l; q++ {
152+
go executeQuery(config.Queries[q])
153+
}
103154
}
104155

105156
time.Sleep(1 * time.Second)
106157
}
107-
108-
wg.Wait()
158+
fmt.Println("wait")
159+
if waitTimeout(&wg, 10*time.Second) {
160+
fmt.Println("Timed out waiting for wait group")
161+
} else {
162+
fmt.Println("Wait group finished")
163+
}
164+
fmt.Println("end wait")
165+
defer db.Close()
109166
}
110167

111-
func loadConfig() {
112-
data, err := ioutil.ReadFile("config.json")
168+
func loadConfig(configFilePath string) {
169+
data, err := ioutil.ReadFile(configFilePath)
170+
if err != nil {
171+
panic(err.Error() + " use -configFile flag")
172+
}
173+
174+
err = json.Unmarshal(data, &config)
113175
if err != nil {
114176
panic(err)
115177
}
178+
fmt.Println("LOADED CONFIG")
179+
pretty, err := json.MarshalIndent(config, "", " ")
180+
if err != nil {
181+
panic(err)
182+
}
183+
fmt.Printf("%s\n", string(pretty))
116184

117-
json.Unmarshal(data, &config)
118-
fmt.Println("Config loaded")
185+
if len(config.Queries) < 1 {
186+
panic("no query specified")
187+
}
119188
}

0 commit comments

Comments
 (0)