stack_wallet/lib/services/ethereum/ethereum_token_service.dart

408 lines
13 KiB
Dart
Raw Normal View History

import 'dart:async';
import 'dart:convert';
import 'package:decimal/decimal.dart';
import 'package:ethereum_addresses/ethereum_addresses.dart';
import 'package:flutter/widgets.dart';
import 'package:http/http.dart';
import 'package:isar/isar.dart';
2023-02-23 22:59:58 +00:00
import 'package:stackwallet/models/ethereum/eth_token.dart';
import 'package:stackwallet/models/isar/models/blockchain_data/address.dart';
import 'package:stackwallet/models/isar/models/blockchain_data/transaction.dart';
import 'package:stackwallet/models/node_model.dart';
2023-01-25 09:29:20 +00:00
import 'package:stackwallet/models/paymint/fee_object_model.dart';
import 'package:stackwallet/models/token_balance.dart';
import 'package:stackwallet/services/coins/ethereum/ethereum_wallet.dart';
2023-02-23 22:59:58 +00:00
import 'package:stackwallet/services/ethereum/ethereum_api.dart';
import 'package:stackwallet/services/event_bus/events/global/updated_in_background_event.dart';
import 'package:stackwallet/services/event_bus/events/global/wallet_sync_status_changed_event.dart';
import 'package:stackwallet/services/event_bus/global_event_bus.dart';
import 'package:stackwallet/services/mixins/eth_token_cache.dart';
import 'package:stackwallet/services/node_service.dart';
import 'package:stackwallet/services/transaction_notification_tracker.dart';
import 'package:stackwallet/utilities/default_nodes.dart';
import 'package:stackwallet/utilities/enums/coin_enum.dart';
import 'package:stackwallet/utilities/enums/fee_rate_type_enum.dart';
import 'package:stackwallet/utilities/eth_commons.dart';
import 'package:stackwallet/utilities/flutter_secure_storage_interface.dart';
import 'package:stackwallet/utilities/format.dart';
import 'package:stackwallet/utilities/logger.dart';
import 'package:tuple/tuple.dart';
2023-02-24 14:45:34 +00:00
import 'package:web3dart/web3dart.dart' as web3dart;
2023-01-25 16:08:27 +00:00
class EthereumTokenService extends ChangeNotifier with EthTokenCache {
2023-03-01 00:36:54 +00:00
final EthContractInfo token;
final EthereumWallet ethWallet;
final TransactionNotificationTracker tracker;
final SecureStorageInterface _secureStore;
2023-02-23 22:59:58 +00:00
2023-02-24 14:45:34 +00:00
late web3dart.EthereumAddress _contractAddress;
late web3dart.EthPrivateKey _credentials;
late web3dart.DeployedContract _contract;
late web3dart.ContractFunction _balanceFunction;
late web3dart.ContractFunction _sendFunction;
2023-01-25 16:08:27 +00:00
late String _tokenAbi;
2023-02-24 14:45:34 +00:00
late web3dart.Web3Client _client;
2023-01-25 09:29:20 +00:00
static const _gasLimit = 200000;
2023-01-25 16:08:27 +00:00
2023-02-23 22:59:58 +00:00
EthereumTokenService({
required this.token,
required this.ethWallet,
2023-01-27 12:32:05 +00:00
required SecureStorageInterface secureStore,
required this.tracker,
}) : _secureStore = secureStore {
2023-02-24 14:45:34 +00:00
_contractAddress = web3dart.EthereumAddress.fromHex(token.contractAddress);
initCache(ethWallet.walletId, token);
}
2023-01-25 09:29:20 +00:00
TokenBalance get balance => _balance ??= getCachedBalance();
TokenBalance? _balance;
2023-01-25 09:29:20 +00:00
Coin get coin => Coin.ethereum;
2023-01-25 09:29:20 +00:00
Future<String> confirmSend({required Map<String, dynamic> txData}) async {
final amount = txData['recipientAmt'];
final decimalAmount =
Format.satoshisToAmount(amount as int, coin: Coin.ethereum);
2023-02-23 22:59:58 +00:00
final bigIntAmount =
amountToBigInt(decimalAmount.toDouble(), token.decimals);
final sentTx = await _client.sendTransaction(
_credentials,
2023-02-24 14:45:34 +00:00
web3dart.Transaction.callContract(
contract: _contract,
function: _sendFunction,
parameters: [
2023-02-24 14:45:34 +00:00
web3dart.EthereumAddress.fromHex(txData['address'] as String),
bigIntAmount
],
maxGas: _gasLimit,
2023-02-24 14:45:34 +00:00
gasPrice: web3dart.EtherAmount.fromUnitAndValue(
web3dart.EtherUnit.wei, txData['feeInWei'])));
return sentTx;
2023-01-25 09:29:20 +00:00
}
Future<String> get currentReceivingAddress async {
final address = await _currentReceivingAddress;
return address?.value ??
checksumEthereumAddress(_credentials.address.toString());
}
2023-01-25 09:29:20 +00:00
Future<Address?> get _currentReceivingAddress => ethWallet.db
.getAddresses(ethWallet.walletId)
.filter()
.typeEqualTo(AddressType.ethereum)
.subTypeEqualTo(AddressSubType.receiving)
.sortByDerivationIndexDesc()
.findFirst();
Future<int> estimateFeeFor(int satoshiAmount, int feeRate) async {
2023-02-23 22:59:58 +00:00
final fee = estimateFee(feeRate, _gasLimit, token.decimals);
return Format.decimalAmountToSatoshis(Decimal.parse(fee.toString()), coin);
2023-01-25 09:29:20 +00:00
}
Future<FeeObject> get fees => _feeObject ??= _getFees();
Future<FeeObject>? _feeObject;
Future<FeeObject> _getFees() async {
2023-02-23 22:59:58 +00:00
return await EthereumAPI.getFees();
}
2023-01-25 09:29:20 +00:00
Future<void> initialize() async {
final storedABI =
await _secureStore.read(key: '${_contractAddress.toString()}_tokenAbi');
if (storedABI == null) {
AbiRequestResponse abi =
await EthereumAPI.fetchTokenAbi(_contractAddress.hex);
//Fetch token ABI so we can call token functions
if (abi.message == "OK") {
_tokenAbi = abi.result;
//Store abi in secure store
await _secureStore.write(
key: '${_contractAddress.hex}_tokenAbi', value: _tokenAbi);
} else {
throw Exception('Failed to load token abi');
}
} else {
_tokenAbi = storedABI;
}
2023-01-27 12:32:05 +00:00
String? mnemonicString = await ethWallet.mnemonicString;
2023-01-25 16:08:27 +00:00
//Get private key for given mnemonic
String privateKey = getPrivateKey(
mnemonicString!, (await ethWallet.mnemonicPassphrase) ?? "");
2023-02-24 14:45:34 +00:00
_credentials = web3dart.EthPrivateKey.fromHex(privateKey);
2023-02-24 14:45:34 +00:00
_contract = web3dart.DeployedContract(
web3dart.ContractAbi.fromJson(_tokenAbi, token.name), _contractAddress);
bool hackInBalanceOf = false, hackInTransfer = false;
try {
_balanceFunction = _contract.function('balanceOf');
} catch (_) {
// function not found so likely a proxy so we need to hack the function in
hackInBalanceOf = true;
}
try {
_sendFunction = _contract.function('transfer');
} catch (_) {
// function not found so likely a proxy so we need to hack the function in
hackInTransfer = true;
}
2023-01-25 09:29:20 +00:00
if (hackInBalanceOf || hackInTransfer) {
final json = jsonDecode(_tokenAbi) as List;
if (hackInBalanceOf) {
json.add({
"constant": true,
"inputs": [
{"name": "", "type": "address"}
],
"name": "balanceOf",
"outputs": [
{"name": "", "type": "uint256"}
],
"payable": false,
"type": "function"
});
}
if (hackInTransfer) {
json.add({
"constant": false,
"inputs": [
{"name": "_to", "type": "address"},
{"name": "_value", "type": "uint256"}
],
"name": "transfer",
"outputs": <dynamic>[],
"payable": false,
"type": "function"
});
}
_tokenAbi = jsonEncode(json);
2023-01-27 12:32:05 +00:00
await _secureStore.write(
2023-02-23 22:59:58 +00:00
key: '${_contractAddress.hex}_tokenAbi', value: _tokenAbi);
2023-01-27 12:32:05 +00:00
_contract = web3dart.DeployedContract(
web3dart.ContractAbi.fromJson(_tokenAbi, token.name),
_contractAddress);
2023-01-25 09:29:20 +00:00
_balanceFunction = _contract.function('balanceOf');
_sendFunction = _contract.function('transfer');
}
_client = await getEthClient();
2023-01-25 09:29:20 +00:00
unawaited(refresh());
}
2023-01-25 09:29:20 +00:00
bool get isRefreshing => _refreshLock;
2023-01-25 09:29:20 +00:00
Future<Map<String, dynamic>> prepareSend(
{required String address,
required int satoshiAmount,
Map<String, dynamic>? args}) async {
final feeRateType = args?["feeRate"];
int fee = 0;
final feeObject = await fees;
switch (feeRateType) {
case FeeRateType.fast:
fee = feeObject.fast;
break;
case FeeRateType.average:
fee = feeObject.medium;
break;
case FeeRateType.slow:
fee = feeObject.slow;
break;
}
final feeEstimate = await estimateFeeFor(satoshiAmount, fee);
Map<String, dynamic> txData = {
"fee": feeEstimate,
"feeInWei": fee,
"address": address,
"recipientAmt": satoshiAmount,
};
return txData;
2023-01-25 09:29:20 +00:00
}
bool _refreshLock = false;
Future<void> refresh() async {
if (!_refreshLock) {
_refreshLock = true;
try {
GlobalEventBus.instance.fire(
WalletSyncStatusChangedEvent(
WalletSyncStatus.syncing,
ethWallet.walletId + token.contractAddress,
coin,
),
);
await refreshCachedBalance();
await _refreshTransactions();
} catch (e, s) {
Logging.instance.log(
"Caught exception in ${token.name} ${ethWallet.walletName} ${ethWallet.walletId} refresh(): $e\n$s",
level: LogLevel.Warning,
);
} finally {
_refreshLock = false;
GlobalEventBus.instance.fire(
WalletSyncStatusChangedEvent(
WalletSyncStatus.synced,
ethWallet.walletId + token.contractAddress,
coin,
),
);
notifyListeners();
}
}
2023-01-25 09:29:20 +00:00
}
Future<void> refreshCachedBalance() async {
final balanceRequest = await _client.call(
contract: _contract,
function: _balanceFunction,
params: [_credentials.address],
);
String _balance = balanceRequest.first.toString();
final newBalance = TokenBalance(
contractAddress: token.contractAddress,
total: int.parse(_balance),
spendable: int.parse(_balance),
blockedTotal: 0,
pendingSpendable: 0,
decimalPlaces: token.decimals,
2023-02-23 22:59:58 +00:00
);
await updateCachedBalance(newBalance);
notifyListeners();
}
2023-01-25 09:29:20 +00:00
Future<List<Transaction>> get transactions => ethWallet.db
.getTransactions(ethWallet.walletId)
.filter()
.otherDataEqualTo(token.contractAddress)
.sortByTimestampDesc()
.findAll();
2023-01-25 09:29:20 +00:00
Future<void> _refreshTransactions() async {
String addressString = await currentReceivingAddress;
2023-01-27 12:32:05 +00:00
final response = await EthereumAPI.getTokenTransactions(
address: addressString,
contractAddress: token.contractAddress,
);
2023-01-27 12:32:05 +00:00
if (response.value == null) {
throw response.exception ??
Exception("Failed to fetch token transactions");
}
2023-01-27 12:32:05 +00:00
final List<Tuple2<Transaction, Address?>> txnsData = [];
2023-01-27 12:32:05 +00:00
for (final tx in response.value!) {
bool isIncoming;
if (checksumEthereumAddress(tx.from) == addressString) {
isIncoming = false;
} else {
isIncoming = true;
}
2023-01-27 12:32:05 +00:00
final txn = Transaction(
walletId: ethWallet.walletId,
txid: tx.hash,
timestamp: tx.timeStamp,
type: isIncoming ? TransactionType.incoming : TransactionType.outgoing,
subType: TransactionSubType.ethToken,
amount: tx.value.toInt(),
fee: tx.gasUsed * tx.gasPrice.toInt(),
height: tx.blockNumber,
isCancelled: false,
isLelantus: false,
slateId: null,
otherData: tx.contractAddress,
inputs: [],
outputs: [],
);
Address? transactionAddress = await ethWallet.db
.getAddresses(ethWallet.walletId)
.filter()
.valueEqualTo(addressString)
.findFirst();
if (transactionAddress == null) {
if (isIncoming) {
transactionAddress = Address(
walletId: ethWallet.walletId,
value: addressString,
publicKey: [],
derivationIndex: 0,
derivationPath: DerivationPath()..value = "$hdPathEthereum/0",
type: AddressType.ethereum,
subType: AddressSubType.receiving,
);
2023-01-27 12:32:05 +00:00
} else {
final myRcvAddr = await currentReceivingAddress;
final isSentToSelf = myRcvAddr == addressString;
transactionAddress = Address(
walletId: ethWallet.walletId,
value: addressString,
publicKey: [],
derivationIndex: isSentToSelf ? 0 : -1,
derivationPath: isSentToSelf
? (DerivationPath()..value = "$hdPathEthereum/0")
: null,
type: AddressType.ethereum,
subType: isSentToSelf
? AddressSubType.receiving
: AddressSubType.nonWallet,
);
2023-01-27 12:32:05 +00:00
}
2023-02-23 22:59:58 +00:00
}
2023-01-27 12:32:05 +00:00
txnsData.add(Tuple2(txn, transactionAddress));
}
await ethWallet.db.addNewTransactionData(txnsData, ethWallet.walletId);
// quick hack to notify manager to call notifyListeners if
// transactions changed
if (txnsData.isNotEmpty) {
GlobalEventBus.instance.fire(
UpdatedInBackgroundEvent(
"${token.name} transactions updated/added for: ${ethWallet.walletId} ${ethWallet.walletName}",
ethWallet.walletId,
),
);
2023-01-27 12:32:05 +00:00
}
}
2023-01-25 09:29:20 +00:00
bool validateAddress(String address) {
return isValidEthereumAddress(address);
}
2023-01-27 12:32:05 +00:00
Future<NodeModel> getCurrentNode() async {
return NodeService(secureStorageInterface: _secureStore)
.getPrimaryNodeFor(coin: coin) ??
DefaultNodes.getNodeFor(coin);
}
2023-02-24 14:45:34 +00:00
Future<web3dart.Web3Client> getEthClient() async {
2023-01-27 12:32:05 +00:00
final node = await getCurrentNode();
2023-02-24 14:45:34 +00:00
return web3dart.Web3Client(node.host, Client());
2023-01-25 09:29:20 +00:00
}
}