2020-06-29 04:44:35 +02:00
|
|
|
using System;
|
2017-09-13 08:47:34 +02:00
|
|
|
using System.Collections.Generic;
|
2020-06-28 10:55:27 +02:00
|
|
|
using System.Globalization;
|
2017-09-13 08:47:34 +02:00
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
2020-06-28 10:55:27 +02:00
|
|
|
using System.Reflection;
|
2017-09-13 08:47:34 +02:00
|
|
|
using System.Text;
|
2024-10-04 15:24:44 +02:00
|
|
|
using System.Threading;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using Microsoft.Extensions.Logging;
|
2020-06-28 10:55:27 +02:00
|
|
|
using NBitcoin;
|
2020-05-31 12:18:29 +02:00
|
|
|
using Newtonsoft.Json;
|
2017-09-13 08:47:34 +02:00
|
|
|
|
2017-09-15 09:06:57 +02:00
|
|
|
namespace BTCPayServer.Services.Rates
|
2017-09-13 08:47:34 +02:00
|
|
|
{
|
2017-10-27 10:53:04 +02:00
|
|
|
public class CurrencyData
|
2017-09-13 08:47:34 +02:00
|
|
|
{
|
2020-05-31 12:18:29 +02:00
|
|
|
public string Name { get; set; }
|
|
|
|
public string Code { get; set; }
|
|
|
|
public int Divisibility { get; set; }
|
|
|
|
public string Symbol { get; set; }
|
2018-05-16 14:19:48 +02:00
|
|
|
public bool Crypto { get; set; }
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
2024-10-04 15:24:44 +02:00
|
|
|
public interface CurrencyDataProvider
|
|
|
|
{
|
|
|
|
Task<CurrencyData[]> LoadCurrencyData(CancellationToken cancellationToken);
|
|
|
|
}
|
|
|
|
public class InMemoryCurrencyDataProvider : CurrencyDataProvider
|
|
|
|
{
|
|
|
|
private readonly CurrencyData[] _currencyData;
|
|
|
|
|
|
|
|
public InMemoryCurrencyDataProvider(CurrencyData[] currencyData)
|
|
|
|
{
|
|
|
|
_currencyData = currencyData;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Task<CurrencyData[]> LoadCurrencyData(CancellationToken cancellationToken) => Task.FromResult(_currencyData);
|
|
|
|
}
|
|
|
|
public class AssemblyCurrencyDataProvider : CurrencyDataProvider
|
|
|
|
{
|
|
|
|
private readonly Assembly _assembly;
|
|
|
|
private readonly string _manifestResourceStream;
|
|
|
|
|
|
|
|
public AssemblyCurrencyDataProvider(Assembly assembly, string manifestResourceStream)
|
|
|
|
{
|
|
|
|
_assembly = assembly;
|
|
|
|
_manifestResourceStream = manifestResourceStream;
|
|
|
|
}
|
|
|
|
public Task<CurrencyData[]> LoadCurrencyData(CancellationToken cancellationToken)
|
|
|
|
{
|
|
|
|
var stream = _assembly.GetManifestResourceStream(_manifestResourceStream);
|
|
|
|
if (stream is null)
|
|
|
|
throw new InvalidOperationException("Unknown manifestResourceStream");
|
|
|
|
string content = null;
|
|
|
|
using (var reader = new StreamReader(stream, Encoding.UTF8))
|
|
|
|
{
|
|
|
|
content = reader.ReadToEnd();
|
|
|
|
}
|
|
|
|
|
|
|
|
var currencies = JsonConvert.DeserializeObject<CurrencyData[]>(content);
|
|
|
|
return Task.FromResult(currencies.ToArray());
|
|
|
|
}
|
|
|
|
}
|
2017-10-27 10:53:04 +02:00
|
|
|
public class CurrencyNameTable
|
|
|
|
{
|
2024-10-04 15:24:44 +02:00
|
|
|
public CurrencyNameTable(IEnumerable<CurrencyDataProvider> currencyDataProviders, ILogger<CurrencyNameTable> logger)
|
2017-10-27 10:53:04 +02:00
|
|
|
{
|
2024-10-04 15:24:44 +02:00
|
|
|
_currencyDataProviders = currencyDataProviders;
|
|
|
|
_logger = logger;
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
|
|
|
|
2024-10-04 15:24:44 +02:00
|
|
|
public async Task ReloadCurrencyData(CancellationToken cancellationToken)
|
|
|
|
{
|
|
|
|
var currencies = new Dictionary<string, CurrencyData>(StringComparer.InvariantCultureIgnoreCase);
|
|
|
|
var loadings = _currencyDataProviders.Select(c => (Task: c.LoadCurrencyData(cancellationToken), Prov: c)).ToList();
|
|
|
|
foreach (var loading in loadings)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
foreach (var curr in await loading.Task)
|
|
|
|
{
|
|
|
|
currencies.TryAdd(curr.Code, curr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
_logger.LogWarning(ex, "Error loading currency data for " + loading.Prov.GetType().FullName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_Currencies = currencies;
|
|
|
|
}
|
2023-10-11 14:12:33 +02:00
|
|
|
static readonly Dictionary<string, IFormatProvider> _CurrencyProviders = new();
|
2019-02-17 08:53:41 +01:00
|
|
|
|
2018-05-20 16:37:18 +02:00
|
|
|
public NumberFormatInfo GetNumberFormatInfo(string currency, bool useFallback)
|
2018-05-15 19:24:59 +02:00
|
|
|
{
|
|
|
|
var data = GetCurrencyProvider(currency);
|
|
|
|
if (data is NumberFormatInfo nfi)
|
|
|
|
return nfi;
|
2018-05-20 16:22:20 +02:00
|
|
|
if (data is CultureInfo ci)
|
|
|
|
return ci.NumberFormat;
|
2018-05-20 16:37:18 +02:00
|
|
|
if (!useFallback)
|
|
|
|
return null;
|
2018-05-20 16:22:20 +02:00
|
|
|
return CreateFallbackCurrencyFormatInfo(currency);
|
2018-05-15 19:24:59 +02:00
|
|
|
}
|
2018-05-20 16:22:20 +02:00
|
|
|
|
|
|
|
private NumberFormatInfo CreateFallbackCurrencyFormatInfo(string currency)
|
|
|
|
{
|
2018-05-20 16:37:18 +02:00
|
|
|
var usd = GetNumberFormatInfo("USD", false);
|
2018-05-20 16:22:20 +02:00
|
|
|
var currencyInfo = (NumberFormatInfo)usd.Clone();
|
|
|
|
currencyInfo.CurrencySymbol = currency;
|
|
|
|
return currencyInfo;
|
|
|
|
}
|
2023-03-13 02:12:58 +01:00
|
|
|
|
2018-12-04 05:04:26 +01:00
|
|
|
public NumberFormatInfo GetNumberFormatInfo(string currency)
|
|
|
|
{
|
|
|
|
var curr = GetCurrencyProvider(currency);
|
|
|
|
if (curr is CultureInfo cu)
|
|
|
|
return cu.NumberFormat;
|
|
|
|
if (curr is NumberFormatInfo ni)
|
|
|
|
return ni;
|
|
|
|
return null;
|
|
|
|
}
|
2023-03-13 02:12:58 +01:00
|
|
|
|
2017-10-27 11:58:43 +02:00
|
|
|
public IFormatProvider GetCurrencyProvider(string currency)
|
|
|
|
{
|
|
|
|
lock (_CurrencyProviders)
|
|
|
|
{
|
|
|
|
if (_CurrencyProviders.Count == 0)
|
|
|
|
{
|
2024-01-18 01:31:35 +01:00
|
|
|
foreach (var culture in CultureInfo.GetCultures(CultureTypes.AllCultures))
|
2017-10-27 11:58:43 +02:00
|
|
|
{
|
2024-01-18 01:31:35 +01:00
|
|
|
// This avoid storms of exception throwing slowing up
|
|
|
|
// startup and debugging sessions
|
|
|
|
if (culture switch
|
|
|
|
{
|
|
|
|
{ LCID: 0x007F or 0x0000 or 0x0c00 or 0x1000 } => true,
|
|
|
|
{ IsNeutralCulture : true } => true,
|
|
|
|
_ => false
|
|
|
|
})
|
|
|
|
continue;
|
2017-10-27 11:58:43 +02:00
|
|
|
try
|
|
|
|
{
|
2024-02-02 09:16:13 +01:00
|
|
|
var symbol = new RegionInfo(culture.LCID).ISOCurrencySymbol;
|
|
|
|
var c = symbol switch
|
|
|
|
{
|
|
|
|
// ARS and COP are officially 2 digits, but due to depreciation,
|
|
|
|
// nobody really use those anymore. (See https://github.com/btcpayserver/btcpayserver/issues/5708)
|
|
|
|
"ARS" or "COP" => ModifyCurrencyDecimalDigit(culture, 0),
|
|
|
|
_ => culture
|
|
|
|
};
|
|
|
|
_CurrencyProviders.TryAdd(symbol, c);
|
2017-10-27 11:58:43 +02:00
|
|
|
}
|
|
|
|
catch { }
|
|
|
|
}
|
2018-05-15 19:24:59 +02:00
|
|
|
|
2020-05-31 12:18:29 +02:00
|
|
|
foreach (var curr in _Currencies.Where(pair => pair.Value.Crypto))
|
2018-05-15 19:24:59 +02:00
|
|
|
{
|
2020-06-28 10:55:27 +02:00
|
|
|
AddCurrency(_CurrencyProviders, curr.Key, curr.Value.Divisibility, curr.Value.Symbol ?? curr.Value.Code);
|
2018-05-15 19:24:59 +02:00
|
|
|
}
|
2017-10-27 11:58:43 +02:00
|
|
|
}
|
2019-06-09 17:46:29 +02:00
|
|
|
return _CurrencyProviders.TryGet(currency.ToUpperInvariant());
|
2017-10-27 11:58:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-02 09:16:13 +01:00
|
|
|
private CultureInfo ModifyCurrencyDecimalDigit(CultureInfo culture, int decimals)
|
|
|
|
{
|
|
|
|
var modifiedCulture = new CultureInfo(culture.Name);
|
|
|
|
NumberFormatInfo modifiedNumberFormat = (NumberFormatInfo)modifiedCulture.NumberFormat.Clone();
|
|
|
|
modifiedNumberFormat.CurrencyDecimalDigits = decimals;
|
|
|
|
modifiedCulture.NumberFormat = modifiedNumberFormat;
|
|
|
|
return modifiedCulture;
|
|
|
|
}
|
|
|
|
|
2017-10-27 11:58:43 +02:00
|
|
|
private void AddCurrency(Dictionary<string, IFormatProvider> currencyProviders, string code, int divisibility, string symbol)
|
|
|
|
{
|
|
|
|
var culture = new CultureInfo("en-US");
|
|
|
|
var number = new NumberFormatInfo();
|
|
|
|
number.CurrencyDecimalDigits = divisibility;
|
|
|
|
number.CurrencySymbol = symbol;
|
|
|
|
number.CurrencyDecimalSeparator = culture.NumberFormat.CurrencyDecimalSeparator;
|
|
|
|
number.CurrencyGroupSeparator = culture.NumberFormat.CurrencyGroupSeparator;
|
|
|
|
number.CurrencyGroupSizes = culture.NumberFormat.CurrencyGroupSizes;
|
|
|
|
number.CurrencyNegativePattern = 8;
|
|
|
|
number.CurrencyPositivePattern = 3;
|
|
|
|
number.NegativeSign = culture.NumberFormat.NegativeSign;
|
|
|
|
currencyProviders.TryAdd(code, number);
|
|
|
|
}
|
2017-09-13 08:47:34 +02:00
|
|
|
|
2024-10-04 15:24:44 +02:00
|
|
|
Dictionary<string, CurrencyData> _Currencies = new();
|
|
|
|
private readonly IEnumerable<CurrencyDataProvider> _currencyDataProviders;
|
|
|
|
private readonly ILogger<CurrencyNameTable> _logger;
|
2017-09-13 08:47:34 +02:00
|
|
|
|
2022-01-24 12:00:13 +01:00
|
|
|
public IEnumerable<CurrencyData> Currencies => _Currencies.Values;
|
|
|
|
|
2018-05-20 16:37:18 +02:00
|
|
|
public CurrencyData GetCurrencyData(string currency, bool useFallback)
|
2017-10-27 10:53:04 +02:00
|
|
|
{
|
2021-12-28 09:39:54 +01:00
|
|
|
ArgumentNullException.ThrowIfNull(currency);
|
2017-10-27 10:53:04 +02:00
|
|
|
CurrencyData result;
|
2018-10-09 16:30:06 +02:00
|
|
|
if (!_Currencies.TryGetValue(currency.ToUpperInvariant(), out result))
|
2018-05-20 16:37:18 +02:00
|
|
|
{
|
2018-10-09 16:30:06 +02:00
|
|
|
if (useFallback)
|
2018-05-20 16:37:18 +02:00
|
|
|
{
|
|
|
|
var usd = GetCurrencyData("USD", false);
|
|
|
|
result = new CurrencyData()
|
|
|
|
{
|
|
|
|
Code = currency,
|
|
|
|
Crypto = true,
|
|
|
|
Name = currency,
|
|
|
|
Divisibility = usd.Divisibility
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
2017-10-27 10:53:04 +02:00
|
|
|
return result;
|
|
|
|
}
|
2017-09-13 08:47:34 +02:00
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
2017-09-13 08:47:34 +02:00
|
|
|
}
|