diff --git a/core/src/main/java/bisq/core/btc/wallet/WalletService.java b/core/src/main/java/bisq/core/btc/wallet/WalletService.java index 8cd19d7dcb3..f2c3d251c6d 100644 --- a/core/src/main/java/bisq/core/btc/wallet/WalletService.java +++ b/core/src/main/java/bisq/core/btc/wallet/WalletService.java @@ -291,7 +291,8 @@ public static void signTx(Wallet wallet, continue; } if (!connectedOutput.isMine(wallet)) { - log.error("connectedOutput is not mine"); + log.info("ConnectedOutput is not mine. This can be the case for BSQ transactions where the " + + "input gets signed by the other wallet. connectedOutput={}", connectedOutput); continue; } diff --git a/core/src/main/java/bisq/core/dao/DaoFacade.java b/core/src/main/java/bisq/core/dao/DaoFacade.java index b4729179d2a..a3ccb9377e6 100644 --- a/core/src/main/java/bisq/core/dao/DaoFacade.java +++ b/core/src/main/java/bisq/core/dao/DaoFacade.java @@ -57,7 +57,6 @@ import bisq.core.dao.state.DaoStateListener; import bisq.core.dao.state.DaoStateService; import bisq.core.dao.state.model.blockchain.BaseTx; -import bisq.core.dao.state.model.blockchain.BaseTxOutput; import bisq.core.dao.state.model.blockchain.Block; import bisq.core.dao.state.model.blockchain.Tx; import bisq.core.dao.state.model.blockchain.TxOutput; @@ -531,10 +530,13 @@ public Optional getBlockAtHeight(int chainHeight) { return daoStateService.getBlockAtHeight(chainHeight); } - public boolean daoStateNeedsRebuilding() { - return daoStateMonitoringService.isInConflictWithSeedNode() || daoStateMonitoringService.isDaoStateBlockChainNotConnecting(); + public boolean isDaoStateReadyAndInSync() { + return daoStateService.isParseBlockChainComplete() && + !daoStateMonitoringService.isInConflictWithSeedNode() && + !daoStateMonitoringService.isDaoStateBlockChainNotConnecting(); } + /////////////////////////////////////////////////////////////////////////////////////////// // Use case: Bonding /////////////////////////////////////////////////////////////////////////////////////////// @@ -578,10 +580,6 @@ public long getTotalAmountOfConfiscatedTxOutputs() { return daoStateService.getTotalAmountOfConfiscatedTxOutputs(); } - public long getTotalAmountOfInvalidatedBsq() { - return daoStateService.getTotalAmountOfInvalidatedBsq(); - } - // Contains burned fee and invalidated bsq due invalid txs public long getTotalAmountOfBurntBsq() { return daoStateService.getTotalAmountOfBurntBsq(); @@ -595,11 +593,6 @@ public List getIrregularTxs() { return daoStateService.getIrregularTxs(); } - public long getTotalAmountOfUnspentTxOutputs() { - // Does not consider confiscated outputs (they stay as utxo) - return daoStateService.getUnspentTxOutputMap().values().stream().mapToLong(BaseTxOutput::getValue).sum(); - } - public Optional getLockTime(String txId) { return daoStateService.getLockTime(txId); } diff --git a/core/src/main/java/bisq/core/dao/burningman/DelayedPayoutTxReceiverService.java b/core/src/main/java/bisq/core/dao/burningman/DelayedPayoutTxReceiverService.java index 661189daf8b..4d2df54ad69 100644 --- a/core/src/main/java/bisq/core/dao/burningman/DelayedPayoutTxReceiverService.java +++ b/core/src/main/java/bisq/core/dao/burningman/DelayedPayoutTxReceiverService.java @@ -78,7 +78,6 @@ public static boolean isHotfixActivated() { // spike when opening arbitration. private static final long DPT_MIN_TX_FEE_RATE = 10; - private final DaoStateService daoStateService; private final BurningManService burningManService; private int currentChainHeight; @@ -127,12 +126,6 @@ public List> getReceivers(int burningManSelectionHeight, burningManService.getActiveBurningManCandidates(burningManSelectionHeight) : burningManService.getBurningManCandidatesByName(burningManSelectionHeight).values(); - - if (burningManCandidates.isEmpty()) { - // If there are no compensation requests (e.g. at dev testing) we fall back to the legacy BM - return List.of(new Tuple2<>(inputAmount, burningManService.getLegacyBurningManAddress(burningManSelectionHeight))); - } - // We need to use the same txFeePerVbyte value for both traders. // We use the tradeTxFee value which is calculated from the average of taker fee tx size and deposit tx size. // Otherwise, we would need to sync the fee rate of both traders. @@ -146,12 +139,19 @@ public List> getReceivers(int burningManSelectionHeight, // Smallest tx size is 246. With additional change output we add 32. To be safe we use the largest expected size. double txSize = 278; long txFeePerVbyte = Math.max(DPT_MIN_TX_FEE_RATE, Math.round(tradeTxFee / txSize)); + + if (burningManCandidates.isEmpty()) { + // If there are no compensation requests (e.g. at dev testing) we fall back to the legacy BM + long spendableAmount = getSpendableAmount(1, inputAmount, txFeePerVbyte); + return List.of(new Tuple2<>(spendableAmount, burningManService.getLegacyBurningManAddress(burningManSelectionHeight))); + } + long spendableAmount = getSpendableAmount(burningManCandidates.size(), inputAmount, txFeePerVbyte); // We only use outputs > 1000 sat or at least 2 times the cost for the output (32 bytes). // If we remove outputs it will be spent as miner fee. long minOutputAmount = Math.max(DPT_MIN_OUTPUT_AMOUNT, txFeePerVbyte * 32 * 2); // Sanity check that max share of a non-legacy BM is 20% over MAX_BURN_SHARE (taking into account potential increase due adjustment) - long maxOutputAmount = Math.round(inputAmount * (BurningManService.MAX_BURN_SHARE * 1.2)); + long maxOutputAmount = Math.round(spendableAmount * (BurningManService.MAX_BURN_SHARE * 1.2)); // We accumulate small amounts which gets filtered out and subtract it from 1 to get an adjustment factor // used later to be applied to the remaining burningmen share. double adjustment = 1 - burningManCandidates.stream() diff --git a/core/src/main/java/bisq/core/dao/node/full/network/FullNodeNetworkService.java b/core/src/main/java/bisq/core/dao/node/full/network/FullNodeNetworkService.java index 20583433cb5..430426dba5c 100644 --- a/core/src/main/java/bisq/core/dao/node/full/network/FullNodeNetworkService.java +++ b/core/src/main/java/bisq/core/dao/node/full/network/FullNodeNetworkService.java @@ -206,7 +206,7 @@ public void onFault(String errorMessage, @Nullable Connection connection) { } private void handleRepublishGovernanceDataRequest() { - log.warn("We received a RepublishGovernanceDataRequest and re-published all proposalPayloads and " + + log.info("We received a RepublishGovernanceDataRequest and re-published all proposalPayloads and " + "blindVotePayloads to the P2P network."); missingDataRequestService.reRepublishAllGovernanceData(); } diff --git a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/BuyerAsMakerProtocol.java b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/BuyerAsMakerProtocol.java index 8f161351aa9..83bda8ac8bf 100644 --- a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/BuyerAsMakerProtocol.java +++ b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/BuyerAsMakerProtocol.java @@ -25,6 +25,7 @@ import bisq.core.trade.protocol.bisq_v1.messages.InputsForDepositTxRequest; import bisq.core.trade.protocol.bisq_v1.messages.PayoutTxPublishedMessage; import bisq.core.trade.protocol.bisq_v1.tasks.ApplyFilter; +import bisq.core.trade.protocol.bisq_v1.tasks.CheckIfDaoStateIsInSync; import bisq.core.trade.protocol.bisq_v1.tasks.TradeTask; import bisq.core.trade.protocol.bisq_v1.tasks.buyer.BuyerFinalizesDelayedPayoutTx; import bisq.core.trade.protocol.bisq_v1.tasks.buyer.BuyerProcessDelayedPayoutTxSignatureRequest; @@ -71,6 +72,7 @@ public void handleTakeOfferRequest(InputsForDepositTxRequest message, .with(message) .from(peer)) .setup(tasks( + CheckIfDaoStateIsInSync.class, MakerProcessesInputsForDepositTxRequest.class, ApplyFilter.class, getVerifyPeersFeePaymentClass(), diff --git a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/BuyerAsTakerProtocol.java b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/BuyerAsTakerProtocol.java index 8a1d927eef4..d7b9413db59 100644 --- a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/BuyerAsTakerProtocol.java +++ b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/BuyerAsTakerProtocol.java @@ -27,6 +27,7 @@ import bisq.core.trade.protocol.bisq_v1.messages.InputsForDepositTxResponse; import bisq.core.trade.protocol.bisq_v1.messages.PayoutTxPublishedMessage; import bisq.core.trade.protocol.bisq_v1.tasks.ApplyFilter; +import bisq.core.trade.protocol.bisq_v1.tasks.CheckIfDaoStateIsInSync; import bisq.core.trade.protocol.bisq_v1.tasks.TradeTask; import bisq.core.trade.protocol.bisq_v1.tasks.buyer.BuyerFinalizesDelayedPayoutTx; import bisq.core.trade.protocol.bisq_v1.tasks.buyer.BuyerProcessDelayedPayoutTxSignatureRequest; @@ -77,6 +78,7 @@ public void onTakeOffer() { expect(phase(Trade.Phase.INIT) .with(TakerEvent.TAKE_OFFER)) .setup(tasks( + CheckIfDaoStateIsInSync.class, ApplyFilter.class, getVerifyPeersFeePaymentClass(), CreateTakerFeeTx.class, diff --git a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/SellerAsMakerProtocol.java b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/SellerAsMakerProtocol.java index 1da30e3e0d8..c88e7de0f87 100644 --- a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/SellerAsMakerProtocol.java +++ b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/SellerAsMakerProtocol.java @@ -27,6 +27,7 @@ import bisq.core.trade.protocol.bisq_v1.messages.DepositTxMessage; import bisq.core.trade.protocol.bisq_v1.messages.InputsForDepositTxRequest; import bisq.core.trade.protocol.bisq_v1.tasks.ApplyFilter; +import bisq.core.trade.protocol.bisq_v1.tasks.CheckIfDaoStateIsInSync; import bisq.core.trade.protocol.bisq_v1.tasks.TradeTask; import bisq.core.trade.protocol.bisq_v1.tasks.maker.MakerCreateAndSignContract; import bisq.core.trade.protocol.bisq_v1.tasks.maker.MakerProcessesInputsForDepositTxRequest; @@ -73,6 +74,7 @@ public void handleTakeOfferRequest(InputsForDepositTxRequest message, .with(message) .from(peer)) .setup(tasks( + CheckIfDaoStateIsInSync.class, MaybeCreateSubAccount.class, MakerProcessesInputsForDepositTxRequest.class, ApplyFilter.class, diff --git a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/SellerAsTakerProtocol.java b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/SellerAsTakerProtocol.java index fae836be769..42804e7cd3d 100644 --- a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/SellerAsTakerProtocol.java +++ b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/SellerAsTakerProtocol.java @@ -26,6 +26,7 @@ import bisq.core.trade.protocol.bisq_v1.messages.DelayedPayoutTxSignatureResponse; import bisq.core.trade.protocol.bisq_v1.messages.InputsForDepositTxResponse; import bisq.core.trade.protocol.bisq_v1.tasks.ApplyFilter; +import bisq.core.trade.protocol.bisq_v1.tasks.CheckIfDaoStateIsInSync; import bisq.core.trade.protocol.bisq_v1.tasks.TradeTask; import bisq.core.trade.protocol.bisq_v1.tasks.seller.MaybeCreateSubAccount; import bisq.core.trade.protocol.bisq_v1.tasks.seller.SellerCreatesDelayedPayoutTx; @@ -73,6 +74,7 @@ public void onTakeOffer() { .with(TakerEvent.TAKE_OFFER) .from(trade.getTradingPeerNodeAddress())) .setup(tasks( + CheckIfDaoStateIsInSync.class, MaybeCreateSubAccount.class, ApplyFilter.class, getVerifyPeersFeePaymentClass(), diff --git a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/CheckIfDaoStateIsInSync.java b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/CheckIfDaoStateIsInSync.java new file mode 100644 index 00000000000..180f9dca9b5 --- /dev/null +++ b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/CheckIfDaoStateIsInSync.java @@ -0,0 +1,46 @@ +/* + * This file is part of Bisq. + * + * Bisq is free software: you can redistribute it and/or modify it + * under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or (at + * your option) any later version. + * + * Bisq is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public + * License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with Bisq. If not, see . + */ + +package bisq.core.trade.protocol.bisq_v1.tasks; + +import bisq.core.trade.model.bisq_v1.Trade; + +import bisq.common.taskrunner.TaskRunner; + +import lombok.extern.slf4j.Slf4j; + +import static com.google.common.base.Preconditions.checkArgument; + +@Slf4j +public class CheckIfDaoStateIsInSync extends TradeTask { + public CheckIfDaoStateIsInSync(TaskRunner taskHandler, Trade trade) { + super(taskHandler, trade); + } + + @Override + protected void run() { + try { + runInterceptHook(); + + checkArgument(processModel.getDaoFacade().isDaoStateReadyAndInSync(), "DAO state is not in sync with seed nodes"); + complete(); + } catch (Throwable t) { + failed(t); + } + } +} + diff --git a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/buyer/BuyerVerifiesFinalDelayedPayoutTx.java b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/buyer/BuyerVerifiesFinalDelayedPayoutTx.java index ba05dacec97..3bf644b1fdb 100644 --- a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/buyer/BuyerVerifiesFinalDelayedPayoutTx.java +++ b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/buyer/BuyerVerifiesFinalDelayedPayoutTx.java @@ -32,7 +32,6 @@ import lombok.extern.slf4j.Slf4j; -import static com.google.common.base.Preconditions.checkArgument; import static com.google.common.base.Preconditions.checkNotNull; @Slf4j @@ -74,8 +73,19 @@ protected void run() { depositTx, delayedPayoutTxReceivers, lockTime); - checkArgument(buyersDelayedPayoutTx.getTxId().equals(finalDelayedPayoutTx.getTxId()), - "TxIds of buyersDelayedPayoutTx and finalDelayedPayoutTx must be the same"); + + if (!buyersDelayedPayoutTx.getTxId().equals(finalDelayedPayoutTx.getTxId())) { + String errorMsg = "TxIds of buyersDelayedPayoutTx and finalDelayedPayoutTx must be the same."; + log.error("{} \nbuyersDelayedPayoutTx={}, \nfinalDelayedPayoutTx={}, " + + "\nBtcWalletService.chainHeight={}, " + + "\nDaoState.chainHeight={}, " + + "\nisDaoStateIsInSync={}", + errorMsg, buyersDelayedPayoutTx, finalDelayedPayoutTx, + processModel.getBtcWalletService().getBestChainHeight(), + processModel.getDaoFacade().getChainHeight(), + processModel.getDaoFacade().isDaoStateReadyAndInSync()); + throw new IllegalArgumentException(errorMsg); + } } complete(); diff --git a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/buyer/BuyerVerifiesPreparedDelayedPayoutTx.java b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/buyer/BuyerVerifiesPreparedDelayedPayoutTx.java index 9f7e9c0d5fe..6fe2321d072 100644 --- a/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/buyer/BuyerVerifiesPreparedDelayedPayoutTx.java +++ b/core/src/main/java/bisq/core/trade/protocol/bisq_v1/tasks/buyer/BuyerVerifiesPreparedDelayedPayoutTx.java @@ -32,7 +32,6 @@ import lombok.extern.slf4j.Slf4j; -import static com.google.common.base.Preconditions.checkArgument; import static com.google.common.base.Preconditions.checkNotNull; @Slf4j @@ -66,8 +65,18 @@ protected void run() { preparedDepositTx, delayedPayoutTxReceivers, lockTime); - checkArgument(buyersPreparedDelayedPayoutTx.getTxId().equals(sellersPreparedDelayedPayoutTx.getTxId()), - "TxIds of buyersPreparedDelayedPayoutTx and sellersPreparedDelayedPayoutTx must be the same"); + if (!buyersPreparedDelayedPayoutTx.getTxId().equals(sellersPreparedDelayedPayoutTx.getTxId())) { + String errorMsg = "TxIds of buyersPreparedDelayedPayoutTx and sellersPreparedDelayedPayoutTx must be the same."; + log.error("{} \nbuyersPreparedDelayedPayoutTx={}, \nsellersPreparedDelayedPayoutTx={}, " + + "\nBtcWalletService.chainHeight={}, " + + "\nDaoState.chainHeight={}, " + + "\nisDaoStateIsInSync={}", + errorMsg, buyersPreparedDelayedPayoutTx, sellersPreparedDelayedPayoutTx, + processModel.getBtcWalletService().getBestChainHeight(), + processModel.getDaoFacade().getChainHeight(), + processModel.getDaoFacade().isDaoStateReadyAndInSync()); + throw new IllegalArgumentException(errorMsg); + } } // If the deposit tx is non-malleable, we already know its final ID, so should check that now diff --git a/desktop/src/main/java/bisq/desktop/main/MainViewModel.java b/desktop/src/main/java/bisq/desktop/main/MainViewModel.java index 3eb0283a320..4c132613c37 100644 --- a/desktop/src/main/java/bisq/desktop/main/MainViewModel.java +++ b/desktop/src/main/java/bisq/desktop/main/MainViewModel.java @@ -310,7 +310,7 @@ public void onSetupComplete() { setupClockWatcherPopup(); marketPricePresentation.setup(); - daoPresentation.setup(); + daoPresentation.init(); accountPresentation.setup(); settingsPresentation.setup(); @@ -505,7 +505,7 @@ private void setupHandlers() { .show()); bisqSetup.getBtcSyncProgress().addListener((observable, oldValue, newValue) -> updateBtcSyncProgress()); - daoPresentation.getBsqSyncProgress().addListener((observable, oldValue, newValue) -> updateBtcSyncProgress()); + daoPresentation.getDaoStateSyncProgress().addListener((observable, oldValue, newValue) -> updateBtcSyncProgress()); bisqSetup.setFilterWarningHandler(warning -> new Popup().warning(warning).show()); @@ -704,7 +704,7 @@ private void updateBtcSyncProgress() { if (btcSyncProgress.doubleValue() < 1) { combinedSyncProgress.set(btcSyncProgress.doubleValue()); } else { - combinedSyncProgress.set(daoPresentation.getBsqSyncProgress().doubleValue()); + combinedSyncProgress.set(daoPresentation.getDaoStateSyncProgress().doubleValue()); } } @@ -783,7 +783,7 @@ StringProperty getBtcInfo() { StringProperty getCombinedFooterInfo() { final StringProperty combinedInfo = new SimpleStringProperty(); - combinedInfo.bind(Bindings.concat(this.footerVersionInfo, " ", daoPresentation.getBsqInfo())); + combinedInfo.bind(Bindings.concat(this.footerVersionInfo, " ", daoPresentation.getDaoStateInfo())); return combinedInfo; } diff --git a/desktop/src/main/java/bisq/desktop/main/presentation/DaoPresentation.java b/desktop/src/main/java/bisq/desktop/main/presentation/DaoPresentation.java index 0afda183c6c..a0d0da6afa4 100644 --- a/desktop/src/main/java/bisq/desktop/main/presentation/DaoPresentation.java +++ b/desktop/src/main/java/bisq/desktop/main/presentation/DaoPresentation.java @@ -1,11 +1,15 @@ package bisq.desktop.main.presentation; import bisq.desktop.Navigation; -import bisq.desktop.util.GUIUtil; +import bisq.desktop.main.MainView; +import bisq.desktop.main.dao.DaoView; +import bisq.desktop.main.dao.monitor.MonitorView; +import bisq.desktop.main.dao.monitor.daostate.DaoStateMonitorView; +import bisq.desktop.main.overlays.popups.Popup; import bisq.core.btc.wallet.BsqWalletService; import bisq.core.btc.wallet.BtcWalletService; -import bisq.core.dao.DaoFacade; +import bisq.core.dao.monitoring.DaoStateMonitoringService; import bisq.core.dao.state.DaoStateListener; import bisq.core.dao.state.DaoStateService; import bisq.core.dao.state.model.blockchain.Block; @@ -28,24 +32,21 @@ import lombok.Getter; @Singleton -public class DaoPresentation implements DaoStateListener { +public class DaoPresentation implements DaoStateListener, DaoStateMonitoringService.Listener { public static final String DAO_NEWS = "daoNews"; - private final Preferences preferences; private final Navigation navigation; private final BtcWalletService btcWalletService; - private final DaoFacade daoFacade; + private final DaoStateMonitoringService daoStateMonitoringService; private final BsqWalletService bsqWalletService; private final DaoStateService daoStateService; private final ChangeListener walletChainHeightListener; - @Getter - private final DoubleProperty bsqSyncProgress = new SimpleDoubleProperty(-1); + private final DoubleProperty daoStateSyncProgress = new SimpleDoubleProperty(-1); @Getter - private final StringProperty bsqInfo = new SimpleStringProperty(""); + private final StringProperty daoStateInfo = new SimpleStringProperty(""); private final SimpleBooleanProperty showNotification = new SimpleBooleanProperty(false); - private boolean daoConflictWarningShown = false; // allow only one conflict warning per session @Inject public DaoPresentation(Preferences preferences, @@ -53,12 +54,11 @@ public DaoPresentation(Preferences preferences, BtcWalletService btcWalletService, BsqWalletService bsqWalletService, DaoStateService daoStateService, - DaoFacade daoFacade) { - this.preferences = preferences; + DaoStateMonitoringService daoStateMonitoringService) { this.navigation = navigation; this.btcWalletService = btcWalletService; this.bsqWalletService = bsqWalletService; - this.daoFacade = daoFacade; + this.daoStateMonitoringService = daoStateMonitoringService; this.daoStateService = daoStateService; preferences.getDontShowAgainMapAsObservable().addListener((MapChangeListener) change -> { @@ -73,34 +73,24 @@ public DaoPresentation(Preferences preferences, } /////////////////////////////////////////////////////////////////////////////////////////// - // Private + // Public /////////////////////////////////////////////////////////////////////////////////////////// - private void onUpdateAnyChainHeight() { - final int bsqBlockChainHeight = daoFacade.getChainHeight(); - final int bsqWalletChainHeight = bsqWalletService.getBestChainHeight(); - if (bsqWalletChainHeight > 0) { - final boolean synced = bsqWalletChainHeight == bsqBlockChainHeight; - if (bsqBlockChainHeight != bsqWalletChainHeight) { - bsqSyncProgress.set(-1); - } else { - bsqSyncProgress.set(0); - } + public void init() { + showNotification.set(false); - if (synced) { - bsqInfo.set(""); - if (daoFacade.daoStateNeedsRebuilding() && !daoConflictWarningShown) { - daoConflictWarningShown = true; // only warn max 1 time per session so as not to annoy - GUIUtil.showDaoNeedsResyncPopup(navigation); - } - } else { - bsqInfo.set(Res.get("mainView.footer.bsqInfo.synchronizing")); - } - } else { - bsqInfo.set(Res.get("mainView.footer.bsqInfo.synchronizing")); - } + btcWalletService.getChainHeightProperty().addListener(walletChainHeightListener); + daoStateService.addDaoStateListener(this); + daoStateMonitoringService.addListener(this); + + onUpdateAnyChainHeight(); + } + + public BooleanProperty getShowDaoUpdatesNotification() { + return showNotification; } + /////////////////////////////////////////////////////////////////////////////////////////// // DaoStateListener /////////////////////////////////////////////////////////////////////////////////////////// @@ -110,22 +100,37 @@ public void onParseBlockCompleteAfterBatchProcessing(Block block) { onUpdateAnyChainHeight(); } + /////////////////////////////////////////////////////////////////////////////////////////// - // Public + // DaoStateMonitoringService.Listener /////////////////////////////////////////////////////////////////////////////////////////// - public BooleanProperty getShowDaoUpdatesNotification() { - return showNotification; + @Override + public void onDaoStateHashesChanged() { + if (!daoStateService.isParseBlockChainComplete()) { + return; + } + + if (daoStateMonitoringService.isInConflictWithSeedNode() || + daoStateMonitoringService.isDaoStateBlockChainNotConnecting()) { + new Popup().warning(Res.get("popup.warning.daoNeedsResync")) + .actionButtonTextWithGoTo("navigation.dao.networkMonitor") + .onAction(() -> navigation.navigateTo(MainView.class, DaoView.class, MonitorView.class, DaoStateMonitorView.class)) + .show(); + } } - public void setup() { - // devs enable this when a news badge is required - //showNotification.set(DevEnv.isDaoActivated() && preferences.showAgain(DAO_NEWS)); - showNotification.set(false); - this.btcWalletService.getChainHeightProperty().addListener(walletChainHeightListener); - daoStateService.addDaoStateListener(this); + /////////////////////////////////////////////////////////////////////////////////////////// + // Private + /////////////////////////////////////////////////////////////////////////////////////////// - onUpdateAnyChainHeight(); + private void onUpdateAnyChainHeight() { + int bsqWalletChainHeight = bsqWalletService.getBestChainHeight(); + int daoStateChainHeight = daoStateService.getChainHeight(); + boolean chainHeightsInSync = bsqWalletChainHeight > 0 && bsqWalletChainHeight == daoStateChainHeight; + boolean isDaoStateReady = chainHeightsInSync && daoStateService.isParseBlockChainComplete(); + daoStateSyncProgress.set(isDaoStateReady ? 0 : -1); + daoStateInfo.set(isDaoStateReady ? "" : Res.get("mainView.footer.bsqInfo.synchronizing")); } } diff --git a/desktop/src/main/java/bisq/desktop/util/GUIUtil.java b/desktop/src/main/java/bisq/desktop/util/GUIUtil.java index 9c64ca9dc82..8c9406795a3 100644 --- a/desktop/src/main/java/bisq/desktop/util/GUIUtil.java +++ b/desktop/src/main/java/bisq/desktop/util/GUIUtil.java @@ -27,9 +27,6 @@ import bisq.desktop.main.MainView; import bisq.desktop.main.account.AccountView; import bisq.desktop.main.account.content.fiataccounts.FiatAccountsView; -import bisq.desktop.main.dao.DaoView; -import bisq.desktop.main.dao.monitor.MonitorView; -import bisq.desktop.main.dao.monitor.daostate.DaoStateMonitorView; import bisq.desktop.main.overlays.popups.Popup; import bisq.core.account.witness.AccountAgeWitness; @@ -806,19 +803,6 @@ public static boolean isBootstrappedOrShowPopup(P2PService p2PService) { return false; } - public static void showDaoNeedsResyncPopup(Navigation navigation) { - String key = "showDaoNeedsResyncPopup"; - if (DontShowAgainLookup.showAgain(key)) { - UserThread.runAfter(() -> new Popup().warning(Res.get("popup.warning.daoNeedsResync")) - .dontShowAgainId(key) - .actionButtonTextWithGoTo("navigation.dao.networkMonitor") - .onAction(() -> { - navigation.navigateTo(MainView.class, DaoView.class, MonitorView.class, DaoStateMonitorView.class); - }) - .show(), 5, TimeUnit.SECONDS); - } - } - public static boolean isReadyForTxBroadcastOrShowPopup(P2PService p2PService, WalletsSetup walletsSetup) { if (!GUIUtil.isBootstrappedOrShowPopup(p2PService)) { return false;