1
0
mirror of https://github.com/mainflux/mainflux.git synced 2025-04-28 13:48:49 +08:00

208 lines
5.1 KiB
Go
Raw Normal View History

// Copyright (c) Mainflux
// SPDX-License-Identifier: Apache-2.0
package bench
import (
"crypto/tls"
"fmt"
"io/ioutil"
"log"
"os"
"strconv"
"sync"
"time"
"github.com/BurntSushi/toml"
)
// Keep struct names exported, otherwise Viper unmarshaling won't work
type mqttBrokerConfig struct {
URL string `toml:"url" mapstructure:"url"`
}
type mqttMessageConfig struct {
Size int `toml:"size" mapstructure:"size"`
Format string `toml:"format" mapstructure:"format"`
QoS int `toml:"qos" mapstructure:"qos"`
Retain bool `toml:"retain" mapstructure:"retain"`
}
type mqttTLSConfig struct {
MTLS bool `toml:"mtls" mapstructure:"mtls"`
SkipTLSVer bool `toml:"skiptlsver" mapstructure:"skiptlsver"`
CA string `toml:"ca" mapstructure:"ca"`
}
type mqttConfig struct {
Broker mqttBrokerConfig `toml:"broker" mapstructure:"broker"`
Message mqttMessageConfig `toml:"message" mapstructure:"message"`
TLS mqttTLSConfig `toml:"tls" mapstructure:"tls"`
}
type testConfig struct {
Count int `toml:"count" mapstructure:"count"`
Pubs int `toml:"pubs" mapstructure:"pubs"`
Subs int `toml:"subs" mapstructure:"subs"`
}
type logConfig struct {
Quiet bool `toml:"quiet" mapstructure:"quiet"`
}
type mainfluxFile struct {
ConnFile string `toml:"connections_file" mapstructure:"connections_file"`
}
type mfThing struct {
ThingID string `toml:"thing_id" mapstructure:"thing_id"`
ThingKey string `toml:"thing_key" mapstructure:"thing_key"`
MTLSCert string `toml:"mtls_cert" mapstructure:"mtls_cert"`
MTLSKey string `toml:"mtls_key" mapstructure:"mtls_key"`
}
type mfChannel struct {
ChannelID string `toml:"channel_id" mapstructure:"channel_id"`
}
type mainflux struct {
Things []mfThing `toml:"things" mapstructure:"things"`
Channels []mfChannel `toml:"channels" mapstructure:"channels"`
}
// Config struct holds benchmark configuration
type Config struct {
MQTT mqttConfig `toml:"mqtt" mapstructure:"mqtt"`
Test testConfig `toml:"test" mapstructure:"test"`
Log logConfig `toml:"log" mapstructure:"log"`
Mf mainfluxFile `toml:"mainflux" mapstructure:"mainflux"`
}
// JSONResults are used to export results as a JSON document
type JSONResults struct {
Runs []*runResults `json:"runs"`
Totals *totalResults `json:"totals"`
}
// Benchmark - main benckhmarking function
func Benchmark(cfg Config) {
var wg sync.WaitGroup
var err error
checkConnection(cfg.MQTT.Broker.URL, 1)
subTimes := make(subTimes)
var caByte []byte
if cfg.MQTT.TLS.MTLS {
caFile, err := os.Open(cfg.MQTT.TLS.CA)
defer caFile.Close()
if err != nil {
fmt.Println(err)
}
caByte, _ = ioutil.ReadAll(caFile)
}
payload := string(make([]byte, cfg.MQTT.Message.Size))
mf := mainflux{}
if _, err := toml.DecodeFile(cfg.Mf.ConnFile, &mf); err != nil {
log.Fatalf("Cannot load Mainflux connections config %s \nuse tools/provision to create file", cfg.Mf.ConnFile)
}
resCh := make(chan *runResults)
done := make(chan bool)
n := len(mf.Channels)
var cert tls.Certificate
// Subscribers
for i := 0; i < cfg.Test.Subs; i++ {
mfChann := mf.Channels[i%n]
mfThing := mf.Things[i%n]
if cfg.MQTT.TLS.MTLS {
cert, err = tls.X509KeyPair([]byte(mfThing.MTLSCert), []byte(mfThing.MTLSKey))
if err != nil {
log.Fatal(err)
}
}
c := &Client{
ID: strconv.Itoa(i),
BrokerURL: cfg.MQTT.Broker.URL,
BrokerUser: mfThing.ThingID,
BrokerPass: mfThing.ThingKey,
MsgTopic: fmt.Sprintf("channels/%s/messages/test", mfChann.ChannelID),
MsgSize: cfg.MQTT.Message.Size,
MsgCount: cfg.Test.Count,
MsgQoS: byte(cfg.MQTT.Message.QoS),
Quiet: cfg.Log.Quiet,
MTLS: cfg.MQTT.TLS.MTLS,
SkipTLSVer: cfg.MQTT.TLS.SkipTLSVer,
CA: caByte,
ClientCert: cert,
Retain: cfg.MQTT.Message.Retain,
Message: payload,
}
wg.Add(1)
go c.runSubscriber(&wg, &subTimes, &done)
}
wg.Wait()
start := time.Now()
// Publishers
for i := 0; i < cfg.Test.Pubs; i++ {
mfChann := mf.Channels[i%n]
mfThing := mf.Things[i%n]
if cfg.MQTT.TLS.MTLS {
cert, err = tls.X509KeyPair([]byte(mfThing.MTLSCert), []byte(mfThing.MTLSKey))
if err != nil {
log.Fatal(err)
}
}
c := &Client{
ID: strconv.Itoa(i),
BrokerURL: cfg.MQTT.Broker.URL,
BrokerUser: mfThing.ThingID,
BrokerPass: mfThing.ThingKey,
MsgTopic: fmt.Sprintf("channels/%s/messages/test", mfChann.ChannelID),
MsgSize: cfg.MQTT.Message.Size,
MsgCount: cfg.Test.Count,
MsgQoS: byte(cfg.MQTT.Message.QoS),
Quiet: cfg.Log.Quiet,
MTLS: cfg.MQTT.TLS.MTLS,
SkipTLSVer: cfg.MQTT.TLS.SkipTLSVer,
CA: caByte,
ClientCert: cert,
Retain: cfg.MQTT.Message.Retain,
Message: payload,
}
go c.runPublisher(resCh)
}
// Collect the results
var results []*runResults
if cfg.Test.Pubs > 0 {
results = make([]*runResults, cfg.Test.Pubs)
}
for i := 0; i < cfg.Test.Pubs; i++ {
results[i] = <-resCh
}
totalTime := time.Now().Sub(start)
totals := calculateTotalResults(results, totalTime, &subTimes)
if totals == nil {
return
}
// Print sats
printResults(results, totals, cfg.MQTT.Message.Format, cfg.Log.Quiet)
}