Publish telemetry in batch with retrying
This commit is contained in:
parent
88e62fa486
commit
1c522d5a61
1 changed files with 48 additions and 21 deletions
|
@ -74,40 +74,67 @@ namespace NucuCar.Domain.Telemetry
|
||||||
|
|
||||||
public override async Task PublishAsync(CancellationToken cancellationToken)
|
public override async Task PublishAsync(CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
|
var data = GetTelemetry();
|
||||||
|
|
||||||
|
var messageString = JsonConvert.SerializeObject(data);
|
||||||
|
Logger?.LogDebug($"Telemetry message: {messageString}");
|
||||||
|
var message = new Message(Encoding.ASCII.GetBytes(messageString));
|
||||||
|
|
||||||
|
await PublishToCloudAsync(message, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Dictionary<string, object> GetTelemetry()
|
||||||
|
{
|
||||||
|
var data = new List<Dictionary<string, object>>();
|
||||||
foreach (var telemeter in RegisteredTelemeters)
|
foreach (var telemeter in RegisteredTelemeters)
|
||||||
{
|
{
|
||||||
var data = telemeter.GetTelemetryData();
|
var telemetryData = telemeter.GetTelemetryData();
|
||||||
if (data == null)
|
if (telemetryData == null)
|
||||||
{
|
{
|
||||||
Logger?.LogWarning($"Warning! Data for {telemeter.GetIdentifier()} is null!");
|
Logger?.LogWarning($"Warning! Data for {telemeter.GetIdentifier()} is null!");
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
telemetryData["_id"] = telemeter.GetIdentifier();
|
||||||
|
data.Add(telemetryData);
|
||||||
|
}
|
||||||
|
|
||||||
var metadata = new Dictionary<string, object>
|
var metadata = new Dictionary<string, object>
|
||||||
{
|
{
|
||||||
["source"] = TelemetrySource ?? nameof(TelemetryPublisherAzure),
|
["source"] = TelemetrySource ?? nameof(TelemetryPublisherAzure),
|
||||||
["id"] = telemeter.GetIdentifier(),
|
|
||||||
["timestamp"] = DateTime.Now,
|
["timestamp"] = DateTime.Now,
|
||||||
["data"] = data,
|
["data"] = data.ToArray()
|
||||||
};
|
};
|
||||||
|
return metadata;
|
||||||
await PublishViaMqtt(metadata, cancellationToken);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task PublishViaMqtt(Dictionary<string, object> data, CancellationToken cancellationToken)
|
private async Task PublishToCloudAsync(Message message, CancellationToken cancellationToken, int maxRetries = 3)
|
||||||
|
{
|
||||||
|
var retry = 0;
|
||||||
|
while (retry < maxRetries)
|
||||||
{
|
{
|
||||||
if (cancellationToken.IsCancellationRequested)
|
if (cancellationToken.IsCancellationRequested)
|
||||||
{
|
{
|
||||||
Logger?.LogInformation("Stopping the AzureTelemetryPublisher, cancellation requested.");
|
Logger?.LogInformation("Publishing telemetry cancelled!");
|
||||||
await DeviceClient.CloseAsync(cancellationToken);
|
break;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var messageString = JsonConvert.SerializeObject(data);
|
try
|
||||||
var message = new Message(Encoding.ASCII.GetBytes(messageString));
|
{
|
||||||
Logger?.LogDebug($"Telemetry message: {messageString}");
|
var cts = new CancellationTokenSource();
|
||||||
await DeviceClient.SendEventAsync(message, cancellationToken);
|
cts.CancelAfter(5000);
|
||||||
|
cts.Token.ThrowIfCancellationRequested();
|
||||||
|
|
||||||
|
/* Should throw OperationCanceledException on timeout or cancel. */
|
||||||
|
await DeviceClient.SendEventAsync(message, cts.Token);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
catch (OperationCanceledException e)
|
||||||
|
{
|
||||||
|
retry += 1;
|
||||||
|
Logger?.LogWarning($"Telemetry not sent! Retry {retry}.");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void Dispose()
|
public override void Dispose()
|
||||||
|
|
Loading…
Reference in a new issue