mirror of
https://github.com/sparrowwallet/sparrow.git
synced 2024-12-24 12:46:45 +00:00
improve wallet loading performance
This commit is contained in:
parent
306f241a4a
commit
9faf036e4d
20 changed files with 129 additions and 57 deletions
2
drongo
2
drongo
|
@ -1 +1 @@
|
||||||
Subproject commit 8dca2ee3f0ba8dbebf88c3629b2a52c7eecf5b89
|
Subproject commit fe61c633ae230c3425d52a0c9ac6264ea77e5041
|
|
@ -880,8 +880,21 @@ public class AppController implements Initializable {
|
||||||
try {
|
try {
|
||||||
Storage storage = new Storage(file);
|
Storage storage = new Storage(file);
|
||||||
if(!storage.isEncrypted()) {
|
if(!storage.isEncrypted()) {
|
||||||
WalletBackupAndKey walletBackupAndKey = storage.loadUnencryptedWallet();
|
Storage.LoadWalletService loadWalletService = new Storage.LoadWalletService(storage);
|
||||||
openWallet(storage, walletBackupAndKey, this, forceSameWindow);
|
loadWalletService.setOnSucceeded(workerStateEvent -> {
|
||||||
|
WalletBackupAndKey walletBackupAndKey = loadWalletService.getValue();
|
||||||
|
openWallet(storage, walletBackupAndKey, this, forceSameWindow);
|
||||||
|
});
|
||||||
|
loadWalletService.setOnFailed(workerStateEvent -> {
|
||||||
|
Throwable exception = workerStateEvent.getSource().getException();
|
||||||
|
if(exception instanceof StorageException) {
|
||||||
|
showErrorDialog("Error Opening Wallet", exception.getMessage());
|
||||||
|
} else if(!attemptImportWallet(file, null)) {
|
||||||
|
log.error("Error opening wallet", exception);
|
||||||
|
showErrorDialog("Error Opening Wallet", exception.getMessage() == null || exception.getMessage().contains("Expected BEGIN_OBJECT") ? "Unsupported wallet file format." : exception.getMessage());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
loadWalletService.start();
|
||||||
} else {
|
} else {
|
||||||
WalletPasswordDialog dlg = new WalletPasswordDialog(storage.getWalletName(null), WalletPasswordDialog.PasswordRequirement.LOAD);
|
WalletPasswordDialog dlg = new WalletPasswordDialog(storage.getWalletName(null), WalletPasswordDialog.PasswordRequirement.LOAD);
|
||||||
Optional<SecureString> optionalPassword = dlg.showAndWait();
|
Optional<SecureString> optionalPassword = dlg.showAndWait();
|
||||||
|
@ -905,9 +918,11 @@ public class AppController implements Initializable {
|
||||||
Platform.runLater(() -> openWalletFile(file, forceSameWindow));
|
Platform.runLater(() -> openWalletFile(file, forceSameWindow));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if(!attemptImportWallet(file, password)) {
|
if(exception instanceof StorageException) {
|
||||||
|
showErrorDialog("Error Opening Wallet", exception.getMessage());
|
||||||
|
} else if(!attemptImportWallet(file, password)) {
|
||||||
log.error("Error Opening Wallet", exception);
|
log.error("Error Opening Wallet", exception);
|
||||||
showErrorDialog("Error Opening Wallet", exception.getMessage() == null ? "Unsupported file format" : exception.getMessage());
|
showErrorDialog("Error Opening Wallet", exception.getMessage() == null || exception.getMessage().contains("Expected BEGIN_OBJECT") ? "Unsupported wallet file format." : exception.getMessage());
|
||||||
}
|
}
|
||||||
password.clear();
|
password.clear();
|
||||||
}
|
}
|
||||||
|
@ -915,8 +930,6 @@ public class AppController implements Initializable {
|
||||||
EventManager.get().post(new StorageEvent(storage.getWalletId(null), TimedEvent.Action.START, "Decrypting wallet..."));
|
EventManager.get().post(new StorageEvent(storage.getWalletId(null), TimedEvent.Action.START, "Decrypting wallet..."));
|
||||||
loadWalletService.start();
|
loadWalletService.start();
|
||||||
}
|
}
|
||||||
} catch(StorageException e) {
|
|
||||||
showErrorDialog("Error Opening Wallet", e.getMessage());
|
|
||||||
} catch(Exception e) {
|
} catch(Exception e) {
|
||||||
if(e instanceof IOException && e.getMessage().startsWith("The process cannot access the file because another process has locked")) {
|
if(e instanceof IOException && e.getMessage().startsWith("The process cannot access the file because another process has locked")) {
|
||||||
log.error("Error opening wallet", e);
|
log.error("Error opening wallet", e);
|
||||||
|
|
|
@ -1,20 +1,26 @@
|
||||||
package com.sparrowwallet.sparrow.control;
|
package com.sparrowwallet.sparrow.control;
|
||||||
|
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
import com.sparrowwallet.drongo.BitcoinUnit;
|
import com.sparrowwallet.drongo.BitcoinUnit;
|
||||||
import com.sparrowwallet.drongo.wallet.Wallet;
|
import com.sparrowwallet.drongo.wallet.Wallet;
|
||||||
import com.sparrowwallet.sparrow.io.Config;
|
import com.sparrowwallet.sparrow.io.Config;
|
||||||
|
import com.sparrowwallet.sparrow.wallet.Entry;
|
||||||
import com.sparrowwallet.sparrow.wallet.TransactionEntry;
|
import com.sparrowwallet.sparrow.wallet.TransactionEntry;
|
||||||
import com.sparrowwallet.sparrow.wallet.WalletTransactionsEntry;
|
import com.sparrowwallet.sparrow.wallet.WalletTransactionsEntry;
|
||||||
import javafx.beans.NamedArg;
|
import javafx.beans.NamedArg;
|
||||||
import javafx.scene.Node;
|
import javafx.scene.Node;
|
||||||
import javafx.scene.chart.*;
|
import javafx.scene.chart.*;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
public class BalanceChart extends LineChart<Number, Number> {
|
public class BalanceChart extends LineChart<Number, Number> {
|
||||||
|
private static final int MAX_VALUES = 500;
|
||||||
|
|
||||||
private XYChart.Series<Number, Number> balanceSeries;
|
private XYChart.Series<Number, Number> balanceSeries;
|
||||||
|
|
||||||
private TransactionEntry selectedEntry;
|
private TransactionEntry selectedEntry;
|
||||||
|
@ -37,7 +43,7 @@ public class BalanceChart extends LineChart<Number, Number> {
|
||||||
setVisible(!walletTransactionsEntry.getChildren().isEmpty());
|
setVisible(!walletTransactionsEntry.getChildren().isEmpty());
|
||||||
balanceSeries.getData().clear();
|
balanceSeries.getData().clear();
|
||||||
|
|
||||||
List<Data<Number, Number>> balanceDataList = walletTransactionsEntry.getChildren().stream()
|
List<Data<Number, Number>> balanceDataList = getTransactionEntries(walletTransactionsEntry)
|
||||||
.map(entry -> (TransactionEntry)entry)
|
.map(entry -> (TransactionEntry)entry)
|
||||||
.filter(txEntry -> txEntry.getBlockTransaction().getHeight() > 0)
|
.filter(txEntry -> txEntry.getBlockTransaction().getHeight() > 0)
|
||||||
.map(txEntry -> new XYChart.Data<>((Number)txEntry.getBlockTransaction().getDate().getTime(), (Number)txEntry.getBalance(), txEntry))
|
.map(txEntry -> new XYChart.Data<>((Number)txEntry.getBlockTransaction().getDate().getTime(), (Number)txEntry.getBalance(), txEntry))
|
||||||
|
@ -74,6 +80,24 @@ public class BalanceChart extends LineChart<Number, Number> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Stream<Entry> getTransactionEntries(WalletTransactionsEntry walletTransactionsEntry) {
|
||||||
|
int total = walletTransactionsEntry.getChildren().size();
|
||||||
|
if(walletTransactionsEntry.getChildren().size() <= MAX_VALUES) {
|
||||||
|
return walletTransactionsEntry.getChildren().stream();
|
||||||
|
}
|
||||||
|
|
||||||
|
int bucketSize = total / MAX_VALUES;
|
||||||
|
List<List<Entry>> buckets = Lists.partition(walletTransactionsEntry.getChildren(), bucketSize);
|
||||||
|
List<Entry> reducedEntries = new ArrayList<>(MAX_VALUES);
|
||||||
|
for(List<Entry> bucket : buckets) {
|
||||||
|
long max = bucket.stream().mapToLong(entry -> Math.abs(entry.getValue())).max().orElse(0);
|
||||||
|
Entry bucketEntry = bucket.stream().filter(entry -> entry.getValue() == max || entry.getValue() == -max).findFirst().orElseThrow();
|
||||||
|
reducedEntries.add(bucketEntry);
|
||||||
|
}
|
||||||
|
|
||||||
|
return reducedEntries.stream();
|
||||||
|
}
|
||||||
|
|
||||||
public void select(TransactionEntry transactionEntry) {
|
public void select(TransactionEntry transactionEntry) {
|
||||||
Set<Node> selectedSymbols = lookupAll(".chart-line-symbol.selected");
|
Set<Node> selectedSymbols = lookupAll(".chart-line-symbol.selected");
|
||||||
for(Node selectedSymbol : selectedSymbols) {
|
for(Node selectedSymbol : selectedSymbols) {
|
||||||
|
|
|
@ -158,6 +158,7 @@ public abstract class FileImportPane extends TitledDescriptionPane {
|
||||||
try {
|
try {
|
||||||
importFile(importer.getName(), null, null);
|
importFile(importer.getName(), null, null);
|
||||||
} catch(ImportException e) {
|
} catch(ImportException e) {
|
||||||
|
log.error("Error importing QR", e);
|
||||||
setError("Import Error", e.getMessage());
|
setError("Import Error", e.getMessage());
|
||||||
}
|
}
|
||||||
} else if(result.outputDescriptor != null) {
|
} else if(result.outputDescriptor != null) {
|
||||||
|
@ -165,6 +166,7 @@ public abstract class FileImportPane extends TitledDescriptionPane {
|
||||||
try {
|
try {
|
||||||
importFile(importer.getName(), null, null);
|
importFile(importer.getName(), null, null);
|
||||||
} catch(ImportException e) {
|
} catch(ImportException e) {
|
||||||
|
log.error("Error importing QR", e);
|
||||||
setError("Import Error", e.getMessage());
|
setError("Import Error", e.getMessage());
|
||||||
}
|
}
|
||||||
} else if(result.payload != null) {
|
} else if(result.payload != null) {
|
||||||
|
|
|
@ -78,7 +78,6 @@ public class CaravanMultisig implements WalletImport, WalletExport {
|
||||||
|
|
||||||
return wallet;
|
return wallet;
|
||||||
} catch(Exception e) {
|
} catch(Exception e) {
|
||||||
log.error("Error importing " + getName() + " wallet", e);
|
|
||||||
throw new ImportException("Error importing " + getName() + " wallet", e);
|
throw new ImportException("Error importing " + getName() + " wallet", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,7 +57,6 @@ public class CoboVaultSinglesig implements KeystoreFileImport, WalletImport {
|
||||||
|
|
||||||
return keystore;
|
return keystore;
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
log.error("Error getting " + getName() + " keystore", e);
|
|
||||||
throw new ImportException("Error getting " + getName() + " keystore", e);
|
throw new ImportException("Error getting " + getName() + " keystore", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -154,7 +154,6 @@ public class ColdcardMultisig implements WalletImport, KeystoreFileImport, Walle
|
||||||
|
|
||||||
return wallet;
|
return wallet;
|
||||||
} catch(Exception e) {
|
} catch(Exception e) {
|
||||||
log.error("Error importing " + getName() + " wallet", e);
|
|
||||||
throw new ImportException("Error importing " + getName() + " wallet", e);
|
throw new ImportException("Error importing " + getName() + " wallet", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -86,7 +86,6 @@ public class ColdcardSinglesig implements KeystoreFileImport, WalletImport {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
log.error("Error getting " + getName() + " keystore", e);
|
|
||||||
throw new ImportException("Error getting " + getName() + " keystore", e);
|
throw new ImportException("Error getting " + getName() + " keystore", e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -264,7 +264,6 @@ public class Electrum implements KeystoreFileImport, WalletImport, WalletExport
|
||||||
|
|
||||||
return wallet;
|
return wallet;
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
log.error("Error importing Electrum Wallet", e);
|
|
||||||
throw new ImportException("Error importing Electrum Wallet", e);
|
throw new ImportException("Error importing Electrum Wallet", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,10 +62,8 @@ public class KeystoneSinglesig implements KeystoreFileImport, WalletImport {
|
||||||
|
|
||||||
return keystore;
|
return keystore;
|
||||||
} catch (IllegalArgumentException e) {
|
} catch (IllegalArgumentException e) {
|
||||||
log.error("Error getting " + getName() + " keystore - not an output descriptor");
|
throw new ImportException("Error getting " + getName() + " keystore - not an output descriptor", e);
|
||||||
throw new ImportException("Error getting " + getName() + " keystore", e);
|
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
log.error("Error getting " + getName() + " keystore", e);
|
|
||||||
throw new ImportException("Error getting " + getName() + " keystore", e);
|
throw new ImportException("Error getting " + getName() + " keystore", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -121,7 +121,6 @@ public class Sparrow implements WalletImport, WalletExport {
|
||||||
|
|
||||||
return wallet;
|
return wallet;
|
||||||
} catch(IOException | StorageException e) {
|
} catch(IOException | StorageException e) {
|
||||||
log.error("Error importing Sparrow wallet", e);
|
|
||||||
throw new ImportException("Error importing Sparrow wallet", e);
|
throw new ImportException("Error importing Sparrow wallet", e);
|
||||||
} finally {
|
} finally {
|
||||||
if(storage != null) {
|
if(storage != null) {
|
||||||
|
|
|
@ -35,7 +35,6 @@ public class SpecterDIY implements KeystoreFileImport, WalletExport {
|
||||||
|
|
||||||
return keystore;
|
return keystore;
|
||||||
} catch(IOException e) {
|
} catch(IOException e) {
|
||||||
log.error("Error getting " + getName() + " keystore", e);
|
|
||||||
throw new ImportException("Error getting " + getName() + " keystore", e);
|
throw new ImportException("Error getting " + getName() + " keystore", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -93,7 +93,6 @@ public class SpecterDesktop implements WalletImport, WalletExport {
|
||||||
return wallet;
|
return wallet;
|
||||||
}
|
}
|
||||||
} catch(Exception e) {
|
} catch(Exception e) {
|
||||||
log.error("Error importing " + getName() + " wallet", e);
|
|
||||||
throw new ImportException("Error importing " + getName() + " wallet", e);
|
throw new ImportException("Error importing " + getName() + " wallet", e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -534,6 +534,11 @@ public class Storage {
|
||||||
private final Storage storage;
|
private final Storage storage;
|
||||||
private final SecureString password;
|
private final SecureString password;
|
||||||
|
|
||||||
|
public LoadWalletService(Storage storage) {
|
||||||
|
this.storage = storage;
|
||||||
|
this.password = null;
|
||||||
|
}
|
||||||
|
|
||||||
public LoadWalletService(Storage storage, SecureString password) {
|
public LoadWalletService(Storage storage, SecureString password) {
|
||||||
this.storage = storage;
|
this.storage = storage;
|
||||||
this.password = password;
|
this.password = password;
|
||||||
|
@ -543,8 +548,15 @@ public class Storage {
|
||||||
protected Task<WalletBackupAndKey> createTask() {
|
protected Task<WalletBackupAndKey> createTask() {
|
||||||
return new Task<>() {
|
return new Task<>() {
|
||||||
protected WalletBackupAndKey call() throws IOException, StorageException {
|
protected WalletBackupAndKey call() throws IOException, StorageException {
|
||||||
WalletBackupAndKey walletBackupAndKey = storage.loadEncryptedWallet(password);
|
WalletBackupAndKey walletBackupAndKey;
|
||||||
password.clear();
|
|
||||||
|
if(password != null) {
|
||||||
|
walletBackupAndKey = storage.loadEncryptedWallet(password);
|
||||||
|
password.clear();
|
||||||
|
} else {
|
||||||
|
walletBackupAndKey = storage.loadUnencryptedWallet();
|
||||||
|
}
|
||||||
|
|
||||||
return walletBackupAndKey;
|
return walletBackupAndKey;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -578,7 +578,7 @@ public class DbPersistence implements Persistence {
|
||||||
if(updateExecutor != null) {
|
if(updateExecutor != null) {
|
||||||
updateExecutor.shutdown();
|
updateExecutor.shutdown();
|
||||||
try {
|
try {
|
||||||
if(!updateExecutor.awaitTermination(10, TimeUnit.SECONDS)) {
|
if(!updateExecutor.awaitTermination(1, TimeUnit.MINUTES)) {
|
||||||
updateExecutor.shutdownNow();
|
updateExecutor.shutdownNow();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -166,7 +166,7 @@ public class ElectrumServer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void addCalculatedScriptHashes(Wallet wallet) {
|
private static void addCalculatedScriptHashes(Wallet wallet) {
|
||||||
getCalculatedScriptHashes(wallet).forEach(retrievedScriptHashes::putIfAbsent);
|
getCalculatedScriptHashes(wallet).forEach(retrievedScriptHashes::putIfAbsent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1248,8 +1248,12 @@ public class ElectrumServer {
|
||||||
protected Task<Boolean> createTask() {
|
protected Task<Boolean> createTask() {
|
||||||
return new Task<>() {
|
return new Task<>() {
|
||||||
protected Boolean call() throws ServerException {
|
protected Boolean call() throws ServerException {
|
||||||
walletSynchronizeLocks.putIfAbsent(wallet, new Object());
|
boolean initial = (walletSynchronizeLocks.putIfAbsent(wallet, new Object()) == null);
|
||||||
synchronized(walletSynchronizeLocks.get(wallet)) {
|
synchronized(walletSynchronizeLocks.get(wallet)) {
|
||||||
|
if(initial) {
|
||||||
|
addCalculatedScriptHashes(wallet);
|
||||||
|
}
|
||||||
|
|
||||||
if(isConnected()) {
|
if(isConnected()) {
|
||||||
ElectrumServer electrumServer = new ElectrumServer();
|
ElectrumServer electrumServer = new ElectrumServer();
|
||||||
Map<String, String> previousScriptHashes = getCalculatedScriptHashes(wallet);
|
Map<String, String> previousScriptHashes = getCalculatedScriptHashes(wallet);
|
||||||
|
|
|
@ -38,7 +38,18 @@ public class TransactionEntry extends Entry implements Comparable<TransactionEnt
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
setConfirmations(calculateConfirmations());
|
confirmations = new IntegerPropertyBase(calculateConfirmations()) {
|
||||||
|
@Override
|
||||||
|
public Object getBean() {
|
||||||
|
return TransactionEntry.this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() {
|
||||||
|
return "confirmations";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
if(isFullyConfirming()) {
|
if(isFullyConfirming()) {
|
||||||
EventManager.get().register(this);
|
EventManager.get().register(this);
|
||||||
}
|
}
|
||||||
|
@ -174,33 +185,17 @@ public class TransactionEntry extends Entry implements Comparable<TransactionEnt
|
||||||
/**
|
/**
|
||||||
* Defines the number of confirmations
|
* Defines the number of confirmations
|
||||||
*/
|
*/
|
||||||
private IntegerProperty confirmations;
|
private final IntegerProperty confirmations;
|
||||||
|
|
||||||
public final void setConfirmations(int value) {
|
public final void setConfirmations(int value) {
|
||||||
if(confirmations != null || value != 0) {
|
confirmations.set(value);
|
||||||
confirmationsProperty().set(value);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public final int getConfirmations() {
|
public final int getConfirmations() {
|
||||||
return confirmations == null ? 0 : confirmations.get();
|
return confirmations.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
public final IntegerProperty confirmationsProperty() {
|
public final IntegerProperty confirmationsProperty() {
|
||||||
if(confirmations == null) {
|
|
||||||
confirmations = new IntegerPropertyBase(0) {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Object getBean() {
|
|
||||||
return TransactionEntry.this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getName() {
|
|
||||||
return "confirmations";
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
return confirmations;
|
return confirmations;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,6 @@ import com.google.common.eventbus.Subscribe;
|
||||||
import com.sparrowwallet.drongo.BitcoinUnit;
|
import com.sparrowwallet.drongo.BitcoinUnit;
|
||||||
import com.sparrowwallet.drongo.protocol.Transaction;
|
import com.sparrowwallet.drongo.protocol.Transaction;
|
||||||
import com.sparrowwallet.sparrow.AppServices;
|
import com.sparrowwallet.sparrow.AppServices;
|
||||||
import com.sparrowwallet.sparrow.CurrencyRate;
|
|
||||||
import com.sparrowwallet.sparrow.EventManager;
|
import com.sparrowwallet.sparrow.EventManager;
|
||||||
import com.sparrowwallet.sparrow.control.*;
|
import com.sparrowwallet.sparrow.control.*;
|
||||||
import com.sparrowwallet.sparrow.event.*;
|
import com.sparrowwallet.sparrow.event.*;
|
||||||
|
|
|
@ -59,7 +59,6 @@ public class WalletForm {
|
||||||
savedPastWallet = backupWallet;
|
savedPastWallet = backupWallet;
|
||||||
|
|
||||||
if(refreshHistory && wallet.isValid()) {
|
if(refreshHistory && wallet.isValid()) {
|
||||||
ElectrumServer.addCalculatedScriptHashes(wallet);
|
|
||||||
refreshHistory(AppServices.getCurrentBlockHeight(), backupWallet);
|
refreshHistory(AppServices.getCurrentBlockHeight(), backupWallet);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -252,11 +251,13 @@ public class WalletForm {
|
||||||
}
|
}
|
||||||
|
|
||||||
private List<WalletNode> getHistoryChangedNodes(Set<WalletNode> previousNodes, Set<WalletNode> currentNodes) {
|
private List<WalletNode> getHistoryChangedNodes(Set<WalletNode> previousNodes, Set<WalletNode> currentNodes) {
|
||||||
|
Map<WalletNode, WalletNode> previousNodeMap = new HashMap<>(previousNodes.size());
|
||||||
|
previousNodes.forEach(walletNode -> previousNodeMap.put(walletNode, walletNode));
|
||||||
|
|
||||||
List<WalletNode> changedNodes = new ArrayList<>();
|
List<WalletNode> changedNodes = new ArrayList<>();
|
||||||
for(WalletNode currentNode : currentNodes) {
|
for(WalletNode currentNode : currentNodes) {
|
||||||
Optional<WalletNode> optPreviousNode = previousNodes.stream().filter(node -> node.equals(currentNode)).findFirst();
|
WalletNode previousNode = previousNodeMap.get(currentNode);
|
||||||
if(optPreviousNode.isPresent()) {
|
if(previousNode != null) {
|
||||||
WalletNode previousNode = optPreviousNode.get();
|
|
||||||
if(!currentNode.getTransactionOutputs().equals(previousNode.getTransactionOutputs())) {
|
if(!currentNode.getTransactionOutputs().equals(previousNode.getTransactionOutputs())) {
|
||||||
changedNodes.add(currentNode);
|
changedNodes.add(currentNode);
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,7 +22,7 @@ public class WalletTransactionsEntry extends Entry {
|
||||||
|
|
||||||
public WalletTransactionsEntry(Wallet wallet) {
|
public WalletTransactionsEntry(Wallet wallet) {
|
||||||
super(wallet, wallet.getName(), getWalletTransactions(wallet).stream().map(WalletTransaction::getTransactionEntry).collect(Collectors.toList()));
|
super(wallet, wallet.getName(), getWalletTransactions(wallet).stream().map(WalletTransaction::getTransactionEntry).collect(Collectors.toList()));
|
||||||
calculateBalances();
|
calculateBalances(false); //No need to resort
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -30,12 +30,14 @@ public class WalletTransactionsEntry extends Entry {
|
||||||
return getBalance();
|
return getBalance();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void calculateBalances() {
|
private void calculateBalances(boolean resort) {
|
||||||
long balance = 0L;
|
long balance = 0L;
|
||||||
long mempoolBalance = 0L;
|
long mempoolBalance = 0L;
|
||||||
|
|
||||||
//Note transaction entries must be in ascending order. This sorting is ultimately done according to BlockTransactions' comparator
|
if(resort) {
|
||||||
getChildren().sort(Comparator.comparing(TransactionEntry.class::cast));
|
//Note transaction entries must be in ascending order. This sorting is ultimately done according to BlockTransactions' comparator
|
||||||
|
getChildren().sort(Comparator.comparing(TransactionEntry.class::cast));
|
||||||
|
}
|
||||||
|
|
||||||
for(Entry entry : getChildren()) {
|
for(Entry entry : getChildren()) {
|
||||||
TransactionEntry transactionEntry = (TransactionEntry)entry;
|
TransactionEntry transactionEntry = (TransactionEntry)entry;
|
||||||
|
@ -66,7 +68,7 @@ public class WalletTransactionsEntry extends Entry {
|
||||||
entriesRemoved.removeAll(current);
|
entriesRemoved.removeAll(current);
|
||||||
getChildren().removeAll(entriesRemoved);
|
getChildren().removeAll(entriesRemoved);
|
||||||
|
|
||||||
calculateBalances();
|
calculateBalances(true);
|
||||||
|
|
||||||
Map<HashIndex, BlockTransactionHashIndex> walletTxos = getWallet().getWalletTxos().entrySet().stream()
|
Map<HashIndex, BlockTransactionHashIndex> walletTxos = getWallet().getWalletTxos().entrySet().stream()
|
||||||
.collect(Collectors.toUnmodifiableMap(entry -> new HashIndex(entry.getKey().getHash(), entry.getKey().getIndex()), Map.Entry::getKey));
|
.collect(Collectors.toUnmodifiableMap(entry -> new HashIndex(entry.getKey().getHash(), entry.getKey().getIndex()), Map.Entry::getKey));
|
||||||
|
@ -86,12 +88,14 @@ public class WalletTransactionsEntry extends Entry {
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Collection<WalletTransaction> getWalletTransactions(Wallet wallet) {
|
private static Collection<WalletTransaction> getWalletTransactions(Wallet wallet) {
|
||||||
Map<BlockTransaction, WalletTransaction> walletTransactionMap = new TreeMap<>();
|
Map<BlockTransaction, WalletTransaction> walletTransactionMap = new HashMap<>(wallet.getTransactions().size());
|
||||||
|
|
||||||
getWalletTransactions(wallet, walletTransactionMap, wallet.getNode(KeyPurpose.RECEIVE));
|
getWalletTransactions(wallet, walletTransactionMap, wallet.getNode(KeyPurpose.RECEIVE));
|
||||||
getWalletTransactions(wallet, walletTransactionMap, wallet.getNode(KeyPurpose.CHANGE));
|
getWalletTransactions(wallet, walletTransactionMap, wallet.getNode(KeyPurpose.CHANGE));
|
||||||
|
|
||||||
return new ArrayList<>(walletTransactionMap.values());
|
List<WalletTransaction> walletTransactions = new ArrayList<>(walletTransactionMap.values());
|
||||||
|
Collections.sort(walletTransactions);
|
||||||
|
return walletTransactions;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void getWalletTransactions(Wallet wallet, Map<BlockTransaction, WalletTransaction> walletTransactionMap, WalletNode purposeNode) {
|
private static void getWalletTransactions(Wallet wallet, Map<BlockTransaction, WalletTransaction> walletTransactionMap, WalletNode purposeNode) {
|
||||||
|
@ -191,7 +195,7 @@ public class WalletTransactionsEntry extends Entry {
|
||||||
return mempoolBalance;
|
return mempoolBalance;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class WalletTransaction {
|
private static class WalletTransaction implements Comparable<WalletTransaction> {
|
||||||
private final Wallet wallet;
|
private final Wallet wallet;
|
||||||
private final BlockTransaction blockTransaction;
|
private final BlockTransaction blockTransaction;
|
||||||
private final Map<BlockTransactionHashIndex, KeyPurpose> incoming = new TreeMap<>();
|
private final Map<BlockTransactionHashIndex, KeyPurpose> incoming = new TreeMap<>();
|
||||||
|
@ -205,5 +209,33 @@ public class WalletTransactionsEntry extends Entry {
|
||||||
public TransactionEntry getTransactionEntry() {
|
public TransactionEntry getTransactionEntry() {
|
||||||
return new TransactionEntry(wallet, blockTransaction, incoming, outgoing);
|
return new TransactionEntry(wallet, blockTransaction, incoming, outgoing);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public long getValue() {
|
||||||
|
long value = 0L;
|
||||||
|
for(BlockTransactionHashIndex in : incoming.keySet()) {
|
||||||
|
value += in.getValue();
|
||||||
|
}
|
||||||
|
for(BlockTransactionHashIndex out : outgoing.keySet()) {
|
||||||
|
value -= out.getValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int compareTo(WalletTransactionsEntry.WalletTransaction other) {
|
||||||
|
//This comparison must be identical to that of TransactionEntry so we can avoid a resort calculating balances when creating WalletTransactionsEntry
|
||||||
|
int blockOrder = blockTransaction.compareBlockOrder(other.blockTransaction);
|
||||||
|
if(blockOrder != 0) {
|
||||||
|
return blockOrder;
|
||||||
|
}
|
||||||
|
|
||||||
|
int valueOrder = Long.compare(other.getValue(), getValue());
|
||||||
|
if(valueOrder != 0) {
|
||||||
|
return valueOrder;
|
||||||
|
}
|
||||||
|
|
||||||
|
return blockTransaction.compareTo(other.blockTransaction);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue