forked from takanorig/mqtt-bench
-
Notifications
You must be signed in to change notification settings - Fork 1
/
mqtt-bench.go
495 lines (411 loc) · 14.4 KB
/
mqtt-bench.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
package main
import (
"bytes"
"crypto/tls"
"crypto/x509"
"flag"
"fmt"
MQTT "github.com/eclipse/paho.mqtt.golang"
"io/ioutil"
"os"
"strconv"
"strings"
"sync"
"time"
)
const BASE_TOPIC string = "/mqtt-bench/benchmark"
var Debug bool = false
var mutex = &sync.Mutex{}
// Stores results of subscribes
var DefaultHandlerResults []*SubscribeResult
// Execution options
type ExecOptions struct {
Broker string // Broker URI
Qos byte // QoS(0|1|2)
Retain bool // Retain
Topic string // Topicのルート
Username string // ユーザID
Password string // パスワード
CertConfig CertConfig // 認証定義
ClientNum int // クライアントの同時実行数
Count int // 1クライアント当たりのメッセージ数
MessageSize int // 1メッセージのサイズ(byte)
UseDefaultHandler bool // Subscriber個別ではなく、デフォルトのMessageHandlerを利用するかどうか
PreTime int // 実行前の待機時間(ms)
IntervalTime int // メッセージ毎の実行間隔時間(ms)
}
// Authentication settings
type CertConfig interface{}
// Server authentication settings
type ServerCertConfig struct {
CertConfig
ServerCertFile string // サーバ証明書ファイル
}
// クライアント認証設定
type ClientCertConfig struct {
CertConfig
RootCAFile string // ルート証明書ファイル
ClientCertFile string // クライアント証明書ファイル
ClientKeyFile string // クライアント公開鍵ファイル
}
// サーバ証明書用のTLS設定を生成する。
// serverCertFile : サーバ証明書のファイル
func CreateServerTlsConfig(serverCertFile string) *tls.Config {
certpool := x509.NewCertPool()
pem, err := ioutil.ReadFile(serverCertFile)
if err == nil {
certpool.AppendCertsFromPEM(pem)
}
return &tls.Config{
RootCAs: certpool,
}
}
// クライアント証明書用のTLS設定を生成する。
// rootCAFile : ルート証明書ファイル
// clientCertFile : クライアント証明書ファイル
// clientKeyFile : クライアント公開鍵ファイル
func CreateClientTlsConfig(rootCAFile string, clientCertFile string, clientKeyFile string) *tls.Config {
certpool := x509.NewCertPool()
rootCA, err := ioutil.ReadFile(rootCAFile)
if err == nil {
certpool.AppendCertsFromPEM(rootCA)
}
cert, err := tls.LoadX509KeyPair(clientCertFile, clientKeyFile)
if err != nil {
panic(err)
}
cert.Leaf, err = x509.ParseCertificate(cert.Certificate[0])
if err != nil {
panic(err)
}
return &tls.Config{
RootCAs: certpool,
ClientAuth: tls.NoClientCert,
ClientCAs: nil,
InsecureSkipVerify: true,
Certificates: []tls.Certificate{cert},
}
}
// 実行する。
func Execute(exec func(clients []MQTT.Client, opts ExecOptions, param ...string) int, opts ExecOptions) {
message := CreateFixedSizeMessage(opts.MessageSize)
// 配列を初期化
DefaultHandlerResults = make([]*SubscribeResult, opts.ClientNum)
clients := make([]MQTT.Client, opts.ClientNum)
hasErr := false
for i := 0; i < opts.ClientNum; i++ {
client := Connect(i, opts)
if client == nil {
hasErr = true
break
}
clients[i] = client
}
// 接続エラーがあれば、接続済みのクライアントの切断処理を行い、処理を終了する。
if hasErr {
for i := 0; i < len(clients); i++ {
client := clients[i]
if client != nil {
Disconnect(client)
}
}
return
}
// 安定させるために、一定時間待機する。
time.Sleep(time.Duration(opts.PreTime) * time.Millisecond)
fmt.Printf("%s Start benchmark\n", time.Now())
startTime := time.Now()
totalCount := exec(clients, opts, message)
endTime := time.Now()
fmt.Printf("%s End benchmark\n", time.Now())
// 処理結果を出力する。
duration := (endTime.Sub(startTime)).Nanoseconds() / int64(1000000) // nanosecond -> millisecond
throughput := float64(totalCount) / float64(duration) * 1000 // messages/sec
fmt.Printf("\nResult : broker=%s, clients=%d, totalCount=%d, duration=%dms, throughput=%.2fmessages/sec\n",
opts.Broker, opts.ClientNum, totalCount, duration, throughput)
// 切断に時間がかかるため、非同期で処理を行う。
AsyncDisconnect(clients)
}
// 全クライアントに対して、publishの処理を行う。
// 送信したメッセージ数を返す(原則、クライアント数分となる)。
func PublishAllClient(clients []MQTT.Client, opts ExecOptions, param ...string) int {
message := param[0]
wg := new(sync.WaitGroup)
totalCount := 0
for id := 0; id < len(clients); id++ {
wg.Add(1)
client := clients[id]
go func(clientId int) {
defer wg.Done()
for index := 0; index < opts.Count; index++ {
topic := fmt.Sprintf(opts.Topic+"/%d", clientId)
if Debug {
fmt.Printf("Publish : id=%d, count=%d, topic=%s\n", clientId, index, topic)
}
Publish(client, topic, opts.Qos, opts.Retain, message)
mutex.Lock()
totalCount++
mutex.Unlock()
if opts.IntervalTime > 0 {
time.Sleep(time.Duration(opts.IntervalTime) * time.Millisecond)
}
}
}(id)
}
wg.Wait()
return totalCount
}
// メッセージを送信する。
func Publish(client MQTT.Client, topic string, qos byte, retain bool, message string) {
token := client.Publish(topic, qos, retain, message)
if token.Wait() && token.Error() != nil {
fmt.Printf("Publish error: %s\n", token.Error())
}
}
// 全クライアントに対して、subscribeの処理を行う。
// 指定されたカウント数分、メッセージを受信待ちする(メッセージが取得できない場合はカウントされない)。
// この処理では、Publishし続けながら、Subscribeの処理を行う。
func SubscribeAllClient(clients []MQTT.Client, opts ExecOptions, param ...string) int {
wg := new(sync.WaitGroup)
results := make([]*SubscribeResult, len(clients))
for id := 0; id < len(clients); id++ {
wg.Add(1)
client := clients[id]
topic := fmt.Sprintf(opts.Topic+"/%d", id)
results[id] = Subscribe(client, topic, opts.Qos)
// DefaultHandlerを利用する場合は、Subscribe個別のHandlerではなく、
// DefaultHandlerの処理結果を参照する。
if opts.UseDefaultHandler == true {
results[id] = DefaultHandlerResults[id]
}
go func(clientId int) {
defer wg.Done()
var loop int = 0
for results[clientId].Count < opts.Count {
loop++
if Debug {
fmt.Printf("Subscribe : id=%d, count=%d, topic=%s\n", clientId, results[clientId].Count, topic)
}
if opts.IntervalTime > 0 {
time.Sleep(time.Duration(opts.IntervalTime) * time.Millisecond)
} else {
// for文による負荷を下げるため、最低でも1000ナノ秒(0.001ミリ秒)は待機する。
time.Sleep(1000 * time.Nanosecond)
}
// 無限ループを避けるため、指定されたCountの100倍に達したら、エラーで終了する。
if loop >= opts.Count*100 {
panic("Subscribe error : Not finished in the max count. It may not be received the message.")
}
}
}(id)
}
wg.Wait()
// 受信メッセージ数をカウント
totalCount := 0
for id := 0; id < len(results); id++ {
totalCount += results[id].Count
}
return totalCount
}
// Subscribeの処理結果
type SubscribeResult struct {
Count int // 受信メッセージ数
}
// メッセージを受信する。
func Subscribe(client MQTT.Client, topic string, qos byte) *SubscribeResult {
var result *SubscribeResult = &SubscribeResult{}
result.Count = 0
var handler MQTT.MessageHandler = func(client MQTT.Client, msg MQTT.Message) {
result.Count++
if Debug {
fmt.Printf("Received message : topic=%s, message=%s\n", msg.Topic(), msg.Payload())
}
}
token := client.Subscribe(topic, qos, handler)
if token.Wait() && token.Error() != nil {
fmt.Printf("Subscribe error: %s\n", token.Error())
}
return result
}
// 固定サイズのメッセージを生成する。
func CreateFixedSizeMessage(size int) string {
var buffer bytes.Buffer
for i := 0; i < size; i++ {
buffer.WriteString(strconv.Itoa(i % 10))
}
message := buffer.String()
return message
}
// 指定されたBrokerへ接続し、そのMQTTクライアントを返す。
// 接続に失敗した場合は nil を返す。
func Connect(id int, execOpts ExecOptions) MQTT.Client {
// 複数プロセスで、ClientIDが重複すると、Broker側で問題となるため、
// プロセスIDを利用して、IDを割り振る。
// mqttbench<プロセスIDの16進数値>-<クライアントの連番>
pid := strconv.FormatInt(int64(os.Getpid()), 16)
clientId := fmt.Sprintf("mqttbench%s-%d", pid, id)
opts := MQTT.NewClientOptions()
opts.AddBroker(execOpts.Broker)
opts.SetClientID(clientId)
if execOpts.Username != "" {
opts.SetUsername(execOpts.Username)
}
if execOpts.Password != "" {
opts.SetPassword(execOpts.Password)
}
// TLSの設定
certConfig := execOpts.CertConfig
switch c := certConfig.(type) {
case ServerCertConfig:
tlsConfig := CreateServerTlsConfig(c.ServerCertFile)
opts.SetTLSConfig(tlsConfig)
case ClientCertConfig:
tlsConfig := CreateClientTlsConfig(c.RootCAFile, c.ClientCertFile, c.ClientKeyFile)
opts.SetTLSConfig(tlsConfig)
default:
// do nothing.
}
if execOpts.UseDefaultHandler == true {
// Apollo(1.7.1利用)の場合、DefaultPublishHandlerを指定しないと、Subscribeできない。
// ただし、指定した場合でもretainされたメッセージは最初の1度しか取得されず、2回目以降のアクセスでは空になる点に注意。
var result *SubscribeResult = &SubscribeResult{}
result.Count = 0
var handler MQTT.MessageHandler = func(client MQTT.Client, msg MQTT.Message) {
result.Count++
if Debug {
fmt.Printf("Received at defaultHandler : topic=%s, message=%s\n", msg.Topic(), msg.Payload())
}
}
opts.SetDefaultPublishHandler(handler)
DefaultHandlerResults[id] = result
}
client := MQTT.NewClient(opts)
token := client.Connect()
if token.Wait() && token.Error() != nil {
fmt.Printf("Connected error: %s\n", token.Error())
return nil
}
return client
}
// 非同期でBrokerとの接続を切断する。
func AsyncDisconnect(clients []MQTT.Client) {
wg := new(sync.WaitGroup)
for _, client := range clients {
wg.Add(1)
go func() {
defer wg.Done()
Disconnect(client)
}()
}
wg.Wait()
}
// Brokerとの接続を切断する。
func Disconnect(client MQTT.Client) {
if client.IsConnected() { // make sure we don't run into a race condition (see https://github.com/eclipse/paho.mqtt.golang/issues/32)
client.Disconnect(10)
}
}
// ファイルの存在チェックを行う。
// ファイルが存在する場合はtrue、存在しない場合はfalseを返す。
// filePath : 存在をチェックするファイルのパス
func FileExists(filePath string) bool {
_, err := os.Stat(filePath)
return err == nil
}
func main() {
broker := flag.String("broker", "tcp://{host}:{port}", "URI of MQTT broker (required)")
action := flag.String("action", "p|pub or s|sub", "Publish or Subscribe or Subscribe(with publishing) (required)")
qos := flag.Int("qos", 0, "MQTT QoS(0|1|2)")
retain := flag.Bool("retain", false, "MQTT Retain")
topic := flag.String("topic", BASE_TOPIC, "Base topic")
username := flag.String("broker-username", "", "Username for connecting to the MQTT broker")
password := flag.String("broker-password", "", "Password for connecting to the MQTT broker")
tls := flag.String("tls", "", "TLS mode. 'server:certFile' or 'client:rootCAFile,clientCertFile,clientKeyFile'")
clients := flag.Int("clients", 10, "Number of clients")
count := flag.Int("count", 100, "Number of loops per client")
size := flag.Int("size", 1024, "Message size per publish (byte)")
useDefaultHandler := flag.Bool("support-unknown-received", false, "Using default messageHandler for a message that does not match any known subscriptions")
preTime := flag.Int("pretime", 3000, "Pre wait time (ms)")
intervalTime := flag.Int("intervaltime", 0, "Interval time per message (ms)")
debug := flag.Bool("x", false, "Debug mode")
flag.Parse()
if len(os.Args) <= 1 {
flag.Usage()
return
}
// validate "broker"
if broker == nil || *broker == "" || *broker == "tcp://{host}:{port}" {
fmt.Printf("Invalid argument : -broker -> %s\n", *broker)
return
}
// validate "action"
var method string = ""
if *action == "p" || *action == "pub" {
method = "pub"
} else if *action == "s" || *action == "sub" {
method = "sub"
}
if method != "pub" && method != "sub" {
fmt.Printf("Invalid argument : -action -> %s\n", *action)
return
}
// parse TLS mode
var certConfig CertConfig = nil
if *tls == "" {
// nil
} else if strings.HasPrefix(*tls, "server:") {
var strArray = strings.Split(*tls, "server:")
serverCertFile := strings.TrimSpace(strArray[1])
if FileExists(serverCertFile) == false {
fmt.Printf("File is not found. : certFile -> %s\n", serverCertFile)
return
}
certConfig = ServerCertConfig{
ServerCertFile: serverCertFile}
} else if strings.HasPrefix(*tls, "client:") {
var strArray = strings.Split(*tls, "client:")
var configArray = strings.Split(strArray[1], ",")
rootCAFile := strings.TrimSpace(configArray[0])
clientCertFile := strings.TrimSpace(configArray[1])
clientKeyFile := strings.TrimSpace(configArray[2])
if FileExists(rootCAFile) == false {
fmt.Printf("File is not found. : rootCAFile -> %s\n", rootCAFile)
return
}
if FileExists(clientCertFile) == false {
fmt.Printf("File is not found. : clientCertFile -> %s\n", clientCertFile)
return
}
if FileExists(clientKeyFile) == false {
fmt.Printf("File is not found. : clientKeyFile -> %s\n", clientKeyFile)
return
}
certConfig = ClientCertConfig{
RootCAFile: rootCAFile,
ClientCertFile: clientCertFile,
ClientKeyFile: clientKeyFile}
} else {
// nil
}
execOpts := ExecOptions{}
execOpts.Broker = *broker
execOpts.Qos = byte(*qos)
execOpts.Retain = *retain
execOpts.Topic = *topic
execOpts.Username = *username
execOpts.Password = *password
execOpts.CertConfig = certConfig
execOpts.ClientNum = *clients
execOpts.Count = *count
execOpts.MessageSize = *size
execOpts.UseDefaultHandler = *useDefaultHandler
execOpts.PreTime = *preTime
execOpts.IntervalTime = *intervalTime
Debug = *debug
switch method {
case "pub":
Execute(PublishAllClient, execOpts)
case "sub":
Execute(SubscribeAllClient, execOpts)
}
}