mirror of
https://github.com/cypherstack/stack_wallet.git
synced 2025-01-10 12:44:31 +00:00
4742 lines
162 KiB
Dart
4742 lines
162 KiB
Dart
import 'dart:async';
|
|
import 'dart:convert';
|
|
import 'dart:io';
|
|
import 'dart:isolate';
|
|
import 'dart:math';
|
|
import 'dart:typed_data';
|
|
|
|
import 'package:bip32/bip32.dart' as bip32;
|
|
import 'package:bip39/bip39.dart' as bip39;
|
|
import 'package:bitcoindart/bitcoindart.dart';
|
|
import 'package:decimal/decimal.dart';
|
|
import 'package:devicelocale/devicelocale.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
|
|
import 'package:http/http.dart';
|
|
import 'package:lelantus/lelantus.dart';
|
|
import 'package:stackwallet/electrumx_rpc/cached_electrumx.dart';
|
|
import 'package:stackwallet/electrumx_rpc/electrumx.dart';
|
|
import 'package:stackwallet/hive/db.dart';
|
|
import 'package:stackwallet/models/lelantus_coin.dart';
|
|
import 'package:stackwallet/models/lelantus_fee_data.dart';
|
|
import 'package:stackwallet/models/models.dart' as models;
|
|
import 'package:stackwallet/models/paymint/fee_object_model.dart';
|
|
import 'package:stackwallet/models/paymint/utxo_model.dart';
|
|
import 'package:stackwallet/services/coins/coin_service.dart';
|
|
import 'package:stackwallet/services/event_bus/events/global/node_connection_status_changed_event.dart';
|
|
import 'package:stackwallet/services/event_bus/events/global/refresh_percent_changed_event.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/node_service.dart';
|
|
import 'package:stackwallet/services/notifications_api.dart';
|
|
import 'package:stackwallet/services/price.dart';
|
|
import 'package:stackwallet/services/transaction_notification_tracker.dart';
|
|
import 'package:stackwallet/utilities/address_utils.dart';
|
|
import 'package:stackwallet/utilities/assets.dart';
|
|
import 'package:stackwallet/utilities/constants.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/flutter_secure_storage_interface.dart';
|
|
import 'package:stackwallet/utilities/format.dart';
|
|
import 'package:stackwallet/utilities/logger.dart';
|
|
import 'package:stackwallet/utilities/prefs.dart';
|
|
import 'package:tuple/tuple.dart';
|
|
import 'package:uuid/uuid.dart';
|
|
|
|
const DUST_LIMIT = 1000;
|
|
const MINIMUM_CONFIRMATIONS = 1;
|
|
const MINT_LIMIT = 100100000000;
|
|
const int LELANTUS_VALUE_SPEND_LIMIT_PER_TRANSACTION = 5001 * 100000000;
|
|
|
|
const JMINT_INDEX = 5;
|
|
const MINT_INDEX = 2;
|
|
const TRANSACTION_LELANTUS = 8;
|
|
const ANONYMITY_SET_EMPTY_ID = 0;
|
|
|
|
const String GENESIS_HASH_MAINNET =
|
|
"4381deb85b1b2c9843c222944b616d997516dcbd6a964e1eaf0def0830695233";
|
|
const String GENESIS_HASH_TESTNET =
|
|
"aa22adcc12becaf436027ffe62a8fb21b234c58c23865291e5dc52cf53f64fca";
|
|
|
|
final firoNetwork = NetworkType(
|
|
messagePrefix: '\x18Zcoin Signed Message:\n',
|
|
bech32: 'bc',
|
|
bip32: Bip32Type(public: 0x0488b21e, private: 0x0488ade4),
|
|
pubKeyHash: 0x52,
|
|
scriptHash: 0x07,
|
|
wif: 0xd2);
|
|
|
|
final firoTestNetwork = NetworkType(
|
|
messagePrefix: '\x18Zcoin Signed Message:\n',
|
|
bech32: 'bc',
|
|
bip32: Bip32Type(public: 0x043587cf, private: 0x04358394),
|
|
pubKeyHash: 0x41,
|
|
scriptHash: 0xb2,
|
|
wif: 0xb9);
|
|
|
|
// isolate
|
|
|
|
Map<ReceivePort, Isolate> isolates = {};
|
|
|
|
Future<ReceivePort> getIsolate(Map<String, dynamic> arguments) async {
|
|
ReceivePort receivePort =
|
|
ReceivePort(); //port for isolate to receive messages.
|
|
arguments['sendPort'] = receivePort.sendPort;
|
|
Logging.instance
|
|
.log("starting isolate ${arguments['function']}", level: LogLevel.Info);
|
|
Isolate isolate = await Isolate.spawn(executeNative, arguments);
|
|
Logging.instance.log("isolate spawned!", level: LogLevel.Info);
|
|
isolates[receivePort] = isolate;
|
|
return receivePort;
|
|
}
|
|
|
|
Future<void> executeNative(Map<String, dynamic> arguments) async {
|
|
await Logging.instance.initInIsolate();
|
|
final sendPort = arguments['sendPort'] as SendPort;
|
|
final function = arguments['function'] as String;
|
|
try {
|
|
if (function == "createJoinSplit") {
|
|
final spendAmount = arguments['spendAmount'] as int;
|
|
final address = arguments['address'] as String;
|
|
final subtractFeeFromAmount = arguments['subtractFeeFromAmount'] as bool;
|
|
final mnemonic = arguments['mnemonic'] as String;
|
|
final index = arguments['index'] as int;
|
|
final price = arguments['price'] as Decimal;
|
|
final lelantusEntries =
|
|
arguments['lelantusEntries'] as List<DartLelantusEntry>;
|
|
final coin = arguments['coin'] as Coin;
|
|
final network = arguments['network'] as NetworkType?;
|
|
final locktime = arguments['locktime'] as int;
|
|
final anonymitySets = arguments['_anonymity_sets'] as List<Map>?;
|
|
final locale = arguments["locale"] as String;
|
|
if (!(network == null || anonymitySets == null)) {
|
|
var joinSplit = await isolateCreateJoinSplitTransaction(
|
|
spendAmount,
|
|
address,
|
|
subtractFeeFromAmount,
|
|
mnemonic,
|
|
index,
|
|
price,
|
|
lelantusEntries,
|
|
locktime,
|
|
coin,
|
|
network,
|
|
anonymitySets,
|
|
locale);
|
|
sendPort.send(joinSplit);
|
|
return;
|
|
}
|
|
} else if (function == "estimateJoinSplit") {
|
|
final spendAmount = arguments['spendAmount'] as int;
|
|
final subtractFeeFromAmount = arguments['subtractFeeFromAmount'] as bool?;
|
|
final lelantusEntries =
|
|
arguments['lelantusEntries'] as List<DartLelantusEntry>;
|
|
final coin = arguments['coin'] as Coin;
|
|
|
|
if (!(subtractFeeFromAmount == null)) {
|
|
var feeData = await isolateEstimateJoinSplitFee(
|
|
spendAmount, subtractFeeFromAmount, lelantusEntries, coin);
|
|
sendPort.send(feeData);
|
|
return;
|
|
}
|
|
} else if (function == "restore") {
|
|
final latestSetId = arguments['latestSetId'] as int;
|
|
final setDataMap = arguments['setDataMap'] as Map;
|
|
final usedSerialNumbers = arguments['usedSerialNumbers'] as List?;
|
|
final mnemonic = arguments['mnemonic'] as String;
|
|
final coin = arguments['coin'] as Coin;
|
|
final network = arguments['network'] as NetworkType?;
|
|
if (!(usedSerialNumbers == null || network == null)) {
|
|
var restoreData = await isolateRestore(
|
|
mnemonic,
|
|
coin,
|
|
latestSetId,
|
|
setDataMap,
|
|
usedSerialNumbers,
|
|
network,
|
|
);
|
|
sendPort.send(restoreData);
|
|
return;
|
|
}
|
|
} else if (function == "isolateDerive") {
|
|
final mnemonic = arguments['mnemonic'] as String;
|
|
final from = arguments['from'] as int;
|
|
final to = arguments['to'] as int;
|
|
final network = arguments['network'] as NetworkType?;
|
|
if (!(network == null)) {
|
|
var derived = await isolateDerive(mnemonic, from, to, network);
|
|
sendPort.send(derived);
|
|
return;
|
|
}
|
|
}
|
|
Logging.instance.log(
|
|
"Error Arguments for $function not formatted correctly",
|
|
level: LogLevel.Fatal);
|
|
sendPort.send("Error");
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"An error was thrown in this isolate $function: $e\n$s",
|
|
level: LogLevel.Error);
|
|
sendPort.send("Error");
|
|
} finally {
|
|
await Logging.instance.isar?.close();
|
|
}
|
|
}
|
|
|
|
void stop(ReceivePort port) {
|
|
Isolate? isolate = isolates.remove(port);
|
|
if (isolate != null) {
|
|
Logging.instance.log('Stopping Isolate...', level: LogLevel.Info);
|
|
isolate.kill(priority: Isolate.immediate);
|
|
isolate = null;
|
|
}
|
|
}
|
|
|
|
Future<Map<String, dynamic>> isolateDerive(
|
|
String mnemonic, int from, int to, NetworkType _network) async {
|
|
Map<String, dynamic> result = {};
|
|
Map<String, dynamic> allReceive = {};
|
|
Map<String, dynamic> allChange = {};
|
|
final root = getBip32Root(mnemonic, _network);
|
|
for (int i = from; i < to; i++) {
|
|
var currentNode = getBip32NodeFromRoot(0, i, root);
|
|
var address = P2PKH(
|
|
network: _network, data: PaymentData(pubkey: currentNode.publicKey))
|
|
.data
|
|
.address!;
|
|
allReceive["$i"] = {
|
|
"publicKey": Format.uint8listToString(currentNode.publicKey),
|
|
"wif": currentNode.toWIF(),
|
|
"address": address,
|
|
};
|
|
|
|
currentNode = getBip32NodeFromRoot(1, i, root);
|
|
address = P2PKH(
|
|
network: _network, data: PaymentData(pubkey: currentNode.publicKey))
|
|
.data
|
|
.address!;
|
|
allChange["$i"] = {
|
|
"publicKey": Format.uint8listToString(currentNode.publicKey),
|
|
"wif": currentNode.toWIF(),
|
|
"address": address,
|
|
};
|
|
if (i % 50 == 0) {
|
|
Logging.instance.log("thread at $i", level: LogLevel.Info);
|
|
}
|
|
}
|
|
result['receive'] = allReceive;
|
|
result['change'] = allChange;
|
|
return result;
|
|
}
|
|
|
|
Future<Map<String, dynamic>> isolateRestore(
|
|
String mnemonic,
|
|
Coin coin,
|
|
int _latestSetId,
|
|
Map<dynamic, dynamic> _setDataMap,
|
|
List<dynamic> _usedSerialNumbers,
|
|
NetworkType network,
|
|
) async {
|
|
List<int> jindexes = [];
|
|
List<Map<dynamic, LelantusCoin>> lelantusCoins = [];
|
|
|
|
final List<String> spendTxIds = [];
|
|
var lastFoundIndex = 0;
|
|
var currentIndex = 0;
|
|
|
|
try {
|
|
final usedSerialNumbers = _usedSerialNumbers;
|
|
Set<dynamic> usedSerialNumbersSet = {};
|
|
for (int ind = 0; ind < usedSerialNumbers.length; ind++) {
|
|
usedSerialNumbersSet.add(usedSerialNumbers[ind]);
|
|
}
|
|
|
|
final root = getBip32Root(mnemonic, network);
|
|
while (currentIndex < lastFoundIndex + 20) {
|
|
final mintKeyPair = getBip32NodeFromRoot(MINT_INDEX, currentIndex, root);
|
|
final mintTag = CreateTag(
|
|
Format.uint8listToString(mintKeyPair.privateKey!),
|
|
currentIndex,
|
|
Format.uint8listToString(mintKeyPair.identifier),
|
|
isTestnet: coin == Coin.firoTestNet);
|
|
|
|
for (var setId = 1; setId <= _latestSetId; setId++) {
|
|
final setData = _setDataMap[setId];
|
|
final foundCoin = setData["coins"].firstWhere(
|
|
(dynamic e) => e[1] == mintTag,
|
|
orElse: () => <Object>[]);
|
|
|
|
if (foundCoin.length == 4) {
|
|
lastFoundIndex = currentIndex;
|
|
if (foundCoin[2] is int) {
|
|
final amount = foundCoin[2] as int;
|
|
final serialNumber = GetSerialNumber(amount,
|
|
Format.uint8listToString(mintKeyPair.privateKey!), currentIndex,
|
|
isTestnet: coin == Coin.firoTestNet);
|
|
String publicCoin = foundCoin[0] as String;
|
|
String txId = foundCoin[3] as String;
|
|
bool isUsed = usedSerialNumbersSet.contains(serialNumber);
|
|
final duplicateCoin = lelantusCoins.firstWhere((element) {
|
|
final coin = element.values.first;
|
|
return coin.txId == txId &&
|
|
coin.index == currentIndex &&
|
|
coin.anonymitySetId != setId;
|
|
}, orElse: () => {});
|
|
if (duplicateCoin.isNotEmpty) {
|
|
debugPrint("removing duplicate: $duplicateCoin");
|
|
lelantusCoins.remove(duplicateCoin);
|
|
}
|
|
lelantusCoins.add({
|
|
publicCoin: LelantusCoin(
|
|
currentIndex,
|
|
amount,
|
|
publicCoin,
|
|
txId,
|
|
setId,
|
|
isUsed,
|
|
)
|
|
});
|
|
Logging.instance
|
|
.log("amount $amount used $isUsed", level: LogLevel.Info);
|
|
} else {
|
|
final keyPath = GetAesKeyPath(foundCoin[0] as String);
|
|
final aesKeyPair = getBip32NodeFromRoot(JMINT_INDEX, keyPath, root);
|
|
if (aesKeyPair.privateKey != null) {
|
|
final aesPrivateKey =
|
|
Format.uint8listToString(aesKeyPair.privateKey!);
|
|
final amount = decryptMintAmount(
|
|
aesPrivateKey,
|
|
foundCoin[2] as String,
|
|
);
|
|
|
|
final serialNumber = GetSerialNumber(
|
|
amount,
|
|
Format.uint8listToString(mintKeyPair.privateKey!),
|
|
currentIndex,
|
|
isTestnet: coin == Coin.firoTestNet);
|
|
String publicCoin = foundCoin[0] as String;
|
|
String txId = foundCoin[3] as String;
|
|
bool isUsed = usedSerialNumbersSet.contains(serialNumber);
|
|
final duplicateCoin = lelantusCoins.firstWhere((element) {
|
|
final coin = element.values.first;
|
|
return coin.txId == txId &&
|
|
coin.index == currentIndex &&
|
|
coin.anonymitySetId != setId;
|
|
}, orElse: () => {});
|
|
if (duplicateCoin.isNotEmpty) {
|
|
debugPrint("removing duplicate: $duplicateCoin");
|
|
lelantusCoins.remove(duplicateCoin);
|
|
}
|
|
lelantusCoins.add({
|
|
'${foundCoin[3]!}': LelantusCoin(
|
|
currentIndex,
|
|
amount,
|
|
publicCoin,
|
|
txId,
|
|
setId,
|
|
isUsed,
|
|
)
|
|
});
|
|
jindexes.add(currentIndex);
|
|
|
|
spendTxIds.add(foundCoin[3] as String);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
currentIndex++;
|
|
}
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown from isolateRestore(): $e\n$s",
|
|
level: LogLevel.Info);
|
|
rethrow;
|
|
}
|
|
|
|
Map<String, dynamic> result = {};
|
|
// Logging.instance.log("mints $lelantusCoins", addToDebugMessagesDB: false);
|
|
// Logging.instance.log("jmints $spendTxIds", addToDebugMessagesDB: false);
|
|
|
|
result['_lelantus_coins'] = lelantusCoins;
|
|
result['mintIndex'] = lastFoundIndex + 1;
|
|
result['jindex'] = jindexes;
|
|
result['spendTxIds'] = spendTxIds;
|
|
|
|
return result;
|
|
}
|
|
|
|
Future<Map<dynamic, dynamic>> staticProcessRestore(
|
|
models.TransactionData data,
|
|
Map<dynamic, dynamic> result,
|
|
) async {
|
|
List<dynamic>? _l = result['_lelantus_coins'] as List?;
|
|
final List<Map<dynamic, LelantusCoin>> lelantusCoins = [];
|
|
for (var el in _l ?? []) {
|
|
lelantusCoins.add({el.keys.first: el.values.first as LelantusCoin});
|
|
}
|
|
|
|
// Edit the receive transactions with the mint fees.
|
|
Map<String, models.Transaction> editedTransactions =
|
|
<String, models.Transaction>{};
|
|
for (var item in lelantusCoins) {
|
|
item.forEach((key, value) {
|
|
String txid = value.txId;
|
|
var tx = data.findTransaction(txid);
|
|
if (tx == null) {
|
|
// This is a jmint.
|
|
return;
|
|
}
|
|
List<models.Transaction> inputs = [];
|
|
for (var element in tx.inputs) {
|
|
var input = data.findTransaction(element.txid);
|
|
if (input != null) {
|
|
inputs.add(input);
|
|
}
|
|
}
|
|
if (inputs.isEmpty) {
|
|
//some error.
|
|
return;
|
|
}
|
|
|
|
int mintfee = tx.fees;
|
|
int sharedfee = mintfee ~/ inputs.length;
|
|
for (var element in inputs) {
|
|
editedTransactions[element.txid] = models.Transaction(
|
|
txid: element.txid,
|
|
confirmedStatus: element.confirmedStatus,
|
|
timestamp: element.timestamp,
|
|
txType: element.txType,
|
|
amount: element.amount,
|
|
aliens: element.aliens,
|
|
worthNow: element.worthNow,
|
|
worthAtBlockTimestamp: element.worthAtBlockTimestamp,
|
|
fees: sharedfee,
|
|
inputSize: element.inputSize,
|
|
outputSize: element.outputSize,
|
|
inputs: element.inputs,
|
|
outputs: element.outputs,
|
|
address: element.address,
|
|
height: element.height,
|
|
confirmations: element.confirmations,
|
|
subType: "mint",
|
|
otherData: txid,
|
|
);
|
|
}
|
|
});
|
|
}
|
|
// Logging.instance.log(editedTransactions, addToDebugMessagesDB: false);
|
|
|
|
Map<String, models.Transaction> transactionMap = data.getAllTransactions();
|
|
// Logging.instance.log(transactionMap, addToDebugMessagesDB: false);
|
|
|
|
editedTransactions.forEach((key, value) {
|
|
transactionMap.update(key, (_value) => value);
|
|
});
|
|
|
|
transactionMap.removeWhere((key, value) =>
|
|
lelantusCoins.any((element) => element.containsKey(key)) ||
|
|
(value.height == -1 && !value.confirmedStatus));
|
|
|
|
result['newTxMap'] = transactionMap;
|
|
return result;
|
|
}
|
|
|
|
Future<LelantusFeeData> isolateEstimateJoinSplitFee(
|
|
int spendAmount,
|
|
bool subtractFeeFromAmount,
|
|
List<DartLelantusEntry> lelantusEntries,
|
|
Coin coin) async {
|
|
Logging.instance.log("estimateJoinsplit fee", level: LogLevel.Info);
|
|
// for (int i = 0; i < lelantusEntries.length; i++) {
|
|
// Logging.instance.log(lelantusEntries[i], addToDebugMessagesDB: false);
|
|
// }
|
|
Logging.instance
|
|
.log("$spendAmount $subtractFeeFromAmount", level: LogLevel.Info);
|
|
|
|
List<int> changeToMint = List.empty(growable: true);
|
|
List<int> spendCoinIndexes = List.empty(growable: true);
|
|
// Logging.instance.log(lelantusEntries, addToDebugMessagesDB: false);
|
|
final fee = estimateFee(
|
|
spendAmount,
|
|
subtractFeeFromAmount,
|
|
lelantusEntries,
|
|
changeToMint,
|
|
spendCoinIndexes,
|
|
isTestnet: coin == Coin.firoTestNet,
|
|
);
|
|
|
|
final estimateFeeData =
|
|
LelantusFeeData(changeToMint[0], fee, spendCoinIndexes);
|
|
Logging.instance.log(
|
|
"estimateFeeData ${estimateFeeData.changeToMint} ${estimateFeeData.fee} ${estimateFeeData.spendCoinIndexes}",
|
|
level: LogLevel.Info);
|
|
return estimateFeeData;
|
|
}
|
|
|
|
Future<dynamic> isolateCreateJoinSplitTransaction(
|
|
int spendAmount,
|
|
String address,
|
|
bool subtractFeeFromAmount,
|
|
String mnemonic,
|
|
int index,
|
|
Decimal price,
|
|
List<DartLelantusEntry> lelantusEntries,
|
|
int locktime,
|
|
Coin coin,
|
|
NetworkType _network,
|
|
List<Map<dynamic, dynamic>> anonymitySetsArg,
|
|
String locale,
|
|
) async {
|
|
final estimateJoinSplitFee = await isolateEstimateJoinSplitFee(
|
|
spendAmount, subtractFeeFromAmount, lelantusEntries, coin);
|
|
var changeToMint = estimateJoinSplitFee.changeToMint;
|
|
var fee = estimateJoinSplitFee.fee;
|
|
var spendCoinIndexes = estimateJoinSplitFee.spendCoinIndexes;
|
|
Logging.instance
|
|
.log("$changeToMint $fee $spendCoinIndexes", level: LogLevel.Info);
|
|
if (spendCoinIndexes.isEmpty) {
|
|
Logging.instance.log("Error, Not enough funds.", level: LogLevel.Error);
|
|
return 1;
|
|
}
|
|
|
|
final tx = TransactionBuilder(network: _network);
|
|
tx.setLockTime(locktime);
|
|
|
|
tx.setVersion(3 | (TRANSACTION_LELANTUS << 16));
|
|
|
|
tx.addInput(
|
|
'0000000000000000000000000000000000000000000000000000000000000000',
|
|
4294967295,
|
|
4294967295,
|
|
Uint8List(0),
|
|
);
|
|
|
|
final jmintKeyPair = getBip32Node(MINT_INDEX, index, mnemonic, _network);
|
|
|
|
final String jmintprivatekey =
|
|
Format.uint8listToString(jmintKeyPair.privateKey!);
|
|
|
|
final keyPath = getMintKeyPath(changeToMint, jmintprivatekey, index,
|
|
isTestnet: coin == Coin.firoTestNet);
|
|
|
|
final aesKeyPair = getBip32Node(JMINT_INDEX, keyPath, mnemonic, _network);
|
|
final aesPrivateKey = Format.uint8listToString(aesKeyPair.privateKey!);
|
|
|
|
final jmintData = createJMintScript(
|
|
changeToMint,
|
|
Format.uint8listToString(jmintKeyPair.privateKey!),
|
|
index,
|
|
Format.uint8listToString(jmintKeyPair.identifier),
|
|
aesPrivateKey,
|
|
isTestnet: coin == Coin.firoTestNet,
|
|
);
|
|
|
|
tx.addOutput(
|
|
Format.stringToUint8List(jmintData),
|
|
0,
|
|
);
|
|
|
|
int amount = spendAmount;
|
|
if (subtractFeeFromAmount) {
|
|
amount -= fee;
|
|
}
|
|
tx.addOutput(
|
|
address,
|
|
amount,
|
|
);
|
|
|
|
final extractedTx = tx.buildIncomplete();
|
|
extractedTx.setPayload(Uint8List(0));
|
|
final txHash = extractedTx.getId();
|
|
|
|
final List<int> setIds = [];
|
|
final List<List<String>> anonymitySets = [];
|
|
final List<String> anonymitySetHashes = [];
|
|
final List<String> groupBlockHashes = [];
|
|
for (var i = 0; i < lelantusEntries.length; i++) {
|
|
final anonymitySetId = lelantusEntries[i].anonymitySetId;
|
|
if (!setIds.contains(anonymitySetId)) {
|
|
setIds.add(anonymitySetId);
|
|
final anonymitySet = anonymitySetsArg.firstWhere(
|
|
(element) => element["setId"] == anonymitySetId,
|
|
orElse: () => <String, dynamic>{});
|
|
if (anonymitySet.isNotEmpty) {
|
|
anonymitySetHashes.add(anonymitySet['setHash'] as String);
|
|
groupBlockHashes.add(anonymitySet['blockHash'] as String);
|
|
List<String> list = [];
|
|
for (int i = 0; i < (anonymitySet['coins'] as List).length; i++) {
|
|
list.add(anonymitySet['coins'][i][0] as String);
|
|
}
|
|
anonymitySets.add(list);
|
|
}
|
|
}
|
|
}
|
|
|
|
final String spendScript = createJoinSplitScript(
|
|
txHash,
|
|
spendAmount,
|
|
subtractFeeFromAmount,
|
|
Format.uint8listToString(jmintKeyPair.privateKey!),
|
|
index,
|
|
lelantusEntries,
|
|
setIds,
|
|
anonymitySets,
|
|
anonymitySetHashes,
|
|
groupBlockHashes,
|
|
isTestnet: coin == Coin.firoTestNet);
|
|
|
|
final finalTx = TransactionBuilder(network: _network);
|
|
finalTx.setLockTime(locktime);
|
|
|
|
finalTx.setVersion(3 | (TRANSACTION_LELANTUS << 16));
|
|
|
|
finalTx.addOutput(
|
|
Format.stringToUint8List(jmintData),
|
|
0,
|
|
);
|
|
|
|
finalTx.addOutput(
|
|
address,
|
|
amount,
|
|
);
|
|
|
|
final extTx = finalTx.buildIncomplete();
|
|
extTx.addInput(
|
|
Format.stringToUint8List(
|
|
'0000000000000000000000000000000000000000000000000000000000000000'),
|
|
4294967295,
|
|
4294967295,
|
|
Format.stringToUint8List("c9"),
|
|
);
|
|
debugPrint("spendscript: $spendScript");
|
|
extTx.setPayload(Format.stringToUint8List(spendScript));
|
|
|
|
final txHex = extTx.toHex();
|
|
final txId = extTx.getId();
|
|
Logging.instance.log("txid $txId", level: LogLevel.Info);
|
|
Logging.instance.log("txHex: $txHex", level: LogLevel.Info);
|
|
return {
|
|
"txid": txId,
|
|
"txHex": txHex,
|
|
"value": amount,
|
|
"fees": Format.satoshisToAmount(fee).toDouble(),
|
|
"fee": fee,
|
|
"jmintValue": changeToMint,
|
|
"publicCoin": "jmintData.publicCoin",
|
|
"spendCoinIndexes": spendCoinIndexes,
|
|
"height": locktime,
|
|
"txType": "Sent",
|
|
"confirmed_status": false,
|
|
"amount": Format.satoshisToAmount(amount).toDouble(),
|
|
"recipientAmt": amount,
|
|
"worthNow": Format.localizedStringAsFixed(
|
|
value: ((Decimal.fromInt(amount) * price) /
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toDecimal(scaleOnInfinitePrecision: 2),
|
|
decimalPlaces: 2,
|
|
locale: locale),
|
|
"address": address,
|
|
"timestamp": DateTime.now().millisecondsSinceEpoch ~/ 1000,
|
|
"subType": "join",
|
|
};
|
|
}
|
|
|
|
Future<int> getBlockHead(ElectrumX client) async {
|
|
try {
|
|
final tip = await client.getBlockHeadTip();
|
|
return tip["height"] as int;
|
|
} catch (e) {
|
|
Logging.instance
|
|
.log("Exception rethrown in getBlockHead(): $e", level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
// end of isolates
|
|
|
|
bip32.BIP32 getBip32Node(
|
|
int chain, int index, String mnemonic, NetworkType network) {
|
|
final root = getBip32Root(mnemonic, network);
|
|
|
|
final node = getBip32NodeFromRoot(chain, index, root);
|
|
return node;
|
|
}
|
|
|
|
/// wrapper for compute()
|
|
bip32.BIP32 getBip32NodeWrapper(
|
|
Tuple4<int, int, String, NetworkType> args,
|
|
) {
|
|
return getBip32Node(
|
|
args.item1,
|
|
args.item2,
|
|
args.item3,
|
|
args.item4,
|
|
);
|
|
}
|
|
|
|
bip32.BIP32 getBip32NodeFromRoot(int chain, int index, bip32.BIP32 root) {
|
|
String coinType;
|
|
switch (root.network.wif) {
|
|
case 0xd2: // firo mainnet wif
|
|
coinType = "136"; // firo mainnet
|
|
break;
|
|
case 0xb9: // firo testnet wif
|
|
coinType = "1"; // firo testnet
|
|
break;
|
|
default:
|
|
throw Exception("Invalid Bitcoin network type used!");
|
|
}
|
|
|
|
final node = root.derivePath("m/44'/$coinType'/0'/$chain/$index");
|
|
return node;
|
|
}
|
|
|
|
/// wrapper for compute()
|
|
bip32.BIP32 getBip32NodeFromRootWrapper(
|
|
Tuple3<int, int, bip32.BIP32> args,
|
|
) {
|
|
return getBip32NodeFromRoot(
|
|
args.item1,
|
|
args.item2,
|
|
args.item3,
|
|
);
|
|
}
|
|
|
|
bip32.BIP32 getBip32Root(String mnemonic, NetworkType network) {
|
|
final seed = bip39.mnemonicToSeed(mnemonic);
|
|
final firoNetworkType = bip32.NetworkType(
|
|
wif: network.wif,
|
|
bip32: bip32.Bip32Type(
|
|
public: network.bip32.public,
|
|
private: network.bip32.private,
|
|
),
|
|
);
|
|
|
|
final root = bip32.BIP32.fromSeed(seed, firoNetworkType);
|
|
return root;
|
|
}
|
|
|
|
/// wrapper for compute()
|
|
bip32.BIP32 getBip32RootWrapper(Tuple2<String, NetworkType> args) {
|
|
return getBip32Root(args.item1, args.item2);
|
|
}
|
|
|
|
Future<String> _getMintScriptWrapper(
|
|
Tuple5<int, String, int, String, bool> data) async {
|
|
String mintHex = getMintScript(data.item1, data.item2, data.item3, data.item4,
|
|
isTestnet: data.item5);
|
|
return mintHex;
|
|
}
|
|
|
|
Future<void> _setTestnetWrapper(bool isTestnet) async {
|
|
// setTestnet(isTestnet);
|
|
}
|
|
|
|
Future<Map<String, dynamic>?> getInitialAnonymitySetCache(
|
|
String groupID,
|
|
) async {
|
|
Logging.instance.log("getInitialAnonymitySetCache", level: LogLevel.Info);
|
|
final Client client = Client();
|
|
try {
|
|
final uri = Uri.parse("$kStackCommunityNodesEndpoint/getAnonymity");
|
|
|
|
final anonSetResult = await client.post(
|
|
uri,
|
|
headers: {'Content-Type': 'application/json'},
|
|
body: jsonEncode({
|
|
"jsonrpc": "2.0",
|
|
"id": "0",
|
|
'aset': groupID,
|
|
}),
|
|
);
|
|
|
|
final response = jsonDecode(anonSetResult.body.toString());
|
|
if (response['status'] == 'success') {
|
|
final anonResponse = jsonDecode(response['result'] as String);
|
|
|
|
final setData = Map<String, dynamic>.from(anonResponse["result"] as Map);
|
|
return setData;
|
|
} else {
|
|
return null;
|
|
}
|
|
} catch (e, s) {
|
|
Logging.instance.log("$e $s", level: LogLevel.Error);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
/// Handles a single instance of a firo wallet
|
|
class FiroWallet extends CoinServiceAPI {
|
|
static const integrationTestFlag =
|
|
bool.fromEnvironment("IS_INTEGRATION_TEST");
|
|
|
|
final _prefs = Prefs.instance;
|
|
|
|
Timer? timer;
|
|
late Coin _coin;
|
|
|
|
bool _shouldAutoSync = false;
|
|
|
|
@override
|
|
bool get shouldAutoSync => _shouldAutoSync;
|
|
|
|
@override
|
|
set shouldAutoSync(bool shouldAutoSync) {
|
|
if (_shouldAutoSync != shouldAutoSync) {
|
|
_shouldAutoSync = shouldAutoSync;
|
|
if (!shouldAutoSync) {
|
|
timer?.cancel();
|
|
timer = null;
|
|
stopNetworkAlivePinging();
|
|
} else {
|
|
startNetworkAlivePinging();
|
|
refresh();
|
|
}
|
|
}
|
|
}
|
|
|
|
NetworkType get _network {
|
|
switch (coin) {
|
|
case Coin.firo:
|
|
return firoNetwork;
|
|
case Coin.firoTestNet:
|
|
return firoTestNetwork;
|
|
default:
|
|
throw Exception("Invalid network type!");
|
|
}
|
|
}
|
|
|
|
@override
|
|
set isFavorite(bool markFavorite) {
|
|
DB.instance.put<dynamic>(
|
|
boxName: walletId, key: "isFavorite", value: markFavorite);
|
|
}
|
|
|
|
@override
|
|
bool get isFavorite {
|
|
try {
|
|
return DB.instance.get<dynamic>(boxName: walletId, key: "isFavorite")
|
|
as bool;
|
|
} catch (e, s) {
|
|
Logging.instance
|
|
.log("isFavorite fetch failed: $e\n$s", level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Coin get coin => _coin;
|
|
|
|
// @override
|
|
// String get coinName =>
|
|
// networkType == BasicNetworkType.main ? "Firo" : "tFiro";
|
|
//
|
|
// @override
|
|
// String get coinTicker =>
|
|
// networkType == BasicNetworkType.main ? "FIRO" : "tFIRO";
|
|
|
|
@override
|
|
Future<List<String>> get mnemonic => _getMnemonicList();
|
|
|
|
// index 0 and 1 for the funds available to spend.
|
|
// index 2 and 3 for all the funds in the wallet (including the undependable ones)
|
|
@override
|
|
Future<Decimal> get availableBalance async {
|
|
final balances = await this.balances;
|
|
return balances[0];
|
|
}
|
|
|
|
// index 0 and 1 for the funds available to spend.
|
|
// index 2 and 3 for all the funds in the wallet (including the undependable ones)
|
|
@override
|
|
Future<Decimal> get pendingBalance async {
|
|
final balances = await this.balances;
|
|
return balances[2] - balances[0];
|
|
}
|
|
|
|
// index 0 and 1 for the funds available to spend.
|
|
// index 2 and 3 for all the funds in the wallet (including the undependable ones)
|
|
@override
|
|
Future<Decimal> get totalBalance async {
|
|
if (!isActive) {
|
|
final totalBalance = DB.instance
|
|
.get<dynamic>(boxName: walletId, key: 'totalBalance') as String?;
|
|
if (totalBalance == null) {
|
|
final balances = await this.balances;
|
|
return balances[2];
|
|
} else {
|
|
return Decimal.parse(totalBalance);
|
|
// the following caused a crash as it seems totalBalance here
|
|
// is a string. Gotta love dynamics
|
|
// return Format.satoshisToAmount(totalBalance);
|
|
}
|
|
}
|
|
final balances = await this.balances;
|
|
return balances[2];
|
|
}
|
|
|
|
/// return spendable balance minus the maximum tx fee
|
|
@override
|
|
Future<Decimal> get balanceMinusMaxFee async {
|
|
final balances = await this.balances;
|
|
final maxFee = await this.maxFee;
|
|
return balances[0] - Format.satoshisToAmount(maxFee);
|
|
}
|
|
|
|
@override
|
|
Future<models.TransactionData> get transactionData => lelantusTransactionData;
|
|
|
|
@override
|
|
bool validateAddress(String address) {
|
|
return Address.validateAddress(address, _network);
|
|
}
|
|
|
|
/// Holds final balances, all utxos under control
|
|
Future<UtxoData>? _utxoData;
|
|
Future<UtxoData> get utxoData => _utxoData ??= _fetchUtxoData();
|
|
|
|
@override
|
|
Future<List<UtxoObject>> get unspentOutputs async =>
|
|
(await utxoData).unspentOutputArray;
|
|
|
|
/// Holds wallet transaction data
|
|
Future<models.TransactionData>? _transactionData;
|
|
Future<models.TransactionData> get _txnData =>
|
|
_transactionData ??= _fetchTransactionData();
|
|
|
|
/// Holds wallet lelantus transaction data
|
|
Future<models.TransactionData>? _lelantusTransactionData;
|
|
Future<models.TransactionData> get lelantusTransactionData =>
|
|
_lelantusTransactionData ??= _getLelantusTransactionData();
|
|
|
|
/// Holds the max fee that can be sent
|
|
Future<int>? _maxFee;
|
|
@override
|
|
Future<int> get maxFee => _maxFee ??= _fetchMaxFee();
|
|
|
|
/// Holds the current balance data
|
|
Future<List<Decimal>>? _balances;
|
|
Future<List<Decimal>> get balances => _balances ??= _getFullBalance();
|
|
|
|
/// Holds all outputs for wallet, used for displaying utxos in app security view
|
|
List<UtxoObject> _outputsList = [];
|
|
|
|
Future<Decimal> get firoPrice async {
|
|
final data =
|
|
await _priceAPI.getPricesAnd24hChange(baseCurrency: _prefs.currency);
|
|
if (coin == Coin.firoTestNet) {
|
|
return data[Coin.firo]!.item1;
|
|
}
|
|
return data[coin]!.item1;
|
|
}
|
|
|
|
// currently isn't used but required due to abstract parent class
|
|
Future<FeeObject>? _feeObject;
|
|
@override
|
|
Future<FeeObject> get fees => _feeObject ??= _getFees();
|
|
|
|
/// Holds updated receiving address
|
|
Future<String>? _currentReceivingAddress;
|
|
@override
|
|
Future<String> get currentReceivingAddress =>
|
|
_currentReceivingAddress ??= _getCurrentAddressForChain(0);
|
|
|
|
// @override
|
|
// Future<String> get currentLegacyReceivingAddress => null;
|
|
|
|
late String _walletName;
|
|
@override
|
|
String get walletName => _walletName;
|
|
|
|
// setter for updating on rename
|
|
@override
|
|
set walletName(String newName) => _walletName = newName;
|
|
|
|
/// unique wallet id
|
|
late String _walletId;
|
|
@override
|
|
String get walletId => _walletId;
|
|
|
|
Future<List<String>>? _allOwnAddresses;
|
|
@override
|
|
Future<List<String>> get allOwnAddresses =>
|
|
_allOwnAddresses ??= _fetchAllOwnAddresses();
|
|
|
|
@override
|
|
Future<bool> testNetworkConnection() async {
|
|
try {
|
|
final result = await _electrumXClient.ping();
|
|
return result;
|
|
} catch (_) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Timer? _networkAliveTimer;
|
|
|
|
void startNetworkAlivePinging() {
|
|
// call once on start right away
|
|
_periodicPingCheck();
|
|
|
|
// then periodically check
|
|
_networkAliveTimer = Timer.periodic(
|
|
Constants.networkAliveTimerDuration,
|
|
(_) async {
|
|
_periodicPingCheck();
|
|
},
|
|
);
|
|
}
|
|
|
|
void _periodicPingCheck() async {
|
|
bool hasNetwork = await testNetworkConnection();
|
|
_isConnected = hasNetwork;
|
|
if (_isConnected != hasNetwork) {
|
|
NodeConnectionStatus status = hasNetwork
|
|
? NodeConnectionStatus.connected
|
|
: NodeConnectionStatus.disconnected;
|
|
GlobalEventBus.instance
|
|
.fire(NodeConnectionStatusChangedEvent(status, walletId, coin));
|
|
}
|
|
}
|
|
|
|
void stopNetworkAlivePinging() {
|
|
_networkAliveTimer?.cancel();
|
|
_networkAliveTimer = null;
|
|
}
|
|
|
|
bool _isConnected = false;
|
|
|
|
@override
|
|
bool get isConnected => _isConnected;
|
|
|
|
Future<Map<String, dynamic>> prepareSendPublic({
|
|
required String address,
|
|
required int satoshiAmount,
|
|
Map<String, dynamic>? args,
|
|
}) async {
|
|
try {
|
|
final feeRateType = args?["feeRate"];
|
|
final feeRateAmount = args?["feeRateAmount"];
|
|
if (feeRateType is FeeRateType || feeRateAmount is int) {
|
|
late final int rate;
|
|
if (feeRateType is FeeRateType) {
|
|
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;
|
|
}
|
|
rate = fee;
|
|
} else {
|
|
rate = feeRateAmount as int;
|
|
}
|
|
|
|
// check for send all
|
|
bool isSendAll = false;
|
|
final balance =
|
|
Format.decimalAmountToSatoshis(await availablePublicBalance());
|
|
if (satoshiAmount == balance) {
|
|
isSendAll = true;
|
|
}
|
|
|
|
final txData =
|
|
await coinSelection(satoshiAmount, rate, address, isSendAll);
|
|
|
|
Logging.instance.log("prepare send: $txData", level: LogLevel.Info);
|
|
try {
|
|
if (txData is int) {
|
|
switch (txData) {
|
|
case 1:
|
|
throw Exception("Insufficient balance!");
|
|
case 2:
|
|
throw Exception(
|
|
"Insufficient funds to pay for transaction fee!");
|
|
default:
|
|
throw Exception("Transaction failed with error code $txData");
|
|
}
|
|
} else {
|
|
final hex = txData["hex"];
|
|
|
|
if (hex is String) {
|
|
final fee = txData["fee"] as int;
|
|
final vSize = txData["vSize"] as int;
|
|
|
|
Logging.instance
|
|
.log("prepared txHex: $hex", level: LogLevel.Info);
|
|
Logging.instance.log("prepared fee: $fee", level: LogLevel.Info);
|
|
Logging.instance
|
|
.log("prepared vSize: $vSize", level: LogLevel.Info);
|
|
|
|
// fee should never be less than vSize sanity check
|
|
if (fee < vSize) {
|
|
throw Exception(
|
|
"Error in fee calculation: Transaction fee cannot be less than vSize");
|
|
}
|
|
|
|
return txData as Map<String, dynamic>;
|
|
} else {
|
|
throw Exception("prepared hex is not a String!!!");
|
|
}
|
|
}
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown from prepareSend(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
} else {
|
|
throw ArgumentError("Invalid fee rate argument provided!");
|
|
}
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown from prepareSend(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<String> confirmSendPublic({dynamic txData}) async {
|
|
try {
|
|
Logging.instance.log("confirmSend txData: $txData", level: LogLevel.Info);
|
|
final txHash = await _electrumXClient.broadcastTransaction(
|
|
rawTx: txData["hex"] as String);
|
|
Logging.instance.log("Sent txHash: $txHash", level: LogLevel.Info);
|
|
return txHash;
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown from confirmSend(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<Map<String, dynamic>> prepareSend({
|
|
required String address,
|
|
required int satoshiAmount,
|
|
Map<String, dynamic>? args,
|
|
}) async {
|
|
try {
|
|
// check for send all
|
|
bool isSendAll = false;
|
|
final balance =
|
|
Format.decimalAmountToSatoshis(await availablePrivateBalance());
|
|
if (satoshiAmount == balance) {
|
|
// print("is send all");
|
|
isSendAll = true;
|
|
}
|
|
dynamic txHexOrError =
|
|
await _createJoinSplitTransaction(satoshiAmount, address, isSendAll);
|
|
Logging.instance.log("txHexOrError $txHexOrError", level: LogLevel.Error);
|
|
if (txHexOrError is int) {
|
|
// Here, we assume that transaction crafting returned an error
|
|
switch (txHexOrError) {
|
|
case 1:
|
|
throw Exception("Insufficient balance!");
|
|
default:
|
|
throw Exception("Error Creating Transaction!");
|
|
}
|
|
} else {
|
|
return txHexOrError as Map<String, dynamic>;
|
|
}
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown in firo prepareSend(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<String> confirmSend({required Map<String, dynamic> txData}) async {
|
|
if (await _submitLelantusToNetwork(txData)) {
|
|
try {
|
|
final txid = txData["txid"] as String;
|
|
|
|
// temporarily update apdate available balance until a full refresh is done
|
|
|
|
// TODO: something here causes an exception to be thrown giving user false info that the tx failed
|
|
Decimal sendTotal = Format.satoshisToAmount(txData["value"] as int);
|
|
sendTotal += Decimal.parse(txData["fees"].toString());
|
|
final bals = await balances;
|
|
bals[0] -= sendTotal;
|
|
_balances = Future(() => bals);
|
|
|
|
return txid;
|
|
} catch (e, s) {
|
|
debugPrint("$e $s");
|
|
return txData["txid"] as String;
|
|
// don't throw anything here or it will tell the user that th tx
|
|
// failed even though it was successfully broadcast to network
|
|
// throw Exception("Transaction failed.");
|
|
}
|
|
} else {
|
|
//TODO provide more info
|
|
throw Exception("Transaction failed.");
|
|
}
|
|
}
|
|
|
|
/// returns txid on successful send
|
|
///
|
|
/// can throw
|
|
@override
|
|
Future<String> send({
|
|
required String toAddress,
|
|
required int amount,
|
|
Map<String, String> args = const {},
|
|
}) async {
|
|
try {
|
|
dynamic txHexOrError =
|
|
await _createJoinSplitTransaction(amount, toAddress, false);
|
|
Logging.instance.log("txHexOrError $txHexOrError", level: LogLevel.Error);
|
|
if (txHexOrError is int) {
|
|
// Here, we assume that transaction crafting returned an error
|
|
switch (txHexOrError) {
|
|
case 1:
|
|
throw Exception("Insufficient balance!");
|
|
default:
|
|
throw Exception("Error Creating Transaction!");
|
|
}
|
|
} else {
|
|
if (await _submitLelantusToNetwork(
|
|
txHexOrError as Map<String, dynamic>)) {
|
|
final txid = txHexOrError["txid"] as String;
|
|
|
|
// temporarily update apdate available balance until a full refresh is done
|
|
Decimal sendTotal =
|
|
Format.satoshisToAmount(txHexOrError["value"] as int);
|
|
sendTotal += Decimal.parse(txHexOrError["fees"].toString());
|
|
final bals = await balances;
|
|
bals[0] -= sendTotal;
|
|
_balances = Future(() => bals);
|
|
|
|
return txid;
|
|
} else {
|
|
//TODO provide more info
|
|
throw Exception("Transaction failed.");
|
|
}
|
|
}
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown in firo send(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<List<String>> _getMnemonicList() async {
|
|
final mnemonicString =
|
|
await _secureStore.read(key: '${_walletId}_mnemonic');
|
|
if (mnemonicString == null) {
|
|
return [];
|
|
}
|
|
final List<String> data = mnemonicString.split(' ');
|
|
return data;
|
|
}
|
|
|
|
late ElectrumX _electrumXClient;
|
|
ElectrumX get electrumXClient => _electrumXClient;
|
|
|
|
late CachedElectrumX _cachedElectrumXClient;
|
|
CachedElectrumX get cachedElectrumXClient => _cachedElectrumXClient;
|
|
|
|
late FlutterSecureStorageInterface _secureStore;
|
|
|
|
late PriceAPI _priceAPI;
|
|
|
|
late TransactionNotificationTracker txTracker;
|
|
|
|
// Constructor
|
|
FiroWallet({
|
|
required String walletId,
|
|
required String walletName,
|
|
required Coin coin,
|
|
required ElectrumX client,
|
|
required CachedElectrumX cachedClient,
|
|
required TransactionNotificationTracker tracker,
|
|
PriceAPI? priceAPI,
|
|
FlutterSecureStorageInterface? secureStore,
|
|
}) {
|
|
txTracker = tracker;
|
|
_walletId = walletId;
|
|
_walletName = walletName;
|
|
_coin = coin;
|
|
_electrumXClient = client;
|
|
_cachedElectrumXClient = cachedClient;
|
|
|
|
_priceAPI = priceAPI ?? PriceAPI(Client());
|
|
_secureStore =
|
|
secureStore ?? const SecureStorageWrapper(FlutterSecureStorage());
|
|
|
|
Logging.instance.log("$walletName isolates length: ${isolates.length}",
|
|
level: LogLevel.Info);
|
|
// investigate possible issues killing shared isolates between multiple firo instances
|
|
for (final isolate in isolates.values) {
|
|
isolate.kill(priority: Isolate.immediate);
|
|
}
|
|
isolates.clear();
|
|
}
|
|
|
|
int estimateTxFee({required int vSize, required int feeRatePerKB}) {
|
|
return vSize * (feeRatePerKB / 1000).ceil();
|
|
}
|
|
|
|
/// The coinselection algorithm decides whether or not the user is eligible to make the transaction
|
|
/// with [satoshiAmountToSend] and [selectedTxFeeRate]. If so, it will call buildTrasaction() and return
|
|
/// a map containing the tx hex along with other important information. If not, then it will return
|
|
/// an integer (1 or 2)
|
|
dynamic coinSelection(
|
|
int satoshiAmountToSend,
|
|
int selectedTxFeeRate,
|
|
String _recipientAddress,
|
|
bool isSendAll, {
|
|
int additionalOutputs = 0,
|
|
List<UtxoObject>? utxos,
|
|
}) async {
|
|
Logging.instance
|
|
.log("Starting coinSelection ----------", level: LogLevel.Info);
|
|
final List<UtxoObject> availableOutputs = utxos ?? _outputsList;
|
|
final List<UtxoObject> spendableOutputs = [];
|
|
int spendableSatoshiValue = 0;
|
|
|
|
// Build list of spendable outputs and totaling their satoshi amount
|
|
for (var i = 0; i < availableOutputs.length; i++) {
|
|
if (availableOutputs[i].blocked == false &&
|
|
availableOutputs[i].status.confirmed == true) {
|
|
spendableOutputs.add(availableOutputs[i]);
|
|
spendableSatoshiValue += availableOutputs[i].value;
|
|
}
|
|
}
|
|
|
|
// sort spendable by age (oldest first)
|
|
spendableOutputs.sort(
|
|
(a, b) => b.status.confirmations.compareTo(a.status.confirmations));
|
|
|
|
Logging.instance.log("spendableOutputs.length: ${spendableOutputs.length}",
|
|
level: LogLevel.Info);
|
|
Logging.instance
|
|
.log("spendableOutputs: $spendableOutputs", level: LogLevel.Info);
|
|
Logging.instance.log("spendableSatoshiValue: $spendableSatoshiValue",
|
|
level: LogLevel.Info);
|
|
Logging.instance
|
|
.log("satoshiAmountToSend: $satoshiAmountToSend", level: LogLevel.Info);
|
|
// If the amount the user is trying to send is smaller than the amount that they have spendable,
|
|
// then return 1, which indicates that they have an insufficient balance.
|
|
if (spendableSatoshiValue < satoshiAmountToSend) {
|
|
return 1;
|
|
// If the amount the user wants to send is exactly equal to the amount they can spend, then return
|
|
// 2, which indicates that they are not leaving enough over to pay the transaction fee
|
|
} else if (spendableSatoshiValue == satoshiAmountToSend && !isSendAll) {
|
|
return 2;
|
|
}
|
|
// If neither of these statements pass, we assume that the user has a spendable balance greater
|
|
// than the amount they're attempting to send. Note that this value still does not account for
|
|
// the added transaction fee, which may require an extra input and will need to be checked for
|
|
// later on.
|
|
|
|
// Possible situation right here
|
|
int satoshisBeingUsed = 0;
|
|
int inputsBeingConsumed = 0;
|
|
List<UtxoObject> utxoObjectsToUse = [];
|
|
|
|
for (var i = 0;
|
|
satoshisBeingUsed < satoshiAmountToSend && i < spendableOutputs.length;
|
|
i++) {
|
|
utxoObjectsToUse.add(spendableOutputs[i]);
|
|
satoshisBeingUsed += spendableOutputs[i].value;
|
|
inputsBeingConsumed += 1;
|
|
}
|
|
for (int i = 0;
|
|
i < additionalOutputs && inputsBeingConsumed < spendableOutputs.length;
|
|
i++) {
|
|
utxoObjectsToUse.add(spendableOutputs[inputsBeingConsumed]);
|
|
satoshisBeingUsed += spendableOutputs[inputsBeingConsumed].value;
|
|
inputsBeingConsumed += 1;
|
|
}
|
|
|
|
Logging.instance
|
|
.log("satoshisBeingUsed: $satoshisBeingUsed", level: LogLevel.Info);
|
|
Logging.instance
|
|
.log("inputsBeingConsumed: $inputsBeingConsumed", level: LogLevel.Info);
|
|
Logging.instance
|
|
.log('utxoObjectsToUse: $utxoObjectsToUse', level: LogLevel.Info);
|
|
|
|
// numberOfOutputs' length must always be equal to that of recipientsArray and recipientsAmtArray
|
|
List<String> recipientsArray = [_recipientAddress];
|
|
List<int> recipientsAmtArray = [satoshiAmountToSend];
|
|
|
|
// gather required signing data
|
|
final utxoSigningData = await fetchBuildTxData(utxoObjectsToUse);
|
|
|
|
if (isSendAll) {
|
|
Logging.instance
|
|
.log("Attempting to send all $coin", level: LogLevel.Info);
|
|
|
|
final int vSizeForOneOutput = (await buildTransaction(
|
|
utxosToUse: utxoObjectsToUse,
|
|
utxoSigningData: utxoSigningData,
|
|
recipients: [_recipientAddress],
|
|
satoshiAmounts: [satoshisBeingUsed - 1],
|
|
))["vSize"] as int;
|
|
int feeForOneOutput = estimateTxFee(
|
|
vSize: vSizeForOneOutput,
|
|
feeRatePerKB: selectedTxFeeRate,
|
|
);
|
|
|
|
if (feeForOneOutput < vSizeForOneOutput + 1) {
|
|
feeForOneOutput = vSizeForOneOutput + 1;
|
|
}
|
|
|
|
final int amount = satoshiAmountToSend - feeForOneOutput;
|
|
dynamic txn = await buildTransaction(
|
|
utxosToUse: utxoObjectsToUse,
|
|
utxoSigningData: utxoSigningData,
|
|
recipients: recipientsArray,
|
|
satoshiAmounts: [amount],
|
|
);
|
|
Map<String, dynamic> transactionObject = {
|
|
"hex": txn["hex"],
|
|
"recipient": recipientsArray[0],
|
|
"recipientAmt": amount,
|
|
"fee": feeForOneOutput,
|
|
"vSize": txn["vSize"],
|
|
};
|
|
return transactionObject;
|
|
}
|
|
|
|
final int vSizeForOneOutput = (await buildTransaction(
|
|
utxosToUse: utxoObjectsToUse,
|
|
utxoSigningData: utxoSigningData,
|
|
recipients: [_recipientAddress],
|
|
satoshiAmounts: [satoshisBeingUsed - 1],
|
|
))["vSize"] as int;
|
|
final int vSizeForTwoOutPuts = (await buildTransaction(
|
|
utxosToUse: utxoObjectsToUse,
|
|
utxoSigningData: utxoSigningData,
|
|
recipients: [
|
|
_recipientAddress,
|
|
await _getCurrentAddressForChain(1),
|
|
],
|
|
satoshiAmounts: [
|
|
satoshiAmountToSend,
|
|
satoshisBeingUsed - satoshiAmountToSend - 1,
|
|
], // dust limit is the minimum amount a change output should be
|
|
))["vSize"] as int;
|
|
debugPrint("vSizeForOneOutput $vSizeForOneOutput");
|
|
debugPrint("vSizeForTwoOutPuts $vSizeForTwoOutPuts");
|
|
|
|
// Assume 1 output, only for recipient and no change
|
|
var feeForOneOutput = estimateTxFee(
|
|
vSize: vSizeForOneOutput,
|
|
feeRatePerKB: selectedTxFeeRate,
|
|
);
|
|
// Assume 2 outputs, one for recipient and one for change
|
|
var feeForTwoOutputs = estimateTxFee(
|
|
vSize: vSizeForTwoOutPuts,
|
|
feeRatePerKB: selectedTxFeeRate,
|
|
);
|
|
|
|
Logging.instance
|
|
.log("feeForTwoOutputs: $feeForTwoOutputs", level: LogLevel.Info);
|
|
Logging.instance
|
|
.log("feeForOneOutput: $feeForOneOutput", level: LogLevel.Info);
|
|
if (feeForOneOutput < (vSizeForOneOutput + 1)) {
|
|
feeForOneOutput = (vSizeForOneOutput + 1);
|
|
}
|
|
if (feeForTwoOutputs < ((vSizeForTwoOutPuts + 1))) {
|
|
feeForTwoOutputs = ((vSizeForTwoOutPuts + 1));
|
|
}
|
|
|
|
Logging.instance
|
|
.log("feeForTwoOutputs: $feeForTwoOutputs", level: LogLevel.Info);
|
|
Logging.instance
|
|
.log("feeForOneOutput: $feeForOneOutput", level: LogLevel.Info);
|
|
|
|
if (satoshisBeingUsed - satoshiAmountToSend > feeForOneOutput) {
|
|
if (satoshisBeingUsed - satoshiAmountToSend >
|
|
feeForOneOutput + DUST_LIMIT) {
|
|
// Here, we know that theoretically, we may be able to include another output(change) but we first need to
|
|
// factor in the value of this output in satoshis.
|
|
int changeOutputSize =
|
|
satoshisBeingUsed - satoshiAmountToSend - feeForTwoOutputs;
|
|
// We check to see if the user can pay for the new transaction with 2 outputs instead of one. If they can and
|
|
// the second output's size > DUST_LIMIT satoshis, we perform the mechanics required to properly generate and use a new
|
|
// change address.
|
|
if (changeOutputSize > DUST_LIMIT &&
|
|
satoshisBeingUsed - satoshiAmountToSend - changeOutputSize ==
|
|
feeForTwoOutputs) {
|
|
// generate new change address if current change address has been used
|
|
await checkChangeAddressForTransactions();
|
|
final String newChangeAddress = await _getCurrentAddressForChain(1);
|
|
|
|
int feeBeingPaid =
|
|
satoshisBeingUsed - satoshiAmountToSend - changeOutputSize;
|
|
|
|
recipientsArray.add(newChangeAddress);
|
|
recipientsAmtArray.add(changeOutputSize);
|
|
// At this point, we have the outputs we're going to use, the amounts to send along with which addresses
|
|
// we intend to send these amounts to. We have enough to send instructions to build the transaction.
|
|
Logging.instance.log('2 outputs in tx', level: LogLevel.Info);
|
|
Logging.instance
|
|
.log('Input size: $satoshisBeingUsed', level: LogLevel.Info);
|
|
Logging.instance.log('Recipient output size: $satoshiAmountToSend',
|
|
level: LogLevel.Info);
|
|
Logging.instance.log('Change Output Size: $changeOutputSize',
|
|
level: LogLevel.Info);
|
|
Logging.instance.log(
|
|
'Difference (fee being paid): $feeBeingPaid sats',
|
|
level: LogLevel.Info);
|
|
Logging.instance
|
|
.log('Estimated fee: $feeForTwoOutputs', level: LogLevel.Info);
|
|
dynamic txn = await buildTransaction(
|
|
utxosToUse: utxoObjectsToUse,
|
|
utxoSigningData: utxoSigningData,
|
|
recipients: recipientsArray,
|
|
satoshiAmounts: recipientsAmtArray,
|
|
);
|
|
|
|
// make sure minimum fee is accurate if that is being used
|
|
if (txn["vSize"] - feeBeingPaid == 1) {
|
|
int changeOutputSize =
|
|
satoshisBeingUsed - satoshiAmountToSend - (txn["vSize"] as int);
|
|
feeBeingPaid =
|
|
satoshisBeingUsed - satoshiAmountToSend - changeOutputSize;
|
|
recipientsAmtArray.removeLast();
|
|
recipientsAmtArray.add(changeOutputSize);
|
|
Logging.instance.log('Adjusted Input size: $satoshisBeingUsed',
|
|
level: LogLevel.Info);
|
|
Logging.instance.log(
|
|
'Adjusted Recipient output size: $satoshiAmountToSend',
|
|
level: LogLevel.Info);
|
|
Logging.instance.log(
|
|
'Adjusted Change Output Size: $changeOutputSize',
|
|
level: LogLevel.Info);
|
|
Logging.instance.log(
|
|
'Adjusted Difference (fee being paid): $feeBeingPaid sats',
|
|
level: LogLevel.Info);
|
|
Logging.instance.log('Adjusted Estimated fee: $feeForTwoOutputs',
|
|
level: LogLevel.Info);
|
|
txn = await buildTransaction(
|
|
utxosToUse: utxoObjectsToUse,
|
|
utxoSigningData: utxoSigningData,
|
|
recipients: recipientsArray,
|
|
satoshiAmounts: recipientsAmtArray,
|
|
);
|
|
}
|
|
|
|
Map<String, dynamic> transactionObject = {
|
|
"hex": txn["hex"],
|
|
"recipient": recipientsArray[0],
|
|
"recipientAmt": recipientsAmtArray[0],
|
|
"fee": feeBeingPaid,
|
|
"vSize": txn["vSize"],
|
|
};
|
|
return transactionObject;
|
|
} else {
|
|
// Something went wrong here. It either overshot or undershot the estimated fee amount or the changeOutputSize
|
|
// is smaller than or equal to [DUST_LIMIT]. Revert to single output transaction.
|
|
Logging.instance.log('1 output in tx', level: LogLevel.Info);
|
|
Logging.instance
|
|
.log('Input size: $satoshisBeingUsed', level: LogLevel.Info);
|
|
Logging.instance.log('Recipient output size: $satoshiAmountToSend',
|
|
level: LogLevel.Info);
|
|
Logging.instance.log(
|
|
'Difference (fee being paid): ${satoshisBeingUsed - satoshiAmountToSend} sats',
|
|
level: LogLevel.Info);
|
|
Logging.instance
|
|
.log('Estimated fee: $feeForOneOutput', level: LogLevel.Info);
|
|
dynamic txn = await buildTransaction(
|
|
utxosToUse: utxoObjectsToUse,
|
|
utxoSigningData: utxoSigningData,
|
|
recipients: recipientsArray,
|
|
satoshiAmounts: recipientsAmtArray,
|
|
);
|
|
Map<String, dynamic> transactionObject = {
|
|
"hex": txn["hex"],
|
|
"recipient": recipientsArray[0],
|
|
"recipientAmt": recipientsAmtArray[0],
|
|
"fee": satoshisBeingUsed - satoshiAmountToSend,
|
|
"vSize": txn["vSize"],
|
|
};
|
|
return transactionObject;
|
|
}
|
|
} else {
|
|
// No additional outputs needed since adding one would mean that it'd be smaller than 546 sats
|
|
// which makes it uneconomical to add to the transaction. Here, we pass data directly to instruct
|
|
// the wallet to begin crafting the transaction that the user requested.
|
|
Logging.instance.log('1 output in tx', level: LogLevel.Info);
|
|
Logging.instance
|
|
.log('Input size: $satoshisBeingUsed', level: LogLevel.Info);
|
|
Logging.instance.log('Recipient output size: $satoshiAmountToSend',
|
|
level: LogLevel.Info);
|
|
Logging.instance.log(
|
|
'Difference (fee being paid): ${satoshisBeingUsed - satoshiAmountToSend} sats',
|
|
level: LogLevel.Info);
|
|
Logging.instance
|
|
.log('Estimated fee: $feeForOneOutput', level: LogLevel.Info);
|
|
dynamic txn = await buildTransaction(
|
|
utxosToUse: utxoObjectsToUse,
|
|
utxoSigningData: utxoSigningData,
|
|
recipients: recipientsArray,
|
|
satoshiAmounts: recipientsAmtArray,
|
|
);
|
|
Map<String, dynamic> transactionObject = {
|
|
"hex": txn["hex"],
|
|
"recipient": recipientsArray[0],
|
|
"recipientAmt": recipientsAmtArray[0],
|
|
"fee": satoshisBeingUsed - satoshiAmountToSend,
|
|
"vSize": txn["vSize"],
|
|
};
|
|
return transactionObject;
|
|
}
|
|
} else if (satoshisBeingUsed - satoshiAmountToSend == feeForOneOutput) {
|
|
// In this scenario, no additional change output is needed since inputs - outputs equal exactly
|
|
// what we need to pay for fees. Here, we pass data directly to instruct the wallet to begin
|
|
// crafting the transaction that the user requested.
|
|
Logging.instance.log('1 output in tx', level: LogLevel.Info);
|
|
Logging.instance
|
|
.log('Input size: $satoshisBeingUsed', level: LogLevel.Info);
|
|
Logging.instance.log('Recipient output size: $satoshiAmountToSend',
|
|
level: LogLevel.Info);
|
|
Logging.instance.log(
|
|
'Fee being paid: ${satoshisBeingUsed - satoshiAmountToSend} sats',
|
|
level: LogLevel.Info);
|
|
Logging.instance
|
|
.log('Estimated fee: $feeForOneOutput', level: LogLevel.Info);
|
|
dynamic txn = await buildTransaction(
|
|
utxosToUse: utxoObjectsToUse,
|
|
utxoSigningData: utxoSigningData,
|
|
recipients: recipientsArray,
|
|
satoshiAmounts: recipientsAmtArray,
|
|
);
|
|
Map<String, dynamic> transactionObject = {
|
|
"hex": txn["hex"],
|
|
"recipient": recipientsArray[0],
|
|
"recipientAmt": recipientsAmtArray[0],
|
|
"fee": feeForOneOutput,
|
|
"vSize": txn["vSize"],
|
|
};
|
|
return transactionObject;
|
|
} else {
|
|
// Remember that returning 2 indicates that the user does not have a sufficient balance to
|
|
// pay for the transaction fee. Ideally, at this stage, we should check if the user has any
|
|
// additional outputs they're able to spend and then recalculate fees.
|
|
Logging.instance.log(
|
|
'Cannot pay tx fee - checking for more outputs and trying again',
|
|
level: LogLevel.Warning);
|
|
// try adding more outputs
|
|
if (spendableOutputs.length > inputsBeingConsumed) {
|
|
return coinSelection(satoshiAmountToSend, selectedTxFeeRate,
|
|
_recipientAddress, isSendAll,
|
|
additionalOutputs: additionalOutputs + 1, utxos: utxos);
|
|
}
|
|
return 2;
|
|
}
|
|
}
|
|
|
|
Future<Map<String, dynamic>> fetchBuildTxData(
|
|
List<UtxoObject> utxosToUse,
|
|
) async {
|
|
// return data
|
|
Map<String, dynamic> results = {};
|
|
Map<String, List<String>> addressTxid = {};
|
|
|
|
// addresses to check
|
|
List<String> addresses = [];
|
|
|
|
try {
|
|
// Populating the addresses to check
|
|
for (var i = 0; i < utxosToUse.length; i++) {
|
|
final txid = utxosToUse[i].txid;
|
|
final tx = await _cachedElectrumXClient.getTransaction(
|
|
txHash: txid,
|
|
coin: coin,
|
|
);
|
|
|
|
for (final output in tx["vout"] as List) {
|
|
final n = output["n"];
|
|
if (n != null && n == utxosToUse[i].vout) {
|
|
final address = output["scriptPubKey"]["addresses"][0] as String;
|
|
|
|
if (!addressTxid.containsKey(address)) {
|
|
addressTxid[address] = <String>[];
|
|
}
|
|
(addressTxid[address] as List).add(txid);
|
|
|
|
addresses.add(address);
|
|
}
|
|
}
|
|
}
|
|
|
|
// p2pkh / bip44
|
|
final addressesLength = addresses.length;
|
|
if (addressesLength > 0) {
|
|
final receiveDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_receiveDerivations");
|
|
final receiveDerivations = Map<String, dynamic>.from(
|
|
jsonDecode(receiveDerivationsString ?? "{}") as Map);
|
|
|
|
final changeDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_changeDerivations");
|
|
final changeDerivations = Map<String, dynamic>.from(
|
|
jsonDecode(changeDerivationsString ?? "{}") as Map);
|
|
|
|
for (int i = 0; i < addressesLength; i++) {
|
|
// receives
|
|
|
|
dynamic receiveDerivation;
|
|
|
|
for (int j = 0; j < receiveDerivations.length; j++) {
|
|
if (receiveDerivations["$j"]["address"] == addresses[i]) {
|
|
receiveDerivation = receiveDerivations["$j"];
|
|
}
|
|
}
|
|
|
|
// receiveDerivation = receiveDerivations[addresses[i]];
|
|
// if a match exists it will not be null
|
|
if (receiveDerivation != null) {
|
|
final data = P2PKH(
|
|
data: PaymentData(
|
|
pubkey: Format.stringToUint8List(
|
|
receiveDerivation["publicKey"] as String)),
|
|
network: _network,
|
|
).data;
|
|
|
|
for (String tx in addressTxid[addresses[i]]!) {
|
|
results[tx] = {
|
|
"output": data.output,
|
|
"keyPair": ECPair.fromWIF(
|
|
receiveDerivation["wif"] as String,
|
|
network: _network,
|
|
),
|
|
};
|
|
}
|
|
} else {
|
|
// if its not a receive, check change
|
|
|
|
dynamic changeDerivation;
|
|
|
|
for (int j = 0; j < changeDerivations.length; j++) {
|
|
if (changeDerivations["$j"]["address"] == addresses[i]) {
|
|
changeDerivation = changeDerivations["$j"];
|
|
}
|
|
}
|
|
|
|
// final changeDerivation = changeDerivations[addresses[i]];
|
|
// if a match exists it will not be null
|
|
if (changeDerivation != null) {
|
|
final data = P2PKH(
|
|
data: PaymentData(
|
|
pubkey: Format.stringToUint8List(
|
|
changeDerivation["publicKey"] as String)),
|
|
network: _network,
|
|
).data;
|
|
|
|
for (String tx in addressTxid[addresses[i]]!) {
|
|
results[tx] = {
|
|
"output": data.output,
|
|
"keyPair": ECPair.fromWIF(
|
|
changeDerivation["wif"] as String,
|
|
network: _network,
|
|
),
|
|
};
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return results;
|
|
} catch (e, s) {
|
|
Logging.instance
|
|
.log("fetchBuildTxData() threw: $e,\n$s", level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
/// Builds and signs a transaction
|
|
Future<Map<String, dynamic>> buildTransaction({
|
|
required List<UtxoObject> utxosToUse,
|
|
required Map<String, dynamic> utxoSigningData,
|
|
required List<String> recipients,
|
|
required List<int> satoshiAmounts,
|
|
}) async {
|
|
Logging.instance
|
|
.log("Starting buildTransaction ----------", level: LogLevel.Info);
|
|
|
|
final txb = TransactionBuilder(network: _network);
|
|
txb.setVersion(1);
|
|
|
|
// Add transaction inputs
|
|
for (var i = 0; i < utxosToUse.length; i++) {
|
|
final txid = utxosToUse[i].txid;
|
|
txb.addInput(txid, utxosToUse[i].vout, null,
|
|
utxoSigningData[txid]["output"] as Uint8List);
|
|
}
|
|
|
|
// Add transaction output
|
|
for (var i = 0; i < recipients.length; i++) {
|
|
txb.addOutput(recipients[i], satoshiAmounts[i]);
|
|
}
|
|
|
|
try {
|
|
// Sign the transaction accordingly
|
|
for (var i = 0; i < utxosToUse.length; i++) {
|
|
final txid = utxosToUse[i].txid;
|
|
txb.sign(
|
|
vin: i,
|
|
keyPair: utxoSigningData[txid]["keyPair"] as ECPair,
|
|
witnessValue: utxosToUse[i].value,
|
|
redeemScript: utxoSigningData[txid]["redeemScript"] as Uint8List?,
|
|
);
|
|
}
|
|
} catch (e, s) {
|
|
Logging.instance.log("Caught exception while signing transaction: $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
|
|
final builtTx = txb.build();
|
|
final vSize = builtTx.virtualSize();
|
|
|
|
return {"hex": builtTx.toHex(), "vSize": vSize};
|
|
}
|
|
|
|
@override
|
|
Future<void> updateNode(bool shouldRefresh) async {
|
|
final failovers = NodeService()
|
|
.failoverNodesFor(coin: coin)
|
|
.map(
|
|
(e) => ElectrumXNode(
|
|
address: e.host,
|
|
port: e.port,
|
|
name: e.name,
|
|
id: e.id,
|
|
useSSL: e.useSSL,
|
|
),
|
|
)
|
|
.toList();
|
|
final newNode = await _getCurrentNode();
|
|
_cachedElectrumXClient = CachedElectrumX.from(
|
|
node: newNode,
|
|
prefs: _prefs,
|
|
failovers: failovers,
|
|
);
|
|
_electrumXClient = ElectrumX.from(
|
|
node: newNode,
|
|
prefs: _prefs,
|
|
failovers: failovers,
|
|
);
|
|
|
|
if (shouldRefresh) {
|
|
unawaited(refresh());
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> initializeNew() async {
|
|
Logging.instance
|
|
.log("Generating new ${coin.prettyName} wallet.", level: LogLevel.Info);
|
|
|
|
if (DB.instance.get<dynamic>(boxName: walletId, key: "id") != null) {
|
|
throw Exception(
|
|
"Attempted to initialize a new wallet using an existing wallet ID!");
|
|
}
|
|
|
|
await _prefs.init();
|
|
try {
|
|
await _generateNewWallet();
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown from initializeNew(): $e\n$s",
|
|
level: LogLevel.Fatal);
|
|
rethrow;
|
|
}
|
|
|
|
await Future.wait([
|
|
DB.instance.put<dynamic>(boxName: walletId, key: "id", value: _walletId),
|
|
_getLelantusTransactionData().then((lelantusTxData) =>
|
|
_lelantusTransactionData = Future(() => lelantusTxData)),
|
|
DB.instance
|
|
.put<dynamic>(boxName: walletId, key: "isFavorite", value: false),
|
|
]);
|
|
}
|
|
|
|
@override
|
|
Future<void> initializeExisting() async {
|
|
Logging.instance.log(
|
|
"Opening existing $_walletId ${coin.prettyName} wallet.",
|
|
level: LogLevel.Info);
|
|
|
|
if ((DB.instance.get<dynamic>(boxName: walletId, key: "id") as String?) ==
|
|
null) {
|
|
throw Exception(
|
|
"Attempted to initialize an existing wallet using an unknown wallet ID!");
|
|
}
|
|
await _prefs.init();
|
|
final data =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: "latest_tx_model")
|
|
as models.TransactionData?;
|
|
if (data != null) {
|
|
_transactionData = Future(() => data);
|
|
}
|
|
}
|
|
|
|
Future<bool> refreshIfThereIsNewData() async {
|
|
if (longMutex) return false;
|
|
if (_hasCalledExit) return false;
|
|
Logging.instance
|
|
.log("$walletName refreshIfThereIsNewData", level: LogLevel.Info);
|
|
|
|
try {
|
|
bool needsRefresh = false;
|
|
Set<String> txnsToCheck = {};
|
|
|
|
for (final String txid in txTracker.pendings) {
|
|
if (!txTracker.wasNotifiedConfirmed(txid)) {
|
|
txnsToCheck.add(txid);
|
|
}
|
|
}
|
|
|
|
for (String txid in txnsToCheck) {
|
|
final txn = await electrumXClient.getTransaction(txHash: txid);
|
|
int confirmations = txn["confirmations"] as int? ?? 0;
|
|
bool isUnconfirmed = confirmations < MINIMUM_CONFIRMATIONS;
|
|
if (!isUnconfirmed) {
|
|
needsRefresh = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!needsRefresh) {
|
|
var allOwnAddresses = await this.allOwnAddresses;
|
|
List<Map<String, dynamic>> allTxs =
|
|
await _fetchHistory(allOwnAddresses);
|
|
models.TransactionData txData = await _txnData;
|
|
for (Map<String, dynamic> transaction in allTxs) {
|
|
if (txData.findTransaction(transaction['tx_hash'] as String) ==
|
|
null) {
|
|
Logging.instance.log(
|
|
" txid not found in address history already ${transaction['tx_hash']}",
|
|
level: LogLevel.Info);
|
|
needsRefresh = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return needsRefresh;
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception caught in refreshIfThereIsNewData: $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<void> getAllTxsToWatch(
|
|
models.TransactionData txData,
|
|
models.TransactionData lTxData,
|
|
) async {
|
|
if (_hasCalledExit) return;
|
|
Logging.instance.log("$walletName periodic", level: LogLevel.Info);
|
|
List<models.Transaction> unconfirmedTxnsToNotifyPending = [];
|
|
List<models.Transaction> unconfirmedTxnsToNotifyConfirmed = [];
|
|
|
|
for (models.TransactionChunk chunk in txData.txChunks) {
|
|
for (models.Transaction tx in chunk.transactions) {
|
|
models.Transaction? lTx = lTxData.findTransaction(tx.txid);
|
|
|
|
if (tx.confirmedStatus) {
|
|
if (txTracker.wasNotifiedPending(tx.txid) &&
|
|
!txTracker.wasNotifiedConfirmed(tx.txid)) {
|
|
// get all transactions that were notified as pending but not as confirmed
|
|
unconfirmedTxnsToNotifyConfirmed.add(tx);
|
|
}
|
|
if (lTx != null &&
|
|
(lTx.inputs.isEmpty || lTx.inputs[0].txid.isEmpty) &&
|
|
lTx.confirmedStatus == false &&
|
|
tx.txType == "Received") {
|
|
// If this is a received that is past 1 or more confirmations and has not been minted,
|
|
if (!txTracker.wasNotifiedPending(tx.txid)) {
|
|
unconfirmedTxnsToNotifyPending.add(tx);
|
|
}
|
|
}
|
|
} else {
|
|
if (!txTracker.wasNotifiedPending(tx.txid)) {
|
|
// get all transactions that were not notified as pending yet
|
|
unconfirmedTxnsToNotifyPending.add(tx);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (models.TransactionChunk chunk in txData.txChunks) {
|
|
for (models.Transaction tx in chunk.transactions) {
|
|
if (!tx.confirmedStatus && tx.inputs[0].txid.isNotEmpty) {
|
|
// Get all normal txs that are at 0 confirmations
|
|
unconfirmedTxnsToNotifyPending
|
|
.removeWhere((e) => e.txid == tx.inputs[0].txid);
|
|
Logging.instance.log("removed tx: ${tx.txid}", level: LogLevel.Info);
|
|
}
|
|
}
|
|
}
|
|
for (models.TransactionChunk chunk in lTxData.txChunks) {
|
|
for (models.Transaction lTX in chunk.transactions) {
|
|
models.Transaction? tx = txData.findTransaction(lTX.txid);
|
|
if (tx == null) {
|
|
// if this is a ltx transaction that is unconfirmed and not represented in the normal transaction set.
|
|
if (!lTX.confirmedStatus) {
|
|
if (!txTracker.wasNotifiedPending(lTX.txid)) {
|
|
unconfirmedTxnsToNotifyPending.add(lTX);
|
|
}
|
|
} else {
|
|
if (txTracker.wasNotifiedPending(lTX.txid) &&
|
|
!txTracker.wasNotifiedConfirmed(lTX.txid)) {
|
|
unconfirmedTxnsToNotifyConfirmed.add(lTX);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
Logging.instance.log(
|
|
"unconfirmedTxnsToNotifyPending $unconfirmedTxnsToNotifyPending",
|
|
level: LogLevel.Info);
|
|
Logging.instance.log(
|
|
"unconfirmedTxnsToNotifyConfirmed $unconfirmedTxnsToNotifyConfirmed",
|
|
level: LogLevel.Info);
|
|
|
|
for (final tx in unconfirmedTxnsToNotifyPending) {
|
|
switch (tx.txType) {
|
|
case "Received":
|
|
unawaited(
|
|
NotificationApi.showNotification(
|
|
title: "Incoming transaction",
|
|
body: walletName,
|
|
walletId: walletId,
|
|
iconAssetName: Assets.svg.iconFor(coin: coin),
|
|
date: DateTime.fromMillisecondsSinceEpoch(tx.timestamp * 1000),
|
|
shouldWatchForUpdates: tx.confirmations < MINIMUM_CONFIRMATIONS,
|
|
coinName: coin.name,
|
|
txid: tx.txid,
|
|
confirmations: tx.confirmations,
|
|
requiredConfirmations: MINIMUM_CONFIRMATIONS,
|
|
),
|
|
);
|
|
await txTracker.addNotifiedPending(tx.txid);
|
|
break;
|
|
case "Sent":
|
|
unawaited(
|
|
NotificationApi.showNotification(
|
|
title: "Outgoing transaction",
|
|
body: walletName,
|
|
walletId: walletId,
|
|
iconAssetName: Assets.svg.iconFor(coin: coin),
|
|
date: DateTime.fromMillisecondsSinceEpoch(tx.timestamp * 1000),
|
|
shouldWatchForUpdates: tx.confirmations < MINIMUM_CONFIRMATIONS,
|
|
coinName: coin.name,
|
|
txid: tx.txid,
|
|
confirmations: tx.confirmations,
|
|
requiredConfirmations: MINIMUM_CONFIRMATIONS,
|
|
),
|
|
);
|
|
await txTracker.addNotifiedPending(tx.txid);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (final tx in unconfirmedTxnsToNotifyConfirmed) {
|
|
if (tx.txType == "Received") {
|
|
unawaited(
|
|
NotificationApi.showNotification(
|
|
title: "Incoming transaction confirmed",
|
|
body: walletName,
|
|
walletId: walletId,
|
|
iconAssetName: Assets.svg.iconFor(coin: coin),
|
|
date: DateTime.fromMillisecondsSinceEpoch(tx.timestamp * 1000),
|
|
shouldWatchForUpdates: false,
|
|
coinName: coin.name,
|
|
),
|
|
);
|
|
await txTracker.addNotifiedConfirmed(tx.txid);
|
|
} else if (tx.txType == "Sent" && tx.subType == "join") {
|
|
unawaited(
|
|
NotificationApi.showNotification(
|
|
title: "Outgoing transaction confirmed",
|
|
body: walletName,
|
|
walletId: walletId,
|
|
iconAssetName: Assets.svg.iconFor(coin: coin),
|
|
date: DateTime.fromMillisecondsSinceEpoch(tx.timestamp * 1000),
|
|
shouldWatchForUpdates: false,
|
|
coinName: coin.name,
|
|
),
|
|
);
|
|
await txTracker.addNotifiedConfirmed(tx.txid);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Generates initial wallet values such as mnemonic, chain (receive/change) arrays and indexes.
|
|
Future<void> _generateNewWallet() async {
|
|
Logging.instance
|
|
.log("IS_INTEGRATION_TEST: $integrationTestFlag", level: LogLevel.Info);
|
|
if (!integrationTestFlag) {
|
|
final features = await electrumXClient.getServerFeatures();
|
|
Logging.instance.log("features: $features", level: LogLevel.Info);
|
|
switch (coin) {
|
|
case Coin.firo:
|
|
if (features['genesis_hash'] != GENESIS_HASH_MAINNET) {
|
|
throw Exception("genesis hash does not match main net!");
|
|
}
|
|
break;
|
|
case Coin.firoTestNet:
|
|
if (features['genesis_hash'] != GENESIS_HASH_TESTNET) {
|
|
throw Exception("genesis hash does not match test net!");
|
|
}
|
|
break;
|
|
default:
|
|
throw Exception(
|
|
"Attempted to generate a FiroWallet using a non firo coin type: ${coin.name}");
|
|
}
|
|
// if (_networkType == BasicNetworkType.main) {
|
|
// if (features['genesis_hash'] != GENESIS_HASH_MAINNET) {
|
|
// throw Exception("genesis hash does not match!");
|
|
// }
|
|
// } else if (_networkType == BasicNetworkType.test) {
|
|
// if (features['genesis_hash'] != GENESIS_HASH_TESTNET) {
|
|
// throw Exception("genesis hash does not match!");
|
|
// }
|
|
// }
|
|
}
|
|
|
|
// this should never fail as overwriting a mnemonic is big bad
|
|
assert((await _secureStore.read(key: '${_walletId}_mnemonic')) == null);
|
|
await _secureStore.write(
|
|
key: '${_walletId}_mnemonic',
|
|
value: bip39.generateMnemonic(strength: 256));
|
|
|
|
// Set relevant indexes
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: 'receivingIndex', value: 0);
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: 'changeIndex', value: 0);
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: 'mintIndex', value: 0);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'blocked_tx_hashes',
|
|
value: [
|
|
"0xdefault"
|
|
]); // A list of transaction hashes to represent frozen utxos in wallet
|
|
// initialize address book entries
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'addressBookEntries',
|
|
value: <String, String>{});
|
|
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: 'jindex', value: <dynamic>[]);
|
|
// Generate and add addresses to relevant arrays
|
|
final initialReceivingAddress = await _generateAddressForChain(0, 0);
|
|
final initialChangeAddress = await _generateAddressForChain(1, 0);
|
|
await addToAddressesArrayForChain(initialReceivingAddress, 0);
|
|
await addToAddressesArrayForChain(initialChangeAddress, 1);
|
|
_currentReceivingAddress = Future(() => initialReceivingAddress);
|
|
}
|
|
|
|
bool refreshMutex = false;
|
|
@override
|
|
bool get isRefreshing => refreshMutex;
|
|
|
|
/// Refreshes display data for the wallet
|
|
@override
|
|
Future<void> refresh() async {
|
|
if (refreshMutex) {
|
|
Logging.instance.log("$walletId $walletName refreshMutex denied",
|
|
level: LogLevel.Info);
|
|
return;
|
|
} else {
|
|
refreshMutex = true;
|
|
}
|
|
Logging.instance
|
|
.log("PROCESSORS ${Platform.numberOfProcessors}", level: LogLevel.Info);
|
|
try {
|
|
GlobalEventBus.instance.fire(
|
|
WalletSyncStatusChangedEvent(
|
|
WalletSyncStatus.syncing,
|
|
walletId,
|
|
coin,
|
|
),
|
|
);
|
|
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.0, walletId));
|
|
|
|
final receiveDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_receiveDerivations");
|
|
if (receiveDerivationsString == null ||
|
|
receiveDerivationsString == "{}") {
|
|
GlobalEventBus.instance
|
|
.fire(RefreshPercentChangedEvent(0.05, walletId));
|
|
final mnemonic = await _secureStore.read(key: '${_walletId}_mnemonic');
|
|
await fillAddresses(mnemonic!,
|
|
numberOfThreads: Platform.numberOfProcessors - isolates.length - 1);
|
|
}
|
|
|
|
await checkReceivingAddressForTransactions();
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.1, walletId));
|
|
|
|
final newUtxoData = _fetchUtxoData();
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.2, walletId));
|
|
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.25, walletId));
|
|
|
|
final newTxData = _fetchTransactionData();
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.35, walletId));
|
|
|
|
final feeObj = _getFees();
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.50, walletId));
|
|
|
|
_utxoData = Future(() => newUtxoData);
|
|
_transactionData = Future(() => newTxData);
|
|
_feeObject = Future(() => feeObj);
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.60, walletId));
|
|
|
|
final lelantusCoins = getLelantusCoinMap();
|
|
Logging.instance.log("_lelantus_coins at refresh: $lelantusCoins",
|
|
level: LogLevel.Warning, printFullLength: true);
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.70, walletId));
|
|
|
|
await _refreshLelantusData();
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.80, walletId));
|
|
|
|
// await autoMint();
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.90, walletId));
|
|
|
|
var balance = await _getFullBalance();
|
|
_balances = Future(() => balance);
|
|
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(0.95, walletId));
|
|
|
|
var txData = (await _txnData);
|
|
var lTxData = (await lelantusTransactionData);
|
|
await getAllTxsToWatch(txData, lTxData);
|
|
|
|
GlobalEventBus.instance.fire(RefreshPercentChangedEvent(1.0, walletId));
|
|
|
|
GlobalEventBus.instance.fire(
|
|
WalletSyncStatusChangedEvent(
|
|
WalletSyncStatus.synced,
|
|
walletId,
|
|
coin,
|
|
),
|
|
);
|
|
refreshMutex = false;
|
|
|
|
if (isActive || shouldAutoSync) {
|
|
timer ??= Timer.periodic(const Duration(seconds: 30), (timer) async {
|
|
bool shouldNotify = await refreshIfThereIsNewData();
|
|
if (shouldNotify) {
|
|
await refresh();
|
|
GlobalEventBus.instance.fire(UpdatedInBackgroundEvent(
|
|
"New data found in $walletId $walletName in background!",
|
|
walletId));
|
|
}
|
|
});
|
|
}
|
|
} catch (error, strace) {
|
|
refreshMutex = false;
|
|
GlobalEventBus.instance.fire(
|
|
NodeConnectionStatusChangedEvent(
|
|
NodeConnectionStatus.disconnected,
|
|
walletId,
|
|
coin,
|
|
),
|
|
);
|
|
GlobalEventBus.instance.fire(
|
|
WalletSyncStatusChangedEvent(
|
|
WalletSyncStatus.unableToSync,
|
|
walletId,
|
|
coin,
|
|
),
|
|
);
|
|
Logging.instance.log(
|
|
"Caught exception in refreshWalletData(): $error\n$strace",
|
|
level: LogLevel.Warning);
|
|
}
|
|
}
|
|
|
|
Future<int> _fetchMaxFee() async {
|
|
final balance = await availableBalance;
|
|
int spendAmount =
|
|
(balance * Decimal.fromInt(Constants.satsPerCoin)).toBigInt().toInt();
|
|
int fee = await estimateJoinSplitFee(spendAmount);
|
|
return fee;
|
|
}
|
|
|
|
Future<List<DartLelantusEntry>> _getLelantusEntry() async {
|
|
final mnemonic = await _secureStore.read(key: '${_walletId}_mnemonic');
|
|
final List<LelantusCoin> lelantusCoins = await _getUnspentCoins();
|
|
final root = await compute(
|
|
getBip32RootWrapper,
|
|
Tuple2(
|
|
mnemonic!,
|
|
_network,
|
|
),
|
|
);
|
|
final waitLelantusEntries = lelantusCoins.map((coin) async {
|
|
final keyPair = await compute(
|
|
getBip32NodeFromRootWrapper,
|
|
Tuple3(
|
|
MINT_INDEX,
|
|
coin.index,
|
|
root,
|
|
),
|
|
);
|
|
if (keyPair.privateKey == null) {
|
|
Logging.instance.log("error bad key", level: LogLevel.Error);
|
|
return DartLelantusEntry(1, 0, 0, 0, 0, '');
|
|
}
|
|
final String privateKey = Format.uint8listToString(keyPair.privateKey!);
|
|
return DartLelantusEntry(coin.isUsed ? 1 : 0, 0, coin.anonymitySetId,
|
|
coin.value, coin.index, privateKey);
|
|
}).toList();
|
|
|
|
final lelantusEntries = await Future.wait(waitLelantusEntries);
|
|
|
|
if (lelantusEntries.isNotEmpty) {
|
|
lelantusEntries.removeWhere((element) => element.amount == 0);
|
|
}
|
|
|
|
return lelantusEntries;
|
|
}
|
|
|
|
List<Map<dynamic, LelantusCoin>> getLelantusCoinMap() {
|
|
final _l = DB.instance
|
|
.get<dynamic>(boxName: walletId, key: '_lelantus_coins') as List?;
|
|
final List<Map<dynamic, LelantusCoin>> lelantusCoins = [];
|
|
for (var el in _l ?? []) {
|
|
lelantusCoins.add({el.keys.first: el.values.first as LelantusCoin});
|
|
}
|
|
return lelantusCoins;
|
|
}
|
|
|
|
Future<List<LelantusCoin>> _getUnspentCoins() async {
|
|
final List<Map<dynamic, LelantusCoin>> lelantusCoins = getLelantusCoinMap();
|
|
if (lelantusCoins.isNotEmpty) {
|
|
lelantusCoins.removeWhere((element) =>
|
|
element.values.any((elementCoin) => elementCoin.value == 0));
|
|
}
|
|
final jindexes =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'jindex') as List?;
|
|
final data = await _txnData;
|
|
final lelantusData = await lelantusTransactionData;
|
|
List<LelantusCoin> coins = [];
|
|
|
|
List<LelantusCoin> lelantusCoinsList =
|
|
lelantusCoins.fold(<LelantusCoin>[], (previousValue, element) {
|
|
previousValue.add(element.values.first);
|
|
return previousValue;
|
|
});
|
|
for (int i = 0; i < lelantusCoinsList.length; i++) {
|
|
// Logging.instance.log("lelantusCoinsList[$i]: ${lelantusCoinsList[i]}");
|
|
final txn = await cachedElectrumXClient.getTransaction(
|
|
txHash: lelantusCoinsList[i].txId,
|
|
verbose: true,
|
|
coin: coin,
|
|
);
|
|
final confirmations = txn["confirmations"];
|
|
bool isUnconfirmed = confirmations is int && confirmations < 1;
|
|
if (!jindexes!.contains(lelantusCoinsList[i].index) &&
|
|
data.findTransaction(lelantusCoinsList[i].txId) == null) {
|
|
isUnconfirmed = true;
|
|
}
|
|
if ((data.findTransaction(lelantusCoinsList[i].txId) != null &&
|
|
!data
|
|
.findTransaction(lelantusCoinsList[i].txId)!
|
|
.confirmedStatus) ||
|
|
(lelantusData.findTransaction(lelantusCoinsList[i].txId) != null &&
|
|
!lelantusData
|
|
.findTransaction(lelantusCoinsList[i].txId)!
|
|
.confirmedStatus)) {
|
|
continue;
|
|
}
|
|
if (!lelantusCoinsList[i].isUsed &&
|
|
lelantusCoinsList[i].anonymitySetId != ANONYMITY_SET_EMPTY_ID &&
|
|
!isUnconfirmed) {
|
|
coins.add(lelantusCoinsList[i]);
|
|
}
|
|
}
|
|
return coins;
|
|
}
|
|
|
|
// index 0 and 1 for the funds available to spend.
|
|
// index 2 and 3 for all the funds in the wallet (including the undependable ones)
|
|
Future<List<Decimal>> _getFullBalance() async {
|
|
try {
|
|
final List<Map<dynamic, LelantusCoin>> lelantusCoins =
|
|
getLelantusCoinMap();
|
|
if (lelantusCoins.isNotEmpty) {
|
|
lelantusCoins.removeWhere((element) =>
|
|
element.values.any((elementCoin) => elementCoin.value == 0));
|
|
}
|
|
final utxos = await utxoData;
|
|
final Decimal price = await firoPrice;
|
|
final data = await _txnData;
|
|
final lData = await lelantusTransactionData;
|
|
final jindexes =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'jindex') as List?;
|
|
int intLelantusBalance = 0;
|
|
int unconfirmedLelantusBalance = 0;
|
|
|
|
for (var element in lelantusCoins) {
|
|
element.forEach((key, value) {
|
|
final tx = data.findTransaction(value.txId);
|
|
models.Transaction? ltx;
|
|
ltx = lData.findTransaction(value.txId);
|
|
// Logging.instance.log("$value $tx $ltx");
|
|
if (!jindexes!.contains(value.index) && tx == null) {
|
|
// This coin is not confirmed and may be replaced
|
|
} else if (jindexes.contains(value.index) &&
|
|
tx == null &&
|
|
!value.isUsed &&
|
|
ltx != null &&
|
|
!ltx.confirmedStatus) {
|
|
unconfirmedLelantusBalance += value.value;
|
|
} else if (jindexes.contains(value.index) && !value.isUsed) {
|
|
intLelantusBalance += value.value;
|
|
} else if (!value.isUsed &&
|
|
(tx == null ? true : tx.confirmedStatus != false)) {
|
|
intLelantusBalance += value.value;
|
|
} else if (tx != null && tx.confirmedStatus == false) {
|
|
unconfirmedLelantusBalance += value.value;
|
|
}
|
|
});
|
|
}
|
|
|
|
final int utxosIntValue = utxos.satoshiBalance;
|
|
final Decimal utxosValue = Format.satoshisToAmount(utxosIntValue);
|
|
|
|
List<Decimal> balances = List.empty(growable: true);
|
|
|
|
Decimal lelantusBalance = Format.satoshisToAmount(intLelantusBalance);
|
|
|
|
balances.add(lelantusBalance);
|
|
|
|
balances.add(lelantusBalance * price);
|
|
|
|
Decimal _unconfirmedLelantusBalance =
|
|
Format.satoshisToAmount(unconfirmedLelantusBalance);
|
|
|
|
balances.add(lelantusBalance + utxosValue + _unconfirmedLelantusBalance);
|
|
|
|
balances.add(
|
|
(lelantusBalance + utxosValue + _unconfirmedLelantusBalance) * price);
|
|
|
|
int availableSats =
|
|
utxos.satoshiBalance - utxos.satoshiBalanceUnconfirmed;
|
|
if (availableSats < 0) {
|
|
availableSats = 0;
|
|
}
|
|
balances.add(Format.satoshisToAmount(availableSats));
|
|
|
|
Logging.instance.log("balances $balances", level: LogLevel.Info);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'totalBalance',
|
|
value: balances[2].toString());
|
|
return balances;
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown in getFullBalance(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<void> anonymizeAllPublicFunds() async {
|
|
try {
|
|
var mintResult = await _mintSelection();
|
|
if (mintResult.isEmpty) {
|
|
Logging.instance.log("nothing to mint", level: LogLevel.Info);
|
|
return;
|
|
}
|
|
await _submitLelantusToNetwork(mintResult);
|
|
unawaited(refresh());
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception caught in anonymizeAllPublicFunds(): $e\n$s",
|
|
level: LogLevel.Warning);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
/// Returns the mint transaction hex to mint all of the available funds.
|
|
Future<Map<String, dynamic>> _mintSelection() async {
|
|
final List<UtxoObject> availableOutputs = _outputsList;
|
|
final List<UtxoObject?> spendableOutputs = [];
|
|
|
|
// Build list of spendable outputs and totaling their satoshi amount
|
|
for (var i = 0; i < availableOutputs.length; i++) {
|
|
if (availableOutputs[i].blocked == false &&
|
|
availableOutputs[i].status.confirmed == true &&
|
|
!(availableOutputs[i].isCoinbase &&
|
|
availableOutputs[i].status.confirmations <= 101)) {
|
|
spendableOutputs.add(availableOutputs[i]);
|
|
}
|
|
}
|
|
|
|
final List<Map<dynamic, LelantusCoin>> lelantusCoins = getLelantusCoinMap();
|
|
if (lelantusCoins.isNotEmpty) {
|
|
lelantusCoins.removeWhere((element) =>
|
|
element.values.any((elementCoin) => elementCoin.value == 0));
|
|
}
|
|
final data = await _txnData;
|
|
final dataMap = data.getAllTransactions();
|
|
dataMap.forEach((key, value) {
|
|
if (value.inputs.isNotEmpty) {
|
|
for (var element in value.inputs) {
|
|
if (lelantusCoins
|
|
.any((element) => element.keys.contains(value.txid)) &&
|
|
spendableOutputs.firstWhere(
|
|
(output) => output?.txid == element.txid,
|
|
orElse: () => null) !=
|
|
null) {
|
|
spendableOutputs
|
|
.removeWhere((output) => output!.txid == element.txid);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
// If there is no Utxos to mint then stop the function.
|
|
if (spendableOutputs.isEmpty) {
|
|
Logging.instance.log("_mintSelection(): No spendable outputs found",
|
|
level: LogLevel.Info);
|
|
return {};
|
|
}
|
|
|
|
int satoshisBeingUsed = 0;
|
|
List<UtxoObject> utxoObjectsToUse = [];
|
|
|
|
for (var i = 0; i < spendableOutputs.length; i++) {
|
|
final spendable = spendableOutputs[i];
|
|
if (spendable != null) {
|
|
utxoObjectsToUse.add(spendable);
|
|
satoshisBeingUsed += spendable.value;
|
|
}
|
|
}
|
|
|
|
var mintsWithoutFee = await createMintsFromAmount(satoshisBeingUsed);
|
|
|
|
var tmpTx = await buildMintTransaction(
|
|
utxoObjectsToUse, satoshisBeingUsed, mintsWithoutFee);
|
|
|
|
int vsize = (tmpTx['transaction'] as Transaction).virtualSize();
|
|
final Decimal dvsize = Decimal.fromInt(vsize);
|
|
|
|
final feesObject = await fees;
|
|
|
|
final Decimal fastFee = Format.satoshisToAmount(feesObject.fast);
|
|
int firoFee =
|
|
(dvsize * fastFee * Decimal.fromInt(100000)).toDouble().ceil();
|
|
// int firoFee = (vsize * feesObject.fast * (1 / 1000.0) * 100000000).ceil();
|
|
|
|
if (firoFee < vsize) {
|
|
firoFee = vsize + 1;
|
|
}
|
|
firoFee = firoFee + 10;
|
|
int satoshiAmountToSend = satoshisBeingUsed - firoFee;
|
|
|
|
var mintsWithFee = await createMintsFromAmount(satoshiAmountToSend);
|
|
|
|
Map<String, dynamic> transaction = await buildMintTransaction(
|
|
utxoObjectsToUse, satoshiAmountToSend, mintsWithFee);
|
|
transaction['transaction'] = "";
|
|
Logging.instance.log(transaction.toString(), level: LogLevel.Info);
|
|
Logging.instance.log(transaction['txHex'], level: LogLevel.Info);
|
|
return transaction;
|
|
}
|
|
|
|
Future<List<Map<String, dynamic>>> createMintsFromAmount(int total) async {
|
|
var tmpTotal = total;
|
|
var index = 0;
|
|
var mints = <Map<String, dynamic>>[];
|
|
final nextFreeMintIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'mintIndex') as int;
|
|
while (tmpTotal > 0) {
|
|
final mintValue = min(tmpTotal, MINT_LIMIT);
|
|
final mint = await _getMintHex(
|
|
mintValue,
|
|
nextFreeMintIndex + index,
|
|
);
|
|
mints.add({
|
|
"value": mintValue,
|
|
"script": mint,
|
|
"index": nextFreeMintIndex + index,
|
|
"publicCoin": "",
|
|
});
|
|
tmpTotal = tmpTotal - MINT_LIMIT;
|
|
index++;
|
|
}
|
|
return mints;
|
|
}
|
|
|
|
/// returns a valid txid if successful
|
|
Future<String> submitHexToNetwork(String hex) async {
|
|
try {
|
|
final txid = await electrumXClient.broadcastTransaction(rawTx: hex);
|
|
return txid;
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Caught exception in submitHexToNetwork(\"$hex\"): $e $s",
|
|
printFullLength: true,
|
|
level: LogLevel.Info);
|
|
// return an invalid tx
|
|
return "transaction submission failed";
|
|
}
|
|
}
|
|
|
|
/// Builds and signs a transaction
|
|
Future<Map<String, dynamic>> buildMintTransaction(List<UtxoObject> utxosToUse,
|
|
int satoshisPerRecipient, List<Map<String, dynamic>> mintsMap) async {
|
|
debugPrint(utxosToUse.toString());
|
|
List<String> addressesToDerive = [];
|
|
|
|
// Populating the addresses to derive
|
|
for (var i = 0; i < utxosToUse.length; i++) {
|
|
final txid = utxosToUse[i].txid;
|
|
final outputIndex = utxosToUse[i].vout;
|
|
|
|
// txid may not work for this as txid may not always be the same as tx_hash?
|
|
final tx = await cachedElectrumXClient.getTransaction(
|
|
txHash: txid,
|
|
verbose: true,
|
|
coin: coin,
|
|
);
|
|
|
|
final vouts = tx["vout"] as List?;
|
|
if (vouts != null && outputIndex < vouts.length) {
|
|
final address =
|
|
vouts[outputIndex]["scriptPubKey"]["addresses"][0] as String?;
|
|
if (address != null) {
|
|
addressesToDerive.add(address);
|
|
}
|
|
}
|
|
}
|
|
|
|
List<ECPair> elipticCurvePairArray = [];
|
|
List<Uint8List> outputDataArray = [];
|
|
|
|
final receiveDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_receiveDerivations");
|
|
final changeDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_changeDerivations");
|
|
|
|
final receiveDerivations = Map<String, dynamic>.from(
|
|
jsonDecode(receiveDerivationsString ?? "{}") as Map);
|
|
final changeDerivations = Map<String, dynamic>.from(
|
|
jsonDecode(changeDerivationsString ?? "{}") as Map);
|
|
|
|
for (var i = 0; i < addressesToDerive.length; i++) {
|
|
final addressToCheckFor = addressesToDerive[i];
|
|
|
|
for (var i = 0; i < receiveDerivations.length; i++) {
|
|
final receive = receiveDerivations["$i"];
|
|
final change = changeDerivations["$i"];
|
|
|
|
if (receive['address'] == addressToCheckFor) {
|
|
Logging.instance
|
|
.log('Receiving found on loop $i', level: LogLevel.Info);
|
|
// Logging.instance.log(
|
|
// 'decoded receive[\'wif\'] version: ${wif.decode(receive['wif'] as String)}, _network: $_network');
|
|
elipticCurvePairArray
|
|
.add(ECPair.fromWIF(receive['wif'] as String, network: _network));
|
|
outputDataArray.add(P2PKH(
|
|
network: _network,
|
|
data: PaymentData(
|
|
pubkey: Format.stringToUint8List(
|
|
receive['publicKey'] as String)))
|
|
.data
|
|
.output!);
|
|
break;
|
|
}
|
|
if (change['address'] == addressToCheckFor) {
|
|
Logging.instance.log('Change found on loop $i', level: LogLevel.Info);
|
|
// Logging.instance.log(
|
|
// 'decoded change[\'wif\'] version: ${wif.decode(change['wif'] as String)}, _network: $_network');
|
|
elipticCurvePairArray
|
|
.add(ECPair.fromWIF(change['wif'] as String, network: _network));
|
|
|
|
outputDataArray.add(P2PKH(
|
|
network: _network,
|
|
data: PaymentData(
|
|
pubkey: Format.stringToUint8List(
|
|
change['publicKey'] as String)))
|
|
.data
|
|
.output!);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
final txb = TransactionBuilder(network: _network);
|
|
txb.setVersion(2);
|
|
|
|
int height = await getBlockHead(electrumXClient);
|
|
txb.setLockTime(height);
|
|
int amount = 0;
|
|
// Add transaction inputs
|
|
for (var i = 0; i < utxosToUse.length; i++) {
|
|
txb.addInput(
|
|
utxosToUse[i].txid, utxosToUse[i].vout, null, outputDataArray[i]);
|
|
amount += utxosToUse[i].value;
|
|
}
|
|
|
|
final index =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'mintIndex') as int;
|
|
Logging.instance.log("index of mint $index", level: LogLevel.Info);
|
|
|
|
for (var mintsElement in mintsMap) {
|
|
Logging.instance.log("using $mintsElement", level: LogLevel.Info);
|
|
Uint8List mintu8 =
|
|
Format.stringToUint8List(mintsElement['script'] as String);
|
|
txb.addOutput(mintu8, mintsElement['value'] as int);
|
|
}
|
|
|
|
for (var i = 0; i < utxosToUse.length; i++) {
|
|
txb.sign(
|
|
vin: i,
|
|
keyPair: elipticCurvePairArray[i],
|
|
witnessValue: utxosToUse[i].value,
|
|
);
|
|
}
|
|
var incomplete = txb.buildIncomplete();
|
|
var txId = incomplete.getId();
|
|
var txHex = incomplete.toHex();
|
|
int fee = amount - incomplete.outs[0].value!;
|
|
|
|
var price = await firoPrice;
|
|
var builtHex = txb.build();
|
|
// return builtHex;
|
|
final locale = await Devicelocale.currentLocale;
|
|
return {
|
|
"transaction": builtHex,
|
|
"txid": txId,
|
|
"txHex": txHex,
|
|
"value": amount - fee,
|
|
"fees": Format.satoshisToAmount(fee).toDouble(),
|
|
"publicCoin": "",
|
|
"height": height,
|
|
"txType": "Sent",
|
|
"confirmed_status": false,
|
|
"amount": Format.satoshisToAmount(amount).toDouble(),
|
|
"worthNow": Format.localizedStringAsFixed(
|
|
value: ((Decimal.fromInt(amount) * price) /
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toDecimal(scaleOnInfinitePrecision: 2),
|
|
decimalPlaces: 2,
|
|
locale: locale!),
|
|
"timestamp": DateTime.now().millisecondsSinceEpoch ~/ 1000,
|
|
"subType": "mint",
|
|
"mintsMap": mintsMap,
|
|
};
|
|
}
|
|
|
|
Future<models.TransactionData> _refreshLelantusData() async {
|
|
final List<Map<dynamic, LelantusCoin>> lelantusCoins = getLelantusCoinMap();
|
|
final jindexes =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'jindex') as List?;
|
|
|
|
// Get all joinsplit transaction ids
|
|
final lelantusTxData = await lelantusTransactionData;
|
|
|
|
final listLelantusTxData = lelantusTxData.getAllTransactions();
|
|
List<String> joinsplits = [];
|
|
for (final tx in listLelantusTxData.values) {
|
|
if (tx.subType == "join") {
|
|
joinsplits.add(tx.txid);
|
|
}
|
|
}
|
|
for (final coin
|
|
in lelantusCoins.fold(<LelantusCoin>[], (previousValue, element) {
|
|
(previousValue as List<LelantusCoin>).add(element.values.first);
|
|
return previousValue;
|
|
})) {
|
|
if (jindexes != null) {
|
|
if (jindexes.contains(coin.index) && !joinsplits.contains(coin.txId)) {
|
|
joinsplits.add(coin.txId);
|
|
}
|
|
}
|
|
}
|
|
|
|
final currentPrice = await firoPrice;
|
|
// Grab the most recent information on all the joinsplits
|
|
|
|
final locale = await Devicelocale.currentLocale;
|
|
final updatedJSplit = await getJMintTransactions(cachedElectrumXClient,
|
|
joinsplits, _prefs.currency, coin, currentPrice, locale!);
|
|
|
|
// update all of joinsplits that are now confirmed.
|
|
for (final tx in updatedJSplit) {
|
|
final currentTx = listLelantusTxData[tx.txid];
|
|
if (currentTx == null) {
|
|
// this send was accidentally not included in the list
|
|
listLelantusTxData[tx.txid] = tx;
|
|
continue;
|
|
}
|
|
if (currentTx.confirmedStatus != tx.confirmedStatus) {
|
|
listLelantusTxData[tx.txid] = tx;
|
|
}
|
|
}
|
|
|
|
final txData = await _txnData;
|
|
|
|
// Logging.instance.log(txData.txChunks);
|
|
final listTxData = txData.getAllTransactions();
|
|
listTxData.forEach((key, value) {
|
|
// ignore change addresses
|
|
// bool hasAtLeastOneReceive = false;
|
|
// int howManyReceiveInputs = 0;
|
|
// for (var element in value.inputs) {
|
|
// if (listLelantusTxData.containsKey(element.txid) &&
|
|
// listLelantusTxData[element.txid]!.txType == "Received"
|
|
// // &&
|
|
// // listLelantusTxData[element.txid].subType != "mint"
|
|
// ) {
|
|
// // hasAtLeastOneReceive = true;
|
|
// // howManyReceiveInputs++;
|
|
// }
|
|
// }
|
|
|
|
if (value.txType == "Received" && value.subType != "mint") {
|
|
// Every receive other than a mint should be shown. Mints will be collected and shown from the send side
|
|
listLelantusTxData[value.txid] = value;
|
|
} else if (value.txType == "Sent") {
|
|
// all sends should be shown, mints will be displayed correctly in the ui
|
|
listLelantusTxData[value.txid] = value;
|
|
}
|
|
});
|
|
|
|
// update the _lelantusTransactionData
|
|
final models.TransactionData newTxData =
|
|
models.TransactionData.fromMap(listLelantusTxData);
|
|
// Logging.instance.log(newTxData.txChunks);
|
|
_lelantusTransactionData = Future(() => newTxData);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'latest_lelantus_tx_model', value: newTxData);
|
|
return newTxData;
|
|
}
|
|
|
|
Future<String> _getMintHex(int amount, int index) async {
|
|
final mnemonic = await _secureStore.read(key: '${_walletId}_mnemonic');
|
|
final mintKeyPair = await compute(
|
|
getBip32NodeWrapper,
|
|
Tuple4(
|
|
MINT_INDEX,
|
|
index,
|
|
mnemonic!,
|
|
_network,
|
|
),
|
|
);
|
|
String keydata = Format.uint8listToString(mintKeyPair.privateKey!);
|
|
String seedID = Format.uint8listToString(mintKeyPair.identifier);
|
|
|
|
String mintHex = await compute(
|
|
_getMintScriptWrapper,
|
|
Tuple5(
|
|
amount,
|
|
keydata,
|
|
index,
|
|
seedID,
|
|
coin == Coin.firoTestNet,
|
|
),
|
|
);
|
|
return mintHex;
|
|
}
|
|
|
|
Future<bool> _submitLelantusToNetwork(
|
|
Map<String, dynamic> transactionInfo) async {
|
|
final latestSetId = await getLatestSetId();
|
|
final txid = await submitHexToNetwork(transactionInfo['txHex'] as String);
|
|
// success if txid matches the generated txid
|
|
Logging.instance.log(
|
|
"_submitLelantusToNetwork txid: ${transactionInfo['txid']}",
|
|
level: LogLevel.Info);
|
|
if (txid == transactionInfo['txid']) {
|
|
final index =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'mintIndex') as int?;
|
|
final List<Map<dynamic, LelantusCoin>> lelantusCoins =
|
|
getLelantusCoinMap();
|
|
List<Map<dynamic, LelantusCoin>> coins;
|
|
if (lelantusCoins.isEmpty) {
|
|
coins = [];
|
|
} else {
|
|
coins = [...lelantusCoins];
|
|
}
|
|
|
|
if (transactionInfo['spendCoinIndexes'] != null) {
|
|
// This is a joinsplit
|
|
|
|
// Update all of the coins that have been spent.
|
|
for (final lCoinMap in coins) {
|
|
final lCoin = lCoinMap.values.first;
|
|
if ((transactionInfo['spendCoinIndexes'] as List<int>)
|
|
.contains(lCoin.index)) {
|
|
lCoinMap[lCoinMap.keys.first] = LelantusCoin(
|
|
lCoin.index,
|
|
lCoin.value,
|
|
lCoin.publicCoin,
|
|
lCoin.txId,
|
|
lCoin.anonymitySetId,
|
|
true);
|
|
}
|
|
}
|
|
|
|
// if a jmint was made add it to the unspent coin index
|
|
LelantusCoin jmint = LelantusCoin(
|
|
index!,
|
|
transactionInfo['jmintValue'] as int? ?? 0,
|
|
transactionInfo['publicCoin'] as String,
|
|
transactionInfo['txid'] as String,
|
|
latestSetId,
|
|
false);
|
|
if (jmint.value > 0) {
|
|
coins.add({jmint.txId: jmint});
|
|
final jindexes = DB.instance
|
|
.get<dynamic>(boxName: walletId, key: 'jindex') as List?;
|
|
jindexes!.add(index);
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: 'jindex', value: jindexes);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'mintIndex', value: index + 1);
|
|
}
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: '_lelantus_coins', value: coins);
|
|
|
|
// add the send transaction
|
|
models.TransactionData data = await lelantusTransactionData;
|
|
Map<String, models.Transaction> transactions =
|
|
data.getAllTransactions();
|
|
transactions[transactionInfo['txid'] as String] =
|
|
models.Transaction.fromLelantusJson(transactionInfo);
|
|
final models.TransactionData newTxData =
|
|
models.TransactionData.fromMap(transactions);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'latest_lelantus_tx_model',
|
|
value: newTxData);
|
|
final ldata = DB.instance.get<dynamic>(
|
|
boxName: walletId,
|
|
key: 'latest_lelantus_tx_model') as models.TransactionData;
|
|
_lelantusTransactionData = Future(() => ldata);
|
|
} else {
|
|
// This is a mint
|
|
Logging.instance.log("this is a mint", level: LogLevel.Info);
|
|
|
|
// TODO: transactionInfo['mintsMap']
|
|
for (final mintMap
|
|
in transactionInfo['mintsMap'] as List<Map<String, dynamic>>) {
|
|
final index = mintMap['index'] as int?;
|
|
LelantusCoin mint = LelantusCoin(
|
|
index!,
|
|
mintMap['value'] as int,
|
|
mintMap['publicCoin'] as String,
|
|
transactionInfo['txid'] as String,
|
|
latestSetId,
|
|
false,
|
|
);
|
|
if (mint.value > 0) {
|
|
coins.add({mint.txId: mint});
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'mintIndex', value: index + 1);
|
|
}
|
|
}
|
|
// Logging.instance.log(coins);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: '_lelantus_coins', value: coins);
|
|
}
|
|
return true;
|
|
} else {
|
|
// Failed to send to network
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Future<FeeObject> _getFees() async {
|
|
try {
|
|
//TODO adjust numbers for different speeds?
|
|
const int f = 1, m = 5, s = 20;
|
|
|
|
final fast = await electrumXClient.estimateFee(blocks: f);
|
|
final medium = await electrumXClient.estimateFee(blocks: m);
|
|
final slow = await electrumXClient.estimateFee(blocks: s);
|
|
|
|
final feeObject = FeeObject(
|
|
numberOfBlocksFast: f,
|
|
numberOfBlocksAverage: m,
|
|
numberOfBlocksSlow: s,
|
|
fast: Format.decimalAmountToSatoshis(fast),
|
|
medium: Format.decimalAmountToSatoshis(medium),
|
|
slow: Format.decimalAmountToSatoshis(slow),
|
|
);
|
|
|
|
Logging.instance.log("fetched fees: $feeObject", level: LogLevel.Info);
|
|
return feeObject;
|
|
|
|
// final result = await electrumXClient.getFeeRate();
|
|
//
|
|
// final locale = await Devicelocale.currentLocale;
|
|
// final String fee =
|
|
// Format.satoshiAmountToPrettyString(result["rate"] as int, locale!);
|
|
//
|
|
// final fees = {
|
|
// "fast": fee,
|
|
// "average": fee,
|
|
// "slow": fee,
|
|
// };
|
|
// final FeeObject feeObject = FeeObject.fromJson(fees);
|
|
// return feeObject;
|
|
} catch (e) {
|
|
Logging.instance
|
|
.log("Exception rethrown from _getFees(): $e", level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<ElectrumXNode> _getCurrentNode() async {
|
|
final node = NodeService().getPrimaryNodeFor(coin: coin) ??
|
|
DefaultNodes.getNodeFor(coin);
|
|
|
|
return ElectrumXNode(
|
|
address: node.host,
|
|
port: node.port,
|
|
name: node.name,
|
|
useSSL: node.useSSL,
|
|
id: node.id,
|
|
);
|
|
}
|
|
|
|
//TODO call get transaction and check each tx to see if it is a "received" tx?
|
|
Future<int> _getReceivedTxCount({required String address}) async {
|
|
try {
|
|
final scripthash = AddressUtils.convertToScriptHash(address, _network);
|
|
final transactions =
|
|
await electrumXClient.getHistory(scripthash: scripthash);
|
|
return transactions.length;
|
|
} catch (e) {
|
|
Logging.instance.log(
|
|
"Exception rethrown in _getReceivedTxCount(address: $address): $e",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<void> checkReceivingAddressForTransactions() async {
|
|
try {
|
|
final String currentExternalAddr = await _getCurrentAddressForChain(0);
|
|
final int numtxs =
|
|
await _getReceivedTxCount(address: currentExternalAddr);
|
|
Logging.instance.log(
|
|
'Number of txs for current receiving: $currentExternalAddr: $numtxs',
|
|
level: LogLevel.Info);
|
|
|
|
if (numtxs >= 1) {
|
|
await incrementAddressIndexForChain(
|
|
0); // First increment the receiving index
|
|
final newReceivingIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'receivingIndex')
|
|
as int; // Check the new receiving index
|
|
final newReceivingAddress = await _generateAddressForChain(0,
|
|
newReceivingIndex); // Use new index to derive a new receiving address
|
|
await addToAddressesArrayForChain(newReceivingAddress,
|
|
0); // Add that new receiving address to the array of receiving addresses
|
|
_currentReceivingAddress = Future(() =>
|
|
newReceivingAddress); // Set the new receiving address that the service
|
|
}
|
|
} on SocketException catch (se, s) {
|
|
Logging.instance.log(
|
|
"SocketException caught in checkReceivingAddressForTransactions(): $se\n$s",
|
|
level: LogLevel.Error);
|
|
return;
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception rethrown from checkReceivingAddressForTransactions(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<void> checkChangeAddressForTransactions() async {
|
|
try {
|
|
final String currentExternalAddr = await _getCurrentAddressForChain(1);
|
|
final int numtxs =
|
|
await _getReceivedTxCount(address: currentExternalAddr);
|
|
Logging.instance.log(
|
|
'Number of txs for current change address: $currentExternalAddr: $numtxs',
|
|
level: LogLevel.Info);
|
|
|
|
if (numtxs >= 1) {
|
|
await incrementAddressIndexForChain(
|
|
0); // First increment the change index
|
|
final newReceivingIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'changeIndex')
|
|
as int; // Check the new change index
|
|
final newReceivingAddress = await _generateAddressForChain(0,
|
|
newReceivingIndex); // Use new index to derive a new change address
|
|
await addToAddressesArrayForChain(newReceivingAddress,
|
|
0); // Add that new receiving address to the array of change addresses
|
|
}
|
|
} on SocketException catch (se, s) {
|
|
Logging.instance.log(
|
|
"SocketException caught in checkChangeAddressForTransactions(): $se\n$s",
|
|
level: LogLevel.Error);
|
|
return;
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception rethrown from checkChangeAddressForTransactions(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<List<String>> _fetchAllOwnAddresses() async {
|
|
final List<String> allAddresses = [];
|
|
final receivingAddresses =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'receivingAddresses')
|
|
as List<dynamic>;
|
|
final changeAddresses =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'changeAddresses')
|
|
as List<dynamic>;
|
|
|
|
for (var i = 0; i < receivingAddresses.length; i++) {
|
|
allAddresses.add(receivingAddresses[i] as String);
|
|
}
|
|
for (var i = 0; i < changeAddresses.length; i++) {
|
|
allAddresses.add(changeAddresses[i] as String);
|
|
}
|
|
return allAddresses;
|
|
}
|
|
|
|
Future<List<Map<String, dynamic>>> _fetchHistory(
|
|
List<String> allAddresses) async {
|
|
try {
|
|
List<Map<String, dynamic>> allTxHashes = [];
|
|
|
|
final Map<int, Map<String, List<dynamic>>> batches = {};
|
|
final Map<String, String> requestIdToAddressMap = {};
|
|
const batchSizeMax = 100;
|
|
int batchNumber = 0;
|
|
for (int i = 0; i < allAddresses.length; i++) {
|
|
if (batches[batchNumber] == null) {
|
|
batches[batchNumber] = {};
|
|
}
|
|
final scripthash =
|
|
AddressUtils.convertToScriptHash(allAddresses[i], _network);
|
|
final id = Logger.isTestEnv ? "$i" : const Uuid().v1();
|
|
requestIdToAddressMap[id] = allAddresses[i];
|
|
batches[batchNumber]!.addAll({
|
|
id: [scripthash]
|
|
});
|
|
if (i % batchSizeMax == batchSizeMax - 1) {
|
|
batchNumber++;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < batches.length; i++) {
|
|
final response =
|
|
await _electrumXClient.getBatchHistory(args: batches[i]!);
|
|
for (final entry in response.entries) {
|
|
for (int j = 0; j < entry.value.length; j++) {
|
|
entry.value[j]["address"] = requestIdToAddressMap[entry.key];
|
|
if (!allTxHashes.contains(entry.value[j])) {
|
|
allTxHashes.add(entry.value[j]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return allTxHashes;
|
|
} catch (e, s) {
|
|
Logging.instance.log("_fetchHistory: $e\n$s", level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<models.TransactionData> _fetchTransactionData() async {
|
|
final changeAddresses =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'changeAddresses')
|
|
as List<dynamic>;
|
|
final List<String> allAddresses = await _fetchAllOwnAddresses();
|
|
// Logging.instance.log("receiving addresses: $receivingAddresses");
|
|
// Logging.instance.log("change addresses: $changeAddresses");
|
|
|
|
List<Map<String, dynamic>> allTxHashes = await _fetchHistory(allAddresses);
|
|
|
|
final cachedTransactions =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'latest_tx_model')
|
|
as models.TransactionData?;
|
|
int latestTxnBlockHeight =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: "storedTxnDataHeight")
|
|
as int? ??
|
|
0;
|
|
|
|
final unconfirmedCachedTransactions =
|
|
cachedTransactions?.getAllTransactions() ?? {};
|
|
unconfirmedCachedTransactions
|
|
.removeWhere((key, value) => value.confirmedStatus);
|
|
|
|
if (cachedTransactions != null) {
|
|
for (final tx in allTxHashes.toList(growable: false)) {
|
|
final txHeight = tx["height"] as int;
|
|
if (txHeight > 0 &&
|
|
txHeight < latestTxnBlockHeight - MINIMUM_CONFIRMATIONS) {
|
|
if (unconfirmedCachedTransactions[tx["tx_hash"] as String] == null) {
|
|
allTxHashes.remove(tx);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
List<String> hashes = [];
|
|
for (var element in allTxHashes) {
|
|
hashes.add(element['tx_hash'] as String);
|
|
}
|
|
List<Map<String, dynamic>> allTransactions = await fastFetch(hashes);
|
|
|
|
Logging.instance.log("allTransactions length: ${allTransactions.length}",
|
|
level: LogLevel.Info);
|
|
|
|
// sort thing stuff
|
|
final currentPrice = await firoPrice;
|
|
final List<Map<String, dynamic>> midSortedArray = [];
|
|
|
|
final locale = await Devicelocale.currentLocale;
|
|
|
|
Logging.instance.log("refresh the txs", level: LogLevel.Info);
|
|
for (final txObject in allTransactions) {
|
|
// Logging.instance.log(txObject);
|
|
List<String> sendersArray = [];
|
|
List<String> recipientsArray = [];
|
|
|
|
// Usually only has value when txType = 'Send'
|
|
int inputAmtSentFromWallet = 0;
|
|
// Usually has value regardless of txType due to change addresses
|
|
int outputAmtAddressedToWallet = 0;
|
|
|
|
Map<String, dynamic> midSortedTx = {};
|
|
List<dynamic> aliens = [];
|
|
|
|
for (final input in txObject["vin"] as List) {
|
|
final address = input["address"] as String?;
|
|
if (address != null) {
|
|
sendersArray.add(address);
|
|
}
|
|
}
|
|
|
|
// Logging.instance.log("sendersArray: $sendersArray");
|
|
|
|
for (final output in txObject["vout"] as List) {
|
|
final addresses = output["scriptPubKey"]["addresses"] as List?;
|
|
if (addresses != null && addresses.isNotEmpty) {
|
|
recipientsArray.add(addresses[0] as String);
|
|
}
|
|
}
|
|
// Logging.instance.log("recipientsArray: $recipientsArray");
|
|
|
|
final foundInSenders =
|
|
allAddresses.any((element) => sendersArray.contains(element));
|
|
// Logging.instance.log("foundInSenders: $foundInSenders");
|
|
|
|
String outAddress = "";
|
|
|
|
int fees = 0;
|
|
|
|
// If txType = Sent, then calculate inputAmtSentFromWallet, calculate who received how much in aliens array (check outputs)
|
|
if (foundInSenders) {
|
|
int outAmount = 0;
|
|
int inAmount = 0;
|
|
bool nFeesUsed = false;
|
|
|
|
for (final input in txObject["vin"] as List) {
|
|
final nFees = input["nFees"];
|
|
if (nFees != null) {
|
|
nFeesUsed = true;
|
|
fees = (Decimal.parse(nFees.toString()) *
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toBigInt()
|
|
.toInt();
|
|
}
|
|
final address = input["address"];
|
|
final value = input["valueSat"];
|
|
if (address != null && value != null) {
|
|
if (allAddresses.contains(address)) {
|
|
inputAmtSentFromWallet += value as int;
|
|
}
|
|
}
|
|
|
|
if (value != null) {
|
|
inAmount += value as int;
|
|
}
|
|
}
|
|
|
|
for (final output in txObject["vout"] as List) {
|
|
final addresses = output["scriptPubKey"]["addresses"] as List?;
|
|
final value = output["value"];
|
|
if (addresses != null && addresses.isNotEmpty) {
|
|
final address = addresses[0] as String;
|
|
if (value != null) {
|
|
if (changeAddresses.contains(address)) {
|
|
inputAmtSentFromWallet -= (Decimal.parse(value.toString()) *
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toBigInt()
|
|
.toInt();
|
|
} else {
|
|
outAddress = address;
|
|
}
|
|
}
|
|
}
|
|
if (value != null) {
|
|
outAmount += (Decimal.parse(value.toString()) *
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toBigInt()
|
|
.toInt();
|
|
}
|
|
}
|
|
|
|
fees = nFeesUsed ? fees : inAmount - outAmount;
|
|
inputAmtSentFromWallet -= inAmount - outAmount;
|
|
} else {
|
|
for (final input in txObject["vin"] as List) {
|
|
final nFees = input["nFees"];
|
|
if (nFees != null) {
|
|
fees += (Decimal.parse(nFees.toString()) *
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toBigInt()
|
|
.toInt();
|
|
}
|
|
}
|
|
|
|
for (final output in txObject["vout"] as List) {
|
|
final addresses = output["scriptPubKey"]["addresses"] as List?;
|
|
if (addresses != null && addresses.isNotEmpty) {
|
|
final address = addresses[0] as String;
|
|
final value = output["value"];
|
|
// Logging.instance.log(address + value.toString());
|
|
|
|
if (allAddresses.contains(address)) {
|
|
outputAmtAddressedToWallet += (Decimal.parse(value.toString()) *
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toBigInt()
|
|
.toInt();
|
|
outAddress = address;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
final int confirms = txObject["confirmations"] as int? ?? 0;
|
|
|
|
// create final tx map
|
|
midSortedTx["txid"] = txObject["txid"];
|
|
midSortedTx["confirmed_status"] = confirms >= MINIMUM_CONFIRMATIONS;
|
|
midSortedTx["confirmations"] = confirms;
|
|
midSortedTx["timestamp"] = txObject["blocktime"] ??
|
|
(DateTime.now().millisecondsSinceEpoch ~/ 1000);
|
|
if (foundInSenders) {
|
|
midSortedTx["txType"] = "Sent";
|
|
midSortedTx["amount"] = inputAmtSentFromWallet;
|
|
final String worthNow = Format.localizedStringAsFixed(
|
|
value: ((currentPrice * Decimal.fromInt(inputAmtSentFromWallet)) /
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toDecimal(scaleOnInfinitePrecision: 2),
|
|
decimalPlaces: 2,
|
|
locale: locale!);
|
|
midSortedTx["worthNow"] = worthNow;
|
|
midSortedTx["worthAtBlockTimestamp"] = worthNow;
|
|
if (txObject["vout"][0]["scriptPubKey"]["type"] == "lelantusmint") {
|
|
midSortedTx["subType"] = "mint";
|
|
}
|
|
} else {
|
|
midSortedTx["txType"] = "Received";
|
|
midSortedTx["amount"] = outputAmtAddressedToWallet;
|
|
final worthNow = Format.localizedStringAsFixed(
|
|
value:
|
|
((currentPrice * Decimal.fromInt(outputAmtAddressedToWallet)) /
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toDecimal(scaleOnInfinitePrecision: 2),
|
|
decimalPlaces: 2,
|
|
locale: locale!);
|
|
midSortedTx["worthNow"] = worthNow;
|
|
midSortedTx["worthAtBlockTimestamp"] = worthNow;
|
|
}
|
|
midSortedTx["aliens"] = aliens;
|
|
midSortedTx["fees"] = fees;
|
|
midSortedTx["address"] = outAddress;
|
|
midSortedTx["inputSize"] = txObject["vin"].length;
|
|
midSortedTx["outputSize"] = txObject["vout"].length;
|
|
midSortedTx["inputs"] = txObject["vin"];
|
|
midSortedTx["outputs"] = txObject["vout"];
|
|
|
|
final int height = txObject["height"] as int? ?? 0;
|
|
midSortedTx["height"] = height;
|
|
|
|
if (height >= latestTxnBlockHeight) {
|
|
latestTxnBlockHeight = height;
|
|
}
|
|
|
|
midSortedArray.add(midSortedTx);
|
|
}
|
|
|
|
// sort by date ---- //TODO not sure if needed
|
|
// shouldn't be any issues with a null timestamp but I got one at some point?
|
|
midSortedArray.sort((a, b) {
|
|
final aT = a["timestamp"];
|
|
final bT = b["timestamp"];
|
|
|
|
if (aT == null && bT == null) {
|
|
return 0;
|
|
} else if (aT == null) {
|
|
return -1;
|
|
} else if (bT == null) {
|
|
return 1;
|
|
} else {
|
|
return (bT as int) - (aT as int);
|
|
}
|
|
});
|
|
|
|
// buildDateTimeChunks
|
|
final Map<String, dynamic> result = {"dateTimeChunks": <dynamic>[]};
|
|
final dateArray = <dynamic>[];
|
|
|
|
for (int i = 0; i < midSortedArray.length; i++) {
|
|
final txObject = midSortedArray[i];
|
|
final date =
|
|
models.extractDateFromTimestamp(txObject["timestamp"] as int);
|
|
final txTimeArray = [txObject["timestamp"], date];
|
|
|
|
if (dateArray.contains(txTimeArray[1])) {
|
|
result["dateTimeChunks"].forEach((dynamic chunk) {
|
|
if (models.extractDateFromTimestamp(chunk["timestamp"] as int) ==
|
|
txTimeArray[1]) {
|
|
if (chunk["transactions"] == null) {
|
|
chunk["transactions"] = <Map<String, dynamic>>[];
|
|
}
|
|
chunk["transactions"].add(txObject);
|
|
}
|
|
});
|
|
} else {
|
|
dateArray.add(txTimeArray[1]);
|
|
final chunk = {
|
|
"timestamp": txTimeArray[0],
|
|
"transactions": [txObject],
|
|
};
|
|
result["dateTimeChunks"].add(chunk);
|
|
}
|
|
}
|
|
|
|
final transactionsMap = cachedTransactions?.getAllTransactions() ?? {};
|
|
transactionsMap
|
|
.addAll(models.TransactionData.fromJson(result).getAllTransactions());
|
|
|
|
final txModel = models.TransactionData.fromMap(transactionsMap);
|
|
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'storedTxnDataHeight',
|
|
value: latestTxnBlockHeight);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'latest_tx_model', value: txModel);
|
|
|
|
return txModel;
|
|
}
|
|
|
|
Future<UtxoData> _fetchUtxoData() async {
|
|
final List<String> allAddresses = await _fetchAllOwnAddresses();
|
|
|
|
try {
|
|
final fetchedUtxoList = <List<Map<String, dynamic>>>[];
|
|
|
|
final Map<int, Map<String, List<dynamic>>> batches = {};
|
|
const batchSizeMax = 100;
|
|
int batchNumber = 0;
|
|
for (int i = 0; i < allAddresses.length; i++) {
|
|
if (batches[batchNumber] == null) {
|
|
batches[batchNumber] = {};
|
|
}
|
|
final scripthash =
|
|
AddressUtils.convertToScriptHash(allAddresses[i], _network);
|
|
batches[batchNumber]!.addAll({
|
|
scripthash: [scripthash]
|
|
});
|
|
if (i % batchSizeMax == batchSizeMax - 1) {
|
|
batchNumber++;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < batches.length; i++) {
|
|
final response =
|
|
await _electrumXClient.getBatchUTXOs(args: batches[i]!);
|
|
for (final entry in response.entries) {
|
|
if (entry.value.isNotEmpty) {
|
|
fetchedUtxoList.add(entry.value);
|
|
}
|
|
}
|
|
}
|
|
final priceData =
|
|
await _priceAPI.getPricesAnd24hChange(baseCurrency: _prefs.currency);
|
|
Decimal currentPrice = priceData[coin]?.item1 ?? Decimal.zero;
|
|
final List<Map<String, dynamic>> outputArray = [];
|
|
int satoshiBalance = 0;
|
|
int satoshiBalancePending = 0;
|
|
|
|
for (int i = 0; i < fetchedUtxoList.length; i++) {
|
|
for (int j = 0; j < fetchedUtxoList[i].length; j++) {
|
|
int value = fetchedUtxoList[i][j]["value"] as int;
|
|
satoshiBalance += value;
|
|
|
|
final txn = await cachedElectrumXClient.getTransaction(
|
|
txHash: fetchedUtxoList[i][j]["tx_hash"] as String,
|
|
verbose: true,
|
|
coin: coin,
|
|
);
|
|
|
|
final Map<String, dynamic> utxo = {};
|
|
final int confirmations = txn["confirmations"] as int? ?? 0;
|
|
final bool confirmed = confirmations >= MINIMUM_CONFIRMATIONS;
|
|
if (!confirmed) {
|
|
satoshiBalancePending += value;
|
|
}
|
|
|
|
utxo["txid"] = txn["txid"];
|
|
utxo["vout"] = fetchedUtxoList[i][j]["tx_pos"];
|
|
utxo["value"] = value;
|
|
|
|
utxo["status"] = <String, dynamic>{};
|
|
utxo["status"]["confirmed"] = confirmed;
|
|
utxo["status"]["confirmations"] = confirmations;
|
|
utxo["status"]["confirmed"] =
|
|
txn["confirmations"] == null ? false : txn["confirmations"] > 0;
|
|
|
|
utxo["status"]["block_height"] = fetchedUtxoList[i][j]["height"];
|
|
utxo["status"]["block_hash"] = txn["blockhash"];
|
|
utxo["status"]["block_time"] = txn["blocktime"];
|
|
|
|
final fiatValue = ((Decimal.fromInt(value) * currentPrice) /
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toDecimal(scaleOnInfinitePrecision: 2);
|
|
utxo["rawWorth"] = fiatValue;
|
|
utxo["fiatWorth"] = fiatValue.toString();
|
|
utxo["is_coinbase"] = txn['vin'][0]['coinbase'] != null;
|
|
outputArray.add(utxo);
|
|
}
|
|
}
|
|
|
|
Decimal currencyBalanceRaw =
|
|
((Decimal.fromInt(satoshiBalance) * currentPrice) /
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toDecimal(scaleOnInfinitePrecision: 2);
|
|
|
|
final Map<String, dynamic> result = {
|
|
"total_user_currency": currencyBalanceRaw.toString(),
|
|
"total_sats": satoshiBalance,
|
|
"total_btc": (Decimal.fromInt(satoshiBalance) /
|
|
Decimal.fromInt(Constants.satsPerCoin))
|
|
.toDecimal(scaleOnInfinitePrecision: Constants.decimalPlaces)
|
|
.toString(),
|
|
"outputArray": outputArray,
|
|
"unconfirmed": satoshiBalancePending,
|
|
};
|
|
|
|
final dataModel = UtxoData.fromJson(result);
|
|
|
|
final List<UtxoObject> allOutputs = dataModel.unspentOutputArray;
|
|
Logging.instance
|
|
.log('Outputs fetched: $allOutputs', level: LogLevel.Info);
|
|
await _sortOutputs(allOutputs);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'latest_utxo_model', value: dataModel);
|
|
// await DB.instance.put<dynamic>(
|
|
// boxName: walletId,
|
|
// key: 'totalBalance',
|
|
// value: dataModel.satoshiBalance);
|
|
return dataModel;
|
|
} catch (e, s) {
|
|
Logging.instance
|
|
.log("Output fetch unsuccessful: $e\n$s", level: LogLevel.Error);
|
|
final latestTxModel =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'latest_utxo_model')
|
|
as models.UtxoData?;
|
|
|
|
if (latestTxModel == null) {
|
|
final emptyModel = {
|
|
"total_user_currency": "0.00",
|
|
"total_sats": 0,
|
|
"total_btc": "0",
|
|
"outputArray": <dynamic>[]
|
|
};
|
|
return UtxoData.fromJson(emptyModel);
|
|
} else {
|
|
Logging.instance
|
|
.log("Old output model located", level: LogLevel.Warning);
|
|
return latestTxModel;
|
|
}
|
|
}
|
|
}
|
|
|
|
Future<models.TransactionData> _getLelantusTransactionData() async {
|
|
final latestModel = DB.instance.get<dynamic>(
|
|
boxName: walletId,
|
|
key: 'latest_lelantus_tx_model') as models.TransactionData?;
|
|
|
|
if (latestModel == null) {
|
|
final emptyModel = {"dateTimeChunks": <dynamic>[]};
|
|
return models.TransactionData.fromJson(emptyModel);
|
|
} else {
|
|
Logging.instance
|
|
.log("Old transaction model located", level: LogLevel.Warning);
|
|
return latestModel;
|
|
}
|
|
}
|
|
|
|
/// Returns the latest receiving/change (external/internal) address for the wallet depending on [chain]
|
|
/// [chain] - Use 0 for receiving (external), 1 for change (internal). Should not be any other value!
|
|
Future<String> _getCurrentAddressForChain(int chain) async {
|
|
if (chain == 0) {
|
|
final externalChainArray = (DB.instance.get<dynamic>(
|
|
boxName: walletId, key: 'receivingAddresses')) as List<dynamic>;
|
|
return externalChainArray.last as String;
|
|
} else {
|
|
// Here, we assume that chain == 1
|
|
final internalChainArray =
|
|
(DB.instance.get<dynamic>(boxName: walletId, key: 'changeAddresses'))
|
|
as List<dynamic>;
|
|
return internalChainArray.last as String;
|
|
}
|
|
}
|
|
|
|
Future<void> fillAddresses(String suppliedMnemonic,
|
|
{int perBatch = 50, int numberOfThreads = 4}) async {
|
|
if (numberOfThreads <= 0) {
|
|
numberOfThreads = 1;
|
|
}
|
|
if (Platform.environment["FLUTTER_TEST"] == "true" || integrationTestFlag) {
|
|
perBatch = 10;
|
|
numberOfThreads = 4;
|
|
}
|
|
|
|
final receiveDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_receiveDerivations");
|
|
final changeDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_changeDerivations");
|
|
|
|
var receiveDerivations = Map<String, dynamic>.from(
|
|
jsonDecode(receiveDerivationsString ?? "{}") as Map);
|
|
var changeDerivations = Map<String, dynamic>.from(
|
|
jsonDecode(changeDerivationsString ?? "{}") as Map);
|
|
|
|
final int start = receiveDerivations.length;
|
|
|
|
List<ReceivePort> ports = List.empty(growable: true);
|
|
for (int i = 0; i < numberOfThreads; i++) {
|
|
ReceivePort receivePort = await getIsolate({
|
|
"function": "isolateDerive",
|
|
"mnemonic": suppliedMnemonic,
|
|
"from": start + i * perBatch,
|
|
"to": start + (i + 1) * perBatch,
|
|
"network": _network,
|
|
});
|
|
ports.add(receivePort);
|
|
}
|
|
for (int i = 0; i < numberOfThreads; i++) {
|
|
ReceivePort receivePort = ports.elementAt(i);
|
|
var message = await receivePort.first;
|
|
if (message is String) {
|
|
Logging.instance.log("this is a string", level: LogLevel.Error);
|
|
stop(receivePort);
|
|
throw Exception("isolateDerive isolate failed");
|
|
}
|
|
stop(receivePort);
|
|
Logging.instance.log('Closing isolateDerive!', level: LogLevel.Info);
|
|
receiveDerivations.addAll(message['receive'] as Map<String, dynamic>);
|
|
changeDerivations.addAll(message['change'] as Map<String, dynamic>);
|
|
}
|
|
Logging.instance.log("isolate derives", level: LogLevel.Info);
|
|
// Logging.instance.log(receiveDerivations);
|
|
// Logging.instance.log(changeDerivations);
|
|
|
|
final newReceiveDerivationsString = jsonEncode(receiveDerivations);
|
|
final newChangeDerivationsString = jsonEncode(changeDerivations);
|
|
|
|
await _secureStore.write(
|
|
key: "${walletId}_receiveDerivations",
|
|
value: newReceiveDerivationsString);
|
|
await _secureStore.write(
|
|
key: "${walletId}_changeDerivations",
|
|
value: newChangeDerivationsString);
|
|
}
|
|
|
|
/// Generates a new internal or external chain address for the wallet using a BIP84 derivation path.
|
|
/// [chain] - Use 0 for receiving (external), 1 for change (internal). Should not be any other value!
|
|
/// [index] - This can be any integer >= 0
|
|
Future<String> _generateAddressForChain(int chain, int index) async {
|
|
// final wallet = await Hive.openBox(this._walletId);
|
|
final mnemonic = await _secureStore.read(key: '${_walletId}_mnemonic');
|
|
Map<String, dynamic>? derivations;
|
|
if (chain == 0) {
|
|
final receiveDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_receiveDerivations");
|
|
derivations = Map<String, dynamic>.from(
|
|
jsonDecode(receiveDerivationsString ?? "{}") as Map);
|
|
} else if (chain == 1) {
|
|
final changeDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_changeDerivations");
|
|
derivations = Map<String, dynamic>.from(
|
|
jsonDecode(changeDerivationsString ?? "{}") as Map);
|
|
}
|
|
|
|
if (derivations!.isNotEmpty) {
|
|
if (derivations["$index"] == null) {
|
|
await fillAddresses(mnemonic!,
|
|
numberOfThreads: Platform.numberOfProcessors - isolates.length - 1);
|
|
Logging.instance.log("calling _generateAddressForChain recursively",
|
|
level: LogLevel.Info);
|
|
return _generateAddressForChain(chain, index);
|
|
}
|
|
return derivations["$index"]['address'] as String;
|
|
} else {
|
|
final node = await compute(
|
|
getBip32NodeWrapper, Tuple4(chain, index, mnemonic!, _network));
|
|
return P2PKH(network: _network, data: PaymentData(pubkey: node.publicKey))
|
|
.data
|
|
.address!;
|
|
}
|
|
}
|
|
|
|
/// Increases the index for either the internal or external chain, depending on [chain].
|
|
/// [chain] - Use 0 for receiving (external), 1 for change (internal). Should not be any other value!
|
|
Future<void> incrementAddressIndexForChain(int chain) async {
|
|
if (chain == 0) {
|
|
final newIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'receivingIndex') +
|
|
1;
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'receivingIndex', value: newIndex);
|
|
} else {
|
|
// Here we assume chain == 1 since it can only be either 0 or 1
|
|
final newIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'changeIndex') + 1;
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: 'changeIndex', value: newIndex);
|
|
}
|
|
}
|
|
|
|
/// Adds [address] to the relevant chain's address array, which is determined by [chain].
|
|
/// [address] - Expects a standard native segwit address
|
|
/// [chain] - Use 0 for receiving (external), 1 for change (internal). Should not be any other value!
|
|
Future<void> addToAddressesArrayForChain(String address, int chain) async {
|
|
String chainArray = '';
|
|
if (chain == 0) {
|
|
chainArray = 'receivingAddresses';
|
|
} else {
|
|
chainArray = 'changeAddresses';
|
|
}
|
|
|
|
final addressArray =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: chainArray);
|
|
if (addressArray == null) {
|
|
Logging.instance.log(
|
|
'Attempting to add the following to array for chain $chain:${[
|
|
address
|
|
]}',
|
|
level: LogLevel.Info);
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: chainArray, value: [address]);
|
|
} else {
|
|
// Make a deep copy of the existing list
|
|
final List<String> newArray = [];
|
|
addressArray
|
|
.forEach((dynamic _address) => newArray.add(_address as String));
|
|
newArray.add(address); // Add the address passed into the method
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: chainArray, value: newArray);
|
|
}
|
|
}
|
|
|
|
/// Takes in a list of UtxoObjects and adds a name (dependent on object index within list)
|
|
/// and checks for the txid associated with the utxo being blocked and marks it accordingly.
|
|
/// Now also checks for output labeling.
|
|
Future<void> _sortOutputs(List<UtxoObject> utxos) async {
|
|
final blockedHashArray =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'blocked_tx_hashes')
|
|
as List<dynamic>?;
|
|
final List<String> lst = [];
|
|
if (blockedHashArray != null) {
|
|
for (var hash in blockedHashArray) {
|
|
lst.add(hash as String);
|
|
}
|
|
}
|
|
final labels =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'labels') as Map? ??
|
|
{};
|
|
|
|
_outputsList = [];
|
|
|
|
for (var i = 0; i < utxos.length; i++) {
|
|
if (labels[utxos[i].txid] != null) {
|
|
utxos[i].txName = labels[utxos[i].txid] as String? ?? "";
|
|
} else {
|
|
utxos[i].txName = 'Output #$i';
|
|
}
|
|
|
|
if (utxos[i].status.confirmed == false) {
|
|
_outputsList.add(utxos[i]);
|
|
} else {
|
|
if (lst.contains(utxos[i].txid)) {
|
|
utxos[i].blocked = true;
|
|
_outputsList.add(utxos[i]);
|
|
} else if (!lst.contains(utxos[i].txid)) {
|
|
_outputsList.add(utxos[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> fullRescan(
|
|
int maxUnusedAddressGap,
|
|
int maxNumberOfIndexesToCheck,
|
|
) async {
|
|
Logging.instance.log("Starting full rescan!", level: LogLevel.Info);
|
|
// timer?.cancel();
|
|
// for (final isolate in isolates.values) {
|
|
// isolate.kill(priority: Isolate.immediate);
|
|
// }
|
|
// isolates.clear();
|
|
longMutex = true;
|
|
GlobalEventBus.instance.fire(
|
|
WalletSyncStatusChangedEvent(
|
|
WalletSyncStatus.syncing,
|
|
walletId,
|
|
coin,
|
|
),
|
|
);
|
|
|
|
// clear cache
|
|
await _cachedElectrumXClient.clearSharedTransactionCache(coin: coin);
|
|
|
|
// back up data
|
|
await _rescanBackup();
|
|
|
|
try {
|
|
final mnemonic = await _secureStore.read(key: '${_walletId}_mnemonic');
|
|
await _recoverWalletFromBIP32SeedPhrase(mnemonic!, maxUnusedAddressGap);
|
|
|
|
longMutex = false;
|
|
Logging.instance.log("Full rescan complete!", level: LogLevel.Info);
|
|
GlobalEventBus.instance.fire(
|
|
WalletSyncStatusChangedEvent(
|
|
WalletSyncStatus.synced,
|
|
walletId,
|
|
coin,
|
|
),
|
|
);
|
|
} catch (e, s) {
|
|
GlobalEventBus.instance.fire(
|
|
WalletSyncStatusChangedEvent(
|
|
WalletSyncStatus.unableToSync,
|
|
walletId,
|
|
coin,
|
|
),
|
|
);
|
|
|
|
// restore from backup
|
|
await _rescanRestore();
|
|
|
|
longMutex = false;
|
|
Logging.instance.log("Exception rethrown from fullRescan(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<void> _rescanBackup() async {
|
|
Logging.instance.log("starting rescan backup", level: LogLevel.Info);
|
|
|
|
// backup current and clear data
|
|
final tempReceivingAddresses =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'receivingAddresses');
|
|
await DB.instance.delete<dynamic>(
|
|
key: 'receivingAddresses',
|
|
boxName: walletId,
|
|
);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'receivingAddresses_BACKUP',
|
|
value: tempReceivingAddresses);
|
|
|
|
final tempChangeAddresses =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'changeAddresses');
|
|
await DB.instance.delete<dynamic>(
|
|
key: 'changeAddresses',
|
|
boxName: walletId,
|
|
);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'changeAddresses_BACKUP',
|
|
value: tempChangeAddresses);
|
|
|
|
final tempReceivingIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'receivingIndex');
|
|
await DB.instance.delete<dynamic>(
|
|
key: 'receivingIndex',
|
|
boxName: walletId,
|
|
);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'receivingIndex_BACKUP',
|
|
value: tempReceivingIndex);
|
|
|
|
final tempChangeIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'changeIndex');
|
|
await DB.instance.delete<dynamic>(
|
|
key: 'changeIndex',
|
|
boxName: walletId,
|
|
);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'changeIndex_BACKUP', value: tempChangeIndex);
|
|
|
|
final receiveDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_receiveDerivations");
|
|
final changeDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_changeDerivations");
|
|
|
|
await _secureStore.write(
|
|
key: "${walletId}_receiveDerivations_BACKUP",
|
|
value: receiveDerivationsString);
|
|
await _secureStore.write(
|
|
key: "${walletId}_changeDerivations_BACKUP",
|
|
value: changeDerivationsString);
|
|
|
|
await _secureStore.write(
|
|
key: "${walletId}_receiveDerivations", value: null);
|
|
await _secureStore.write(key: "${walletId}_changeDerivations", value: null);
|
|
|
|
// back up but no need to delete
|
|
final tempMintIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'mintIndex');
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'mintIndex_BACKUP', value: tempMintIndex);
|
|
|
|
final tempLelantusCoins =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: '_lelantus_coins');
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: '_lelantus_coins_BACKUP',
|
|
value: tempLelantusCoins);
|
|
|
|
final tempJIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'jindex');
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'jindex_BACKUP', value: tempJIndex);
|
|
|
|
final tempLelantusTxModel = DB.instance
|
|
.get<dynamic>(boxName: walletId, key: 'latest_lelantus_tx_model');
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'latest_lelantus_tx_model_BACKUP',
|
|
value: tempLelantusTxModel);
|
|
|
|
Logging.instance.log("rescan backup complete", level: LogLevel.Info);
|
|
}
|
|
|
|
Future<void> _rescanRestore() async {
|
|
Logging.instance.log("starting rescan restore", level: LogLevel.Info);
|
|
|
|
// restore from backup
|
|
final tempReceivingAddresses = DB.instance
|
|
.get<dynamic>(boxName: walletId, key: 'receivingAddresses_BACKUP');
|
|
final tempChangeAddresses = DB.instance
|
|
.get<dynamic>(boxName: walletId, key: 'changeAddresses_BACKUP');
|
|
final tempReceivingIndex = DB.instance
|
|
.get<dynamic>(boxName: walletId, key: 'receivingIndex_BACKUP');
|
|
final tempChangeIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'changeIndex_BACKUP');
|
|
final tempMintIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'mintIndex_BACKUP');
|
|
final tempLelantusCoins = DB.instance
|
|
.get<dynamic>(boxName: walletId, key: '_lelantus_coins_BACKUP');
|
|
final tempJIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'jindex_BACKUP');
|
|
final tempLelantusTxModel = DB.instance.get<dynamic>(
|
|
boxName: walletId, key: 'latest_lelantus_tx_model_BACKUP');
|
|
|
|
final receiveDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_receiveDerivations_BACKUP");
|
|
final changeDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_changeDerivations_BACKUP");
|
|
|
|
await _secureStore.write(
|
|
key: "${walletId}_receiveDerivations", value: receiveDerivationsString);
|
|
await _secureStore.write(
|
|
key: "${walletId}_changeDerivations", value: changeDerivationsString);
|
|
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'receivingAddresses',
|
|
value: tempReceivingAddresses);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'changeAddresses', value: tempChangeAddresses);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'receivingIndex', value: tempReceivingIndex);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'changeIndex', value: tempChangeIndex);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'mintIndex', value: tempMintIndex);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: '_lelantus_coins', value: tempLelantusCoins);
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: 'jindex', value: tempJIndex);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'latest_lelantus_tx_model',
|
|
value: tempLelantusTxModel);
|
|
|
|
Logging.instance.log("rescan restore complete", level: LogLevel.Info);
|
|
}
|
|
|
|
/// wrapper for _recoverWalletFromBIP32SeedPhrase()
|
|
@override
|
|
Future<void> recoverFromMnemonic({
|
|
required String mnemonic,
|
|
required int maxUnusedAddressGap,
|
|
required int maxNumberOfIndexesToCheck,
|
|
required int height,
|
|
}) async {
|
|
try {
|
|
await compute(
|
|
_setTestnetWrapper,
|
|
coin == Coin.firoTestNet,
|
|
);
|
|
Logging.instance.log("IS_INTEGRATION_TEST: $integrationTestFlag",
|
|
level: LogLevel.Info);
|
|
if (!integrationTestFlag) {
|
|
final features = await electrumXClient.getServerFeatures();
|
|
Logging.instance.log("features: $features", level: LogLevel.Info);
|
|
switch (coin) {
|
|
case Coin.firo:
|
|
if (features['genesis_hash'] != GENESIS_HASH_MAINNET) {
|
|
throw Exception("genesis hash does not match main net!");
|
|
}
|
|
break;
|
|
case Coin.firoTestNet:
|
|
if (features['genesis_hash'] != GENESIS_HASH_TESTNET) {
|
|
throw Exception("genesis hash does not match test net!");
|
|
}
|
|
break;
|
|
default:
|
|
throw Exception(
|
|
"Attempted to generate a FiroWallet using a non firo coin type: ${coin.name}");
|
|
}
|
|
// if (_networkType == BasicNetworkType.main) {
|
|
// if (features['genesis_hash'] != GENESIS_HASH_MAINNET) {
|
|
// throw Exception("genesis hash does not match main net!");
|
|
// }
|
|
// } else if (_networkType == BasicNetworkType.test) {
|
|
// if (features['genesis_hash'] != GENESIS_HASH_TESTNET) {
|
|
// throw Exception("genesis hash does not match test net!");
|
|
// }
|
|
// }
|
|
}
|
|
// this should never fail
|
|
if ((await _secureStore.read(key: '${_walletId}_mnemonic')) != null) {
|
|
throw Exception("Attempted to overwrite mnemonic on restore!");
|
|
}
|
|
await _secureStore.write(
|
|
key: '${_walletId}_mnemonic', value: mnemonic.trim());
|
|
await _recoverWalletFromBIP32SeedPhrase(
|
|
mnemonic.trim(), maxUnusedAddressGap);
|
|
|
|
await compute(
|
|
_setTestnetWrapper,
|
|
false,
|
|
);
|
|
} catch (e, s) {
|
|
await compute(
|
|
_setTestnetWrapper,
|
|
false,
|
|
);
|
|
Logging.instance.log(
|
|
"Exception rethrown from recoverFromMnemonic(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
bool longMutex = false;
|
|
|
|
Future<Map<int, dynamic>> getSetDataMap(int latestSetId) async {
|
|
final Map<int, dynamic> setDataMap = {};
|
|
final anonymitySets = await fetchAnonymitySets();
|
|
for (int setId = 1; setId <= latestSetId; setId++) {
|
|
final setData = anonymitySets
|
|
.firstWhere((element) => element["setId"] == setId, orElse: () => {});
|
|
|
|
if (setData.isNotEmpty) {
|
|
setDataMap[setId] = setData;
|
|
}
|
|
}
|
|
return setDataMap;
|
|
}
|
|
|
|
Future<void> _makeDerivations(
|
|
String suppliedMnemonic, int maxUnusedAddressGap) async {
|
|
List<String> receivingAddressArray = [];
|
|
List<String> changeAddressArray = [];
|
|
|
|
int receivingIndex = -1;
|
|
int changeIndex = -1;
|
|
|
|
// The gap limit will be capped at 20
|
|
int receivingGapCounter = 0;
|
|
int changeGapCounter = 0;
|
|
|
|
await fillAddresses(suppliedMnemonic,
|
|
numberOfThreads: Platform.numberOfProcessors - isolates.length - 1);
|
|
|
|
final receiveDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_receiveDerivations");
|
|
final changeDerivationsString =
|
|
await _secureStore.read(key: "${walletId}_changeDerivations");
|
|
|
|
final receiveDerivations = Map<String, dynamic>.from(
|
|
jsonDecode(receiveDerivationsString ?? "{}") as Map);
|
|
final changeDerivations = Map<String, dynamic>.from(
|
|
jsonDecode(changeDerivationsString ?? "{}") as Map);
|
|
|
|
// log("rcv: $receiveDerivations");
|
|
// log("chg: $changeDerivations");
|
|
|
|
// Deriving and checking for receiving addresses
|
|
for (var i = 0; i < receiveDerivations.length; i++) {
|
|
// Break out of loop when receivingGapCounter hits maxUnusedAddressGap
|
|
// Same gap limit for change as for receiving, breaks when it hits maxUnusedAddressGap
|
|
if (receivingGapCounter >= maxUnusedAddressGap &&
|
|
changeGapCounter >= maxUnusedAddressGap) {
|
|
break;
|
|
}
|
|
|
|
final receiveDerivation = receiveDerivations["$i"];
|
|
final address = receiveDerivation['address'] as String;
|
|
|
|
final changeDerivation = changeDerivations["$i"];
|
|
final _address = changeDerivation['address'] as String;
|
|
Future<int>? futureNumTxs;
|
|
Future<int>? _futureNumTxs;
|
|
if (receivingGapCounter < maxUnusedAddressGap) {
|
|
futureNumTxs = _getReceivedTxCount(address: address);
|
|
}
|
|
if (changeGapCounter < maxUnusedAddressGap) {
|
|
_futureNumTxs = _getReceivedTxCount(address: _address);
|
|
}
|
|
try {
|
|
if (futureNumTxs != null) {
|
|
int numTxs = await futureNumTxs;
|
|
if (numTxs >= 1) {
|
|
receivingIndex = i;
|
|
receivingAddressArray.add(address);
|
|
} else if (numTxs == 0) {
|
|
receivingGapCounter += 1;
|
|
}
|
|
}
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception rethrown from recoverWalletFromBIP32SeedPhrase(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
|
|
try {
|
|
if (_futureNumTxs != null) {
|
|
int numTxs = await _futureNumTxs;
|
|
if (numTxs >= 1) {
|
|
changeIndex = i;
|
|
changeAddressArray.add(_address);
|
|
} else if (numTxs == 0) {
|
|
changeGapCounter += 1;
|
|
}
|
|
}
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception rethrown from recoverWalletFromBIP32SeedPhrase(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
// If restoring a wallet that never received any funds, then set receivingArray manually
|
|
// If we didn't do this, it'd store an empty array
|
|
if (receivingIndex == -1) {
|
|
final String receivingAddress = await _generateAddressForChain(0, 0);
|
|
receivingAddressArray.add(receivingAddress);
|
|
}
|
|
|
|
// If restoring a wallet that never sent any funds with change, then set changeArray
|
|
// manually. If we didn't do this, it'd store an empty array.
|
|
if (changeIndex == -1) {
|
|
final String changeAddress = await _generateAddressForChain(1, 0);
|
|
changeAddressArray.add(changeAddress);
|
|
}
|
|
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'receivingAddresses',
|
|
value: receivingAddressArray);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'changeAddresses', value: changeAddressArray);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'receivingIndex',
|
|
value: receivingIndex == -1 ? 0 : receivingIndex);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: 'changeIndex',
|
|
value: changeIndex == -1 ? 0 : changeIndex);
|
|
}
|
|
|
|
/// Recovers wallet from [suppliedMnemonic]. Expects a valid mnemonic.
|
|
Future<void> _recoverWalletFromBIP32SeedPhrase(
|
|
String suppliedMnemonic, int maxUnusedAddressGap) async {
|
|
longMutex = true;
|
|
Logging.instance
|
|
.log("PROCESSORS ${Platform.numberOfProcessors}", level: LogLevel.Info);
|
|
try {
|
|
final latestSetId = await getLatestSetId();
|
|
final setDataMap = getSetDataMap(latestSetId);
|
|
final usedSerialNumbers = getUsedCoinSerials();
|
|
final makeDerivations =
|
|
_makeDerivations(suppliedMnemonic, maxUnusedAddressGap);
|
|
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: "id", value: _walletId);
|
|
await DB.instance
|
|
.put<dynamic>(boxName: walletId, key: "isFavorite", value: false);
|
|
|
|
await Future.wait([usedSerialNumbers, setDataMap, makeDerivations]);
|
|
|
|
await _restore(latestSetId, await setDataMap, await usedSerialNumbers);
|
|
longMutex = false;
|
|
} catch (e, s) {
|
|
longMutex = false;
|
|
Logging.instance.log(
|
|
"Exception rethrown from recoverWalletFromBIP32SeedPhrase(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<void> _restore(int latestSetId, Map<dynamic, dynamic> setDataMap,
|
|
dynamic usedSerialNumbers) async {
|
|
final mnemonic = await _secureStore.read(key: '${_walletId}_mnemonic');
|
|
final dataFuture = _txnData;
|
|
final String currency = _prefs.currency;
|
|
final Decimal currentPrice = await firoPrice;
|
|
|
|
ReceivePort receivePort = await getIsolate({
|
|
"function": "restore",
|
|
"mnemonic": mnemonic,
|
|
"coin": coin,
|
|
"latestSetId": latestSetId,
|
|
"setDataMap": setDataMap,
|
|
"usedSerialNumbers": usedSerialNumbers,
|
|
"network": _network,
|
|
});
|
|
|
|
await Future.wait([dataFuture]);
|
|
var result = await receivePort.first;
|
|
if (result is String) {
|
|
Logging.instance
|
|
.log("restore() ->> this is a string", level: LogLevel.Error);
|
|
stop(receivePort);
|
|
throw Exception("isolate restore failed.");
|
|
}
|
|
stop(receivePort);
|
|
|
|
final message = await staticProcessRestore(
|
|
(await dataFuture), result as Map<dynamic, dynamic>);
|
|
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'mintIndex', value: message['mintIndex']);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId,
|
|
key: '_lelantus_coins',
|
|
value: message['_lelantus_coins']);
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'jindex', value: message['jindex']);
|
|
|
|
final transactionMap =
|
|
message["newTxMap"] as Map<String, models.Transaction>;
|
|
|
|
// Create the joinsplit transactions.
|
|
final spendTxs = await getJMintTransactions(
|
|
_cachedElectrumXClient,
|
|
message["spendTxIds"] as List<String>,
|
|
currency,
|
|
coin,
|
|
currentPrice,
|
|
(await Devicelocale.currentLocale)!);
|
|
Logging.instance.log(spendTxs, level: LogLevel.Info);
|
|
for (var element in spendTxs) {
|
|
transactionMap[element.txid] = element;
|
|
}
|
|
|
|
final models.TransactionData newTxData =
|
|
models.TransactionData.fromMap(transactionMap);
|
|
|
|
_lelantusTransactionData = Future(() => newTxData);
|
|
|
|
await DB.instance.put<dynamic>(
|
|
boxName: walletId, key: 'latest_lelantus_tx_model', value: newTxData);
|
|
}
|
|
|
|
Future<List<Map<String, dynamic>>> fetchAnonymitySets() async {
|
|
try {
|
|
final latestSetId = await getLatestSetId();
|
|
|
|
final List<Map<String, dynamic>> sets = [];
|
|
List<Future<Map<String, dynamic>>> anonFutures = [];
|
|
for (int i = 1; i <= latestSetId; i++) {
|
|
final set = cachedElectrumXClient.getAnonymitySet(
|
|
groupId: "$i",
|
|
coin: coin,
|
|
);
|
|
anonFutures.add(set);
|
|
}
|
|
await Future.wait(anonFutures);
|
|
for (int i = 1; i <= latestSetId; i++) {
|
|
Map<String, dynamic> set = (await anonFutures[i - 1]);
|
|
set["setId"] = i;
|
|
sets.add(set);
|
|
}
|
|
return sets;
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception rethrown from refreshAnonymitySets: $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<dynamic> _createJoinSplitTransaction(
|
|
int spendAmount, String address, bool subtractFeeFromAmount) async {
|
|
final price = await firoPrice;
|
|
final mnemonic = await _secureStore.read(key: '${_walletId}_mnemonic');
|
|
final index = DB.instance.get<dynamic>(boxName: walletId, key: 'mintIndex');
|
|
final lelantusEntry = await _getLelantusEntry();
|
|
final anonymitySets = await fetchAnonymitySets();
|
|
final locktime = await getBlockHead(electrumXClient);
|
|
final locale = await Devicelocale.currentLocale;
|
|
|
|
ReceivePort receivePort = await getIsolate({
|
|
"function": "createJoinSplit",
|
|
"spendAmount": spendAmount,
|
|
"address": address,
|
|
"subtractFeeFromAmount": subtractFeeFromAmount,
|
|
"mnemonic": mnemonic,
|
|
"index": index,
|
|
"price": price,
|
|
"lelantusEntries": lelantusEntry,
|
|
"locktime": locktime,
|
|
"coin": coin,
|
|
"network": _network,
|
|
"_anonymity_sets": anonymitySets,
|
|
"locale": locale,
|
|
});
|
|
var message = await receivePort.first;
|
|
if (message is String) {
|
|
Logging.instance
|
|
.log("Error in CreateJoinSplit: $message", level: LogLevel.Error);
|
|
stop(receivePort);
|
|
return 3;
|
|
}
|
|
if (message is int) {
|
|
stop(receivePort);
|
|
return message;
|
|
}
|
|
stop(receivePort);
|
|
Logging.instance.log('Closing createJoinSplit!', level: LogLevel.Info);
|
|
return message;
|
|
}
|
|
|
|
Future<int> getLatestSetId() async {
|
|
try {
|
|
final id = await electrumXClient.getLatestCoinId();
|
|
return id;
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown in firo_wallet.dart: $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
Future<List<dynamic>> getUsedCoinSerials() async {
|
|
try {
|
|
final response = await cachedElectrumXClient.getUsedCoinSerials(
|
|
coin: coin,
|
|
);
|
|
return response;
|
|
} catch (e, s) {
|
|
Logging.instance.log("Exception rethrown in firo_wallet.dart: $e\n$s",
|
|
level: LogLevel.Error);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<void> exit() async {
|
|
_hasCalledExit = true;
|
|
timer?.cancel();
|
|
timer = null;
|
|
stopNetworkAlivePinging();
|
|
for (final isolate in isolates.values) {
|
|
isolate.kill(priority: Isolate.immediate);
|
|
}
|
|
isolates.clear();
|
|
Logging.instance
|
|
.log("$walletName firo_wallet exit finished", level: LogLevel.Info);
|
|
}
|
|
|
|
bool _hasCalledExit = false;
|
|
|
|
@override
|
|
bool get hasCalledExit => _hasCalledExit;
|
|
|
|
bool isActive = false;
|
|
|
|
@override
|
|
void Function(bool)? get onIsActiveWalletChanged => (isActive) async {
|
|
timer?.cancel();
|
|
timer = null;
|
|
if (isActive) {
|
|
await compute(
|
|
_setTestnetWrapper,
|
|
coin == Coin.firoTestNet,
|
|
);
|
|
} else {
|
|
await compute(
|
|
_setTestnetWrapper,
|
|
false,
|
|
);
|
|
}
|
|
this.isActive = isActive;
|
|
};
|
|
|
|
Future<dynamic> getCoinsToJoinSplit(
|
|
int required,
|
|
) async {
|
|
List<DartLelantusEntry> coins = await _getLelantusEntry();
|
|
if (required > LELANTUS_VALUE_SPEND_LIMIT_PER_TRANSACTION) {
|
|
return false;
|
|
}
|
|
|
|
int availableBalance = coins.fold(
|
|
0, (previousValue, element) => previousValue + element.amount);
|
|
|
|
if (required > availableBalance) {
|
|
return false;
|
|
}
|
|
|
|
// sort by biggest amount. if it is same amount we will prefer the older block
|
|
coins.sort((a, b) =>
|
|
(a.amount != b.amount ? a.amount > b.amount : a.height < b.height)
|
|
? -1
|
|
: 1);
|
|
int spendVal = 0;
|
|
|
|
List<DartLelantusEntry> coinsToSpend = [];
|
|
|
|
while (spendVal < required) {
|
|
if (coins.isEmpty) {
|
|
break;
|
|
}
|
|
|
|
DartLelantusEntry? chosen;
|
|
int need = required - spendVal;
|
|
|
|
var itr = coins.first;
|
|
if (need >= itr.amount) {
|
|
chosen = itr;
|
|
coins.remove(itr);
|
|
} else {
|
|
for (int index = coins.length - 1; index != 0; index--) {
|
|
var coinIt = coins[index];
|
|
var nextItr = coins[index - 1];
|
|
|
|
if (coinIt.amount >= need &&
|
|
(index - 1 == 0 || nextItr.amount != coinIt.amount)) {
|
|
chosen = coinIt;
|
|
coins.remove(chosen);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
spendVal += chosen!.amount;
|
|
coinsToSpend.insert(coinsToSpend.length, chosen);
|
|
}
|
|
|
|
// sort by group id ay ascending order. it is mandatory for creating proper joinsplit
|
|
coinsToSpend.sort((a, b) => a.anonymitySetId < b.anonymitySetId ? 1 : -1);
|
|
|
|
int changeToMint = spendVal - required;
|
|
List<int> indices = [];
|
|
for (var l in coinsToSpend) {
|
|
indices.add(l.index);
|
|
}
|
|
List<DartLelantusEntry> coinsToBeSpentOut = [];
|
|
coinsToBeSpentOut.addAll(coinsToSpend);
|
|
|
|
return {"changeToMint": changeToMint, "coinsToSpend": coinsToBeSpentOut};
|
|
}
|
|
|
|
Future<int> estimateJoinSplitFee(
|
|
int spendAmount,
|
|
) async {
|
|
int fee;
|
|
int size;
|
|
|
|
for (fee = 0;;) {
|
|
int currentRequired = spendAmount;
|
|
|
|
var map = await getCoinsToJoinSplit(currentRequired);
|
|
if (map is bool && !map) {
|
|
return 0;
|
|
}
|
|
|
|
List<DartLelantusEntry> coinsToBeSpent =
|
|
map['coinsToSpend'] as List<DartLelantusEntry>;
|
|
|
|
// 1054 is constant part, mainly Schnorr and Range proofs, 2560 is for each sigma/aux data
|
|
// 179 other parts of tx, assuming 1 utxo and 1 jmint
|
|
size = 1054 + 2560 * coinsToBeSpent.length + 180;
|
|
// uint64_t feeNeeded = GetMinimumFee(size, DEFAULT_TX_CONFIRM_TARGET);
|
|
int feeNeeded =
|
|
size; //TODO(Levon) temporary, use real estimation methods here
|
|
|
|
if (fee >= feeNeeded) {
|
|
break;
|
|
}
|
|
|
|
fee = feeNeeded;
|
|
}
|
|
|
|
return fee;
|
|
}
|
|
|
|
@override
|
|
Future<int> estimateFeeFor(int satoshiAmount, int feeRate) async {
|
|
int fee = await estimateJoinSplitFee(satoshiAmount);
|
|
return fee;
|
|
}
|
|
|
|
Future<int> estimateFeeForPublic(int satoshiAmount, int feeRate) async {
|
|
final available =
|
|
Format.decimalAmountToSatoshis(await availablePublicBalance());
|
|
|
|
if (available == satoshiAmount) {
|
|
return satoshiAmount - sweepAllEstimate(feeRate);
|
|
} else if (satoshiAmount <= 0 || satoshiAmount > available) {
|
|
return roughFeeEstimate(1, 2, feeRate);
|
|
}
|
|
|
|
int runningBalance = 0;
|
|
int inputCount = 0;
|
|
for (final output in _outputsList) {
|
|
runningBalance += output.value;
|
|
inputCount++;
|
|
if (runningBalance > satoshiAmount) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
final oneOutPutFee = roughFeeEstimate(inputCount, 1, feeRate);
|
|
final twoOutPutFee = roughFeeEstimate(inputCount, 2, feeRate);
|
|
|
|
if (runningBalance - satoshiAmount > oneOutPutFee) {
|
|
if (runningBalance - satoshiAmount > oneOutPutFee + DUST_LIMIT) {
|
|
final change = runningBalance - satoshiAmount - twoOutPutFee;
|
|
if (change > DUST_LIMIT &&
|
|
runningBalance - satoshiAmount - change == twoOutPutFee) {
|
|
return runningBalance - satoshiAmount - change;
|
|
} else {
|
|
return runningBalance - satoshiAmount;
|
|
}
|
|
} else {
|
|
return runningBalance - satoshiAmount;
|
|
}
|
|
} else if (runningBalance - satoshiAmount == oneOutPutFee) {
|
|
return oneOutPutFee;
|
|
} else {
|
|
return twoOutPutFee;
|
|
}
|
|
}
|
|
|
|
// TODO: correct formula for firo?
|
|
int roughFeeEstimate(int inputCount, int outputCount, int feeRatePerKB) {
|
|
return ((181 * inputCount) + (34 * outputCount) + 10) *
|
|
(feeRatePerKB / 1000).ceil();
|
|
}
|
|
|
|
int sweepAllEstimate(int feeRate) {
|
|
int available = 0;
|
|
int inputCount = 0;
|
|
for (final output in _outputsList) {
|
|
if (output.status.confirmed) {
|
|
available += output.value;
|
|
inputCount++;
|
|
}
|
|
}
|
|
|
|
// transaction will only have 1 output minus the fee
|
|
final estimatedFee = roughFeeEstimate(inputCount, 1, feeRate);
|
|
|
|
return available - estimatedFee;
|
|
}
|
|
|
|
Future<List<Map<String, dynamic>>> fastFetch(List<String> allTxHashes) async {
|
|
List<Map<String, dynamic>> allTransactions = [];
|
|
|
|
const futureLimit = 30;
|
|
List<Future<Map<String, dynamic>>> transactionFutures = [];
|
|
int currentFutureCount = 0;
|
|
for (final txHash in allTxHashes) {
|
|
Future<Map<String, dynamic>> transactionFuture =
|
|
cachedElectrumXClient.getTransaction(
|
|
txHash: txHash,
|
|
verbose: true,
|
|
coin: coin,
|
|
);
|
|
transactionFutures.add(transactionFuture);
|
|
currentFutureCount++;
|
|
if (currentFutureCount > futureLimit) {
|
|
currentFutureCount = 0;
|
|
await Future.wait(transactionFutures);
|
|
for (final fTx in transactionFutures) {
|
|
final tx = await fTx;
|
|
// delete unused large parts
|
|
tx.remove("hex");
|
|
tx.remove("lelantusData");
|
|
|
|
allTransactions.add(tx);
|
|
}
|
|
}
|
|
}
|
|
if (currentFutureCount != 0) {
|
|
currentFutureCount = 0;
|
|
await Future.wait(transactionFutures);
|
|
for (final fTx in transactionFutures) {
|
|
final tx = await fTx;
|
|
// delete unused large parts
|
|
tx.remove("hex");
|
|
tx.remove("lelantusData");
|
|
|
|
allTransactions.add(tx);
|
|
}
|
|
}
|
|
return allTransactions;
|
|
}
|
|
|
|
Future<List<models.Transaction>> getJMintTransactions(
|
|
CachedElectrumX cachedClient,
|
|
List<String> transactions,
|
|
String currency,
|
|
Coin coin,
|
|
Decimal currentPrice,
|
|
String locale,
|
|
) async {
|
|
try {
|
|
List<models.Transaction> txs = [];
|
|
List<Map<String, dynamic>> allTransactions =
|
|
await fastFetch(transactions);
|
|
|
|
for (int i = 0; i < allTransactions.length; i++) {
|
|
try {
|
|
final tx = allTransactions[i];
|
|
|
|
tx["confirmed_status"] =
|
|
tx["confirmations"] != null && tx["confirmations"] as int > 0;
|
|
tx["timestamp"] = tx["time"];
|
|
tx["txType"] = "Sent";
|
|
|
|
var sendIndex = 1;
|
|
if (tx["vout"][0]["value"] != null &&
|
|
Decimal.parse(tx["vout"][0]["value"].toString()) > Decimal.zero) {
|
|
sendIndex = 0;
|
|
}
|
|
tx["amount"] = tx["vout"][sendIndex]["value"];
|
|
|
|
tx["address"] = tx["vout"][sendIndex]["scriptPubKey"]["addresses"][0];
|
|
|
|
tx["fees"] = tx["vin"][0]["nFees"];
|
|
tx["inputSize"] = tx["vin"].length;
|
|
tx["outputSize"] = tx["vout"].length;
|
|
|
|
final decimalAmount = Decimal.parse(tx["amount"].toString());
|
|
|
|
tx["worthNow"] = Format.localizedStringAsFixed(
|
|
value: currentPrice * decimalAmount,
|
|
locale: locale,
|
|
decimalPlaces: 2,
|
|
);
|
|
tx["worthAtBlockTimestamp"] = tx["worthNow"];
|
|
|
|
tx["subType"] = "join";
|
|
txs.add(models.Transaction.fromLelantusJson(tx));
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception caught in getJMintTransactions(): $e\n$s",
|
|
level: LogLevel.Info);
|
|
rethrow;
|
|
}
|
|
}
|
|
return txs;
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception rethrown in getJMintTransactions(): $e\n$s",
|
|
level: LogLevel.Info);
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
@override
|
|
Future<bool> generateNewAddress() async {
|
|
try {
|
|
await incrementAddressIndexForChain(
|
|
0); // First increment the receiving index
|
|
final newReceivingIndex =
|
|
DB.instance.get<dynamic>(boxName: walletId, key: 'receivingIndex')
|
|
as int; // Check the new receiving index
|
|
final newReceivingAddress = await _generateAddressForChain(0,
|
|
newReceivingIndex); // Use new index to derive a new receiving address
|
|
await addToAddressesArrayForChain(newReceivingAddress,
|
|
0); // Add that new receiving address to the array of receiving addresses
|
|
_currentReceivingAddress = Future(() =>
|
|
newReceivingAddress); // Set the new receiving address that the service
|
|
|
|
return true;
|
|
} catch (e, s) {
|
|
Logging.instance.log(
|
|
"Exception rethrown from generateNewAddress(): $e\n$s",
|
|
level: LogLevel.Error);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Future<Decimal> availablePrivateBalance() async {
|
|
return (await balances)[0];
|
|
}
|
|
|
|
Future<Decimal> availablePublicBalance() async {
|
|
return (await balances)[4];
|
|
}
|
|
}
|