import 'dart:convert'; import 'package:cake_wallet/anonpay/anonpay_donation_link_info.dart'; import 'package:cake_wallet/anonpay/anonpay_invoice_info.dart'; import 'package:cake_wallet/anonpay/anonpay_request.dart'; import 'package:cake_wallet/anonpay/anonpay_status_response.dart'; import 'package:cake_wallet/core/fiat_conversion_service.dart'; import 'package:cake_wallet/entities/exchange_api_mode.dart'; import 'package:cake_wallet/entities/fiat_api_mode.dart'; import 'package:cake_wallet/entities/fiat_currency.dart'; import 'package:cake_wallet/exchange/limits.dart'; import 'package:cw_core/wallet_base.dart'; import 'package:http/http.dart'; import 'package:cw_core/crypto_currency.dart'; import 'package:cake_wallet/.secrets.g.dart' as secrets; class AnonPayApi { const AnonPayApi({ this.apiMode = ExchangeApiMode.enabled, required this.wallet, }); final ExchangeApiMode apiMode; final WalletBase wallet; static const anonpayRef = secrets.anonPayReferralCode; static const onionApiAuthority = 'trocadorfyhlu27aefre5u7zri66gudtzdyelymftvr4yjwcxhfaqsid.onion'; static const clearNetAuthority = 'trocador.app'; static const markup = secrets.trocadorExchangeMarkup; static const anonPayPath = '/anonpay'; static const anonPayStatus = '/anonpay/status'; static const coinPath = 'api/coin'; static const apiKey = secrets.trocadorApiKey; Future paymentStatus(String id) async { final authority = await _getAuthority(); final response = await get(Uri.https(authority, "$anonPayStatus/$id")); final responseJSON = json.decode(response.body) as Map; final status = responseJSON['Status'] as String; final fiatAmount = responseJSON['Fiat_Amount'] as double?; final fiatEquiv = responseJSON['Fiat_Equiv'] as String?; final amountTo = responseJSON['AmountTo'] as double?; final coinTo = responseJSON['CoinTo'] as String; final address = responseJSON['Address'] as String; return AnonpayStatusResponse( status: status, fiatAmount: fiatAmount, amountTo: amountTo, coinTo: coinTo, address: address, fiatEquiv: fiatEquiv, ); } Future createInvoice(AnonPayRequest request) async { final description = Uri.encodeComponent(request.description); final body = { 'ticker_to': request.cryptoCurrency.title.toLowerCase(), 'network_to': _networkFor(request.cryptoCurrency), 'address': request.address, 'name': request.name, 'description': description, 'email': request.email, 'ref': anonpayRef, 'markup': markup, 'direct': 'False', }; if (request.amount != null) { body['amount'] = request.amount; } if (request.fiatEquivalent != null) { body['fiat_equiv'] = request.fiatEquivalent; } final authority = await _getAuthority(); final response = await get(Uri.https(authority, anonPayPath, body)); final responseJSON = json.decode(response.body) as Map; final id = responseJSON['ID'] as String; final url = responseJSON['url'] as String; final urlOnion = responseJSON['url_onion'] as String; final statusUrl = responseJSON['status_url'] as String; final statusUrlOnion = responseJSON['status_url_onion'] as String; final statusInfo = await paymentStatus(id); return AnonpayInvoiceInfo( invoiceId: id, clearnetUrl: url, onionUrl: urlOnion, status: statusInfo.status, fiatAmount: statusInfo.fiatAmount, fiatEquiv: statusInfo.fiatEquiv, amountTo: statusInfo.amountTo, coinTo: statusInfo.coinTo, address: statusInfo.address, clearnetStatusUrl: statusUrl, onionStatusUrl: statusUrlOnion, walletId: wallet.id, createdAt: DateTime.now(), provider: 'Trocador AnonPay invoice', ); } Future generateDonationLink(AnonPayRequest request) async { final body = { 'ticker_to': request.cryptoCurrency.title.toLowerCase(), 'network_to': _networkFor(request.cryptoCurrency), 'address': request.address, 'ref': anonpayRef, 'direct': 'True', }; if (request.name.isNotEmpty) { body['name'] = request.name; } if (request.description.isNotEmpty) { body['description'] = request.description; } if (request.email.isNotEmpty) { body['email'] = request.email; } final clearnetUrl = Uri.https(clearNetAuthority, anonPayPath, body); final onionUrl = Uri.https(onionApiAuthority, anonPayPath, body); return AnonpayDonationLinkInfo( clearnetUrl: clearnetUrl.toString(), onionUrl: onionUrl.toString(), address: request.address, ); } Future fetchLimits({ FiatCurrency? fiatCurrency, required CryptoCurrency cryptoCurrency, }) async { double fiatRate = 0.0; if (fiatCurrency != null) { fiatRate = await FiatConversionService.fetchPrice( crypto: cryptoCurrency, fiat: fiatCurrency, apiMode: apiMode, ); } final params = { 'api_key': apiKey, 'ticker': cryptoCurrency.title.toLowerCase(), 'name': cryptoCurrency.name, }; final String apiAuthority = await _getAuthority(); final uri = Uri.https(apiAuthority, 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; final minimum = coinJson['minimum'] as double; final maximum = coinJson['maximum'] as double; if (fiatCurrency != null) { return Limits( min: double.tryParse((minimum * fiatRate).toStringAsFixed(2)), max: double.tryParse((maximum * fiatRate).toStringAsFixed(2)), ); } return Limits( min: minimum, max: maximum, ); } String _networkFor(CryptoCurrency currency) { switch (currency) { case CryptoCurrency.usdt: return CryptoCurrency.btc.title.toLowerCase(); case CryptoCurrency.eth: return 'ERC20'; default: return currency.tag != null ? _normalizeTag(currency.tag!) : 'Mainnet'; } } String _normalizeTag(String tag) { switch (tag) { case 'ETH': return 'ERC20'; default: return tag.toLowerCase(); } } Future _getAuthority() async { try { if (apiMode == ExchangeApiMode.torOnly) { return onionApiAuthority; } final uri = Uri.https(onionApiAuthority, '/anonpay'); await get(uri); return onionApiAuthority; } catch (e) { return clearNetAuthority; } } }