update fiat conversion service

This commit is contained in:
Serhii 2024-05-08 14:16:44 +03:00
parent cbedd8e2c6
commit 97877c4c7f
2 changed files with 97 additions and 56 deletions

View file

@ -50,19 +50,19 @@ Future<double> _fetchPrice(Map<String, dynamic> args) async {
} }
} }
Future<double?> _fetchHistoricalPrice(Map<String, dynamic> args) async { Future<Map<String, dynamic>?> _fetchHistoricalPrice(Map<String, dynamic> args) async {
final crypto = args['crypto'] as CryptoCurrency; final crypto = args['crypto'] as CryptoCurrency;
final fiat = args['fiat'] as FiatCurrency; final fiat = args['fiat'] as FiatCurrency;
final torOnly = args['torOnly'] as bool; final torOnly = args['torOnly'] as bool;
final date = args['date'] as DateTime; final intervalCount = args['intervalCount'] as int;
final intervalFromNow = DateTime.now().difference(date).inMinutes; final intervalMinutes = args['intervalMinutes'] as int;
final Map<String, String> queryParams = { final Map<String, String> queryParams = {
'interval_count': '2', 'interval_count': intervalCount.toString(),
'base': crypto.toString(), 'base': crypto.toString(),
'quote': fiat.toString(), 'quote': fiat.toString(),
'key': secrets.fiatApiKey, 'key': secrets.fiatApiKey,
'interval_minutes': intervalFromNow.toString() 'interval_minutes': intervalMinutes.toString()
}; };
try { try {
@ -78,19 +78,12 @@ Future<double?> _fetchHistoricalPrice(Map<String, dynamic> args) async {
if (response.statusCode != 200) return null; if (response.statusCode != 200) return null;
final data = json.decode(response.body) as Map<String, dynamic>; final data = json.decode(response.body) as Map<String, dynamic>;
final errors = data['errors'] as Map<String, dynamic>;
if (errors.isNotEmpty) return null;
final results = data['results'] as Map<String, dynamic>; final results = data['results'] as Map<String, dynamic>;
if (results.isNotEmpty) { if (results.isNotEmpty) return results;
return (results.values.first as double) / 100000000;
} else {
return null; return null;
}
} catch (e) { } catch (e) {
print(e.toString()); print(e.toString());
return null; return null;
@ -104,10 +97,15 @@ Future<double> _fetchPriceAsync(CryptoCurrency crypto, FiatCurrency fiat, bool t
'torOnly': torOnly, 'torOnly': torOnly,
}); });
Future<double?> _fetchHistoricalAsync( Future<Map<String, dynamic>?> _fetchHistoricalAsync(CryptoCurrency crypto, FiatCurrency fiat, bool torOnly,
CryptoCurrency crypto, FiatCurrency fiat, bool torOnly, DateTime date) async => int intervalCount, int intervalMinutes) async =>
compute( compute(_fetchHistoricalPrice, {
_fetchHistoricalPrice, {'fiat': fiat, 'crypto': crypto, 'torOnly': torOnly, 'date': date}); 'fiat': fiat,
'crypto': crypto,
'torOnly': torOnly,
'intervalCount': intervalCount,
'intervalMinutes': intervalMinutes
});
class FiatConversionService { class FiatConversionService {
static Future<double> fetchPrice({ static Future<double> fetchPrice({
@ -117,11 +115,12 @@ class FiatConversionService {
}) async => }) async =>
await _fetchPriceAsync(crypto, fiat, torOnly); await _fetchPriceAsync(crypto, fiat, torOnly);
static Future<double?> fetchHistoricalPrice({ static Future<Map<String, dynamic>?> fetchHistoricalPrice({
required CryptoCurrency crypto, required CryptoCurrency crypto,
required FiatCurrency fiat, required FiatCurrency fiat,
required bool torOnly, required bool torOnly,
required DateTime date, required int intervalCount,
required int intervalMinutes,
}) async => }) async =>
await _fetchHistoricalAsync(crypto, fiat, torOnly, date); await _fetchHistoricalAsync(crypto, fiat, torOnly, intervalCount, intervalMinutes);
} }

View file

@ -4,7 +4,9 @@ import 'package:cake_wallet/entities/fiat_api_mode.dart';
import 'package:cake_wallet/entities/transaction_description.dart'; import 'package:cake_wallet/entities/transaction_description.dart';
import 'package:cake_wallet/store/dashboard/fiat_conversion_store.dart'; import 'package:cake_wallet/store/dashboard/fiat_conversion_store.dart';
import 'package:cake_wallet/store/settings_store.dart'; import 'package:cake_wallet/store/settings_store.dart';
import 'package:cw_core/amount_converter.dart';
import 'package:cw_core/wallet_base.dart'; import 'package:cw_core/wallet_base.dart';
import 'package:cw_core/wallet_type.dart';
import 'package:hive/hive.dart'; import 'package:hive/hive.dart';
Future<void> historicalRateUpdate( Future<void> historicalRateUpdate(
@ -12,26 +14,56 @@ Future<void> historicalRateUpdate(
SettingsStore settingsStore, SettingsStore settingsStore,
FiatConversionStore fiatConversionStore, FiatConversionStore fiatConversionStore,
Box<TransactionDescription> transactionDescription) async { Box<TransactionDescription> transactionDescription) async {
final transactions = wallet.transactionHistory.transactions.values.toList(); int accuracyMinutes = 0;
const int batchSize = 10; switch (wallet.type) {
const Duration delayBetweenBatches = Duration(milliseconds: 2); case WalletType.monero:
case WalletType.polygon:
case WalletType.nano:
case WalletType.solana:
case WalletType.haven:
case WalletType.tron:
accuracyMinutes = 540; // 9 hours
break;
case WalletType.ethereum:
accuracyMinutes = 360; // 6 hours
break;
case WalletType.bitcoin:
case WalletType.bitcoinCash:
case WalletType.litecoin:
accuracyMinutes = 180; // 3 hours
break;
default:
accuracyMinutes = 180; // 3 hours
}
int nextBatchStart = 0; final historicalRateStorageDurationMinutes = 86400; // 2 months
final intervalCount = historicalRateStorageDurationMinutes ~/ accuracyMinutes;
final totalAllowedAgeMinutes = historicalRateStorageDurationMinutes + accuracyMinutes;
final currentTime = DateTime.now();
for (int i = 0; i < transactions.length; i += batchSize) { final result = await FiatConversionService.fetchHistoricalPrice(
final batch = transactions.skip(i).take(batchSize); crypto: wallet.currency,
fiat: settingsStore.fiatCurrency,
torOnly: settingsStore.fiatApiMode == FiatApiMode.torOnly,
intervalCount: intervalCount,
intervalMinutes: accuracyMinutes);
bool needsProcessing = batch.any((tx) { if (result == null) return;
var description = transactionDescription.get(tx.id);
final fiatName = settingsStore.fiatCurrency.toString(); Map<DateTime, double> convertedRates = {};
return description == null ||
description.historicalRates.isEmpty || result.forEach((key, value) {
!description.historicalRates.containsKey(fiatName); DateTime keyAsDateTime = DateTime.parse(key).toUtc();
convertedRates[keyAsDateTime] = value as double;
}); });
if (needsProcessing) { final transactions = wallet.transactionHistory.transactions.values.toList();
await Future.wait(batch.map((tx) async {
for (var tx in transactions) {
final txAgeMinutes = currentTime.difference(tx.date).inMinutes;
if (txAgeMinutes > totalAllowedAgeMinutes) continue;
var description = transactionDescription.get(tx.id); var description = transactionDescription.get(tx.id);
final fiatName = settingsStore.fiatCurrency.toString(); final fiatName = settingsStore.fiatCurrency.toString();
@ -39,29 +71,39 @@ Future<void> historicalRateUpdate(
description.historicalRates.isEmpty || description.historicalRates.isEmpty ||
!description.historicalRates.containsKey(fiatName)) { !description.historicalRates.containsKey(fiatName)) {
try { try {
final result = await FiatConversionService.fetchHistoricalPrice( List<DateTime> historyTimestamps = convertedRates.keys.toList();
crypto: wallet.currency, final txHistoryTimestamps = tx.date.toUtc();
fiat: settingsStore.fiatCurrency,
torOnly: settingsStore.fiatApiMode == FiatApiMode.torOnly,
date: tx.date);
if (result == null) return; final closestTimestamp = findClosestTimestamp(historyTimestamps, txHistoryTimestamps);
if (closestTimestamp != null &&
txHistoryTimestamps.difference(closestTimestamp).abs() <=
Duration(minutes: accuracyMinutes)) {
final rate = convertedRates[closestTimestamp];
if (rate != null) {
description ??= TransactionDescription(id: tx.id); description ??= TransactionDescription(id: tx.id);
Map<String, String> rates = description.historicalRates; Map<String, String> rates = description.historicalRates;
rates[fiatName] = (result * tx.amount).toString(); rates[fiatName] =
(rate * AmountConverter.amountIntToDouble(wallet.currency, tx.amount)).toString();
description.historicalRates = rates; description.historicalRates = rates;
await transactionDescription.put(tx.id, description); await transactionDescription.put(tx.id, description);
}
}
} catch (e) { } catch (e) {
print("Error fetching historical price: $e"); print("Error fetching historical price: $e");
} }
} }
})); }
}
nextBatchStart = i + batchSize; DateTime? findClosestTimestamp(List<DateTime> timestamps, DateTime target) {
if (nextBatchStart < transactions.length) { DateTime? closest;
await Future.delayed(delayBetweenBatches); for (var timestamp in timestamps) {
} if (closest == null ||
(target.difference(timestamp).abs() < target.difference(closest).abs())) {
closest = timestamp;
} }
} }
return closest;
} }