mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-01-24 11:36:21 +00:00
update fiat conversion service
This commit is contained in:
parent
cbedd8e2c6
commit
97877c4c7f
2 changed files with 97 additions and 56 deletions
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue