import 'dart:convert'; import 'package:cake_wallet/.secrets.g.dart' as secrets; 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'; import 'package:cw_core/crypto_currency.dart'; import 'package:http/http.dart'; double lightningDoubleToBitcoinDouble({required double amount}) { return amount / 100000000; } double bitcoinDoubleToLightningDouble({required double amount}) { return amount * 100000000; } class TrocadorExchangeProvider extends ExchangeProvider { TrocadorExchangeProvider({this.useTorOnly = false, this.providerStates = const {}}) : _lastUsedRateId = '', _provider = [], super(pairList: supportedPairs(_notSupported)); bool useTorOnly; final Map providerStates; static const List availableProviders = [ 'Swapter', 'StealthEx', 'Simpleswap', 'Swapuz', 'ChangeNow', 'Changehero', 'FixedFloat', 'LetsExchange', 'Exolix', 'Godex', 'Exch', 'CoinCraddle', 'Alfacash', 'LocalMonero', 'XChange', 'NeroSwap', 'Changee', 'BitcoinVN', 'EasyBit', 'WizardSwap', 'Quantex', 'SwapSpace', ]; static const List _notSupported = [ CryptoCurrency.stx, CryptoCurrency.zaddr, ]; static const apiKey = secrets.trocadorApiKey; static const onionApiAuthority = 'trocadorfyhlu27aefre5u7zri66gudtzdyelymftvr4yjwcxhfaqsid.onion'; static const clearNetAuthority = 'trocador.app'; static const markup = secrets.trocadorExchangeMarkup; static const newRatePath = '/api/new_rate'; static const createTradePath = 'api/new_trade'; static const tradePath = 'api/trade'; static const coinPath = 'api/coin'; String _lastUsedRateId; List _provider; @override String get title => 'Trocador'; @override bool get isAvailable => true; @override bool get isEnabled => true; @override bool get supportsFixedRate => true; @override bool get supportsOnionAddress => true; @override ExchangeProviderDescription get description => ExchangeProviderDescription.trocador; @override Future checkIsAvailable() async => true; @override Future fetchLimits( {required CryptoCurrency from, required CryptoCurrency to, required bool isFixedRateMode}) async { final params = { 'api_key': apiKey, 'ticker': _normalizeCurrency(from), 'name': from.name, }; final uri = await _getUri(coinPath, params); final response = await get(uri); if (response.statusCode != 200) throw Exception('Unexpected http status: ${response.statusCode}'); final responseJSON = json.decode(response.body) as List; if (responseJSON.isEmpty) throw Exception('No data'); final coinJson = responseJSON.first as Map; // trocador treats btcln as just bitcoin amounts: if (from == CryptoCurrency.btcln) { return Limits( min: bitcoinDoubleToLightningDouble(amount: (coinJson['minimum'] as double)), max: bitcoinDoubleToLightningDouble(amount: (coinJson['maximum'] as double)), ); } return Limits( min: coinJson['minimum'] as double, max: coinJson['maximum'] as double, ); } @override Future fetchRate( {required CryptoCurrency from, required CryptoCurrency to, required double amount, required bool isFixedRateMode, required bool isReceiveAmount}) async { try { if (amount == 0) return 0.0; double amt = amount; if (from == CryptoCurrency.btcln) { amt = lightningDoubleToBitcoinDouble(amount: amount); } final params = { 'api_key': apiKey, 'ticker_from': _normalizeCurrency(from), 'ticker_to': _normalizeCurrency(to), 'network_from': _networkFor(from), 'network_to': _networkFor(to), if (!isFixedRateMode) 'amount_from': amt.toString(), if (isFixedRateMode) 'amount_to': amt.toString(), 'payment': isFixedRateMode ? 'True' : 'False', 'min_kycrating': 'C', 'markup': markup, }; final uri = await _getUri(newRatePath, params); final response = await get(uri); final responseJSON = json.decode(response.body) as Map; final fromAmount = double.parse(responseJSON['amount_from'].toString()); final toAmount = double.parse(responseJSON['amount_to'].toString()); final rateId = responseJSON['trade_id'] as String? ?? ''; var quotes = responseJSON['quotes']['quotes'] as List; _provider = quotes.map((quote) => quote['provider']).toList(); if (rateId.isNotEmpty) _lastUsedRateId = rateId; return isReceiveAmount ? (amount / fromAmount) : (toAmount / amount); } catch (e) { print(e.toString()); return 0.0; } } @override Future createTrade({ required TradeRequest request, required bool isFixedRateMode, required bool isSendAll, }) async { double fromAmt = double.parse(request.fromAmount); double toAmt = double.parse(request.toAmount); if (request.fromCurrency == CryptoCurrency.btcln) { fromAmt = lightningDoubleToBitcoinDouble(amount: fromAmt); } if (request.toCurrency == CryptoCurrency.btcln) { toAmt = lightningDoubleToBitcoinDouble(amount: toAmt); } final params = { 'api_key': apiKey, 'ticker_from': _normalizeCurrency(request.fromCurrency), 'ticker_to': _normalizeCurrency(request.toCurrency), 'network_from': _networkFor(request.fromCurrency), 'network_to': _networkFor(request.toCurrency), 'payment': isFixedRateMode ? 'True' : 'False', 'min_kycrating': 'C', 'markup': markup, if (!isFixedRateMode) 'amount_from': fromAmt.toString(), if (isFixedRateMode) 'amount_to': toAmt.toString(), 'address': request.toAddress, 'refund': request.refundAddress }; double amt = double.tryParse(request.toAmount) ?? 0; if (request.fromCurrency == CryptoCurrency.btcln) { amt = lightningDoubleToBitcoinDouble(amount: amt); } if (isFixedRateMode) { await fetchRate( from: request.fromCurrency, to: request.toCurrency, amount: amt, isFixedRateMode: true, isReceiveAmount: true, ); params['id'] = _lastUsedRateId; } String firstAvailableProvider = ''; for (var provider in _provider) { if (providerStates.containsKey(provider) && providerStates[provider] == true) { firstAvailableProvider = provider as String; break; } } if (firstAvailableProvider.isEmpty) { throw Exception('No available provider is enabled'); } params['provider'] = firstAvailableProvider; final uri = await _getUri(createTradePath, params); final response = await get(uri); if (response.statusCode == 400) { final responseJSON = json.decode(response.body) as Map; final error = responseJSON['error'] as String; final message = responseJSON['message'] as String; throw Exception('${error}\n$message'); } if (response.statusCode != 200) throw Exception('Unexpected http status: ${response.statusCode}'); final responseJSON = json.decode(response.body) as Map; final id = responseJSON['trade_id'] as String; final inputAddress = responseJSON['address_provider'] as String; final refundAddress = responseJSON['refund_address'] as String; final status = responseJSON['status'] as String; final payoutAddress = responseJSON['address_user'] as String; final date = responseJSON['date'] as String; final password = responseJSON['password'] as String; final providerId = responseJSON['id_provider'] as String; final providerName = responseJSON['provider'] as String; String? responseAmount = responseJSON['amount_from']?.toString(); if (request.fromCurrency == CryptoCurrency.btcln && responseAmount != null) { responseAmount = bitcoinDoubleToLightningDouble(amount: double.parse(responseAmount)).toString(); } responseAmount ??= fromAmt.toString(); return Trade( id: id, from: request.fromCurrency, to: request.toCurrency, provider: description, inputAddress: inputAddress, refundAddress: refundAddress, state: TradeState.deserialize(raw: status), password: password, providerId: providerId, providerName: providerName, createdAt: DateTime.tryParse(date)?.toLocal(), amount: responseAmount, payoutAddress: payoutAddress, isSendAll: isSendAll); } @override Future findTradeById({required String id}) async { final uri = await _getUri(tradePath, {'api_key': apiKey, 'id': id}); return get(uri).then((response) { if (response.statusCode != 200) throw Exception('Unexpected http status: ${response.statusCode}'); final responseListJson = json.decode(response.body) as List; final responseJSON = responseListJson.first; final id = responseJSON['trade_id'] as String; final payoutAddress = responseJSON['address_user'] as String; final refundAddress = responseJSON['refund_address'] as String; final inputAddress = responseJSON['address_provider'] as String; final fromAmount = responseJSON['amount_from']?.toString() ?? '0'; final password = responseJSON['password'] as String; final providerId = responseJSON['id_provider'] as String; final providerName = responseJSON['provider'] as String; return Trade( id: id, from: CryptoCurrency.fromString(responseJSON['ticker_from'] as String), to: CryptoCurrency.fromString(responseJSON['ticker_to'] as String), provider: description, inputAddress: inputAddress, refundAddress: refundAddress, createdAt: DateTime.parse(responseJSON['date'] as String), amount: fromAmount, state: TradeState.deserialize(raw: responseJSON['status'] as String), payoutAddress: payoutAddress, password: password, providerId: providerId, providerName: providerName, ); }); } String _networkFor(CryptoCurrency currency) { switch (currency) { case CryptoCurrency.eth: return 'ERC20'; case CryptoCurrency.maticpoly: return 'Mainnet'; case CryptoCurrency.usdcpoly: case CryptoCurrency.usdtPoly: case CryptoCurrency.usdcEPoly: return 'MATIC'; case CryptoCurrency.zec: return 'Mainnet'; case CryptoCurrency.btcln: return 'Lightning'; default: return currency.tag != null ? _normalizeTag(currency.tag!) : 'Mainnet'; } } String _normalizeCurrency(CryptoCurrency currency) { switch (currency) { case CryptoCurrency.zec: return 'zec'; case CryptoCurrency.usdcEPoly: return 'usdce'; case CryptoCurrency.btcln: return 'btc'; default: return currency.title.toLowerCase(); } } String _normalizeTag(String tag) { switch (tag) { case 'ETH': return 'ERC20'; case 'TRX': return 'TRC20'; case 'LN': return 'Lightning'; default: return tag.toLowerCase(); } } Future _getUri(String path, Map queryParams) async { final uri = Uri.http(onionApiAuthority, path, queryParams); if (useTorOnly) return uri; try { await get(uri); return uri; } catch (e) { return Uri.https(clearNetAuthority, path, queryParams); } } }