mirror of
https://github.com/cypherstack/stack_wallet.git
synced 2024-11-06 04:17:42 +00:00
374 lines
12 KiB
Dart
374 lines
12 KiB
Dart
/*
|
|
* This file is part of Stack Wallet.
|
|
*
|
|
* Copyright (c) 2023 Cypher Stack
|
|
* All Rights Reserved.
|
|
* The code is distributed under GPLv3 license, see LICENSE file for details.
|
|
* Generated by Cypher Stack on 2023-05-26
|
|
*
|
|
*/
|
|
|
|
import 'dart:convert';
|
|
|
|
import 'package:bip47/src/util.dart';
|
|
import 'package:decimal/decimal.dart';
|
|
import 'package:stackwallet/electrumx_rpc/cached_electrumx_client.dart';
|
|
import 'package:stackwallet/models/isar/models/blockchain_data/v2/input_v2.dart';
|
|
import 'package:stackwallet/models/isar/models/blockchain_data/v2/output_v2.dart';
|
|
import 'package:stackwallet/models/isar/models/blockchain_data/v2/transaction_v2.dart';
|
|
import 'package:stackwallet/models/isar/models/isar_models.dart';
|
|
import 'package:stackwallet/utilities/amount/amount.dart';
|
|
import 'package:stackwallet/utilities/enums/coin_enum.dart';
|
|
import 'package:stackwallet/utilities/util.dart' as util;
|
|
import 'package:stackwallet/wallets/wallet/wallet_mixin_interfaces/paynym_interface.dart';
|
|
import 'package:tuple/tuple.dart';
|
|
|
|
mixin ElectrumXParsing {
|
|
Future<TransactionV2> getTransaction(
|
|
String txHash,
|
|
Coin coin,
|
|
String walletId,
|
|
CachedElectrumXClient cachedElectrumX, [
|
|
String? debugTitle,
|
|
]) async {
|
|
final jsonTx = await cachedElectrumX.getTransaction(
|
|
txHash: txHash,
|
|
coin: coin,
|
|
);
|
|
print("+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++");
|
|
util.Util.printJson(jsonTx, debugTitle);
|
|
print("+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++");
|
|
|
|
// parse inputs
|
|
final List<InputV2> inputs = [];
|
|
for (final jsonInput in jsonTx["vin"] as List) {
|
|
final map = Map<String, dynamic>.from(jsonInput as Map);
|
|
|
|
final List<String> addresses = [];
|
|
String valueStringSats = "0";
|
|
OutpointV2? outpoint;
|
|
|
|
final coinbase = map["coinbase"] as String?;
|
|
|
|
if (coinbase == null) {
|
|
final txid = map["txid"] as String;
|
|
final vout = map["vout"] as int;
|
|
|
|
final inputTx =
|
|
await cachedElectrumX.getTransaction(txHash: txid, coin: coin);
|
|
|
|
final prevOutJson = Map<String, dynamic>.from(
|
|
(inputTx["vout"] as List).firstWhere((e) => e["n"] == vout) as Map);
|
|
|
|
final prevOut = OutputV2.fromElectrumXJson(
|
|
prevOutJson,
|
|
decimalPlaces: coin.decimals,
|
|
walletOwns: false,
|
|
);
|
|
|
|
outpoint = OutpointV2.isarCantDoRequiredInDefaultConstructor(
|
|
txid: txid,
|
|
vout: vout,
|
|
);
|
|
valueStringSats = prevOut.valueStringSats;
|
|
addresses.addAll(prevOut.addresses);
|
|
}
|
|
|
|
final input = InputV2.isarCantDoRequiredInDefaultConstructor(
|
|
scriptSigHex: map["scriptSig"]?["hex"] as String?,
|
|
sequence: map["sequence"] as int?,
|
|
outpoint: outpoint,
|
|
valueStringSats: valueStringSats,
|
|
addresses: addresses,
|
|
witness: map["witness"] as String?,
|
|
coinbase: coinbase,
|
|
innerRedeemScriptAsm: map["innerRedeemscriptAsm"] as String?,
|
|
walletOwns: false,
|
|
);
|
|
|
|
inputs.add(input);
|
|
}
|
|
|
|
// parse outputs
|
|
final List<OutputV2> outputs = [];
|
|
for (final outputJson in jsonTx["vout"] as List) {
|
|
final output = OutputV2.fromElectrumXJson(
|
|
Map<String, dynamic>.from(outputJson as Map),
|
|
decimalPlaces: coin.decimals,
|
|
walletOwns: false,
|
|
);
|
|
outputs.add(output);
|
|
}
|
|
|
|
return TransactionV2(
|
|
walletId: walletId,
|
|
blockHash: jsonTx["blockhash"] as String?,
|
|
hash: jsonTx["hash"] as String,
|
|
txid: jsonTx["txid"] as String,
|
|
height: jsonTx["height"] as int?,
|
|
version: jsonTx["version"] as int,
|
|
timestamp: jsonTx["blocktime"] as int? ??
|
|
DateTime.timestamp().millisecondsSinceEpoch ~/ 1000,
|
|
inputs: List.unmodifiable(inputs),
|
|
outputs: List.unmodifiable(outputs),
|
|
subType: TransactionSubType.none,
|
|
type: TransactionType.unknown,
|
|
otherData: null,
|
|
);
|
|
}
|
|
|
|
Future<Tuple2<Transaction, Address>> parseTransaction(
|
|
Map<String, dynamic> txData,
|
|
dynamic electrumxClient,
|
|
List<Address> myAddresses,
|
|
Coin coin,
|
|
int minConfirms,
|
|
String walletId,
|
|
) async {
|
|
Set<String> receivingAddresses = myAddresses
|
|
.where((e) =>
|
|
e.subType == AddressSubType.receiving ||
|
|
e.subType == AddressSubType.paynymReceive ||
|
|
e.subType == AddressSubType.paynymNotification)
|
|
.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> outputAddresses = {};
|
|
|
|
Amount totalInputValue = Amount(
|
|
rawValue: BigInt.zero,
|
|
fractionDigits: coin.decimals,
|
|
);
|
|
Amount totalOutputValue = Amount(
|
|
rawValue: BigInt.zero,
|
|
fractionDigits: coin.decimals,
|
|
);
|
|
|
|
Amount amountSentFromWallet = Amount(
|
|
rawValue: BigInt.zero,
|
|
fractionDigits: coin.decimals,
|
|
);
|
|
Amount amountReceivedInWallet = Amount(
|
|
rawValue: BigInt.zero,
|
|
fractionDigits: coin.decimals,
|
|
);
|
|
Amount changeAmount = Amount(
|
|
rawValue: BigInt.zero,
|
|
fractionDigits: coin.decimals,
|
|
);
|
|
|
|
// parse inputs
|
|
for (final input in txData["vin"] as List) {
|
|
final prevTxid = input["txid"] as String;
|
|
final prevOut = input["vout"] as int;
|
|
|
|
// fetch input tx to get address
|
|
final inputTx = await electrumxClient.getTransaction(
|
|
txHash: prevTxid,
|
|
coin: coin,
|
|
);
|
|
|
|
for (final output in inputTx["vout"] as List) {
|
|
// check matching output
|
|
if (prevOut == output["n"]) {
|
|
// get value
|
|
final value = Amount.fromDecimal(
|
|
Decimal.parse(output["value"].toString()),
|
|
fractionDigits: coin.decimals,
|
|
);
|
|
|
|
// add value to total
|
|
totalInputValue += value;
|
|
|
|
// get input(prevOut) address
|
|
final address = output["scriptPubKey"]?["addresses"]?[0] as String? ??
|
|
output["scriptPubKey"]?["address"] as String?;
|
|
|
|
if (address != null) {
|
|
inputAddresses.add(address);
|
|
|
|
// if input was from my wallet, add value to amount sent
|
|
if (receivingAddresses.contains(address) ||
|
|
changeAddresses.contains(address)) {
|
|
amountSentFromWallet += value;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// parse outputs
|
|
for (final output in txData["vout"] as List) {
|
|
// get value
|
|
final value = Amount.fromDecimal(
|
|
Decimal.parse(output["value"].toString()),
|
|
fractionDigits: coin.decimals,
|
|
);
|
|
|
|
// add value to total
|
|
totalOutputValue += value;
|
|
|
|
// get output address
|
|
final address = output["scriptPubKey"]?["addresses"]?[0] as String? ??
|
|
output["scriptPubKey"]?["address"] as String?;
|
|
if (address != null) {
|
|
outputAddresses.add(address);
|
|
|
|
// if output was to my wallet, add value to amount received
|
|
if (receivingAddresses.contains(address)) {
|
|
amountReceivedInWallet += value;
|
|
} else if (changeAddresses.contains(address)) {
|
|
changeAmount += value;
|
|
}
|
|
}
|
|
}
|
|
|
|
final mySentFromAddresses = [
|
|
...receivingAddresses.intersection(inputAddresses),
|
|
...changeAddresses.intersection(inputAddresses)
|
|
];
|
|
final myReceivedOnAddresses =
|
|
receivingAddresses.intersection(outputAddresses);
|
|
final myChangeReceivedOnAddresses =
|
|
changeAddresses.intersection(outputAddresses);
|
|
|
|
final fee = totalInputValue - totalOutputValue;
|
|
|
|
// this is the address initially used to fetch the txid
|
|
Address transactionAddress = txData["address"] as Address;
|
|
|
|
TransactionType type;
|
|
Amount amount;
|
|
if (mySentFromAddresses.isNotEmpty && myReceivedOnAddresses.isNotEmpty) {
|
|
// tx is sent to self
|
|
type = TransactionType.sentToSelf;
|
|
|
|
// should be 0
|
|
amount =
|
|
amountSentFromWallet - amountReceivedInWallet - fee - changeAmount;
|
|
} else if (mySentFromAddresses.isNotEmpty) {
|
|
// outgoing tx
|
|
type = TransactionType.outgoing;
|
|
amount = amountSentFromWallet - changeAmount - fee;
|
|
|
|
// non wallet addresses found in tx outputs
|
|
final nonWalletOutAddresses = outputAddresses.difference(
|
|
myChangeReceivedOnAddresses,
|
|
);
|
|
|
|
if (nonWalletOutAddresses.isNotEmpty) {
|
|
final possible = nonWalletOutAddresses.first;
|
|
|
|
if (transactionAddress.value != possible) {
|
|
transactionAddress = Address(
|
|
walletId: walletId,
|
|
value: possible,
|
|
derivationIndex: -1,
|
|
derivationPath: null,
|
|
subType: AddressSubType.nonWallet,
|
|
type: AddressType.nonWallet,
|
|
publicKey: [],
|
|
);
|
|
}
|
|
} else {
|
|
// some other type of tx where the receiving address is
|
|
// one of my change addresses
|
|
|
|
type = TransactionType.sentToSelf;
|
|
amount = changeAmount;
|
|
}
|
|
} else {
|
|
// incoming tx
|
|
type = TransactionType.incoming;
|
|
amount = amountReceivedInWallet;
|
|
}
|
|
|
|
List<Output> outs = [];
|
|
List<Input> ins = [];
|
|
|
|
for (final json in txData["vin"] as List) {
|
|
bool isCoinBase = json['coinbase'] != null;
|
|
String? witness;
|
|
if (json['witness'] != null && json['witness'] is String) {
|
|
witness = json['witness'] as String;
|
|
} else if (json['txinwitness'] != null) {
|
|
if (json['txinwitness'] is List) {
|
|
witness = jsonEncode(json['txinwitness']);
|
|
}
|
|
}
|
|
final input = Input(
|
|
txid: json['txid'] as String,
|
|
vout: json['vout'] as int? ?? -1,
|
|
scriptSig: json['scriptSig']?['hex'] as String?,
|
|
scriptSigAsm: json['scriptSig']?['asm'] as String?,
|
|
isCoinbase: isCoinBase ? isCoinBase : json['is_coinbase'] as bool?,
|
|
sequence: json['sequence'] as int?,
|
|
innerRedeemScriptAsm: json['innerRedeemscriptAsm'] as String?,
|
|
witness: witness,
|
|
);
|
|
ins.add(input);
|
|
}
|
|
|
|
for (final json in txData["vout"] as List) {
|
|
final output = Output(
|
|
scriptPubKey: json['scriptPubKey']?['hex'] as String?,
|
|
scriptPubKeyAsm: json['scriptPubKey']?['asm'] as String?,
|
|
scriptPubKeyType: json['scriptPubKey']?['type'] as String?,
|
|
scriptPubKeyAddress:
|
|
json["scriptPubKey"]?["addresses"]?[0] as String? ??
|
|
json['scriptPubKey']?['type'] as String? ??
|
|
"",
|
|
value: Amount.fromDecimal(
|
|
Decimal.parse(json["value"].toString()),
|
|
fractionDigits: coin.decimals,
|
|
).raw.toInt(),
|
|
);
|
|
outs.add(output);
|
|
}
|
|
|
|
TransactionSubType txSubType = TransactionSubType.none;
|
|
if (this is PaynymInterface && outs.length > 1 && ins.isNotEmpty) {
|
|
for (int i = 0; i < outs.length; i++) {
|
|
List<String>? scriptChunks = outs[i].scriptPubKeyAsm?.split(" ");
|
|
if (scriptChunks?.length == 2 && scriptChunks?[0] == "OP_RETURN") {
|
|
final blindedPaymentCode = scriptChunks![1];
|
|
final bytes = blindedPaymentCode.fromHex;
|
|
|
|
// https://en.bitcoin.it/wiki/BIP_0047#Sending
|
|
if (bytes.length == 80 && bytes.first == 1) {
|
|
txSubType = TransactionSubType.bip47Notification;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
final tx = Transaction(
|
|
walletId: walletId,
|
|
txid: txData["txid"] as String,
|
|
timestamp: txData["blocktime"] as int? ??
|
|
(DateTime.now().millisecondsSinceEpoch ~/ 1000),
|
|
type: type,
|
|
subType: txSubType,
|
|
// amount may overflow. Deprecated. Use amountString
|
|
amount: amount.raw.toInt(),
|
|
amountString: amount.toJsonString(),
|
|
fee: fee.raw.toInt(),
|
|
height: txData["height"] as int?,
|
|
isCancelled: false,
|
|
isLelantus: false,
|
|
slateId: null,
|
|
otherData: null,
|
|
nonce: null,
|
|
inputs: ins,
|
|
outputs: outs,
|
|
numberOfMessages: null,
|
|
);
|
|
|
|
return Tuple2(tx, transactionAddress);
|
|
}
|
|
}
|