mirror of
https://github.com/JKorf/CryptoExchange.Net
synced 2026-02-16 22:23:54 +00:00
Performance update: Authentication Added Ed25519 signing support for NET8.0 and newer Added static methods on ApiCredentials to create credentials of a specific type Added static ApiCredentials.ReadFromFile method to read a key from file Added required abstract SupportedCredentialTypes property on AuthenticationProvider base class General Performance Added checks before logging statements to prevent overhead of building the log string if logging is not needed Added ExchangeHelpers.ProcessQueuedAsync method to process updates async Replaced locking object types from object to Lock in NET9.0 and newer Replaced some Task response types with ValueTask to prevent allocation overhead on hot paths Updated Json ArrayConverter to reduce some allocation overhead Updated Json BoolConverter to prevent boxing Updated Json DateTimeConverter to prevent boxing Updated Json EnumConverter caching to reduce lookup overhead Updated ExtensionMethods.CreateParamString to reduce allocations Updated ExtensionMethods.AppendPath to reduce overhead REST Refactored REST message processing to separate IRestMessageHandler instance Split RestApiClient.PrepareAsync into CheckTimeSync and RateLimitAsync Updated IRequest.Accept type from string to MediaTypeWithQualityHeaderValue to prevent creation on each request Updated IRequest.GetHeaders response type from KeyValuePair<string, string[]>[] to HttpRequestHeaders to prevent additional mapping Updated IResponse.ResponseHeaders type from KeyValuePair<string, string[]>[] to HttpResponseHeaders to prevent additional mapping Updated WebCallResult RequestHeaders and ResponseHeaders types to HttpRequestHeaders and HttpResponseHeaders Removed unnecessary empty dictionary initializations for each request Removed CallResult creation in internal methods to prevent having to create multiple versions for different result types Socket Added HighPerformance websocket client implementation which significantly reduces memory overhead and improves speed but with certain limitations Added MaxIndividualSubscriptionsPerConnection setting in SocketApiClient to limit the number of individual stream subscriptions on a connection Added SocketIndividualSubscriptionCombineTarget option to set the target number of individual stream subscriptions per connection Added new websocket message handling logic which is faster and reduces memory allocation Added UseUpdatedDeserialization option to toggle between updated deserialization and old deserialization Added Exchange property to DataEvent to prevent additional mapping overhead for Shared apis Refactored message callback to be sync instead of async to prevent async overhead Refactored CryptoExchangeWebSocketClient.IncomingKbps calculation to significantly reduce overhead Moved websocket client creation from SocketApiClient to SocketConnection Removed DataEvent.As and DataEvent.ToCallResult methods in favor of single ToType method Removed DataEvent creation on lower levels to prevent having to create multiple versions for different result types Removed Subscription<TSubResponse, TUnsubResponse> as its no longer used Other Added null check to ParameterCollection for required parameters Added Net10.0 target framework Updated dependency versions Updated Shared asset aliases check to be culture invariant Updated Error string representation Updated some namespaces Updated SymbolOrderBook processing of buffered updates to prevent additional allocation Removed ExchangeEvent type which is no longer needed Removed unused usings
133 lines
4.4 KiB
C#
133 lines
4.4 KiB
C#
using CryptoExchange.Net.Authentication;
|
|
using CryptoExchange.Net.Objects.Options;
|
|
using Microsoft.Extensions.Logging;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Threading;
|
|
|
|
namespace CryptoExchange.Net.Clients
|
|
{
|
|
/// <summary>
|
|
/// The base for all clients, websocket client and rest client
|
|
/// </summary>
|
|
public abstract class BaseClient : IDisposable
|
|
{
|
|
/// <summary>
|
|
/// Version of the CryptoExchange.Net base library
|
|
/// </summary>
|
|
public Version CryptoExchangeLibVersion { get; } = typeof(BaseClient).Assembly.GetName().Version!;
|
|
|
|
/// <summary>
|
|
/// Version of the client implementation
|
|
/// </summary>
|
|
public Version ExchangeLibVersion
|
|
{
|
|
get
|
|
{
|
|
lock(_versionLock)
|
|
{
|
|
if (_exchangeVersion == null)
|
|
_exchangeVersion = GetType().Assembly.GetName().Version!;
|
|
|
|
return _exchangeVersion;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// The name of the API the client is for
|
|
/// </summary>
|
|
public string Exchange { get; }
|
|
|
|
/// <summary>
|
|
/// Api clients in this client
|
|
/// </summary>
|
|
internal List<BaseApiClient> ApiClients { get; } = new List<BaseApiClient>();
|
|
|
|
/// <summary>
|
|
/// The log object
|
|
/// </summary>
|
|
protected internal ILogger _logger;
|
|
|
|
#if NET9_0_OR_GREATER
|
|
private readonly Lock _versionLock = new Lock();
|
|
#else
|
|
private readonly object _versionLock = new object();
|
|
#endif
|
|
private Version _exchangeVersion;
|
|
|
|
/// <summary>
|
|
/// Provided client options
|
|
/// </summary>
|
|
public ExchangeOptions ClientOptions { get; private set; }
|
|
|
|
/// <summary>
|
|
/// ctor
|
|
/// </summary>
|
|
/// <param name="logger">Logger</param>
|
|
/// <param name="exchange">The name of the exchange this client is for</param>
|
|
#pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable.
|
|
protected BaseClient(ILoggerFactory? logger, string exchange)
|
|
#pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable.
|
|
{
|
|
Exchange = exchange;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Initialize the client with the specified options
|
|
/// </summary>
|
|
/// <param name="options"></param>
|
|
/// <exception cref="ArgumentNullException"></exception>
|
|
protected virtual void Initialize(ExchangeOptions options)
|
|
{
|
|
if (options == null)
|
|
throw new ArgumentNullException(nameof(options));
|
|
|
|
ClientOptions = options;
|
|
_logger.Log(LogLevel.Trace, $"Client configuration: {options}, CryptoExchange.Net: v{CryptoExchangeLibVersion}, {Exchange}.Net: v{ExchangeLibVersion}");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Set the API credentials for this client. All Api clients in this client will use the new credentials, regardless of earlier set options.
|
|
/// </summary>
|
|
/// <param name="credentials">The credentials to set</param>
|
|
protected virtual void SetApiCredentials<T>(T credentials) where T : ApiCredentials
|
|
{
|
|
foreach (var apiClient in ApiClients)
|
|
apiClient.SetApiCredentials(credentials);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Register an API client
|
|
/// </summary>
|
|
/// <param name="apiClient">The client</param>
|
|
protected T AddApiClient<T>(T apiClient) where T : BaseApiClient
|
|
{
|
|
if (ClientOptions == null)
|
|
throw new InvalidOperationException("Client should have called Initialize before adding API clients");
|
|
|
|
ApiClients.Add(apiClient);
|
|
return apiClient;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Apply the options delegate to a new options instance
|
|
/// </summary>
|
|
protected static T ApplyOptionsDelegate<T>(Action<T>? del) where T: new()
|
|
{
|
|
var opts = new T();
|
|
del?.Invoke(opts);
|
|
return opts;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Dispose
|
|
/// </summary>
|
|
public virtual void Dispose()
|
|
{
|
|
foreach (var client in ApiClients)
|
|
client.Dispose();
|
|
}
|
|
}
|
|
}
|