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;
|
2020-08-01 14:55:24 +00:00
|
|
|
using NucuCar.Domain.Http;
|
2020-08-01 13:03:06 +00:00
|
|
|
using NucuCar.Domain.Utilities;
|
2020-08-01 15:07:13 +00:00
|
|
|
using NucuCar.Telemetry.Abstractions;
|
2020-10-25 13:24:34 +00:00
|
|
|
using HttpClient = NucuCar.Domain.Http.HttpClient;
|
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-04-20 15:19:32 +00:00
|
|
|
protected HttpClient 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(
|
|
|
|
$"Can't start {nameof(TelemetryPublisherFirestore)}! Malformed connection string! " +
|
|
|
|
$"Missing ProjectId!");
|
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(
|
|
|
|
$"Can't start {nameof(TelemetryPublisherFirestore)}! Malformed connection string! " +
|
|
|
|
$"Missing CollectionName!");
|
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-04-20 15:19:32 +00:00
|
|
|
HttpClient = new HttpClient(requestUrl) {Timeout = timeout, Logger = Logger};
|
2020-02-08 17:47:44 +00:00
|
|
|
Logger?.LogInformation($"Initialized {nameof(TelemetryPublisherFirestore)}");
|
2020-04-17 15:11:07 +00:00
|
|
|
Logger?.LogInformation($"ProjectId: {firestoreProjectId}; CollectionName: {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-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
|
|
|
|
};
|
2020-04-19 10:08:34 +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-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
|
|
|
{
|
2020-04-19 10:08:34 +00:00
|
|
|
Logger?.LogError($"Firestore authentication request failed! {response?.StatusCode}!");
|
|
|
|
Logger?.LogDebug($"{response?.Content}");
|
2020-04-17 14:04:13 +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;
|
|
|
|
}
|
|
|
|
// Check if the token is about to expire in the next 5 minutes.
|
|
|
|
if (DateTime.UtcNow.AddMinutes(5) < _authorizationExpiryTime)
|
|
|
|
{
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
Logger?.LogWarning(e.Message);
|
|
|
|
}
|
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
|
|
|
{
|
2020-04-19 10:08:34 +00:00
|
|
|
Logger?.LogError($"Failed to publish telemetry data! {responseMessage.StatusCode}. Retrying...");
|
|
|
|
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
|
|
|
|
{
|
|
|
|
Logger?.LogError($"Failed to publish telemetry data! {responseMessage?.StatusCode}");
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2020-04-17 14:04:13 +00:00
|
|
|
}
|
2020-04-19 10:08:34 +00:00
|
|
|
default:
|
|
|
|
Logger?.LogError($"Failed to publish telemetry data! {responseMessage?.StatusCode}");
|
|
|
|
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
|
|
|
}
|