|
- using System;
- using System.Diagnostics;
- using System.Linq;
- using System.Text;
- using System.Threading;
- using System.Threading.Tasks;
- using MQTTnet.Client;
- using MQTTnet.Client.Options;
- using MQTTnet.Protocol;
- using MQTTnet.Server;
-
- namespace MQTTnet.TestApp.NetCore
- {
- public static class PerformanceTest
- {
- public static void RunClientOnly()
- {
- try
- {
- var options = new MqttClientOptions
- {
- ChannelOptions = new MqttClientTcpOptions
- {
- Server = "127.0.0.1"
- },
- CleanSession = true
- };
-
- var client = new MqttFactory().CreateMqttClient();
- client.ConnectAsync(options).GetAwaiter().GetResult();
-
- var message = CreateMessage();
- var stopwatch = new Stopwatch();
-
- for (var i = 0; i < 10; i++)
- {
- var sentMessagesCount = 0;
-
- stopwatch.Restart();
- while (stopwatch.ElapsedMilliseconds < 1000)
- {
- client.PublishAsync(message).GetAwaiter().GetResult();
- sentMessagesCount++;
- }
-
- Console.WriteLine($"Sending {sentMessagesCount} messages per second. #" + (i + 1));
-
- GC.Collect();
- }
- }
- catch (Exception exception)
- {
- Console.WriteLine(exception);
- }
- }
-
- public static void RunClientAndServer()
- {
- try
- {
- var mqttServer = new MqttFactory().CreateMqttServer();
- mqttServer.StartAsync(new MqttServerOptions()).GetAwaiter().GetResult();
-
- var options = new MqttClientOptions
- {
- ChannelOptions = new MqttClientTcpOptions
- {
- Server = "127.0.0.1"
- },
- CleanSession = true
- };
-
- var client = new MqttFactory().CreateMqttClient();
- client.ConnectAsync(options).GetAwaiter().GetResult();
-
- var message = CreateMessage();
- var stopwatch = new Stopwatch();
-
- for (var i = 0; i < 10; i++)
- {
- stopwatch.Restart();
-
- var sentMessagesCount = 0;
- while (stopwatch.ElapsedMilliseconds < 1000)
- {
- client.PublishAsync(message).GetAwaiter().GetResult();
- sentMessagesCount++;
- }
-
- Console.WriteLine($"Sending {sentMessagesCount} messages per second. #" + (i + 1));
- }
- }
- catch (Exception exception)
- {
- Console.WriteLine(exception);
- }
- }
-
- private static Task RunClientsAsync(int msgChunkSize, TimeSpan interval, bool concurrent)
- {
- return Task.WhenAll(Enumerable.Range(0, 3).Select(i => Task.Run(() => RunClientAsync(msgChunkSize, interval, concurrent))));
- }
-
- private static async Task RunClientAsync(int msgChunkSize, TimeSpan interval, bool concurrent)
- {
- try
- {
- var options = new MqttClientOptions
- {
- ChannelOptions = new MqttClientTcpOptions { Server = "localhost" },
- ClientId = "Client1",
- CleanSession = true,
- CommunicationTimeout = TimeSpan.FromMinutes(10)
- };
-
- var client = new MqttFactory().CreateMqttClient();
-
- try
- {
- await client.ConnectAsync(options).ConfigureAwait(false);
- }
- catch (Exception exception)
- {
- Console.WriteLine("### CONNECTING FAILED ###" + Environment.NewLine + exception);
- }
-
- var message = CreateMessage();
- var stopwatch = Stopwatch.StartNew();
-
- var testMessageCount = 10000;
- for (var i = 0; i < testMessageCount; i++)
- {
- await client.PublishAsync(message);
- }
-
- stopwatch.Stop();
- Console.WriteLine($"Sent 10.000 messages within {stopwatch.ElapsedMilliseconds} ms ({stopwatch.ElapsedMilliseconds / (float)testMessageCount} ms / message).");
-
- var last = DateTime.Now;
- var msgCount = 0;
-
- while (true)
- {
- var msgs = Enumerable.Range(0, msgChunkSize)
- .Select(i => CreateMessage())
- .ToList();
-
- if (concurrent)
- {
- //send concurrent (test for raceconditions)
- var sendTasks = msgs
- .Select(msg => PublishSingleMessage(client, msg, ref msgCount))
- .ToList();
-
- await Task.WhenAll(sendTasks);
- }
- else
- {
- await client.PublishAsync(msgs);
- msgCount += msgs.Count;
- //send multiple
- }
-
- var now = DateTime.Now;
- if (last < now - TimeSpan.FromSeconds(1))
- {
- Console.WriteLine($"sending {msgCount} intended {msgChunkSize / interval.TotalSeconds}");
- msgCount = 0;
- last = now;
- }
-
- await Task.Delay(interval).ConfigureAwait(false);
- }
- }
- catch (Exception exception)
- {
- Console.WriteLine(exception);
- }
- }
-
- private static MqttApplicationMessage CreateMessage()
- {
- return new MqttApplicationMessage
- {
- Topic = "A/B/C",
- Payload = Encoding.UTF8.GetBytes("Hello World"),
- QualityOfServiceLevel = MqttQualityOfServiceLevel.AtMostOnce
- };
- }
-
- private static Task PublishSingleMessage(IMqttClient client, MqttApplicationMessage applicationMessage, ref int count)
- {
- Interlocked.Increment(ref count);
- return Task.Run(() => client.PublishAsync(applicationMessage));
- }
- }
- }
|