mirror of
https://github.com/JKorf/CryptoExchange.Net
synced 2025-06-08 00:16:27 +00:00
242 lines
11 KiB
C#
242 lines
11 KiB
C#
using Microsoft.Extensions.Primitives;
|
|
using Newtonsoft.Json;
|
|
using System;
|
|
using System.Diagnostics;
|
|
using System.Diagnostics.CodeAnalysis;
|
|
using System.Globalization;
|
|
|
|
namespace CryptoExchange.Net.Converters.JsonNet
|
|
{
|
|
/// <summary>
|
|
/// Datetime converter. Supports converting from string/long/double to DateTime and back. Numbers are assumed to be the time since 1970-01-01.
|
|
/// </summary>
|
|
public class DateTimeConverter: JsonConverter
|
|
{
|
|
private static readonly DateTime _epoch = new(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc);
|
|
private const long _ticksPerSecond = TimeSpan.TicksPerMillisecond * 1000;
|
|
private const decimal _ticksPerMicrosecond = TimeSpan.TicksPerMillisecond / 1000;
|
|
private const decimal _ticksPerNanosecond = TimeSpan.TicksPerMillisecond / 1000m / 1000;
|
|
|
|
/// <inheritdoc />
|
|
public override bool CanConvert(Type objectType)
|
|
{
|
|
return objectType == typeof(DateTime) || objectType == typeof(DateTime?);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public override object? ReadJson(JsonReader reader, Type objectType, object? existingValue, JsonSerializer serializer)
|
|
{
|
|
if (reader.Value == null)
|
|
{
|
|
if (objectType == typeof(DateTime))
|
|
return default(DateTime);
|
|
|
|
return null;
|
|
}
|
|
|
|
if(reader.TokenType is JsonToken.Integer)
|
|
{
|
|
var longValue = (long)reader.Value;
|
|
if (longValue == 0 || longValue == -1)
|
|
return objectType == typeof(DateTime) ? default(DateTime): null;
|
|
|
|
return ParseFromLong(longValue);
|
|
}
|
|
else if (reader.TokenType is JsonToken.Float)
|
|
{
|
|
var doubleValue = (double)reader.Value;
|
|
if (doubleValue == 0 || doubleValue == -1)
|
|
return objectType == typeof(DateTime) ? default(DateTime) : null;
|
|
|
|
if (doubleValue < 19999999999)
|
|
return ConvertFromSeconds(doubleValue);
|
|
|
|
return ConvertFromMilliseconds(doubleValue);
|
|
}
|
|
else if(reader.TokenType is JsonToken.String)
|
|
{
|
|
var stringValue = (string)reader.Value;
|
|
if (string.IsNullOrWhiteSpace(stringValue)
|
|
|| stringValue == "-1"
|
|
|| (double.TryParse(stringValue, out var doubleVal) && doubleVal == 0))
|
|
{
|
|
return objectType == typeof(DateTime) ? default(DateTime) : null;
|
|
}
|
|
|
|
return ParseFromString(stringValue);
|
|
}
|
|
else if(reader.TokenType == JsonToken.Date)
|
|
{
|
|
return (DateTime)reader.Value;
|
|
}
|
|
else
|
|
{
|
|
Trace.WriteLine("{DateTime.Now:yyyy/MM/dd HH:mm:ss:fff} | Warning | Unknown DateTime format: " + reader.Value);
|
|
return default;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parse a long value to datetime
|
|
/// </summary>
|
|
/// <param name="longValue"></param>
|
|
/// <returns></returns>
|
|
public static DateTime ParseFromLong(long longValue)
|
|
{
|
|
if (longValue < 19999999999)
|
|
return ConvertFromSeconds(longValue);
|
|
if (longValue < 19999999999999)
|
|
return ConvertFromMilliseconds(longValue);
|
|
if (longValue < 19999999999999999)
|
|
return ConvertFromMicroseconds(longValue);
|
|
|
|
return ConvertFromNanoseconds(longValue);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Parse a string value to datetime
|
|
/// </summary>
|
|
/// <param name="stringValue"></param>
|
|
/// <returns></returns>
|
|
public static DateTime ParseFromString(string stringValue)
|
|
{
|
|
if (stringValue.Length == 12 && stringValue.StartsWith("202"))
|
|
{
|
|
// Parse 202303261200 format
|
|
if (!int.TryParse(stringValue.Substring(0, 4), out var year)
|
|
|| !int.TryParse(stringValue.Substring(4, 2), out var month)
|
|
|| !int.TryParse(stringValue.Substring(6, 2), out var day)
|
|
|| !int.TryParse(stringValue.Substring(8, 2), out var hour)
|
|
|| !int.TryParse(stringValue.Substring(10, 2), out var minute))
|
|
{
|
|
Trace.WriteLine($"{DateTime.Now:yyyy/MM/dd HH:mm:ss:fff} | Warning | Unknown DateTime format: " + stringValue);
|
|
return default;
|
|
}
|
|
return new DateTime(year, month, day, hour, minute, 0, DateTimeKind.Utc);
|
|
}
|
|
|
|
if (stringValue.Length == 8)
|
|
{
|
|
// Parse 20211103 format
|
|
if (!int.TryParse(stringValue.Substring(0, 4), out var year)
|
|
|| !int.TryParse(stringValue.Substring(4, 2), out var month)
|
|
|| !int.TryParse(stringValue.Substring(6, 2), out var day))
|
|
{
|
|
Trace.WriteLine($"{DateTime.Now:yyyy/MM/dd HH:mm:ss:fff} | Warning | Unknown DateTime format: " + stringValue);
|
|
return default;
|
|
}
|
|
return new DateTime(year, month, day, 0, 0, 0, DateTimeKind.Utc);
|
|
}
|
|
|
|
if (stringValue.Length == 6)
|
|
{
|
|
// Parse 211103 format
|
|
if (!int.TryParse(stringValue.Substring(0, 2), out var year)
|
|
|| !int.TryParse(stringValue.Substring(2, 2), out var month)
|
|
|| !int.TryParse(stringValue.Substring(4, 2), out var day))
|
|
{
|
|
Trace.WriteLine("{DateTime.Now:yyyy/MM/dd HH:mm:ss:fff} | Warning | Unknown DateTime format: " + stringValue);
|
|
return default;
|
|
}
|
|
return new DateTime(year + 2000, month, day, 0, 0, 0, DateTimeKind.Utc);
|
|
}
|
|
|
|
if (double.TryParse(stringValue, NumberStyles.Float, CultureInfo.InvariantCulture, out var doubleValue))
|
|
{
|
|
// Parse 1637745563.000 format
|
|
if (doubleValue < 19999999999)
|
|
return ConvertFromSeconds(doubleValue);
|
|
if (doubleValue < 19999999999999)
|
|
return ConvertFromMilliseconds((long)doubleValue);
|
|
if (doubleValue < 19999999999999999)
|
|
return ConvertFromMicroseconds((long)doubleValue);
|
|
|
|
return ConvertFromNanoseconds((long)doubleValue);
|
|
}
|
|
|
|
if (stringValue.Length == 10)
|
|
{
|
|
// Parse 2021-11-03 format
|
|
var values = stringValue.Split('-');
|
|
if (!int.TryParse(values[0], out var year)
|
|
|| !int.TryParse(values[1], out var month)
|
|
|| !int.TryParse(values[2], out var day))
|
|
{
|
|
Trace.WriteLine("{DateTime.Now:yyyy/MM/dd HH:mm:ss:fff} | Warning | Unknown DateTime format: " + stringValue);
|
|
return default;
|
|
}
|
|
|
|
return new DateTime(year, month, day, 0, 0, 0, DateTimeKind.Utc);
|
|
}
|
|
|
|
return DateTime.Parse(stringValue, CultureInfo.InvariantCulture, DateTimeStyles.AdjustToUniversal | DateTimeStyles.AssumeUniversal);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Convert a seconds since epoch (01-01-1970) value to DateTime
|
|
/// </summary>
|
|
/// <param name="seconds"></param>
|
|
/// <returns></returns>
|
|
public static DateTime ConvertFromSeconds(double seconds) => _epoch.AddTicks((long)Math.Round(seconds * _ticksPerSecond));
|
|
/// <summary>
|
|
/// Convert a milliseconds since epoch (01-01-1970) value to DateTime
|
|
/// </summary>
|
|
/// <param name="milliseconds"></param>
|
|
/// <returns></returns>
|
|
public static DateTime ConvertFromMilliseconds(double milliseconds) => _epoch.AddTicks((long)Math.Round(milliseconds * TimeSpan.TicksPerMillisecond));
|
|
/// <summary>
|
|
/// Convert a microseconds since epoch (01-01-1970) value to DateTime
|
|
/// </summary>
|
|
/// <param name="microseconds"></param>
|
|
/// <returns></returns>
|
|
public static DateTime ConvertFromMicroseconds(long microseconds) => _epoch.AddTicks((long)Math.Round(microseconds * _ticksPerMicrosecond));
|
|
/// <summary>
|
|
/// Convert a nanoseconds since epoch (01-01-1970) value to DateTime
|
|
/// </summary>
|
|
/// <param name="nanoseconds"></param>
|
|
/// <returns></returns>
|
|
public static DateTime ConvertFromNanoseconds(long nanoseconds) => _epoch.AddTicks((long)Math.Round(nanoseconds * _ticksPerNanosecond));
|
|
/// <summary>
|
|
/// Convert a DateTime value to seconds since epoch (01-01-1970) value
|
|
/// </summary>
|
|
/// <param name="time"></param>
|
|
/// <returns></returns>
|
|
[return: NotNullIfNotNull("time")]
|
|
public static long? ConvertToSeconds(DateTime? time) => time == null ? null: (long)Math.Round((time.Value - _epoch).TotalSeconds);
|
|
/// <summary>
|
|
/// Convert a DateTime value to milliseconds since epoch (01-01-1970) value
|
|
/// </summary>
|
|
/// <param name="time"></param>
|
|
/// <returns></returns>
|
|
[return: NotNullIfNotNull("time")]
|
|
public static long? ConvertToMilliseconds(DateTime? time) => time == null ? null : (long)Math.Round((time.Value - _epoch).TotalMilliseconds);
|
|
/// <summary>
|
|
/// Convert a DateTime value to microseconds since epoch (01-01-1970) value
|
|
/// </summary>
|
|
/// <param name="time"></param>
|
|
/// <returns></returns>
|
|
[return: NotNullIfNotNull("time")]
|
|
public static long? ConvertToMicroseconds(DateTime? time) => time == null ? null : (long)Math.Round((time.Value - _epoch).Ticks / _ticksPerMicrosecond);
|
|
/// <summary>
|
|
/// Convert a DateTime value to nanoseconds since epoch (01-01-1970) value
|
|
/// </summary>
|
|
/// <param name="time"></param>
|
|
/// <returns></returns>
|
|
[return: NotNullIfNotNull("time")]
|
|
public static long? ConvertToNanoseconds(DateTime? time) => time == null ? null : (long)Math.Round((time.Value - _epoch).Ticks / _ticksPerNanosecond);
|
|
|
|
|
|
/// <inheritdoc />
|
|
public override void WriteJson(JsonWriter writer, object? value, JsonSerializer serializer)
|
|
{
|
|
var datetimeValue = (DateTime?)value;
|
|
if (datetimeValue == null)
|
|
writer.WriteValue((DateTime?)null);
|
|
if(datetimeValue == default(DateTime))
|
|
writer.WriteValue((DateTime?)null);
|
|
else
|
|
writer.WriteValue((long)Math.Round(((DateTime)value! - new DateTime(1970, 1, 1)).TotalMilliseconds));
|
|
}
|
|
}
|
|
}
|