mirror of
https://github.com/JKorf/CryptoExchange.Net
synced 2025-06-07 16:06:15 +00:00
* Added support for Native AOT compilation * Updated all IEnumerable response types to array response types * Added Pass support for ApiCredentials, removing the need for most implementations to add their own ApiCredentials type * Added KeepAliveTimeout setting setting ping frame timeouts for SocketApiClient * Added IBookTickerRestClient Shared interface for requesting book tickers * Added ISpotTriggerOrderRestClient Shared interface for managing spot trigger orders * Added ISpotOrderClientIdClient Shared interface for managing spot orders by client order id * Added IFuturesTriggerOrderRestClient Shared interface for managing futures trigger orders * Added IFuturesOrderClientIdClient Shared interface for managing futures orders by client order id * Added IFuturesTpSlRestClient Shared interface for setting TP/SL on open futures positions * Added GenerateClientOrderId to ISpotOrderRestClient and IFuturesOrderRestClient interface * Added OptionalExchangeParameters and Supported properties to EndpointOptions * Refactor Shared interfaces quantity parameters and properties to use SharedQuantity * Added SharedSymbol property to Shared interface models returning a symbol * Added TriggerPrice, IsTriggerOrder, TakeProfitPrice, StopLossPrice and IsCloseOrder to SharedFuturesOrder response model * Added MaxShortLeverage and MaxLongLeverage to SharedFuturesSymbol response model * Added StopLossPrice and TakeProfitPrice to SharedPosition response model * Added TriggerPrice and IsTriggerOrder to SharedSpotOrder response model * Added QuoteVolume property to SharedSpotTicker response model * Added AssetAlias configuration models * Added static ExchangeSymbolCache for tracking symbol information from exchanges * Added static CallResult.SuccessResult to be used instead of constructing success CallResult instance * Added static ApplyRules, RandomHexString and RandomLong helper methods to ExchangeHelpers class * Added AsErrorWithData To CallResult * Added OriginalData property to CallResult * Added support for adjusting the rate limit key per call, allowing for ratelimiting depending on request parameters * Added implementation for integration testing ISymbolOrderBook instances * Added implementation for integration testing socket subscriptions * Added implementation for testing socket queries * Updated request cancellation logging to Debug level * Updated logging SourceContext to include the client type * Updated some logging logic, errors no longer contain any data, exception are not logged as string but instead forwarded to structured logging * Fixed warning for Enum parsing throwing exception and output warnings for each object in a response to only once to prevent slowing down execution * Fixed memory leak in AsyncAutoRestEvent * Fixed logging for ping frame timeout * Fixed warning getting logged when user stops SymbolOrderBook instance * Fixed socket client `UnsubscribeAll` not unsubscribing dedicated connections * Fixed memory leak in Rest client cache * Fixed integers bigger than int16 not getting correctly parsed to enums * Fixed issue where the default options were overridden when using SetApiCredentials * Removed Newtonsoft.Json dependency * Removed legacy Rest client code * Removed legacy ISpotClient and IFuturesClient support
237 lines
11 KiB
C#
237 lines
11 KiB
C#
using System;
|
|
using System.Collections.Concurrent;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Reflection;
|
|
using System.Text.Json.Serialization;
|
|
using System.Text.Json;
|
|
using CryptoExchange.Net.Attributes;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics.CodeAnalysis;
|
|
using System.Threading;
|
|
using System.Diagnostics;
|
|
|
|
namespace CryptoExchange.Net.Converters.SystemTextJson
|
|
{
|
|
/// <summary>
|
|
/// Converter for arrays to objects. Can deserialize data like [0.1, 0.2, "test"] to an object. Mapping is done by marking the class with [JsonConverter(typeof(ArrayConverter))] and the properties
|
|
/// with [ArrayProperty(x)] where x is the index of the property in the array
|
|
/// </summary>
|
|
#if NET5_0_OR_GREATER
|
|
public class ArrayConverter<[DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicProperties)] T> : JsonConverter<T> where T : new()
|
|
#else
|
|
public class ArrayConverter<T> : JsonConverter<T> where T : new()
|
|
#endif
|
|
{
|
|
private static readonly Lazy<List<ArrayPropertyInfo>> _typePropertyInfo = new Lazy<List<ArrayPropertyInfo>>(CacheTypeAttributes, LazyThreadSafetyMode.PublicationOnly);
|
|
|
|
private static readonly ConcurrentDictionary<JsonConverter, JsonSerializerOptions> _converterOptionsCache = new ConcurrentDictionary<JsonConverter, JsonSerializerOptions>();
|
|
|
|
/// <inheritdoc />
|
|
#if NET5_0_OR_GREATER
|
|
[UnconditionalSuppressMessage("AssemblyLoadTrimming", "IL3050:RequiresUnreferencedCode", Justification = "JsonSerializerOptions provided here has TypeInfoResolver set")]
|
|
[UnconditionalSuppressMessage("AssemblyLoadTrimming", "IL2026:RequiresUnreferencedCode", Justification = "JsonSerializerOptions provided here has TypeInfoResolver set")]
|
|
#endif
|
|
public override void Write(Utf8JsonWriter writer, T value, JsonSerializerOptions options)
|
|
{
|
|
if (value == null)
|
|
{
|
|
writer.WriteNullValue();
|
|
return;
|
|
}
|
|
|
|
writer.WriteStartArray();
|
|
|
|
var ordered = _typePropertyInfo.Value.Where(x => x.ArrayProperty != null).OrderBy(p => p.ArrayProperty.Index);
|
|
var last = -1;
|
|
foreach (var prop in ordered)
|
|
{
|
|
if (prop.ArrayProperty.Index == last)
|
|
continue;
|
|
|
|
while (prop.ArrayProperty.Index != last + 1)
|
|
{
|
|
writer.WriteNullValue();
|
|
last += 1;
|
|
}
|
|
|
|
last = prop.ArrayProperty.Index;
|
|
|
|
var objValue = prop.PropertyInfo.GetValue(value);
|
|
if (objValue == null)
|
|
{
|
|
writer.WriteNullValue();
|
|
continue;
|
|
}
|
|
|
|
JsonSerializerOptions? typeOptions = null;
|
|
if (prop.JsonConverter != null)
|
|
{
|
|
typeOptions = new JsonSerializerOptions
|
|
{
|
|
NumberHandling = JsonNumberHandling.AllowReadingFromString | JsonNumberHandling.AllowNamedFloatingPointLiterals,
|
|
PropertyNameCaseInsensitive = false,
|
|
TypeInfoResolver = options.TypeInfoResolver,
|
|
};
|
|
typeOptions.Converters.Add(prop.JsonConverter);
|
|
}
|
|
|
|
if (prop.JsonConverter == null && IsSimple(prop.PropertyInfo.PropertyType))
|
|
{
|
|
if (prop.TargetType == typeof(string))
|
|
writer.WriteStringValue(Convert.ToString(objValue, CultureInfo.InvariantCulture));
|
|
else if (prop.TargetType == typeof(bool))
|
|
writer.WriteBooleanValue((bool)objValue);
|
|
else
|
|
writer.WriteRawValue(Convert.ToString(objValue, CultureInfo.InvariantCulture)!);
|
|
}
|
|
else
|
|
{
|
|
JsonSerializer.Serialize(writer, objValue, typeOptions ?? options);
|
|
}
|
|
}
|
|
|
|
writer.WriteEndArray();
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public override T? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
|
|
{
|
|
if (reader.TokenType == JsonTokenType.Null)
|
|
return default;
|
|
|
|
var result = Activator.CreateInstance(typeof(T))!;
|
|
return (T)ParseObject(ref reader, result, typeof(T), options);
|
|
}
|
|
|
|
|
|
#if NET5_0_OR_GREATER
|
|
[UnconditionalSuppressMessage("AssemblyLoadTrimming", "IL3050:RequiresUnreferencedCode", Justification = "JsonSerializerOptions provided here has TypeInfoResolver set")]
|
|
[UnconditionalSuppressMessage("AssemblyLoadTrimming", "IL2026:RequiresUnreferencedCode", Justification = "JsonSerializerOptions provided here has TypeInfoResolver set")]
|
|
private static object ParseObject(ref Utf8JsonReader reader, object result, [DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicProperties)] Type objectType, JsonSerializerOptions options)
|
|
#else
|
|
private static object ParseObject(ref Utf8JsonReader reader, object result, Type objectType, JsonSerializerOptions options)
|
|
#endif
|
|
{
|
|
if (reader.TokenType != JsonTokenType.StartArray)
|
|
throw new Exception("Not an array");
|
|
|
|
int index = 0;
|
|
while (reader.Read())
|
|
{
|
|
if (reader.TokenType == JsonTokenType.EndArray)
|
|
break;
|
|
|
|
var indexAttributes = _typePropertyInfo.Value.Where(a => a.ArrayProperty.Index == index);
|
|
if (!indexAttributes.Any())
|
|
{
|
|
index++;
|
|
continue;
|
|
}
|
|
|
|
foreach (var attribute in indexAttributes)
|
|
{
|
|
var targetType = attribute.TargetType;
|
|
object? value = null;
|
|
if (attribute.JsonConverter != null)
|
|
{
|
|
if (!_converterOptionsCache.TryGetValue(attribute.JsonConverter, out var newOptions))
|
|
{
|
|
newOptions = new JsonSerializerOptions
|
|
{
|
|
NumberHandling = JsonNumberHandling.AllowReadingFromString | JsonNumberHandling.AllowNamedFloatingPointLiterals,
|
|
PropertyNameCaseInsensitive = false,
|
|
Converters = { attribute.JsonConverter },
|
|
TypeInfoResolver = options.TypeInfoResolver,
|
|
};
|
|
_converterOptionsCache.TryAdd(attribute.JsonConverter, newOptions);
|
|
}
|
|
|
|
var doc = JsonDocument.ParseValue(ref reader);
|
|
value = doc.Deserialize(attribute.PropertyInfo.PropertyType, newOptions);
|
|
}
|
|
else if (attribute.DefaultDeserialization)
|
|
{
|
|
value = JsonDocument.ParseValue(ref reader).Deserialize(options.GetTypeInfo(attribute.PropertyInfo.PropertyType));
|
|
}
|
|
else
|
|
{
|
|
value = reader.TokenType switch
|
|
{
|
|
JsonTokenType.Null => null,
|
|
JsonTokenType.False => false,
|
|
JsonTokenType.True => true,
|
|
JsonTokenType.String => reader.GetString(),
|
|
JsonTokenType.Number => reader.GetDecimal(),
|
|
JsonTokenType.StartObject => JsonSerializer.Deserialize(ref reader, attribute.TargetType, options),
|
|
_ => throw new NotImplementedException($"Array deserialization of type {reader.TokenType} not supported"),
|
|
};
|
|
}
|
|
|
|
if (targetType.IsAssignableFrom(value?.GetType()))
|
|
attribute.PropertyInfo.SetValue(result, value);
|
|
else
|
|
attribute.PropertyInfo.SetValue(result, value == null ? null : Convert.ChangeType(value, targetType, CultureInfo.InvariantCulture));
|
|
}
|
|
|
|
index++;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
private static bool IsSimple(Type type)
|
|
{
|
|
if (type.IsGenericType && type.GetGenericTypeDefinition() == typeof(Nullable<>))
|
|
{
|
|
// nullable type, check if the nested type is simple.
|
|
return IsSimple(type.GetGenericArguments()[0]);
|
|
}
|
|
return type.IsPrimitive
|
|
|| type.IsEnum
|
|
|| type == typeof(string)
|
|
|| type == typeof(decimal);
|
|
}
|
|
|
|
#if NET5_0_OR_GREATER
|
|
[UnconditionalSuppressMessage("AssemblyLoadTrimming", "IL3050:RequiresUnreferencedCode", Justification = "JsonSerializerOptions provided here has TypeInfoResolver set")]
|
|
[UnconditionalSuppressMessage("AssemblyLoadTrimming", "IL2026:RequiresUnreferencedCode", Justification = "JsonSerializerOptions provided here has TypeInfoResolver set")]
|
|
private static List<ArrayPropertyInfo> CacheTypeAttributes()
|
|
#else
|
|
private static List<ArrayPropertyInfo> CacheTypeAttributes()
|
|
#endif
|
|
{
|
|
var attributes = new List<ArrayPropertyInfo>();
|
|
var properties = typeof(T).GetProperties();
|
|
foreach (var property in properties)
|
|
{
|
|
var att = property.GetCustomAttribute<ArrayPropertyAttribute>();
|
|
if (att == null)
|
|
continue;
|
|
|
|
var targetType = Nullable.GetUnderlyingType(property.PropertyType) ?? property.PropertyType;
|
|
var converterType = property.GetCustomAttribute<JsonConverterAttribute>()?.ConverterType ?? targetType.GetCustomAttribute<JsonConverterAttribute>()?.ConverterType;
|
|
attributes.Add(new ArrayPropertyInfo
|
|
{
|
|
ArrayProperty = att,
|
|
PropertyInfo = property,
|
|
DefaultDeserialization = property.GetCustomAttribute<CryptoExchange.Net.Attributes.JsonConversionAttribute>() != null,
|
|
JsonConverter = converterType == null ? null : (JsonConverter)Activator.CreateInstance(converterType)!,
|
|
TargetType = targetType
|
|
});
|
|
}
|
|
|
|
return attributes;
|
|
}
|
|
|
|
private class ArrayPropertyInfo
|
|
{
|
|
public PropertyInfo PropertyInfo { get; set; } = null!;
|
|
public ArrayPropertyAttribute ArrayProperty { get; set; } = null!;
|
|
public JsonConverter? JsonConverter { get; set; }
|
|
public bool DefaultDeserialization { get; set; }
|
|
public Type TargetType { get; set; } = null!;
|
|
}
|
|
}
|
|
}
|