2024-01-25 20:35:58 +00:00
|
|
|
import 'dart:convert';
|
|
|
|
|
2024-01-29 11:40:40 +00:00
|
|
|
import 'package:cake_wallet/entities/preferences_key.dart';
|
2024-01-25 20:35:58 +00:00
|
|
|
import 'package:cake_wallet/exchange/exchange_provider_description.dart';
|
|
|
|
import 'package:cake_wallet/exchange/limits.dart';
|
|
|
|
import 'package:cake_wallet/exchange/provider/exchange_provider.dart';
|
|
|
|
import 'package:cake_wallet/exchange/trade.dart';
|
|
|
|
import 'package:cake_wallet/exchange/trade_request.dart';
|
|
|
|
import 'package:cake_wallet/exchange/trade_state.dart';
|
|
|
|
import 'package:cake_wallet/exchange/utils/currency_pairs_utils.dart';
|
2024-01-29 13:48:36 +00:00
|
|
|
import 'package:collection/collection.dart';
|
2024-01-28 13:02:59 +00:00
|
|
|
import 'package:cw_core/amount_converter.dart';
|
2024-01-25 20:35:58 +00:00
|
|
|
import 'package:cw_core/crypto_currency.dart';
|
2024-01-29 11:40:40 +00:00
|
|
|
import 'package:hive/hive.dart';
|
2024-01-25 20:35:58 +00:00
|
|
|
import 'package:http/http.dart' as http;
|
2024-01-29 11:40:40 +00:00
|
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
2024-01-25 20:35:58 +00:00
|
|
|
|
|
|
|
class ThorChainExchangeProvider extends ExchangeProvider {
|
2024-01-29 11:40:40 +00:00
|
|
|
ThorChainExchangeProvider({required this.tradesStore})
|
|
|
|
: super(pairList: supportedPairs(_notSupported));
|
2024-01-25 20:35:58 +00:00
|
|
|
|
|
|
|
static final List<CryptoCurrency> _notSupported = [
|
|
|
|
...(CryptoCurrency.all
|
2024-01-28 13:02:59 +00:00
|
|
|
.where((element) => ![
|
|
|
|
CryptoCurrency.btc,
|
|
|
|
CryptoCurrency.eth,
|
|
|
|
CryptoCurrency.ltc,
|
|
|
|
].contains(element))
|
2024-01-25 20:35:58 +00:00
|
|
|
.toList())
|
|
|
|
];
|
|
|
|
|
|
|
|
static const _baseURL = 'https://thornode.ninerealms.com';
|
|
|
|
static const _quotePath = '/thorchain/quote/swap';
|
2024-01-28 13:02:59 +00:00
|
|
|
static const _affiliateName = 'cakewallet';
|
2024-01-29 13:48:36 +00:00
|
|
|
static const _affiliateBps = '10';
|
2024-01-25 20:35:58 +00:00
|
|
|
|
2024-01-29 11:40:40 +00:00
|
|
|
final Box<Trade> tradesStore;
|
|
|
|
|
2024-01-25 20:35:58 +00:00
|
|
|
@override
|
|
|
|
String get title => 'ThorChain';
|
|
|
|
|
|
|
|
@override
|
|
|
|
bool get isAvailable => true;
|
|
|
|
|
|
|
|
@override
|
|
|
|
bool get isEnabled => true;
|
|
|
|
|
|
|
|
@override
|
2024-01-30 11:55:54 +00:00
|
|
|
bool get supportsFixedRate => false;
|
2024-01-25 20:35:58 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
ExchangeProviderDescription get description => ExchangeProviderDescription.thorChain;
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<bool> checkIsAvailable() async => true;
|
|
|
|
|
2024-01-28 13:02:59 +00:00
|
|
|
Future<Map<String, dynamic>> _getSwapQuote(Map<String, String> params) async {
|
|
|
|
final uri = Uri.parse('$_baseURL$_quotePath${Uri(queryParameters: params)}');
|
|
|
|
final response = await http.get(uri);
|
|
|
|
|
|
|
|
if (response.statusCode != 200) {
|
|
|
|
throw Exception('Unexpected HTTP status: ${response.statusCode}');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (response.body.contains('error')) {
|
|
|
|
throw Exception('Unexpected response: ${response.body}');
|
|
|
|
}
|
|
|
|
|
|
|
|
return json.decode(response.body) as Map<String, dynamic>;
|
|
|
|
}
|
|
|
|
|
2024-01-25 20:35:58 +00:00
|
|
|
@override
|
|
|
|
Future<Limits> fetchLimits(
|
|
|
|
{required CryptoCurrency from,
|
|
|
|
required CryptoCurrency to,
|
|
|
|
required bool isFixedRateMode}) async {
|
|
|
|
final params = {
|
|
|
|
'from_asset': _normalizeCurrency(from),
|
|
|
|
'to_asset': _normalizeCurrency(to),
|
2024-01-28 13:02:59 +00:00
|
|
|
'amount': AmountConverter.amountToSmallestUnit(cryptoCurrency: from, amount: 1).toString(),
|
2024-01-30 13:09:46 +00:00
|
|
|
'affiliate': _affiliateName,
|
|
|
|
'affiliate_bps': _affiliateBps
|
2024-01-25 20:35:58 +00:00
|
|
|
};
|
|
|
|
|
2024-01-28 13:02:59 +00:00
|
|
|
final responseJSON = await _getSwapQuote(params);
|
2024-01-25 20:35:58 +00:00
|
|
|
final minAmountIn = responseJSON['recommended_min_amount_in'] as String?;
|
2024-01-28 13:02:59 +00:00
|
|
|
final formattedMinAmountIn = minAmountIn != null ? int.parse(minAmountIn) / 1e8 : 0.0;
|
2024-01-25 20:35:58 +00:00
|
|
|
|
|
|
|
return Limits(min: formattedMinAmountIn);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<Trade> createTrade({required TradeRequest request, required bool isFixedRateMode}) async {
|
|
|
|
double amountBTC = double.parse(request.fromAmount);
|
2024-01-28 13:02:59 +00:00
|
|
|
String formattedAmount = AmountConverter.amountToSmallestUnit(
|
|
|
|
cryptoCurrency: request.fromCurrency, amount: amountBTC)
|
|
|
|
.toString();
|
2024-01-25 20:35:58 +00:00
|
|
|
|
|
|
|
final params = {
|
|
|
|
'from_asset': _normalizeCurrency(request.fromCurrency),
|
|
|
|
'to_asset': _normalizeCurrency(request.toCurrency),
|
|
|
|
'amount': formattedAmount,
|
|
|
|
'destination': request.toAddress,
|
2024-01-28 13:02:59 +00:00
|
|
|
'affiliate': _affiliateName,
|
2024-01-29 11:40:40 +00:00
|
|
|
'affiliate_bps': _affiliateBps
|
|
|
|
};
|
2024-01-25 20:35:58 +00:00
|
|
|
|
2024-01-28 13:02:59 +00:00
|
|
|
final responseJSON = await _getSwapQuote(params);
|
2024-01-25 20:35:58 +00:00
|
|
|
|
|
|
|
final inputAddress = responseJSON['inbound_address'] as String?;
|
2024-01-28 13:02:59 +00:00
|
|
|
final memo = responseJSON['memo'] as String?;
|
2024-01-29 11:40:40 +00:00
|
|
|
final tradeId = await getNextTradeCounter();
|
2024-01-25 20:35:58 +00:00
|
|
|
|
|
|
|
return Trade(
|
2024-01-29 11:40:40 +00:00
|
|
|
id: tradeId.toString(),
|
2024-01-25 20:35:58 +00:00
|
|
|
from: request.fromCurrency,
|
|
|
|
to: request.toCurrency,
|
|
|
|
provider: description,
|
|
|
|
inputAddress: inputAddress,
|
|
|
|
createdAt: DateTime.now(),
|
|
|
|
amount: request.fromAmount,
|
|
|
|
state: TradeState.created,
|
2024-01-28 13:02:59 +00:00
|
|
|
payoutAddress: request.toAddress,
|
|
|
|
memo: memo);
|
2024-01-25 20:35:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<double> fetchRate(
|
|
|
|
{required CryptoCurrency from,
|
|
|
|
required CryptoCurrency to,
|
|
|
|
required double amount,
|
|
|
|
required bool isFixedRateMode,
|
|
|
|
required bool isReceiveAmount}) async {
|
|
|
|
try {
|
|
|
|
if (amount == 0) return 0.0;
|
|
|
|
|
|
|
|
final params = {
|
|
|
|
'from_asset': _normalizeCurrency(from),
|
|
|
|
'to_asset': _normalizeCurrency(to),
|
2024-01-28 13:02:59 +00:00
|
|
|
'amount':
|
|
|
|
AmountConverter.amountToSmallestUnit(cryptoCurrency: from, amount: amount).toString(),
|
2024-01-25 20:35:58 +00:00
|
|
|
};
|
|
|
|
|
2024-01-28 13:02:59 +00:00
|
|
|
final responseJSON = await _getSwapQuote(params);
|
2024-01-25 20:35:58 +00:00
|
|
|
|
|
|
|
final expectedAmountOutString = responseJSON['expected_amount_out'] as String? ?? '0';
|
|
|
|
final expectedAmountOut = double.parse(expectedAmountOutString);
|
2024-01-28 13:02:59 +00:00
|
|
|
double formattedAmountOut = 0.0;
|
2024-01-25 20:35:58 +00:00
|
|
|
|
2024-01-28 13:02:59 +00:00
|
|
|
if (to == CryptoCurrency.eth) {
|
|
|
|
formattedAmountOut = expectedAmountOut / 1e9;
|
|
|
|
} else {
|
|
|
|
formattedAmountOut = AmountConverter.amountIntToDouble(to, expectedAmountOut.toInt());
|
|
|
|
}
|
2024-01-25 20:35:58 +00:00
|
|
|
|
|
|
|
return formattedAmountOut;
|
|
|
|
} catch (e) {
|
|
|
|
print(e.toString());
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-29 11:40:40 +00:00
|
|
|
Future<Trade> findTradeById({required String id}) async {
|
|
|
|
final foundTrade = tradesStore.values.firstWhereOrNull((element) => element.id == id);
|
|
|
|
if (foundTrade == null) {
|
|
|
|
throw Exception('Trade with id $id not found');
|
|
|
|
}
|
|
|
|
return foundTrade;
|
2024-01-25 20:35:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
String _normalizeCurrency(CryptoCurrency currency) {
|
|
|
|
switch (currency) {
|
|
|
|
case CryptoCurrency.btc:
|
|
|
|
return 'BTC.BTC';
|
|
|
|
case CryptoCurrency.eth:
|
|
|
|
return 'ETH.ETH';
|
2024-01-28 13:02:59 +00:00
|
|
|
case CryptoCurrency.ltc:
|
|
|
|
return 'LTC.LTC';
|
|
|
|
case CryptoCurrency.bch:
|
|
|
|
return 'BCH.BCH';
|
2024-01-25 20:35:58 +00:00
|
|
|
default:
|
|
|
|
return currency.title.toLowerCase();
|
|
|
|
}
|
|
|
|
}
|
2024-01-29 11:40:40 +00:00
|
|
|
|
|
|
|
Future<int> getNextTradeCounter() async {
|
|
|
|
final prefs = await SharedPreferences.getInstance();
|
|
|
|
int currentCounter = prefs.getInt(PreferencesKey.thorChainTradeCounter) ?? 0;
|
|
|
|
currentCounter++;
|
|
|
|
await prefs.setInt(PreferencesKey.thorChainTradeCounter, currentCounter);
|
|
|
|
return currentCounter;
|
|
|
|
}
|
2024-01-25 20:35:58 +00:00
|
|
|
}
|