mirror of
https://github.com/bitwarden/server.git
synced 2025-06-07 19:50:32 -05:00

* Enable NRT for Core/Jobs files * Enable NRT for Core/HostedServices files * Enable NRT for Core/Exceptions files * Enable NRT for Core/NotificationHub files --------- Co-authored-by: Bernd Schoolmann <mail@quexten.com>
173 lines
5.7 KiB
C#
173 lines
5.7 KiB
C#
using System.Diagnostics.CodeAnalysis;
|
|
using System.Security.Cryptography;
|
|
using System.Text;
|
|
using System.Web;
|
|
using Bit.Core.Settings;
|
|
using Bit.Core.Utilities;
|
|
using Microsoft.Azure.NotificationHubs;
|
|
|
|
namespace Bit.Core.NotificationHub;
|
|
|
|
#nullable enable
|
|
|
|
public class NotificationHubConnection
|
|
{
|
|
public string? HubName { get; init; }
|
|
public string? ConnectionString { get; init; }
|
|
private Lazy<NotificationHubConnectionStringBuilder> _parsedConnectionString;
|
|
public Uri Endpoint => _parsedConnectionString.Value.Endpoint;
|
|
private string SasKey => _parsedConnectionString.Value.SharedAccessKey;
|
|
private string SasKeyName => _parsedConnectionString.Value.SharedAccessKeyName;
|
|
public bool EnableSendTracing { get; init; }
|
|
private NotificationHubClient? _hubClient;
|
|
/// <summary>
|
|
/// Gets the NotificationHubClient for this connection.
|
|
///
|
|
/// If the client is null, it will be initialized.
|
|
///
|
|
/// <throws>Exception</throws> if the connection is invalid.
|
|
/// </summary>
|
|
public NotificationHubClient HubClient
|
|
{
|
|
get
|
|
{
|
|
if (_hubClient == null)
|
|
{
|
|
if (!IsValid)
|
|
{
|
|
throw new Exception("Invalid notification hub settings");
|
|
}
|
|
Init();
|
|
}
|
|
return _hubClient;
|
|
}
|
|
private set
|
|
{
|
|
_hubClient = value;
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Gets the start date for registration.
|
|
///
|
|
/// If null, registration is always disabled.
|
|
/// </summary>
|
|
public DateTime? RegistrationStartDate { get; init; }
|
|
/// <summary>
|
|
/// Gets the end date for registration.
|
|
///
|
|
/// If null, registration has no end date.
|
|
/// </summary>
|
|
public DateTime? RegistrationEndDate { get; init; }
|
|
/// <summary>
|
|
/// Gets whether all data needed to generate a connection to Notification Hub is present.
|
|
/// </summary>
|
|
public bool IsValid
|
|
{
|
|
get
|
|
{
|
|
{
|
|
var invalid = string.IsNullOrWhiteSpace(HubName) || string.IsNullOrWhiteSpace(ConnectionString);
|
|
return !invalid;
|
|
}
|
|
}
|
|
}
|
|
|
|
public string LogString
|
|
{
|
|
get
|
|
{
|
|
return $"HubName: {HubName}, EnableSendTracing: {EnableSendTracing}, RegistrationStartDate: {RegistrationStartDate}, RegistrationEndDate: {RegistrationEndDate}";
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets whether registration is enabled for the given comb ID.
|
|
/// This is based off of the generation time encoded in the comb ID.
|
|
/// </summary>
|
|
/// <param name="comb"></param>
|
|
/// <returns></returns>
|
|
public bool RegistrationEnabled(Guid comb)
|
|
{
|
|
var combTime = CoreHelpers.DateFromComb(comb);
|
|
return RegistrationEnabled(combTime);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets whether registration is enabled for the given time.
|
|
/// </summary>
|
|
/// <param name="queryTime">The time to check</param>
|
|
/// <returns></returns>
|
|
public bool RegistrationEnabled(DateTime queryTime)
|
|
{
|
|
if (queryTime >= RegistrationEndDate || RegistrationStartDate == null)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return RegistrationStartDate < queryTime;
|
|
}
|
|
|
|
public HttpRequestMessage CreateRequest(HttpMethod method, string pathUri, params string[] queryParameters)
|
|
{
|
|
var uriBuilder = new UriBuilder(Endpoint)
|
|
{
|
|
Scheme = "https",
|
|
Path = $"{HubName}/{pathUri.TrimStart('/')}",
|
|
Query = string.Join('&', [.. queryParameters, "api-version=2015-01"]),
|
|
};
|
|
|
|
var result = new HttpRequestMessage(method, uriBuilder.Uri);
|
|
result.Headers.Add("Authorization", GenerateSasToken(uriBuilder.Uri));
|
|
result.Headers.Add("TrackingId", Guid.NewGuid().ToString());
|
|
return result;
|
|
}
|
|
|
|
private string GenerateSasToken(Uri uri)
|
|
{
|
|
string targetUri = Uri.EscapeDataString(uri.ToString().ToLower()).ToLower();
|
|
long expires = DateTime.UtcNow.AddMinutes(1).Ticks / TimeSpan.TicksPerSecond;
|
|
string stringToSign = targetUri + "\n" + expires;
|
|
|
|
using (var hmac = new HMACSHA256(Encoding.UTF8.GetBytes(SasKey)))
|
|
{
|
|
var signature = Convert.ToBase64String(hmac.ComputeHash(Encoding.UTF8.GetBytes(stringToSign)));
|
|
return $"SharedAccessSignature sr={targetUri}&sig={HttpUtility.UrlEncode(signature)}&se={expires}&skn={SasKeyName}";
|
|
}
|
|
}
|
|
|
|
private NotificationHubConnection()
|
|
{
|
|
_parsedConnectionString = new(() => new NotificationHubConnectionStringBuilder(ConnectionString));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a new NotificationHubConnection from the given settings.
|
|
/// </summary>
|
|
/// <param name="settings"></param>
|
|
/// <returns></returns>
|
|
public static NotificationHubConnection From(GlobalSettings.NotificationHubSettings settings)
|
|
{
|
|
return new()
|
|
{
|
|
HubName = settings.HubName,
|
|
ConnectionString = settings.ConnectionString,
|
|
EnableSendTracing = settings.EnableSendTracing,
|
|
// Comb time is not precise enough for millisecond accuracy
|
|
RegistrationStartDate = settings.RegistrationStartDate.HasValue ? Truncate(settings.RegistrationStartDate.Value, TimeSpan.FromMilliseconds(10)) : null,
|
|
RegistrationEndDate = settings.RegistrationEndDate
|
|
};
|
|
}
|
|
|
|
[MemberNotNull(nameof(_hubClient))]
|
|
private NotificationHubConnection Init()
|
|
{
|
|
_hubClient = NotificationHubClient.CreateClientFromConnectionString(ConnectionString, HubName, EnableSendTracing);
|
|
return this;
|
|
}
|
|
|
|
private static DateTime Truncate(DateTime dateTime, TimeSpan resolution)
|
|
{
|
|
return dateTime.AddTicks(-(dateTime.Ticks % resolution.Ticks));
|
|
}
|
|
}
|