mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-01-18 16:55:58 +00:00
5c9f176d18
* fix: Improve exchange flow by adding a timeout to the call to fetch rate from providers * fix: Adjust time limit for fetching rate to 7 seconds and add timelimit to fetching limits * fix: Make fetch limits a Future.wait * feat: Add currency for amount and estimated receive amount to confirm sending page for exchange * fix: Remove unneeded code * fix: Modify receive amount to reflect value coming from the individual exchange providers if available and ensure receiveAmount is calculated based on selected exchange provider's rate
254 lines
8.2 KiB
Dart
254 lines
8.2 KiB
Dart
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_not_created_exception.dart';
|
|
import 'package:cake_wallet/exchange/trade_not_found_exception.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';
|
|
|
|
class QuantexExchangeProvider extends ExchangeProvider {
|
|
QuantexExchangeProvider() : super(pairList: supportedPairs(_notSupported));
|
|
|
|
static final List<CryptoCurrency> _notSupported = [
|
|
...(CryptoCurrency.all
|
|
.where((element) => ![
|
|
CryptoCurrency.btc,
|
|
CryptoCurrency.sol,
|
|
CryptoCurrency.eth,
|
|
CryptoCurrency.ltc,
|
|
CryptoCurrency.ada,
|
|
CryptoCurrency.bch,
|
|
CryptoCurrency.usdt,
|
|
CryptoCurrency.bnb,
|
|
CryptoCurrency.xmr,
|
|
].contains(element))
|
|
.toList())
|
|
];
|
|
|
|
static final markup = secrets.quantexExchangeMarkup;
|
|
|
|
static const apiAuthority = 'api.myquantex.com';
|
|
static const getRate = '/api/swap/get-rate';
|
|
static const getCoins = '/api/swap/get-coins';
|
|
static const createOrder = '/api/swap/create-order';
|
|
|
|
@override
|
|
String get title => 'Quantex';
|
|
|
|
@override
|
|
bool get isAvailable => true;
|
|
|
|
@override
|
|
bool get isEnabled => true;
|
|
|
|
@override
|
|
bool get supportsFixedRate => false;
|
|
|
|
@override
|
|
ExchangeProviderDescription get description => ExchangeProviderDescription.quantex;
|
|
|
|
@override
|
|
Future<bool> checkIsAvailable() async => true;
|
|
|
|
@override
|
|
Future<Limits> fetchLimits({
|
|
required CryptoCurrency from,
|
|
required CryptoCurrency to,
|
|
required bool isFixedRateMode,
|
|
}) async {
|
|
try {
|
|
final uri = Uri.https(apiAuthority, getCoins);
|
|
final response = await get(uri);
|
|
|
|
final responseJSON = json.decode(response.body) as Map<String, dynamic>;
|
|
|
|
if (response.statusCode != 200)
|
|
throw Exception('Unexpected http status: ${response.statusCode}');
|
|
|
|
final coinsInfo = responseJSON['data'] as List<dynamic>;
|
|
|
|
for (var coin in coinsInfo) {
|
|
if (coin['id'].toString().toUpperCase() == _normalizeCurrency(from)) {
|
|
return Limits(
|
|
min: double.parse(coin['min'].toString()),
|
|
max: double.parse(coin['max'].toString()),
|
|
);
|
|
}
|
|
}
|
|
|
|
// coin not found:
|
|
return Limits(min: 0, max: 0);
|
|
} catch (e) {
|
|
print(e.toString());
|
|
return Limits(min: 0, max: 0);
|
|
}
|
|
}
|
|
|
|
@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 headers = <String, String>{};
|
|
final params = <String, dynamic>{};
|
|
final body = <String, String>{
|
|
'coin_send': _normalizeCurrency(from),
|
|
'coin_receive': _normalizeCurrency(to),
|
|
'ref': 'cake',
|
|
};
|
|
|
|
final uri = Uri.https(apiAuthority, getRate, params);
|
|
final response = await post(uri, body: body, headers: headers);
|
|
final responseBody = json.decode(response.body) as Map<String, dynamic>;
|
|
|
|
if (response.statusCode != 200)
|
|
throw Exception('Unexpected http status: ${response.statusCode}');
|
|
|
|
final data = responseBody['data'] as Map<String, dynamic>;
|
|
double rate = double.parse(data['price'].toString());
|
|
return rate;
|
|
} catch (e) {
|
|
print("error fetching rate: ${e.toString()}");
|
|
return 0.0;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<Trade> createTrade({
|
|
required TradeRequest request,
|
|
required bool isFixedRateMode,
|
|
required bool isSendAll,
|
|
}) async {
|
|
try {
|
|
final headers = <String, String>{};
|
|
final params = <String, dynamic>{};
|
|
var body = <String, dynamic>{
|
|
'coin_send': _normalizeCurrency(request.fromCurrency),
|
|
'coin_receive': _normalizeCurrency(request.toCurrency),
|
|
'amount_send': request.fromAmount,
|
|
'recipient': request.toAddress,
|
|
'ref': 'cake',
|
|
'markup': markup,
|
|
};
|
|
|
|
String? fromNetwork = _networkFor(request.fromCurrency);
|
|
String? toNetwork = _networkFor(request.toCurrency);
|
|
if (fromNetwork != null) body['coin_send_network'] = fromNetwork;
|
|
if (toNetwork != null) body['coin_receive_network'] = toNetwork;
|
|
|
|
final uri = Uri.https(apiAuthority, createOrder, params);
|
|
final response = await post(uri, body: body, headers: headers);
|
|
final responseBody = json.decode(response.body) as Map<String, dynamic>;
|
|
|
|
if (response.statusCode == 400 || responseBody["success"] == false) {
|
|
final error = responseBody['errors'][0]['msg'] as String;
|
|
throw TradeNotCreatedException(description, description: error);
|
|
}
|
|
|
|
if (response.statusCode != 200)
|
|
throw Exception('Unexpected http status: ${response.statusCode}');
|
|
|
|
final responseData = responseBody['data'] as Map<String, dynamic>;
|
|
final receiveAmount = responseData["amount_receive"]?.toString();
|
|
|
|
return Trade(
|
|
id: responseData["order_id"] as String,
|
|
inputAddress: responseData["server_address"] as String,
|
|
amount: request.fromAmount,
|
|
receiveAmount: receiveAmount ?? request.toAmount,
|
|
from: request.fromCurrency,
|
|
to: request.toCurrency,
|
|
provider: description,
|
|
createdAt: DateTime.now(),
|
|
state: TradeState.created,
|
|
payoutAddress: request.toAddress,
|
|
isSendAll: isSendAll,
|
|
);
|
|
} catch (e) {
|
|
print("error creating trade: ${e.toString()}");
|
|
throw TradeNotCreatedException(description, description: e.toString());
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<Trade> findTradeById({required String id}) async {
|
|
try {
|
|
final headers = <String, String>{};
|
|
final params = <String, dynamic>{};
|
|
var body = <String, dynamic>{
|
|
'order_id': id,
|
|
};
|
|
|
|
final uri = Uri.https(apiAuthority, createOrder, params);
|
|
final response = await post(uri, body: body, headers: headers);
|
|
final responseBody = json.decode(response.body) as Map<String, dynamic>;
|
|
|
|
if (response.statusCode == 400 || responseBody["success"] == false) {
|
|
final error = responseBody['errors'][0]['msg'] as String;
|
|
throw TradeNotCreatedException(description, description: error);
|
|
}
|
|
|
|
if (response.statusCode != 200)
|
|
throw Exception('Unexpected http status: ${response.statusCode}');
|
|
|
|
final responseData = responseBody['data'] as Map<String, dynamic>;
|
|
final fromCurrency = responseData['coin_send'] as String;
|
|
final from = CryptoCurrency.fromString(fromCurrency);
|
|
final toCurrency = responseData['coin_receive'] as String;
|
|
final to = CryptoCurrency.fromString(toCurrency);
|
|
final inputAddress = responseData['server_address'] as String;
|
|
final status = responseData['status'] as String;
|
|
final state = TradeState.deserialize(raw: status);
|
|
final response_id = responseData['order_id'] as String;
|
|
final expectedSendAmount = responseData['amount_send'] as String;
|
|
|
|
return Trade(
|
|
id: response_id,
|
|
from: from,
|
|
to: to,
|
|
provider: description,
|
|
inputAddress: inputAddress,
|
|
amount: expectedSendAmount,
|
|
state: state,
|
|
);
|
|
} catch (e) {
|
|
print("error getting trade: ${e.toString()}");
|
|
throw TradeNotFoundException(
|
|
id,
|
|
provider: description,
|
|
description: e.toString(),
|
|
);
|
|
}
|
|
}
|
|
|
|
String _normalizeCurrency(CryptoCurrency currency) {
|
|
switch (currency) {
|
|
default:
|
|
return currency.title.toUpperCase();
|
|
}
|
|
}
|
|
|
|
String? _networkFor(CryptoCurrency currency) {
|
|
switch (currency) {
|
|
case CryptoCurrency.usdt:
|
|
return "USDT_ERC20";
|
|
case CryptoCurrency.bnb:
|
|
return "BNB_BSC";
|
|
default:
|
|
return null;
|
|
}
|
|
}
|
|
}
|