2018-08-02 18:14:33 +02:00
|
|
|
|
using System;
|
2018-08-02 23:23:37 +02:00
|
|
|
|
using System.Linq;
|
2018-08-02 18:14:33 +02:00
|
|
|
|
using System.Threading;
|
|
|
|
|
using System.Threading.Tasks;
|
2018-08-02 23:23:37 +02:00
|
|
|
|
using Bit.Core;
|
|
|
|
|
using Bit.Core.Models;
|
2018-08-02 18:14:33 +02:00
|
|
|
|
using Microsoft.AspNetCore.SignalR;
|
|
|
|
|
using Microsoft.Extensions.Hosting;
|
|
|
|
|
using Microsoft.Extensions.Logging;
|
2018-08-02 23:23:37 +02:00
|
|
|
|
using Microsoft.WindowsAzure.Storage;
|
|
|
|
|
using Microsoft.WindowsAzure.Storage.Queue;
|
|
|
|
|
using Newtonsoft.Json;
|
2018-08-02 18:14:33 +02:00
|
|
|
|
|
|
|
|
|
namespace Bit.Hub
|
|
|
|
|
{
|
|
|
|
|
public class TimedHostedService : IHostedService, IDisposable
|
|
|
|
|
{
|
|
|
|
|
private readonly ILogger _logger;
|
|
|
|
|
private readonly IHubContext<SyncHub> _hubContext;
|
2018-08-02 23:23:37 +02:00
|
|
|
|
private readonly GlobalSettings _globalSettings;
|
2018-08-02 18:14:33 +02:00
|
|
|
|
|
2018-08-02 23:23:37 +02:00
|
|
|
|
private Task _executingTask;
|
|
|
|
|
private CancellationTokenSource _cts;
|
|
|
|
|
private CloudQueue _queue;
|
|
|
|
|
|
|
|
|
|
public TimedHostedService(ILogger<TimedHostedService> logger, IHubContext<SyncHub> hubContext,
|
|
|
|
|
GlobalSettings globalSettings)
|
2018-08-02 18:14:33 +02:00
|
|
|
|
{
|
|
|
|
|
_logger = logger;
|
|
|
|
|
_hubContext = hubContext;
|
2018-08-02 23:23:37 +02:00
|
|
|
|
_globalSettings = globalSettings;
|
2018-08-02 18:14:33 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Task StartAsync(CancellationToken cancellationToken)
|
|
|
|
|
{
|
2018-08-02 23:23:37 +02:00
|
|
|
|
_cts = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken);
|
|
|
|
|
_executingTask = ExecuteAsync(_cts.Token);
|
|
|
|
|
return _executingTask.IsCompleted ? _executingTask : Task.CompletedTask;
|
2018-08-02 18:14:33 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-08-02 23:23:37 +02:00
|
|
|
|
public async Task StopAsync(CancellationToken cancellationToken)
|
2018-08-02 18:14:33 +02:00
|
|
|
|
{
|
2018-08-02 23:23:37 +02:00
|
|
|
|
if(_executingTask == null)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
_cts.Cancel();
|
|
|
|
|
await Task.WhenAny(_executingTask, Task.Delay(-1, cancellationToken));
|
|
|
|
|
cancellationToken.ThrowIfCancellationRequested();
|
2018-08-02 18:14:33 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-08-02 23:23:37 +02:00
|
|
|
|
public void Dispose()
|
2018-08-02 18:14:33 +02:00
|
|
|
|
{
|
2018-08-02 23:23:37 +02:00
|
|
|
|
// TODO
|
2018-08-02 18:14:33 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-08-02 23:23:37 +02:00
|
|
|
|
private async Task ExecuteAsync(CancellationToken cancellationToken)
|
2018-08-02 18:14:33 +02:00
|
|
|
|
{
|
2018-08-02 23:23:37 +02:00
|
|
|
|
var storageAccount = CloudStorageAccount.Parse(_globalSettings.Events.ConnectionString);
|
|
|
|
|
var queueClient = storageAccount.CreateCloudQueueClient();
|
|
|
|
|
_queue = queueClient.GetQueueReference("sync");
|
|
|
|
|
|
|
|
|
|
while(!cancellationToken.IsCancellationRequested)
|
|
|
|
|
{
|
|
|
|
|
var messages = await _queue.GetMessagesAsync(32, TimeSpan.FromMinutes(1),
|
|
|
|
|
null, null, cancellationToken);
|
|
|
|
|
if(messages.Any())
|
|
|
|
|
{
|
|
|
|
|
foreach(var message in messages)
|
|
|
|
|
{
|
|
|
|
|
var notification = JsonConvert.DeserializeObject<PushNotificationData<object>>(
|
|
|
|
|
message.AsString);
|
|
|
|
|
switch(notification.Type)
|
|
|
|
|
{
|
|
|
|
|
case Core.Enums.PushType.SyncCipherUpdate:
|
|
|
|
|
case Core.Enums.PushType.SyncCipherCreate:
|
|
|
|
|
case Core.Enums.PushType.SyncCipherDelete:
|
|
|
|
|
case Core.Enums.PushType.SyncLoginDelete:
|
|
|
|
|
var cipherNotification =
|
|
|
|
|
JsonConvert.DeserializeObject<PushNotificationData<SyncCipherPushNotification>>(
|
|
|
|
|
message.AsString);
|
|
|
|
|
if(cipherNotification.Payload.UserId.HasValue)
|
|
|
|
|
{
|
|
|
|
|
await _hubContext.Clients.User(cipherNotification.Payload.UserId.ToString())
|
|
|
|
|
.SendAsync("ReceiveMessage", notification, cancellationToken);
|
|
|
|
|
}
|
|
|
|
|
else if(cipherNotification.Payload.OrganizationId.HasValue)
|
|
|
|
|
{
|
|
|
|
|
await _hubContext.Clients.Group(
|
|
|
|
|
$"Organization_{cipherNotification.Payload.OrganizationId}")
|
|
|
|
|
.SendAsync("ReceiveMessage", notification, cancellationToken);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case Core.Enums.PushType.SyncFolderUpdate:
|
|
|
|
|
case Core.Enums.PushType.SyncFolderCreate:
|
|
|
|
|
case Core.Enums.PushType.SyncFolderDelete:
|
|
|
|
|
var folderNotification =
|
|
|
|
|
JsonConvert.DeserializeObject<PushNotificationData<SyncFolderPushNotification>>(
|
|
|
|
|
message.AsString);
|
|
|
|
|
await _hubContext.Clients.User(folderNotification.Payload.UserId.ToString())
|
|
|
|
|
.SendAsync("ReceiveMessage", notification, cancellationToken);
|
|
|
|
|
break;
|
|
|
|
|
case Core.Enums.PushType.SyncCiphers:
|
|
|
|
|
case Core.Enums.PushType.SyncVault:
|
|
|
|
|
case Core.Enums.PushType.SyncOrgKeys:
|
|
|
|
|
case Core.Enums.PushType.SyncSettings:
|
|
|
|
|
var userNotification =
|
|
|
|
|
JsonConvert.DeserializeObject<PushNotificationData<SyncUserPushNotification>>(
|
|
|
|
|
message.AsString);
|
|
|
|
|
await _hubContext.Clients.User(userNotification.Payload.UserId.ToString())
|
|
|
|
|
.SendAsync("ReceiveMessage", notification, cancellationToken);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
await _queue.DeleteMessageAsync(message);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
await Task.Delay(TimeSpan.FromSeconds(5), cancellationToken);
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-08-02 18:14:33 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|