2020-04-20 15:19:32 +00:00
|
|
|
using System;
|
2020-02-08 17:47:44 +00:00
|
|
|
using System.Collections.Generic;
|
2020-04-17 14:04:13 +00:00
|
|
|
using System.Net;
|
2020-10-25 13:24:34 +00:00
|
|
|
using System.Net.Http;
|
2020-02-08 17:47:44 +00:00
|
|
|
using System.Threading;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using Microsoft.Extensions.Logging;
|
2021-08-02 18:57:52 +00:00
|
|
|
using NucuCar.Core.Http;
|
|
|
|
using NucuCar.Core.Utilities;
|
2020-08-01 15:07:13 +00:00
|
|
|
using NucuCar.Telemetry.Abstractions;
|
2020-02-08 17:47:44 +00:00
|
|
|
|
2020-08-01 15:07:13 +00:00
|
|
|
namespace NucuCar.Telemetry.Publishers
|
2020-02-08 17:47:44 +00:00
|
|
|
{
|
|
|
|
/// <summary>
|
|
|
|
/// This class is used to publish the telemetry data to Google's Cloud Firestore.
|
|
|
|
/// Requires the environment variable: GOOGLE_APPLICATION_CREDENTIALS to be set.
|
|
|
|
/// See: https://cloud.google.com/docs/authentication/getting-started
|
2020-02-16 13:49:31 +00:00
|
|
|
/// or Firebase > Project Settings > Service Accounts (Authentication is not implemented!)
|
2020-02-08 17:47:44 +00:00
|
|
|
/// <remarks>
|
|
|
|
/// The connection string has the following parameters:
|
|
|
|
/// ProjectId (required) — The string for the Firestore project id.
|
|
|
|
/// CollectionName (required) — The string for the Firestore collection name.
|
2020-04-17 14:04:13 +00:00
|
|
|
/// WebApiKey (optional) — The web api key of the firebase project.
|
|
|
|
/// WebApiEmail (optional) — An email to use when requesting id tokens.
|
|
|
|
/// WebApiPassword (optional) — The password to use when requesting id tokens.
|
2020-02-08 17:47:44 +00:00
|
|
|
/// Timeout (optional) — The number in milliseconds in which to timeout if publishing fails. Default: 10000
|
|
|
|
/// </remarks>
|
|
|
|
/// </summary>
|
|
|
|
public class TelemetryPublisherFirestore : TelemetryPublisher
|
|
|
|
{
|
2020-11-25 18:49:15 +00:00
|
|
|
protected MinimalHttpClient HttpClient;
|
2020-02-08 17:47:44 +00:00
|
|
|
|
2020-04-17 14:04:13 +00:00
|
|
|
private string _idToken;
|
2020-11-24 17:03:02 +00:00
|
|
|
private DateTime _authorizationExpiryTime;
|
2020-04-17 14:04:13 +00:00
|
|
|
|
|
|
|
// Variables used for authentication
|
|
|
|
private readonly string _webEmail;
|
|
|
|
private readonly string _webPassword;
|
|
|
|
private readonly string _webApiKey;
|
|
|
|
|
2020-08-01 15:07:13 +00:00
|
|
|
public TelemetryPublisherFirestore(TelemetryPublisherOptions opts) : base(opts)
|
2020-02-08 17:47:44 +00:00
|
|
|
{
|
2020-04-19 10:08:34 +00:00
|
|
|
// Parse Options
|
2020-02-08 17:47:44 +00:00
|
|
|
var options = ConnectionStringParser.Parse(opts.ConnectionString);
|
|
|
|
if (!options.TryGetValue("ProjectId", out var firestoreProjectId))
|
|
|
|
{
|
|
|
|
Logger?.LogCritical(
|
2021-08-01 18:01:19 +00:00
|
|
|
"Can't start {Name}! Malformed connection string! Missing ProjectId!",
|
|
|
|
nameof(TelemetryPublisherFirestore));
|
2020-04-20 15:19:32 +00:00
|
|
|
throw new ArgumentException("Malformed connection string!");
|
2020-02-08 17:47:44 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 13:45:41 +00:00
|
|
|
if (!options.TryGetValue("CollectionName", out var firestoreCollection))
|
2020-02-08 17:47:44 +00:00
|
|
|
{
|
|
|
|
Logger?.LogCritical(
|
2021-08-01 18:01:19 +00:00
|
|
|
"Can't start {Name}! Malformed connection string! Missing CollectionName!",
|
|
|
|
nameof(TelemetryPublisherFirestore));
|
2020-04-20 15:19:32 +00:00
|
|
|
throw new ArgumentException("Malformed connection string!");
|
2020-02-08 17:47:44 +00:00
|
|
|
}
|
|
|
|
|
2020-04-19 10:08:34 +00:00
|
|
|
var timeout = int.Parse(options.GetValueOrDefault("Timeout", "10000") ?? "10000");
|
2020-04-17 14:04:13 +00:00
|
|
|
_webApiKey = options.GetValueOrDefault("WebApiKey", null);
|
|
|
|
_webEmail = options.GetValueOrDefault("WebApiEmail", null);
|
|
|
|
_webPassword = options.GetValueOrDefault("WebApiPassword", null);
|
2020-02-08 17:47:44 +00:00
|
|
|
|
2020-04-19 10:08:34 +00:00
|
|
|
// Setup HttpClient
|
2020-02-16 13:45:41 +00:00
|
|
|
var requestUrl = $"https://firestore.googleapis.com/v1/projects/{firestoreProjectId}/" +
|
|
|
|
$"databases/(default)/documents/{firestoreCollection}/";
|
2020-11-25 18:49:15 +00:00
|
|
|
HttpClient = new MinimalHttpClient(requestUrl) {Timeout = timeout, Logger = Logger};
|
2021-08-01 18:01:19 +00:00
|
|
|
Logger?.LogInformation("Initialized {Name}", nameof(TelemetryPublisherFirestore));
|
|
|
|
Logger?.LogInformation("ProjectId: {FirestoreProjectId}; CollectionName: {FirestoreCollection}",
|
|
|
|
firestoreProjectId, firestoreCollection);
|
2020-02-08 17:47:44 +00:00
|
|
|
}
|
|
|
|
|
2020-04-19 10:08:34 +00:00
|
|
|
private async Task SetupAuthorization()
|
2020-04-17 14:04:13 +00:00
|
|
|
{
|
2020-11-25 18:49:15 +00:00
|
|
|
HttpClient.ClearAuthorizationHeader();
|
2021-08-01 18:01:19 +00:00
|
|
|
|
2020-10-31 14:55:16 +00:00
|
|
|
// https://cloud.google.com/identity-platform/docs/use-rest-api#section-sign-in-email-password
|
2020-04-17 14:04:13 +00:00
|
|
|
var requestUrl = $"https://identitytoolkit.googleapis.com/v1/accounts:signInWithPassword?key={_webApiKey}";
|
|
|
|
var data = new Dictionary<string, object>()
|
|
|
|
{
|
|
|
|
["email"] = _webEmail,
|
|
|
|
["password"] = _webPassword,
|
|
|
|
["returnSecureToken"] = true
|
|
|
|
};
|
2021-08-01 18:01:19 +00:00
|
|
|
|
2020-04-20 15:19:32 +00:00
|
|
|
var response = await HttpClient.PostAsync(requestUrl, data);
|
2020-04-19 10:08:34 +00:00
|
|
|
|
|
|
|
if (response?.StatusCode == HttpStatusCode.OK)
|
2020-04-17 14:04:13 +00:00
|
|
|
{
|
2020-11-24 21:39:37 +00:00
|
|
|
Logger?.LogInformation("Firestore authentication OK!");
|
2020-04-19 10:08:34 +00:00
|
|
|
var jsonContent = await response.GetJson();
|
|
|
|
_idToken = jsonContent.GetProperty("idToken").ToString();
|
2020-10-31 14:55:16 +00:00
|
|
|
// Setup next expire.
|
|
|
|
var expiresIn = double.Parse(jsonContent.GetProperty("expiresIn").ToString());
|
2020-11-24 17:03:02 +00:00
|
|
|
_authorizationExpiryTime = DateTime.UtcNow.AddSeconds(expiresIn);
|
2020-04-20 15:19:32 +00:00
|
|
|
HttpClient.Authorization(_idToken);
|
2020-04-17 14:04:13 +00:00
|
|
|
}
|
2020-04-19 10:08:34 +00:00
|
|
|
else
|
2020-04-17 14:04:13 +00:00
|
|
|
{
|
2021-08-01 18:01:19 +00:00
|
|
|
Logger?.LogError("Firestore authentication request failed! {StatusCode}!", response?.StatusCode);
|
2020-11-25 17:10:24 +00:00
|
|
|
if (response != null)
|
|
|
|
{
|
|
|
|
var contentBody = await response.Content.ReadAsStringAsync();
|
2021-08-01 18:01:19 +00:00
|
|
|
Logger?.LogDebug("{Body}", contentBody);
|
2020-11-25 17:10:24 +00:00
|
|
|
}
|
2020-04-17 14:04:13 +00:00
|
|
|
}
|
|
|
|
}
|
2020-11-24 17:03:32 +00:00
|
|
|
|
2020-11-24 17:03:02 +00:00
|
|
|
private async Task CheckAndSetupAuthorization()
|
|
|
|
{
|
|
|
|
// If there are no credentials or partial credentials supplies there must be no authorization.
|
|
|
|
if (_webApiKey == null || _webEmail == null || _webPassword == null)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2020-11-24 17:03:32 +00:00
|
|
|
|
2020-11-24 21:39:37 +00:00
|
|
|
// Check if the token is about to expire in the next 15 minutes.
|
|
|
|
if (DateTime.UtcNow.AddMinutes(15) < _authorizationExpiryTime)
|
2020-11-24 17:03:02 +00:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
await SetupAuthorization();
|
|
|
|
}
|
2020-04-17 14:04:13 +00:00
|
|
|
|
2020-02-08 17:47:44 +00:00
|
|
|
public override async Task PublishAsync(CancellationToken cancellationToken)
|
|
|
|
{
|
|
|
|
if (cancellationToken.IsCancellationRequested)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2020-04-17 14:04:13 +00:00
|
|
|
|
2020-04-19 10:08:34 +00:00
|
|
|
var data = FirebaseRestTranslator.Translator.Translate(null, GetTelemetry());
|
2020-10-25 13:24:34 +00:00
|
|
|
|
|
|
|
HttpResponseMessage responseMessage = null;
|
|
|
|
try
|
|
|
|
{
|
2020-11-24 17:03:02 +00:00
|
|
|
await CheckAndSetupAuthorization();
|
2020-10-25 13:24:34 +00:00
|
|
|
responseMessage = await HttpClient.PostAsync("", data);
|
|
|
|
}
|
|
|
|
// ArgumentException occurs during json serialization errors.
|
|
|
|
catch (ArgumentException e)
|
|
|
|
{
|
2021-08-01 18:01:19 +00:00
|
|
|
Logger?.LogWarning("{Message}", e.Message);
|
2020-10-25 13:24:34 +00:00
|
|
|
}
|
2020-10-31 14:55:16 +00:00
|
|
|
|
2020-04-17 14:04:13 +00:00
|
|
|
|
2020-04-19 10:08:34 +00:00
|
|
|
switch (responseMessage?.StatusCode)
|
|
|
|
{
|
|
|
|
case HttpStatusCode.OK:
|
|
|
|
Logger?.LogInformation("Published data to Firestore!");
|
|
|
|
break;
|
|
|
|
case HttpStatusCode.Forbidden:
|
2020-11-23 17:56:48 +00:00
|
|
|
case HttpStatusCode.Unauthorized:
|
2020-04-17 14:04:13 +00:00
|
|
|
{
|
2021-08-01 18:01:19 +00:00
|
|
|
Logger?.LogError("Failed to publish telemetry data! {StatusCode}. Retrying...",
|
|
|
|
responseMessage.StatusCode);
|
2020-04-19 10:08:34 +00:00
|
|
|
await SetupAuthorization();
|
2020-04-20 15:19:32 +00:00
|
|
|
responseMessage = await HttpClient.PostAsync("", data);
|
2020-04-19 10:08:34 +00:00
|
|
|
if (responseMessage != null && responseMessage.IsSuccessStatusCode)
|
|
|
|
{
|
|
|
|
Logger?.LogInformation("Published data to Firestore on retry!");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-08-01 18:01:19 +00:00
|
|
|
Logger?.LogError("Failed to publish telemetry data! {StatusCode}", responseMessage?.StatusCode);
|
2020-04-19 10:08:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2020-04-17 14:04:13 +00:00
|
|
|
}
|
2020-04-19 10:08:34 +00:00
|
|
|
default:
|
2021-08-01 18:01:19 +00:00
|
|
|
Logger?.LogError("Failed to publish telemetry data! {StatusCode}", responseMessage?.StatusCode);
|
2020-04-19 10:08:34 +00:00
|
|
|
break;
|
2020-04-17 14:04:13 +00:00
|
|
|
}
|
2020-02-08 17:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public override void Dispose()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
2020-02-16 13:45:41 +00:00
|
|
|
}
|