* add base grpc service and swap auth service to streaming * remove Authorize from hub itself * remove unused usings * heave files server to net 7, add exception handling in grpc auth stream Co-authored-by: rootdarkarchon <root.darkarchon@outlook.com>
110 lines
3.1 KiB
C#
110 lines
3.1 KiB
C#
using System.Collections.Concurrent;
|
|
using System.Security.Cryptography;
|
|
using MareSynchronosShared.Protos;
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
namespace MareSynchronosShared.Services;
|
|
|
|
public class GrpcAuthenticationService : GrpcBaseService
|
|
{
|
|
private record AuthRequestInternal
|
|
{
|
|
public AuthRequest Request { get; set; }
|
|
public long Id { get; set; }
|
|
}
|
|
|
|
private readonly AuthService.AuthServiceClient _authClient;
|
|
private readonly ConcurrentQueue<AuthRequestInternal> _requestQueue = new();
|
|
private readonly ConcurrentDictionary<long, AuthReply> _authReplies = new();
|
|
private long _requestId = 0;
|
|
|
|
public GrpcAuthenticationService(ILogger<GrpcAuthenticationService> logger, AuthService.AuthServiceClient authClient) : base(logger)
|
|
{
|
|
_authClient = authClient;
|
|
}
|
|
|
|
public async Task<AuthReply> AuthorizeAsync(string ip, string secretKey)
|
|
{
|
|
using var sha1 = SHA1.Create();
|
|
var id = Interlocked.Increment(ref _requestId);
|
|
_requestQueue.Enqueue(new AuthRequestInternal()
|
|
{
|
|
Id = id,
|
|
Request = new AuthRequest()
|
|
{
|
|
Ip = ip,
|
|
SecretKey = secretKey,
|
|
}
|
|
});
|
|
|
|
using CancellationTokenSource cts = new(TimeSpan.FromSeconds(30));
|
|
AuthReply response = null;
|
|
|
|
while (!GrpcIsFaulty && !cts.IsCancellationRequested && !_authReplies.TryRemove(id, out response))
|
|
{
|
|
await Task.Delay(10, cts.Token).ConfigureAwait(false);
|
|
}
|
|
|
|
return response ?? new AuthReply
|
|
{
|
|
Success = false,
|
|
};
|
|
}
|
|
|
|
public async Task GrpcAuthStream(CancellationToken token)
|
|
{
|
|
try
|
|
{
|
|
using var stream = _authClient.Authorize(cancellationToken: token);
|
|
while (!token.IsCancellationRequested)
|
|
{
|
|
while (_requestQueue.TryDequeue(out var request))
|
|
{
|
|
await stream.RequestStream.WriteAsync(request.Request, token).ConfigureAwait(false);
|
|
await stream.ResponseStream.MoveNext(token).ConfigureAwait(false);
|
|
_authReplies[request.Id] = stream.ResponseStream.Current;
|
|
}
|
|
|
|
await Task.Delay(10, token).ConfigureAwait(false);
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
SetGrpcFaulty();
|
|
}
|
|
}
|
|
|
|
protected override Task OnGrpcRestore()
|
|
{
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
protected override Task PostStartStream()
|
|
{
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
protected override Task PreStartStream()
|
|
{
|
|
_requestQueue.Clear();
|
|
_authReplies.Clear();
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
protected override Task StartAsyncInternal(CancellationToken cancellationToken)
|
|
{
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
protected override Task StartStream(CancellationToken ct)
|
|
{
|
|
_ = GrpcAuthStream(ct);
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
protected override Task StopAsyncInternal(CancellationToken cancellationToken)
|
|
{
|
|
return Task.CompletedTask;
|
|
}
|
|
}
|