mirror of
https://github.com/JKorf/CryptoExchange.Net
synced 2025-06-10 01:16:24 +00:00
Websocket connection performance improvements, Added multiple options, SymbolOrderBook improvements
This commit is contained in:
parent
c29b2d0cf1
commit
acfd6310d9
@ -25,7 +25,8 @@ namespace CryptoExchange.Net.Converters
|
||||
return new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc).AddSeconds(d);
|
||||
|
||||
var t = double.Parse(reader.Value.ToString(), CultureInfo.InvariantCulture);
|
||||
return new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc).AddSeconds(t);
|
||||
// Set ticks instead of seconds or milliseconds, because AddSeconds/AddMilliseconds rounds to nearest millisecond
|
||||
return new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc).AddTicks((long)(t * TimeSpan.TicksPerSecond));
|
||||
}
|
||||
|
||||
/// <inheritdoc />
|
||||
|
@ -1320,6 +1320,12 @@
|
||||
<member name="P:CryptoExchange.Net.Interfaces.ISocketClient.SocketCombineTarget">
|
||||
<inheritdoc cref="P:CryptoExchange.Net.Objects.SocketClientOptions.SocketSubscriptionsCombineTarget"/>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Interfaces.ISocketClient.MaxReconnectTries">
|
||||
<inheritdoc cref="P:CryptoExchange.Net.Objects.SocketClientOptions.MaxReconnectTries"/>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Interfaces.ISocketClient.MaxResubscribeTries">
|
||||
<inheritdoc cref="P:CryptoExchange.Net.Objects.SocketClientOptions.MaxResubscribeTries"/>
|
||||
</member>
|
||||
<member name="M:CryptoExchange.Net.Interfaces.ISocketClient.UnsubscribeAsync(CryptoExchange.Net.Sockets.UpdateSubscription)">
|
||||
<summary>
|
||||
Unsubscribe from a stream
|
||||
@ -2219,6 +2225,11 @@
|
||||
The name of the order book implementation
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Objects.OrderBookOptions.ChecksumValidationEnabled">
|
||||
<summary>
|
||||
Whether or not checksum validation is enabled. Default is true, disabling will ignore checksum messages.
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Objects.OrderBookOptions.SequenceNumbersAreConsecutive">
|
||||
<summary>
|
||||
Whether each update should have a consecutive id number. Used to identify and reconnect when numbers are skipped.
|
||||
@ -2339,6 +2350,21 @@
|
||||
Time to wait between reconnect attempts
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Objects.SocketClientOptions.MaxReconnectTries">
|
||||
<summary>
|
||||
The maximum number of times to try to reconnect
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Objects.SocketClientOptions.MaxResubscribeTries">
|
||||
<summary>
|
||||
The maximum number of times to try to resubscribe after reconnecting
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Objects.SocketClientOptions.MaxConcurrentResubscriptionsPerSocket">
|
||||
<summary>
|
||||
Max number of concurrent resubscription tasks per socket after reconnecting a socket
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Objects.SocketClientOptions.SocketResponseTimeout">
|
||||
<summary>
|
||||
The time to wait for a socket response before giving a timeout
|
||||
@ -3000,6 +3026,15 @@
|
||||
<member name="P:CryptoExchange.Net.SocketClient.SocketCombineTarget">
|
||||
<inheritdoc cref="P:CryptoExchange.Net.Objects.SocketClientOptions.SocketSubscriptionsCombineTarget"/>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.SocketClient.MaxReconnectTries">
|
||||
<inheritdoc cref="P:CryptoExchange.Net.Objects.SocketClientOptions.MaxReconnectTries"/>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.SocketClient.MaxResubscribeTries">
|
||||
<inheritdoc cref="P:CryptoExchange.Net.Objects.SocketClientOptions.MaxResubscribeTries"/>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.SocketClient.MaxConcurrentResubscriptionsPerSocket">
|
||||
<inheritdoc cref="!:SocketClientOptions.MaxConcurrentResubscriptions"/>
|
||||
</member>
|
||||
<member name="F:CryptoExchange.Net.SocketClient.dataInterpreterBytes">
|
||||
<summary>
|
||||
Delegate used for processing byte data received from socket connections before it is processed by handlers
|
||||
@ -3621,6 +3656,16 @@
|
||||
If the socket should be reconnected upon closing
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Sockets.SocketConnection.ReconnectTry">
|
||||
<summary>
|
||||
Current reconnect try
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Sockets.SocketConnection.ResubscribeTry">
|
||||
<summary>
|
||||
Current resubscribe try
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:CryptoExchange.Net.Sockets.SocketConnection.DisconnectTime">
|
||||
<summary>
|
||||
Time of disconnecting
|
||||
@ -3807,6 +3852,18 @@
|
||||
</summary>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:CryptoExchange.Net.Sockets.UpdateSubscription.UnsubscribeAsync">
|
||||
<summary>
|
||||
Unsubscribe a subscription
|
||||
</summary>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:CryptoExchange.Net.Sockets.UpdateSubscription.ResubscribeAsync">
|
||||
<summary>
|
||||
Resubscribe this subscription
|
||||
</summary>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="T:CryptoExchange.Net.Sockets.WebsocketFactory">
|
||||
<summary>
|
||||
Default weboscket factory implementation
|
||||
|
@ -43,6 +43,12 @@ namespace CryptoExchange.Net.Interfaces
|
||||
|
||||
/// <inheritdoc cref="SocketClientOptions.SocketSubscriptionsCombineTarget"/>
|
||||
int SocketCombineTarget { get; }
|
||||
/// <inheritdoc cref="SocketClientOptions.MaxReconnectTries"/>
|
||||
int? MaxReconnectTries { get; }
|
||||
/// <inheritdoc cref="SocketClientOptions.MaxResubscribeTries"/>
|
||||
int? MaxResubscribeTries { get; }
|
||||
/// <inheritdoc cref="SocketClientOptions.MaxConcurrentResubscriptionsPerSocket"/>
|
||||
int MaxConcurrentResubscriptionsPerSocket { get; }
|
||||
|
||||
/// <summary>
|
||||
/// Unsubscribe from a stream
|
||||
|
@ -45,6 +45,11 @@ namespace CryptoExchange.Net.Objects
|
||||
/// </summary>
|
||||
public string OrderBookName { get; }
|
||||
|
||||
/// <summary>
|
||||
/// Whether or not checksum validation is enabled. Default is true, disabling will ignore checksum messages.
|
||||
/// </summary>
|
||||
public bool ChecksumValidationEnabled { get; set; } = true;
|
||||
|
||||
/// <summary>
|
||||
/// Whether each update should have a consecutive id number. Used to identify and reconnect when numbers are skipped.
|
||||
/// </summary>
|
||||
@ -220,6 +225,21 @@ namespace CryptoExchange.Net.Objects
|
||||
/// </summary>
|
||||
public TimeSpan ReconnectInterval { get; set; } = TimeSpan.FromSeconds(5);
|
||||
|
||||
/// <summary>
|
||||
/// The maximum number of times to try to reconnect
|
||||
/// </summary>
|
||||
public int? MaxReconnectTries { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// The maximum number of times to try to resubscribe after reconnecting
|
||||
/// </summary>
|
||||
public int? MaxResubscribeTries { get; set; } = 5;
|
||||
|
||||
/// <summary>
|
||||
/// Max number of concurrent resubscription tasks per socket after reconnecting a socket
|
||||
/// </summary>
|
||||
public int MaxConcurrentResubscriptionsPerSocket { get; set; } = 5;
|
||||
|
||||
/// <summary>
|
||||
/// The time to wait for a socket response before giving a timeout
|
||||
/// </summary>
|
||||
|
@ -37,6 +37,7 @@ namespace CryptoExchange.Net.OrderBook
|
||||
private UpdateSubscription? subscription;
|
||||
private readonly bool sequencesAreConsecutive;
|
||||
private readonly bool strictLevels;
|
||||
private readonly bool validateChecksum;
|
||||
|
||||
private Task? _processTask;
|
||||
private readonly AutoResetEvent _queueEvent;
|
||||
@ -214,6 +215,7 @@ namespace CryptoExchange.Net.OrderBook
|
||||
|
||||
sequencesAreConsecutive = options.SequenceNumbersAreConsecutive;
|
||||
strictLevels = options.StrictLevels;
|
||||
validateChecksum = options.ChecksumValidationEnabled;
|
||||
Symbol = symbol;
|
||||
Status = OrderBookStatus.Disconnected;
|
||||
|
||||
@ -233,7 +235,7 @@ namespace CryptoExchange.Net.OrderBook
|
||||
{
|
||||
log.Write(LogLevel.Debug, $"{Id} order book {Symbol} starting");
|
||||
Status = OrderBookStatus.Connecting;
|
||||
_processTask = Task.Run(ProcessQueue);
|
||||
_processTask = Task.Factory.StartNew(ProcessQueue, TaskCreationOptions.LongRunning);
|
||||
|
||||
var startResult = await DoStartAsync().ConfigureAwait(false);
|
||||
if (!startResult)
|
||||
@ -243,7 +245,14 @@ namespace CryptoExchange.Net.OrderBook
|
||||
}
|
||||
|
||||
subscription = startResult.Data;
|
||||
subscription.ConnectionLost += Reset;
|
||||
subscription.ConnectionLost += () =>
|
||||
{
|
||||
|
||||
log.Write(LogLevel.Warning, $"{Id} order book {Symbol} connection lost");
|
||||
Status = OrderBookStatus.Reconnecting;
|
||||
Reset();
|
||||
};
|
||||
|
||||
subscription.ConnectionRestored += async time => await ResyncAsync().ConfigureAwait(false);
|
||||
Status = OrderBookStatus.Synced;
|
||||
return new CallResult<bool>(true, null);
|
||||
@ -288,8 +297,6 @@ namespace CryptoExchange.Net.OrderBook
|
||||
|
||||
private void Reset()
|
||||
{
|
||||
log.Write(LogLevel.Warning, $"{Id} order book {Symbol} connection lost");
|
||||
Status = OrderBookStatus.Reconnecting;
|
||||
_queueEvent.Set();
|
||||
// Clear queue
|
||||
while (_processQueue.TryDequeue(out _)) { }
|
||||
@ -380,9 +387,7 @@ namespace CryptoExchange.Net.OrderBook
|
||||
{
|
||||
lock (bookLock)
|
||||
{
|
||||
if (Status == OrderBookStatus.Connecting || Status == OrderBookStatus.Disconnected)
|
||||
return;
|
||||
|
||||
log.Write(LogLevel.Warning, $"{Symbol} bookSet");
|
||||
bookSet = true;
|
||||
asks.Clear();
|
||||
foreach (var ask in item.Asks)
|
||||
@ -408,9 +413,6 @@ namespace CryptoExchange.Net.OrderBook
|
||||
{
|
||||
lock (bookLock)
|
||||
{
|
||||
if (Status == OrderBookStatus.Connecting || Status == OrderBookStatus.Disconnected)
|
||||
return;
|
||||
|
||||
if (!bookSet)
|
||||
{
|
||||
processBuffer.Add(new ProcessBufferRangeSequenceEntry()
|
||||
@ -434,7 +436,7 @@ namespace CryptoExchange.Net.OrderBook
|
||||
if (asks.First().Key < bids.First().Key)
|
||||
{
|
||||
log.Write(LogLevel.Warning, $"{Id} order book {Symbol} detected out of sync order book. Resyncing");
|
||||
_ = subscription?.ReconnectAsync();
|
||||
Resubscribe();
|
||||
return;
|
||||
}
|
||||
|
||||
@ -448,6 +450,9 @@ namespace CryptoExchange.Net.OrderBook
|
||||
{
|
||||
lock (bookLock)
|
||||
{
|
||||
if (!validateChecksum)
|
||||
return;
|
||||
|
||||
bool checksumResult = false;
|
||||
try
|
||||
{
|
||||
@ -467,12 +472,31 @@ namespace CryptoExchange.Net.OrderBook
|
||||
// Should maybe only reconnect the specific subscription?
|
||||
|
||||
log.Write(LogLevel.Warning, $"{Id} order book {Symbol} out of sync. Resyncing");
|
||||
_ = subscription?.ReconnectAsync();
|
||||
Resubscribe();
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void Resubscribe()
|
||||
{
|
||||
Status = OrderBookStatus.Syncing;
|
||||
_ = Task.Run(async () =>
|
||||
{
|
||||
await subscription!.UnsubscribeAsync().ConfigureAwait(false);
|
||||
Reset();
|
||||
if (!await subscription!.ResubscribeAsync().ConfigureAwait(false))
|
||||
{
|
||||
// Resubscribing failed, reconnect the socket
|
||||
log.Write(LogLevel.Warning, $"{Id} order book {Symbol} resync failed, reconnecting socket");
|
||||
Status = OrderBookStatus.Reconnecting;
|
||||
_ = subscription!.ReconnectAsync();
|
||||
}
|
||||
else
|
||||
await ResyncAsync().ConfigureAwait(false);
|
||||
});
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Set the initial data for the order book
|
||||
/// </summary>
|
||||
@ -591,9 +615,6 @@ namespace CryptoExchange.Net.OrderBook
|
||||
/// <param name="entry">The entry</param>
|
||||
protected virtual bool ProcessUpdate(long sequence, OrderBookEntryType type, ISymbolOrderBookEntry entry)
|
||||
{
|
||||
if (Status != OrderBookStatus.Syncing && Status != OrderBookStatus.Synced)
|
||||
return false;
|
||||
|
||||
if (sequence <= LastSequenceNumber)
|
||||
{
|
||||
log.Write(LogLevel.Debug, $"{Id} order book {Symbol} update skipped #{sequence}");
|
||||
@ -604,7 +625,7 @@ namespace CryptoExchange.Net.OrderBook
|
||||
{
|
||||
// Out of sync
|
||||
log.Write(LogLevel.Warning, $"{Id} order book {Symbol} out of sync (expected { LastSequenceNumber + 1}, was {sequence}), reconnecting");
|
||||
subscription?.ReconnectAsync();
|
||||
Resubscribe();
|
||||
return false;
|
||||
}
|
||||
|
||||
|
@ -49,6 +49,12 @@ namespace CryptoExchange.Net
|
||||
public int MaxSocketConnections { get; protected set; } = 9999;
|
||||
/// <inheritdoc cref="SocketClientOptions.SocketSubscriptionsCombineTarget"/>
|
||||
public int SocketCombineTarget { get; protected set; }
|
||||
/// <inheritdoc cref="SocketClientOptions.MaxReconnectTries"/>
|
||||
public int? MaxReconnectTries { get; protected set; }
|
||||
/// <inheritdoc cref="SocketClientOptions.MaxResubscribeTries"/>
|
||||
public int? MaxResubscribeTries { get; protected set; }
|
||||
/// <inheritdoc cref="SocketClientOptions.MaxConcurrentResubscriptions"/>
|
||||
public int MaxConcurrentResubscriptionsPerSocket { get; protected set; }
|
||||
/// <summary>
|
||||
/// Delegate used for processing byte data received from socket connections before it is processed by handlers
|
||||
/// </summary>
|
||||
@ -102,6 +108,9 @@ namespace CryptoExchange.Net
|
||||
ResponseTimeout = exchangeOptions.SocketResponseTimeout;
|
||||
SocketNoDataTimeout = exchangeOptions.SocketNoDataTimeout;
|
||||
SocketCombineTarget = exchangeOptions.SocketSubscriptionsCombineTarget ?? 1;
|
||||
MaxReconnectTries = exchangeOptions.MaxReconnectTries;
|
||||
MaxResubscribeTries = exchangeOptions.MaxResubscribeTries;
|
||||
MaxConcurrentResubscriptionsPerSocket = exchangeOptions.MaxConcurrentResubscriptionsPerSocket;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
@ -5,6 +5,7 @@ using Microsoft.Extensions.Logging;
|
||||
using System;
|
||||
using System.Collections.Concurrent;
|
||||
using System.Collections.Generic;
|
||||
using System.Diagnostics;
|
||||
using System.IO;
|
||||
using System.Net;
|
||||
using System.Net.WebSockets;
|
||||
@ -215,14 +216,19 @@ namespace CryptoExchange.Net.Sockets
|
||||
return false;
|
||||
}
|
||||
|
||||
_sendTask = Task.Run(SendLoopAsync);
|
||||
_receiveTask = Task.Run(ReceiveLoopAsync);
|
||||
log.Write(LogLevel.Trace, $"Socket {Id} connection succeeded, starting communication");
|
||||
_sendTask = Task.Factory.StartNew(SendLoopAsync, TaskCreationOptions.LongRunning);
|
||||
_receiveTask = Task.Factory.StartNew(ReceiveLoopAsync, TaskCreationOptions.LongRunning);
|
||||
if (Timeout != default)
|
||||
_timeoutTask = Task.Run(CheckTimeoutAsync);
|
||||
|
||||
var sw = Stopwatch.StartNew();
|
||||
while (!_startedSent || !_startedReceive)
|
||||
// Wait for the tasks to have actually started
|
||||
await Task.Delay(10).ConfigureAwait(false);
|
||||
|
||||
log.Write(LogLevel.Warning, $"Socket {Id} waited for {sw.ElapsedMilliseconds}ms for tasks to start");
|
||||
|
||||
log.Write(LogLevel.Debug, $"Socket {Id} connected");
|
||||
return true;
|
||||
}
|
||||
@ -237,6 +243,7 @@ namespace CryptoExchange.Net.Sockets
|
||||
throw new InvalidOperationException("Can't send data when socket is not connected");
|
||||
|
||||
var bytes = _encoding.GetBytes(data);
|
||||
log.Write(LogLevel.Trace, $"Socket {Id} Adding {bytes.Length} to sent buffer");
|
||||
_sendBuffer.Enqueue(bytes);
|
||||
_sendEvent.Set();
|
||||
}
|
||||
@ -278,6 +285,7 @@ namespace CryptoExchange.Net.Sockets
|
||||
if (_timeoutTask != null)
|
||||
tasksToAwait.Add(_timeoutTask);
|
||||
|
||||
log.Write(LogLevel.Trace, $"Socket {Id} waiting for communication loops to finish");
|
||||
await Task.WhenAll(tasksToAwait).ConfigureAwait(false);
|
||||
log.Write(LogLevel.Debug, $"Socket {Id} closed");
|
||||
Handle(closeHandlers);
|
||||
@ -296,6 +304,7 @@ namespace CryptoExchange.Net.Sockets
|
||||
openHandlers.Clear();
|
||||
closeHandlers.Clear();
|
||||
messageHandlers.Clear();
|
||||
log.Write(LogLevel.Trace, $"Socket {Id} disposed");
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
@ -9,6 +9,7 @@ using CryptoExchange.Net.Logging;
|
||||
using Newtonsoft.Json;
|
||||
using Newtonsoft.Json.Linq;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using CryptoExchange.Net.Objects;
|
||||
|
||||
namespace CryptoExchange.Net.Sockets
|
||||
{
|
||||
@ -68,7 +69,14 @@ namespace CryptoExchange.Net.Sockets
|
||||
/// If the socket should be reconnected upon closing
|
||||
/// </summary>
|
||||
public bool ShouldReconnect { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Current reconnect try
|
||||
/// </summary>
|
||||
public int ReconnectTry { get; set; }
|
||||
/// <summary>
|
||||
/// Current resubscribe try
|
||||
/// </summary>
|
||||
public int ResubscribeTry { get; set; }
|
||||
/// <summary>
|
||||
/// Time of disconnecting
|
||||
/// </summary>
|
||||
@ -133,6 +141,7 @@ namespace CryptoExchange.Net.Sockets
|
||||
Socket.OnClose += SocketOnClose;
|
||||
Socket.OnOpen += () =>
|
||||
{
|
||||
ReconnectTry = 0;
|
||||
PausedActivity = false;
|
||||
Connected = true;
|
||||
};
|
||||
@ -325,7 +334,21 @@ namespace CryptoExchange.Net.Sockets
|
||||
Socket.Reset();
|
||||
if (!await Socket.ConnectAsync().ConfigureAwait(false))
|
||||
{
|
||||
log.Write(LogLevel.Debug, $"Socket {Socket.Id} failed to reconnect");
|
||||
ReconnectTry++;
|
||||
if(socketClient.MaxReconnectTries != null
|
||||
&& ReconnectTry >= socketClient.MaxReconnectTries)
|
||||
{
|
||||
log.Write(LogLevel.Debug, $"Socket {Socket.Id} failed to reconnect after {ReconnectTry} tries, closing");
|
||||
ShouldReconnect = false;
|
||||
|
||||
if (socketClient.sockets.ContainsKey(Socket.Id))
|
||||
socketClient.sockets.TryRemove(Socket.Id, out _);
|
||||
|
||||
Closed?.Invoke();
|
||||
break;
|
||||
}
|
||||
|
||||
log.Write(LogLevel.Debug, $"Socket {Socket.Id} failed to reconnect{(socketClient.MaxReconnectTries != null ? $", try {ReconnectTry}/{socketClient.MaxReconnectTries}": "")}");
|
||||
continue;
|
||||
}
|
||||
|
||||
@ -337,9 +360,28 @@ namespace CryptoExchange.Net.Sockets
|
||||
|
||||
var reconnectResult = await ProcessReconnectAsync().ConfigureAwait(false);
|
||||
if (!reconnectResult)
|
||||
{
|
||||
ResubscribeTry++;
|
||||
|
||||
if (socketClient.MaxResubscribeTries != null &&
|
||||
ResubscribeTry >= socketClient.MaxResubscribeTries)
|
||||
{
|
||||
log.Write(LogLevel.Debug, $"Socket {Socket.Id} failed to resubscribe after {ResubscribeTry} tries, closing");
|
||||
ShouldReconnect = false;
|
||||
|
||||
if (socketClient.sockets.ContainsKey(Socket.Id))
|
||||
socketClient.sockets.TryRemove(Socket.Id, out _);
|
||||
|
||||
Closed?.Invoke();
|
||||
}
|
||||
else
|
||||
log.Write(LogLevel.Debug, $"Socket {Socket.Id} resubscribing all subscriptions failed on reconnected socket{(socketClient.MaxResubscribeTries != null ? $", try {ResubscribeTry}/{socketClient.MaxResubscribeTries}" : "")}. Disconnecting and reconnecting.");
|
||||
|
||||
await Socket.CloseAsync().ConfigureAwait(false);
|
||||
}
|
||||
else
|
||||
{
|
||||
ResubscribeTry = 0;
|
||||
if (lostTriggered)
|
||||
{
|
||||
lostTriggered = false;
|
||||
@ -389,29 +431,39 @@ namespace CryptoExchange.Net.Sockets
|
||||
lock (subscriptionLock)
|
||||
subscriptionList = subscriptions.Where(h => h.Request != null).ToList();
|
||||
|
||||
var success = true;
|
||||
var taskList = new List<Task>();
|
||||
// Foreach subscription which is subscribed by a subscription request we will need to resend that request to resubscribe
|
||||
foreach (var subscription in subscriptionList)
|
||||
for (var i = 0; i < subscriptionList.Count; i += socketClient.MaxConcurrentResubscriptionsPerSocket)
|
||||
{
|
||||
var task = socketClient.SubscribeAndWaitAsync(this, subscription.Request!, subscription).ContinueWith(t =>
|
||||
var success = true;
|
||||
var taskList = new List<Task>();
|
||||
foreach (var subscription in subscriptionList.Skip(i).Take(socketClient.MaxConcurrentResubscriptionsPerSocket))
|
||||
{
|
||||
if (!t.Result)
|
||||
success = false;
|
||||
});
|
||||
taskList.Add(task);
|
||||
}
|
||||
var task = socketClient.SubscribeAndWaitAsync(this, subscription.Request!, subscription).ContinueWith(t =>
|
||||
{
|
||||
if (!t.Result)
|
||||
success = false;
|
||||
});
|
||||
taskList.Add(task);
|
||||
}
|
||||
|
||||
await Task.WhenAll(taskList).ConfigureAwait(false);
|
||||
if (!success)
|
||||
{
|
||||
log.Write(LogLevel.Debug, $"Socket {Socket.Id} resubscribing all subscriptions failed on reconnected socket. Disconnecting and reconnecting.");
|
||||
return false;
|
||||
}
|
||||
await Task.WhenAll(taskList).ConfigureAwait(false);
|
||||
if (!success)
|
||||
return false;
|
||||
}
|
||||
|
||||
log.Write(LogLevel.Debug, $"Socket {Socket.Id} all subscription successfully resubscribed on reconnected socket.");
|
||||
return true;
|
||||
}
|
||||
|
||||
internal async Task UnsubscribeAsync(SocketSubscription socketSubscription)
|
||||
{
|
||||
await socketClient.UnsubscribeAsync(this, socketSubscription).ConfigureAwait(false);
|
||||
}
|
||||
|
||||
internal async Task<CallResult<bool>> ResubscribeAsync(SocketSubscription socketSubscription)
|
||||
{
|
||||
return await socketClient.SubscribeAndWaitAsync(this, socketSubscription.Request!, socketSubscription).ConfigureAwait(false);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Close the connection
|
||||
|
@ -1,4 +1,5 @@
|
||||
using System;
|
||||
using CryptoExchange.Net.Objects;
|
||||
using System;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace CryptoExchange.Net.Sockets
|
||||
@ -91,5 +92,23 @@ namespace CryptoExchange.Net.Sockets
|
||||
{
|
||||
return connection.Socket.CloseAsync();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Unsubscribe a subscription
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
internal async Task UnsubscribeAsync()
|
||||
{
|
||||
await connection.UnsubscribeAsync(subscription).ConfigureAwait(false);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Resubscribe this subscription
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
internal async Task<CallResult<bool>> ResubscribeAsync()
|
||||
{
|
||||
return await connection.ResubscribeAsync(subscription).ConfigureAwait(false);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user