mirror of
https://github.com/JKorf/CryptoExchange.Net
synced 2025-06-08 16:36:15 +00:00
240 lines
9.5 KiB
C#
240 lines
9.5 KiB
C#
using CryptoExchange.Net.Converters;
|
|
using CryptoExchange.Net.Objects;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.Net.Http;
|
|
using System.Security.Cryptography;
|
|
using System.Text;
|
|
|
|
namespace CryptoExchange.Net.Authentication
|
|
{
|
|
/// <summary>
|
|
/// Base class for authentication providers
|
|
/// </summary>
|
|
public abstract class AuthenticationProvider
|
|
{
|
|
/// <summary>
|
|
/// The provided credentials
|
|
/// </summary>
|
|
public ApiCredentials Credentials { get; }
|
|
|
|
/// <summary>
|
|
/// </summary>
|
|
protected byte[] _sBytes;
|
|
|
|
/// <summary>
|
|
/// ctor
|
|
/// </summary>
|
|
/// <param name="credentials"></param>
|
|
protected AuthenticationProvider(ApiCredentials credentials)
|
|
{
|
|
if (credentials.Secret == null)
|
|
throw new ArgumentException("ApiKey/Secret needed");
|
|
|
|
Credentials = credentials;
|
|
_sBytes = Encoding.UTF8.GetBytes(credentials.Secret.GetString());
|
|
}
|
|
|
|
/// <summary>
|
|
/// Authenticate a request. Output parameters should include the providedParameters input
|
|
/// </summary>
|
|
/// <param name="apiClient">The Api client sending the request</param>
|
|
/// <param name="uri">The uri for the request</param>
|
|
/// <param name="method">The method of the request</param>
|
|
/// <param name="providedParameters">The request parameters</param>
|
|
/// <param name="auth">If the requests should be authenticated</param>
|
|
/// <param name="arraySerialization">Array serialization type</param>
|
|
/// <param name="parameterPosition">The position where the providedParameters should go</param>
|
|
/// <param name="uriParameters">Parameters that need to be in the Uri of the request. Should include the provided parameters if they should go in the uri</param>
|
|
/// <param name="bodyParameters">Parameters that need to be in the body of the request. Should include the provided parameters if they should go in the body</param>
|
|
/// <param name="headers">The headers that should be send with the request</param>
|
|
public abstract void AuthenticateRequest(
|
|
RestApiClient apiClient,
|
|
Uri uri,
|
|
HttpMethod method,
|
|
Dictionary<string, object> providedParameters,
|
|
bool auth,
|
|
ArrayParametersSerialization arraySerialization,
|
|
HttpMethodParameterPosition parameterPosition,
|
|
out SortedDictionary<string, object> uriParameters,
|
|
out SortedDictionary<string, object> bodyParameters,
|
|
out Dictionary<string, string> headers
|
|
);
|
|
|
|
/// <summary>
|
|
/// SHA256 sign the data and return the bytes
|
|
/// </summary>
|
|
/// <param name="data"></param>
|
|
/// <returns></returns>
|
|
protected static byte[] SignSHA256Bytes(string data)
|
|
{
|
|
using var encryptor = SHA256.Create();
|
|
return encryptor.ComputeHash(Encoding.UTF8.GetBytes(data));
|
|
}
|
|
|
|
/// <summary>
|
|
/// SHA256 sign the data and return the hash
|
|
/// </summary>
|
|
/// <param name="data">Data to sign</param>
|
|
/// <param name="outputType">String type</param>
|
|
/// <returns></returns>
|
|
protected static string SignSHA256(string data, SignOutputType? outputType = null)
|
|
{
|
|
using var encryptor = SHA256.Create();
|
|
var resultBytes = encryptor.ComputeHash(Encoding.UTF8.GetBytes(data));
|
|
return outputType == SignOutputType.Base64 ? BytesToBase64String(resultBytes): BytesToHexString(resultBytes);
|
|
}
|
|
|
|
/// <summary>
|
|
/// SHA384 sign the data and return the hash
|
|
/// </summary>
|
|
/// <param name="data">Data to sign</param>
|
|
/// <param name="outputType">String type</param>
|
|
/// <returns></returns>
|
|
protected static string SignSHA384(string data, SignOutputType? outputType = null)
|
|
{
|
|
using var encryptor = SHA384.Create();
|
|
var resultBytes = encryptor.ComputeHash(Encoding.UTF8.GetBytes(data));
|
|
return outputType == SignOutputType.Base64 ? BytesToBase64String(resultBytes) : BytesToHexString(resultBytes);
|
|
}
|
|
|
|
/// <summary>
|
|
/// SHA512 sign the data and return the hash
|
|
/// </summary>
|
|
/// <param name="data">Data to sign</param>
|
|
/// <param name="outputType">String type</param>
|
|
/// <returns></returns>
|
|
protected static string SignSHA512(string data, SignOutputType? outputType = null)
|
|
{
|
|
using var encryptor = SHA512.Create();
|
|
var resultBytes = encryptor.ComputeHash(Encoding.UTF8.GetBytes(data));
|
|
return outputType == SignOutputType.Base64 ? BytesToBase64String(resultBytes) : BytesToHexString(resultBytes);
|
|
}
|
|
|
|
/// <summary>
|
|
/// MD5 sign the data and return the hash
|
|
/// </summary>
|
|
/// <param name="data">Data to sign</param>
|
|
/// <param name="outputType">String type</param>
|
|
/// <returns></returns>
|
|
protected static string SignMD5(string data, SignOutputType? outputType = null)
|
|
{
|
|
using var encryptor = MD5.Create();
|
|
var resultBytes = encryptor.ComputeHash(Encoding.UTF8.GetBytes(data));
|
|
return outputType == SignOutputType.Base64 ? BytesToBase64String(resultBytes) : BytesToHexString(resultBytes);
|
|
}
|
|
|
|
/// <summary>
|
|
/// HMACSHA256 sign the data and return the hash
|
|
/// </summary>
|
|
/// <param name="data">Data to sign</param>
|
|
/// <param name="outputType">String type</param>
|
|
/// <returns></returns>
|
|
protected string SignHMACSHA256(string data, SignOutputType? outputType = null)
|
|
{
|
|
using var encryptor = new HMACSHA256(_sBytes);
|
|
var resultBytes = encryptor.ComputeHash(Encoding.UTF8.GetBytes(data));
|
|
return outputType == SignOutputType.Base64 ? BytesToBase64String(resultBytes) : BytesToHexString(resultBytes);
|
|
}
|
|
|
|
/// <summary>
|
|
/// HMACSHA384 sign the data and return the hash
|
|
/// </summary>
|
|
/// <param name="data">Data to sign</param>
|
|
/// <param name="outputType">String type</param>
|
|
/// <returns></returns>
|
|
protected string SignHMACSHA384(string data, SignOutputType? outputType = null)
|
|
{
|
|
using var encryptor = new HMACSHA384(_sBytes);
|
|
var resultBytes = encryptor.ComputeHash(Encoding.UTF8.GetBytes(data));
|
|
return outputType == SignOutputType.Base64 ? BytesToBase64String(resultBytes) : BytesToHexString(resultBytes);
|
|
}
|
|
|
|
/// <summary>
|
|
/// HMACSHA512 sign the data and return the hash
|
|
/// </summary>
|
|
/// <param name="data">Data to sign</param>
|
|
/// <param name="outputType">String type</param>
|
|
/// <returns></returns>
|
|
protected string SignHMACSHA512(string data, SignOutputType? outputType = null)
|
|
=> SignHMACSHA512(Encoding.UTF8.GetBytes(data), outputType);
|
|
|
|
/// <summary>
|
|
/// HMACSHA512 sign the data and return the hash
|
|
/// </summary>
|
|
/// <param name="data">Data to sign</param>
|
|
/// <param name="outputType">String type</param>
|
|
/// <returns></returns>
|
|
protected string SignHMACSHA512(byte[] data, SignOutputType? outputType = null)
|
|
{
|
|
using var encryptor = new HMACSHA512(_sBytes);
|
|
var resultBytes = encryptor.ComputeHash(data);
|
|
return outputType == SignOutputType.Base64 ? BytesToBase64String(resultBytes) : BytesToHexString(resultBytes);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sign a string
|
|
/// </summary>
|
|
/// <param name="toSign"></param>
|
|
/// <returns></returns>
|
|
public virtual string Sign(string toSign)
|
|
{
|
|
return toSign;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sign a byte array
|
|
/// </summary>
|
|
/// <param name="toSign"></param>
|
|
/// <returns></returns>
|
|
public virtual byte[] Sign(byte[] toSign)
|
|
{
|
|
return toSign;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Convert byte array to hex string
|
|
/// </summary>
|
|
/// <param name="buff"></param>
|
|
/// <returns></returns>
|
|
protected static string BytesToHexString(byte[] buff)
|
|
{
|
|
var result = string.Empty;
|
|
foreach (var t in buff)
|
|
result += t.ToString("X2");
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Convert byte array to base64 string
|
|
/// </summary>
|
|
/// <param name="buff"></param>
|
|
/// <returns></returns>
|
|
protected static string BytesToBase64String(byte[] buff)
|
|
{
|
|
return Convert.ToBase64String(buff);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get current timestamp including the time sync offset from the api client
|
|
/// </summary>
|
|
/// <param name="apiClient"></param>
|
|
/// <returns></returns>
|
|
protected static DateTime GetTimestamp(RestApiClient apiClient)
|
|
{
|
|
return DateTime.UtcNow.Add(apiClient?.GetTimeOffset() ?? TimeSpan.Zero)!;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get millisecond timestamp as a string including the time sync offset from the api client
|
|
/// </summary>
|
|
/// <param name="apiClient"></param>
|
|
/// <returns></returns>
|
|
protected static string GetMillisecondTimestamp(RestApiClient apiClient)
|
|
{
|
|
return DateTimeConverter.ConvertToMilliseconds(GetTimestamp(apiClient)).Value.ToString(CultureInfo.InvariantCulture);
|
|
}
|
|
}
|
|
}
|