mirror of
https://github.com/cypherstack/stack_wallet.git
synced 2025-01-11 05:04:35 +00:00
modify experimental parseTransaction function to return the new Isar Transaction model
This commit is contained in:
parent
e7877358da
commit
d27af7243c
1 changed files with 86 additions and 49 deletions
|
@ -3,12 +3,14 @@ import 'dart:typed_data';
|
||||||
|
|
||||||
import 'package:bip47/bip47.dart';
|
import 'package:bip47/bip47.dart';
|
||||||
import 'package:bip47/src/util.dart';
|
import 'package:bip47/src/util.dart';
|
||||||
import 'package:bitcoindart/bitcoindart.dart';
|
import 'package:bitcoindart/bitcoindart.dart' as btc_dart;
|
||||||
import 'package:bitcoindart/src/utils/constants/op.dart' as op;
|
import 'package:bitcoindart/src/utils/constants/op.dart' as op;
|
||||||
import 'package:bitcoindart/src/utils/script.dart' as bscript;
|
import 'package:bitcoindart/src/utils/script.dart' as bscript;
|
||||||
|
import 'package:dart_numerics/dart_numerics.dart';
|
||||||
import 'package:decimal/decimal.dart';
|
import 'package:decimal/decimal.dart';
|
||||||
import 'package:pointycastle/digests/sha256.dart';
|
import 'package:pointycastle/digests/sha256.dart';
|
||||||
import 'package:stackwallet/hive/db.dart';
|
import 'package:stackwallet/hive/db.dart';
|
||||||
|
import 'package:stackwallet/models/isar/models/isar_models.dart';
|
||||||
import 'package:stackwallet/models/models.dart' as models;
|
import 'package:stackwallet/models/models.dart' as models;
|
||||||
import 'package:stackwallet/models/paymint/utxo_model.dart';
|
import 'package:stackwallet/models/paymint/utxo_model.dart';
|
||||||
import 'package:stackwallet/services/coins/dogecoin/dogecoin_wallet.dart';
|
import 'package:stackwallet/services/coins/dogecoin/dogecoin_wallet.dart';
|
||||||
|
@ -59,7 +61,8 @@ extension PayNym on DogecoinWallet {
|
||||||
Future<Uint8List> signWithNotificationKey(Uint8List data) async {
|
Future<Uint8List> signWithNotificationKey(Uint8List data) async {
|
||||||
final node = getBip32Root((await mnemonic).join(" "), network)
|
final node = getBip32Root((await mnemonic).join(" "), network)
|
||||||
.derivePath("m/47'/0'/0'");
|
.derivePath("m/47'/0'/0'");
|
||||||
final pair = ECPair.fromPrivateKey(node.privateKey!, network: network);
|
final pair =
|
||||||
|
btc_dart.ECPair.fromPrivateKey(node.privateKey!, network: network);
|
||||||
final signed = pair.sign(SHA256Digest().process(data));
|
final signed = pair.sign(SHA256Digest().process(data));
|
||||||
return signed;
|
return signed;
|
||||||
}
|
}
|
||||||
|
@ -420,7 +423,7 @@ extension PayNym on DogecoinWallet {
|
||||||
final buffer = rev.buffer.asByteData();
|
final buffer = rev.buffer.asByteData();
|
||||||
buffer.setUint32(txPoint.length, txPointIndex, Endian.little);
|
buffer.setUint32(txPoint.length, txPointIndex, Endian.little);
|
||||||
|
|
||||||
final myKeyPair = utxoSigningData[utxo.txid]["keyPair"] as ECPair;
|
final myKeyPair = utxoSigningData[utxo.txid]["keyPair"] as btc_dart.ECPair;
|
||||||
|
|
||||||
final S = SecretPoint(
|
final S = SecretPoint(
|
||||||
myKeyPair.privateKey!,
|
myKeyPair.privateKey!,
|
||||||
|
@ -440,7 +443,7 @@ extension PayNym on DogecoinWallet {
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// build a notification tx
|
// build a notification tx
|
||||||
final txb = TransactionBuilder(network: network);
|
final txb = btc_dart.TransactionBuilder(network: network);
|
||||||
txb.setVersion(1);
|
txb.setVersion(1);
|
||||||
|
|
||||||
txb.addInput(
|
txb.addInput(
|
||||||
|
@ -454,7 +457,7 @@ extension PayNym on DogecoinWallet {
|
||||||
// TODO: add possible change output and mark output as dangerous
|
// TODO: add possible change output and mark output as dangerous
|
||||||
if (change > 0) {
|
if (change > 0) {
|
||||||
// generate new change address if current change address has been used
|
// generate new change address if current change address has been used
|
||||||
await checkChangeAddressForTransactions(DerivePathType.bip44);
|
await checkChangeAddressForTransactions();
|
||||||
final String changeAddress =
|
final String changeAddress =
|
||||||
await getCurrentAddressForChain(1, DerivePathType.bip44);
|
await getCurrentAddressForChain(1, DerivePathType.bip44);
|
||||||
txb.addOutput(changeAddress, change);
|
txb.addOutput(changeAddress, change);
|
||||||
|
@ -470,7 +473,7 @@ extension PayNym on DogecoinWallet {
|
||||||
final txid = utxosToUse[i].txid;
|
final txid = utxosToUse[i].txid;
|
||||||
txb.sign(
|
txb.sign(
|
||||||
vin: i,
|
vin: i,
|
||||||
keyPair: utxoSigningData[txid]["keyPair"] as ECPair,
|
keyPair: utxoSigningData[txid]["keyPair"] as btc_dart.ECPair,
|
||||||
// witnessValue: utxosToUse[i].value,
|
// witnessValue: utxosToUse[i].value,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -563,15 +566,22 @@ extension PayNym on DogecoinWallet {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<Map<String, dynamic>> parseTransaction(
|
Future<Transaction> parseTransaction(
|
||||||
Map<String, dynamic> txData,
|
Map<String, dynamic> txData,
|
||||||
dynamic electrumxClient,
|
dynamic electrumxClient,
|
||||||
Set<String> myAddresses,
|
List<Address> myAddresses,
|
||||||
Set<String> myChangeAddresses,
|
|
||||||
Coin coin,
|
Coin coin,
|
||||||
int minConfirms,
|
int minConfirms,
|
||||||
Decimal currentPrice,
|
|
||||||
) async {
|
) async {
|
||||||
|
Set<String> receivingAddresses = myAddresses
|
||||||
|
.where((e) => e.subType == AddressSubType.receiving)
|
||||||
|
.map((e) => e.value)
|
||||||
|
.toSet();
|
||||||
|
Set<String> changeAddresses = myAddresses
|
||||||
|
.where((e) => e.subType == AddressSubType.change)
|
||||||
|
.map((e) => e.value)
|
||||||
|
.toSet();
|
||||||
|
|
||||||
Set<String> inputAddresses = {};
|
Set<String> inputAddresses = {};
|
||||||
Set<String> outputAddresses = {};
|
Set<String> outputAddresses = {};
|
||||||
|
|
||||||
|
@ -580,6 +590,7 @@ Future<Map<String, dynamic>> parseTransaction(
|
||||||
|
|
||||||
int amountSentFromWallet = 0;
|
int amountSentFromWallet = 0;
|
||||||
int amountReceivedInWallet = 0;
|
int amountReceivedInWallet = 0;
|
||||||
|
int changeAmount = 0;
|
||||||
|
|
||||||
// parse inputs
|
// parse inputs
|
||||||
for (final input in txData["vin"] as List) {
|
for (final input in txData["vin"] as List) {
|
||||||
|
@ -612,7 +623,8 @@ Future<Map<String, dynamic>> parseTransaction(
|
||||||
inputAddresses.add(address);
|
inputAddresses.add(address);
|
||||||
|
|
||||||
// if input was from my wallet, add value to amount sent
|
// if input was from my wallet, add value to amount sent
|
||||||
if (myAddresses.contains(address)) {
|
if (receivingAddresses.contains(address) ||
|
||||||
|
changeAddresses.contains(address)) {
|
||||||
amountSentFromWallet += value;
|
amountSentFromWallet += value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -637,61 +649,86 @@ Future<Map<String, dynamic>> parseTransaction(
|
||||||
if (address != null) {
|
if (address != null) {
|
||||||
outputAddresses.add(address);
|
outputAddresses.add(address);
|
||||||
|
|
||||||
// if output was from my wallet, add value to amount received
|
// if output was to my wallet, add value to amount received
|
||||||
if (myAddresses.contains(address)) {
|
if (receivingAddresses.contains(address)) {
|
||||||
amountReceivedInWallet += value;
|
amountReceivedInWallet += value;
|
||||||
|
} else if (changeAddresses.contains(address)) {
|
||||||
|
changeAmount += value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
final mySentFromAddresses = myAddresses.intersection(inputAddresses);
|
final mySentFromAddresses = [
|
||||||
final myReceivedOnAddresses = myAddresses.intersection(outputAddresses);
|
...receivingAddresses.intersection(inputAddresses),
|
||||||
|
...changeAddresses.intersection(inputAddresses)
|
||||||
|
];
|
||||||
|
final myReceivedOnAddresses =
|
||||||
|
receivingAddresses.intersection(outputAddresses);
|
||||||
|
final myChangeReceivedOnAddresses =
|
||||||
|
changeAddresses.intersection(outputAddresses);
|
||||||
|
|
||||||
final fee = totalInputValue - totalOutputValue;
|
final fee = totalInputValue - totalOutputValue;
|
||||||
|
|
||||||
// create normalized tx data map
|
final tx = Transaction();
|
||||||
Map<String, dynamic> normalizedTx = {};
|
tx.txid = txData["txid"] as String;
|
||||||
|
tx.timestamp = txData["blocktime"] as int? ??
|
||||||
final int confirms = txData["confirmations"] as int? ?? 0;
|
|
||||||
|
|
||||||
normalizedTx["txid"] = txData["txid"] as String;
|
|
||||||
normalizedTx["confirmed_status"] = confirms >= minConfirms;
|
|
||||||
normalizedTx["confirmations"] = confirms;
|
|
||||||
normalizedTx["timestamp"] = txData["blocktime"] as int? ??
|
|
||||||
(DateTime.now().millisecondsSinceEpoch ~/ 1000);
|
(DateTime.now().millisecondsSinceEpoch ~/ 1000);
|
||||||
normalizedTx["aliens"] = <dynamic>[];
|
|
||||||
normalizedTx["fees"] = fee;
|
|
||||||
normalizedTx["address"] = txData["address"] as String;
|
|
||||||
normalizedTx["inputSize"] = txData["vin"].length;
|
|
||||||
normalizedTx["outputSize"] = txData["vout"].length;
|
|
||||||
normalizedTx["inputs"] = txData["vin"];
|
|
||||||
normalizedTx["outputs"] = txData["vout"];
|
|
||||||
normalizedTx["height"] = txData["height"] as int;
|
|
||||||
|
|
||||||
int amount;
|
|
||||||
String type;
|
|
||||||
if (mySentFromAddresses.isNotEmpty && myReceivedOnAddresses.isNotEmpty) {
|
if (mySentFromAddresses.isNotEmpty && myReceivedOnAddresses.isNotEmpty) {
|
||||||
// tx is sent to self
|
// tx is sent to self
|
||||||
type = "Sent to self";
|
tx.type = TransactionType.sendToSelf;
|
||||||
amount = amountSentFromWallet - amountReceivedInWallet - fee;
|
tx.amount =
|
||||||
|
amountSentFromWallet - amountReceivedInWallet - fee - changeAmount;
|
||||||
} else if (mySentFromAddresses.isNotEmpty) {
|
} else if (mySentFromAddresses.isNotEmpty) {
|
||||||
// outgoing tx
|
// outgoing tx
|
||||||
type = "Sent";
|
tx.type = TransactionType.outgoing;
|
||||||
amount = amountSentFromWallet;
|
tx.amount = amountSentFromWallet - changeAmount - fee;
|
||||||
} else {
|
} else {
|
||||||
// incoming tx
|
// incoming tx
|
||||||
type = "Received";
|
tx.type = TransactionType.incoming;
|
||||||
amount = amountReceivedInWallet;
|
tx.amount = amountReceivedInWallet;
|
||||||
}
|
}
|
||||||
|
|
||||||
normalizedTx["txType"] = type;
|
// TODO: other subtypes
|
||||||
normalizedTx["amount"] = amount;
|
tx.subType = TransactionSubType.none;
|
||||||
normalizedTx["worthNow"] = (Format.satoshisToAmount(
|
|
||||||
amount,
|
|
||||||
coin: coin,
|
|
||||||
) *
|
|
||||||
currentPrice)
|
|
||||||
.toStringAsFixed(2);
|
|
||||||
|
|
||||||
return normalizedTx;
|
tx.fee = fee;
|
||||||
|
tx.address = txData["address"] as String;
|
||||||
|
|
||||||
|
for (final json in txData["vin"] as List) {
|
||||||
|
bool isCoinBase = json['coinbase'] != null;
|
||||||
|
final input = Input();
|
||||||
|
input.txid = json['txid'] as String;
|
||||||
|
input.vout = json['vout'] as int? ?? -1;
|
||||||
|
input.scriptSig = json['scriptSig']?['hex'] as String?;
|
||||||
|
input.scriptSigAsm = json['scriptSig']?['asm'] as String?;
|
||||||
|
input.isCoinbase = isCoinBase ? isCoinBase : json['is_coinbase'] as bool?;
|
||||||
|
input.sequence = json['sequence'] as int?;
|
||||||
|
input.innerRedeemScriptAsm = json['innerRedeemscriptAsm'] as String?;
|
||||||
|
tx.inputs.add(input);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (final json in txData["vout"] as List) {
|
||||||
|
final output = Output();
|
||||||
|
output.scriptPubKey = json['scriptPubKey']?['hex'] as String?;
|
||||||
|
output.scriptPubKeyAsm = json['scriptPubKey']?['asm'] as String?;
|
||||||
|
output.scriptPubKeyType = json['scriptPubKey']?['type'] as String?;
|
||||||
|
output.scriptPubKeyAddress =
|
||||||
|
json["scriptPubKey"]?["addresses"]?[0] as String? ??
|
||||||
|
json['scriptPubKey']['type'] as String;
|
||||||
|
output.value = Format.decimalAmountToSatoshis(
|
||||||
|
Decimal.parse(json["value"].toString()),
|
||||||
|
coin,
|
||||||
|
);
|
||||||
|
tx.outputs.add(output);
|
||||||
|
}
|
||||||
|
|
||||||
|
tx.height = txData["height"] as int? ?? int64MaxValue;
|
||||||
|
|
||||||
|
//TODO: change these for epic (or other coins that need it)
|
||||||
|
tx.cancelled = false;
|
||||||
|
tx.slateId = null;
|
||||||
|
tx.otherData = null;
|
||||||
|
|
||||||
|
return tx;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue