2024-05-14 02:07:16 +00:00
|
|
|
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>;
|
2024-07-23 00:20:55 +00:00
|
|
|
final receiveAmount = responseData["amount_receive"]?.toString();
|
2024-05-14 02:07:16 +00:00
|
|
|
|
|
|
|
return Trade(
|
|
|
|
id: responseData["order_id"] as String,
|
|
|
|
inputAddress: responseData["server_address"] as String,
|
|
|
|
amount: request.fromAmount,
|
2024-07-23 00:20:55 +00:00
|
|
|
receiveAmount: receiveAmount ?? request.toAmount,
|
2024-05-14 02:07:16 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|