You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
150 lines
5.5 KiB
150 lines
5.5 KiB
4 years ago
|
using System;
|
||
|
using System.Collections.Concurrent;
|
||
|
#if DEBUG_LIMITS
|
||
|
using System.Diagnostics;
|
||
|
#endif
|
||
|
using System.IO;
|
||
|
using System.Linq;
|
||
|
using System.Threading;
|
||
|
using System.Threading.Tasks;
|
||
|
|
||
|
namespace Discord.Net.Queue
|
||
|
{
|
||
|
internal class RequestQueue : IDisposable
|
||
|
{
|
||
|
public event Func<string, RateLimitInfo?, Task> RateLimitTriggered;
|
||
|
|
||
|
private readonly ConcurrentDictionary<string, RequestBucket> _buckets;
|
||
|
private readonly SemaphoreSlim _tokenLock;
|
||
|
private readonly CancellationTokenSource _cancelTokenSource; //Dispose token
|
||
|
private CancellationTokenSource _clearToken;
|
||
|
private CancellationToken _parentToken;
|
||
|
private CancellationTokenSource _requestCancelTokenSource;
|
||
|
private CancellationToken _requestCancelToken; //Parent token + Clear token
|
||
|
private DateTimeOffset _waitUntil;
|
||
|
|
||
|
private Task _cleanupTask;
|
||
|
|
||
|
public RequestQueue()
|
||
|
{
|
||
|
_tokenLock = new SemaphoreSlim(1, 1);
|
||
|
|
||
|
_clearToken = new CancellationTokenSource();
|
||
|
_cancelTokenSource = new CancellationTokenSource();
|
||
|
_requestCancelToken = CancellationToken.None;
|
||
|
_parentToken = CancellationToken.None;
|
||
|
|
||
|
_buckets = new ConcurrentDictionary<string, RequestBucket>();
|
||
|
|
||
|
_cleanupTask = RunCleanup();
|
||
|
}
|
||
|
|
||
|
public async Task SetCancelTokenAsync(CancellationToken cancelToken)
|
||
|
{
|
||
|
await _tokenLock.WaitAsync().ConfigureAwait(false);
|
||
|
try
|
||
|
{
|
||
|
_parentToken = cancelToken;
|
||
|
_requestCancelTokenSource?.Dispose();
|
||
|
_requestCancelTokenSource = CancellationTokenSource.CreateLinkedTokenSource(cancelToken, _clearToken.Token);
|
||
|
_requestCancelToken = _requestCancelTokenSource.Token;
|
||
|
}
|
||
|
finally { _tokenLock.Release(); }
|
||
|
}
|
||
|
public async Task ClearAsync()
|
||
|
{
|
||
|
await _tokenLock.WaitAsync().ConfigureAwait(false);
|
||
|
try
|
||
|
{
|
||
|
_clearToken?.Cancel();
|
||
|
_clearToken?.Dispose();
|
||
|
_clearToken = new CancellationTokenSource();
|
||
|
if (_parentToken != null)
|
||
|
{
|
||
|
_requestCancelTokenSource?.Dispose();
|
||
|
_requestCancelTokenSource = CancellationTokenSource.CreateLinkedTokenSource(_clearToken.Token, _parentToken);
|
||
|
_requestCancelToken = _requestCancelTokenSource.Token;
|
||
|
}
|
||
|
else
|
||
|
_requestCancelToken = _clearToken.Token;
|
||
|
}
|
||
|
finally { _tokenLock.Release(); }
|
||
|
}
|
||
|
|
||
|
public async Task<Stream> SendAsync(RestRequest request)
|
||
|
{
|
||
|
CancellationTokenSource createdTokenSource = null;
|
||
|
if (request.Options.CancelToken.CanBeCanceled)
|
||
|
{
|
||
|
createdTokenSource = CancellationTokenSource.CreateLinkedTokenSource(_requestCancelToken, request.Options.CancelToken);
|
||
|
request.Options.CancelToken = createdTokenSource.Token;
|
||
|
}
|
||
|
else
|
||
|
request.Options.CancelToken = _requestCancelToken;
|
||
|
|
||
|
var bucket = GetOrCreateBucket(request.Options.BucketId, request);
|
||
|
var result = await bucket.SendAsync(request).ConfigureAwait(false);
|
||
|
createdTokenSource?.Dispose();
|
||
|
return result;
|
||
|
}
|
||
|
public async Task SendAsync(WebSocketRequest request)
|
||
|
{
|
||
|
//TODO: Re-impl websocket buckets
|
||
|
request.CancelToken = _requestCancelToken;
|
||
|
await request.SendAsync().ConfigureAwait(false);
|
||
|
}
|
||
|
|
||
|
internal async Task EnterGlobalAsync(int id, RestRequest request)
|
||
|
{
|
||
|
int millis = (int)Math.Ceiling((_waitUntil - DateTimeOffset.UtcNow).TotalMilliseconds);
|
||
|
if (millis > 0)
|
||
|
{
|
||
|
#if DEBUG_LIMITS
|
||
|
Debug.WriteLine($"[{id}] Sleeping {millis} ms (Pre-emptive) [Global]");
|
||
|
#endif
|
||
|
await Task.Delay(millis).ConfigureAwait(false);
|
||
|
}
|
||
|
}
|
||
|
internal void PauseGlobal(RateLimitInfo info)
|
||
|
{
|
||
|
_waitUntil = DateTimeOffset.UtcNow.AddMilliseconds(info.RetryAfter.Value + (info.Lag?.TotalMilliseconds ?? 0.0));
|
||
|
}
|
||
|
|
||
|
private RequestBucket GetOrCreateBucket(string id, RestRequest request)
|
||
|
{
|
||
|
return _buckets.GetOrAdd(id, x => new RequestBucket(this, request, x));
|
||
|
}
|
||
|
internal async Task RaiseRateLimitTriggered(string bucketId, RateLimitInfo? info)
|
||
|
{
|
||
|
await RateLimitTriggered(bucketId, info).ConfigureAwait(false);
|
||
|
}
|
||
|
|
||
|
private async Task RunCleanup()
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
while (!_cancelTokenSource.IsCancellationRequested)
|
||
|
{
|
||
|
var now = DateTimeOffset.UtcNow;
|
||
|
foreach (var bucket in _buckets.Select(x => x.Value))
|
||
|
{
|
||
|
if ((now - bucket.LastAttemptAt).TotalMinutes > 1.0)
|
||
|
_buckets.TryRemove(bucket.Id, out _);
|
||
|
}
|
||
|
await Task.Delay(60000, _cancelTokenSource.Token).ConfigureAwait(false); //Runs each minute
|
||
|
}
|
||
|
}
|
||
|
catch (OperationCanceledException) { }
|
||
|
catch (ObjectDisposedException) { }
|
||
|
}
|
||
|
|
||
|
public void Dispose()
|
||
|
{
|
||
|
_cancelTokenSource?.Dispose();
|
||
|
_tokenLock?.Dispose();
|
||
|
_clearToken?.Dispose();
|
||
|
_requestCancelTokenSource?.Dispose();
|
||
|
}
|
||
|
}
|
||
|
}
|