diff --git a/README.md b/README.md index 483c7e7a637..465e04f912d 100644 --- a/README.md +++ b/README.md @@ -71,14 +71,16 @@ To bring Haveno to life, we need resources. If you have the possibility, please ### Monero -`42sjokkT9FmiWPqVzrWPFE5NCJXwt96bkBozHf4vgLR9hXyJDqKHEHKVscAARuD7in5wV1meEcSTJTanCTDzidTe2cFXS1F` - - +

+ Donate Monero
+ 42sjokkT9FmiWPqVzrWPFE5NCJXwt96bkBozHf4vgLR9hXyJDqKHEHKVscAARuD7in5wV1meEcSTJTanCTDzidTe2cFXS1F +

If you are using a wallet that supports OpenAlias (like the 'official' CLI and GUI wallets), you can simply put `fund@haveno.exchange` as the "receiver" address. ### Bitcoin -`1AKq3CE1yBAnxGmHXbNFfNYStcByNDc5gQ` - - +

+ Donate Bitcoin
+ 1AKq3CE1yBAnxGmHXbNFfNYStcByNDc5gQ +

diff --git a/build.gradle b/build.gradle index 060205573bd..83d42efbbe9 100644 --- a/build.gradle +++ b/build.gradle @@ -317,6 +317,8 @@ configure(project(':common')) { exclude(module: 'animal-sniffer-annotations') } + // override transitive dependency and use latest version from bisq + implementation(group: 'com.github.bisq-network', name: 'jtorctl') { version { strictly "[b2a172f44edcd8deaa5ed75d936dcbb007f0d774]" } } implementation "org.openjfx:javafx-base:$javafxVersion:$os" implementation "org.openjfx:javafx-graphics:$javafxVersion:$os" } @@ -607,7 +609,7 @@ configure(project(':desktop')) { apply plugin: 'com.github.johnrengelman.shadow' apply from: 'package/package.gradle' - version = '1.0.10-SNAPSHOT' + version = '1.0.11-SNAPSHOT' jar.manifest.attributes( "Implementation-Title": project.name, diff --git a/common/src/main/java/haveno/common/app/Version.java b/common/src/main/java/haveno/common/app/Version.java index c8ab2551d24..e224c13d106 100644 --- a/common/src/main/java/haveno/common/app/Version.java +++ b/common/src/main/java/haveno/common/app/Version.java @@ -28,7 +28,7 @@ public class Version { // The application versions // We use semantic versioning with major, minor and patch - public static final String VERSION = "1.0.10"; + public static final String VERSION = "1.0.11"; /** * Holds a list of the tagged resource files for optimizing the getData requests. diff --git a/common/src/main/java/haveno/common/file/FileUtil.java b/common/src/main/java/haveno/common/file/FileUtil.java index 60ac2c3abc5..449faea64bc 100644 --- a/common/src/main/java/haveno/common/file/FileUtil.java +++ b/common/src/main/java/haveno/common/file/FileUtil.java @@ -40,10 +40,13 @@ @Slf4j public class FileUtil { + + private static final String BACKUP_DIR = "backup"; + public static void rollingBackup(File dir, String fileName, int numMaxBackupFiles) { if (numMaxBackupFiles <= 0) return; if (dir.exists()) { - File backupDir = new File(Paths.get(dir.getAbsolutePath(), "backup").toString()); + File backupDir = new File(Paths.get(dir.getAbsolutePath(), BACKUP_DIR).toString()); if (!backupDir.exists()) if (!backupDir.mkdir()) log.warn("make dir failed.\nBackupDir=" + backupDir.getAbsolutePath()); @@ -72,8 +75,21 @@ public static void rollingBackup(File dir, String fileName, int numMaxBackupFile } } + public static File getLatestBackupFile(File dir, String fileName) { + File backupDir = new File(Paths.get(dir.getAbsolutePath(), BACKUP_DIR).toString()); + if (!backupDir.exists()) return null; + String dirName = "backups_" + fileName; + if (dirName.contains(".")) dirName = dirName.replace(".", "_"); + File backupFileDir = new File(Paths.get(backupDir.getAbsolutePath(), dirName).toString()); + if (!backupFileDir.exists()) return null; + File[] files = backupFileDir.listFiles(); + if (files == null || files.length == 0) return null; + Arrays.sort(files, Comparator.comparing(File::getName)); + return files[files.length - 1]; + } + public static void deleteRollingBackup(File dir, String fileName) { - File backupDir = new File(Paths.get(dir.getAbsolutePath(), "backup").toString()); + File backupDir = new File(Paths.get(dir.getAbsolutePath(), BACKUP_DIR).toString()); if (!backupDir.exists()) return; String dirName = "backups_" + fileName; if (dirName.contains(".")) dirName = dirName.replace(".", "_"); diff --git a/core/src/main/java/haveno/core/api/XmrConnectionService.java b/core/src/main/java/haveno/core/api/XmrConnectionService.java index 48538ef48f5..4465fd2a358 100644 --- a/core/src/main/java/haveno/core/api/XmrConnectionService.java +++ b/core/src/main/java/haveno/core/api/XmrConnectionService.java @@ -99,9 +99,10 @@ public final class XmrConnectionService { @Getter private MoneroDaemonInfo lastInfo; private Long lastLogPollErrorTimestamp; - private Long syncStartHeight = null; + private long lastLogDaemonNotSyncedTimestamp; + private Long syncStartHeight; private TaskLooper daemonPollLooper; - private long lastRefreshPeriodMs = 0; + private long lastRefreshPeriodMs; @Getter private boolean isShutDownStarted; private List listeners = new ArrayList<>(); @@ -371,7 +372,6 @@ public boolean isSyncedWithinTolerance() { Long targetHeight = getTargetHeight(); if (targetHeight == null) return false; if (targetHeight - chainHeight.get() <= 3) return true; // synced if within 3 blocks of target height - log.warn("Our chain height: {} is out of sync with peer nodes chain height: {}", chainHeight.get(), targetHeight); return false; } @@ -720,7 +720,7 @@ private void doPollDaemon() { } // log error message periodically - if ((lastLogPollErrorTimestamp == null || System.currentTimeMillis() - lastLogPollErrorTimestamp > HavenoUtils.LOG_POLL_ERROR_PERIOD_MS)) { + if (lastLogPollErrorTimestamp == null || System.currentTimeMillis() - lastLogPollErrorTimestamp > HavenoUtils.LOG_POLL_ERROR_PERIOD_MS) { log.warn("Failed to fetch daemon info, trying to switch to best connection: " + e.getMessage()); if (DevEnv.isDevMode()) e.printStackTrace(); lastLogPollErrorTimestamp = System.currentTimeMillis(); @@ -734,6 +734,12 @@ private void doPollDaemon() { // connected to daemon isConnected = true; + // throttle warnings if daemon not synced + if (!isSyncedWithinTolerance() && System.currentTimeMillis() - lastLogDaemonNotSyncedTimestamp > HavenoUtils.LOG_DAEMON_NOT_SYNCED_WARN_PERIOD_MS) { + log.warn("Our chain height: {} is out of sync with peer nodes chain height: {}", chainHeight.get(), getTargetHeight()); + lastLogDaemonNotSyncedTimestamp = System.currentTimeMillis(); + } + // announce connection change if refresh period changes if (getRefreshPeriodMs() != lastRefreshPeriodMs) { lastRefreshPeriodMs = getRefreshPeriodMs(); diff --git a/core/src/main/java/haveno/core/offer/placeoffer/tasks/MakerReserveOfferFunds.java b/core/src/main/java/haveno/core/offer/placeoffer/tasks/MakerReserveOfferFunds.java index 7bfbda31ebe..4b44a8102d1 100644 --- a/core/src/main/java/haveno/core/offer/placeoffer/tasks/MakerReserveOfferFunds.java +++ b/core/src/main/java/haveno/core/offer/placeoffer/tasks/MakerReserveOfferFunds.java @@ -87,7 +87,7 @@ protected void run() { //if (true) throw new RuntimeException("Pretend error"); reserveTx = model.getXmrWalletService().createReserveTx(penaltyFee, makerFee, sendAmount, securityDeposit, returnAddress, openOffer.isReserveExactAmount(), preferredSubaddressIndex); } catch (Exception e) { - log.warn("Error creating reserve tx, offerId={}, attempt={}/{}, error={}", i + 1, TradeProtocol.MAX_ATTEMPTS, openOffer.getShortId(), e.getMessage()); + log.warn("Error creating reserve tx, offerId={}, attempt={}/{}, error={}", openOffer.getShortId(), i + 1, TradeProtocol.MAX_ATTEMPTS, e.getMessage()); model.getXmrWalletService().handleWalletError(e, sourceConnection); verifyPending(); if (i == TradeProtocol.MAX_ATTEMPTS - 1) throw e; diff --git a/core/src/main/java/haveno/core/support/dispute/arbitration/ArbitrationManager.java b/core/src/main/java/haveno/core/support/dispute/arbitration/ArbitrationManager.java index 0ac6aa432b8..8082aad475e 100644 --- a/core/src/main/java/haveno/core/support/dispute/arbitration/ArbitrationManager.java +++ b/core/src/main/java/haveno/core/support/dispute/arbitration/ArbitrationManager.java @@ -511,6 +511,7 @@ private MoneroTxSet signAndPublishDisputePayoutTx(Trade trade) { break; } catch (Exception e) { if (trade.isPayoutPublished()) throw new IllegalStateException("Payout tx already published for " + trade.getClass().getSimpleName() + " " + trade.getShortId()); + if (HavenoUtils.isNotEnoughSigners(e)) throw new IllegalArgumentException(e); log.warn("Failed to submit dispute payout tx, tradeId={}, attempt={}/{}, error={}", trade.getShortId(), i + 1, TradeProtocol.MAX_ATTEMPTS, e.getMessage()); if (i == TradeProtocol.MAX_ATTEMPTS - 1) throw e; if (trade.getXmrConnectionService().isConnected()) trade.requestSwitchToNextBestConnection(sourceConnection); diff --git a/core/src/main/java/haveno/core/trade/HavenoUtils.java b/core/src/main/java/haveno/core/trade/HavenoUtils.java index b1ad9da8395..3db5331c139 100644 --- a/core/src/main/java/haveno/core/trade/HavenoUtils.java +++ b/core/src/main/java/haveno/core/trade/HavenoUtils.java @@ -81,6 +81,7 @@ public class HavenoUtils { // other configuration public static final long LOG_POLL_ERROR_PERIOD_MS = 1000 * 60 * 4; // log poll errors up to once every 4 minutes + public static final long LOG_DAEMON_NOT_SYNCED_WARN_PERIOD_MS = 1000 * 30; // log warnings when daemon not synced once every 30s // synchronize requests to the daemon private static boolean SYNC_DAEMON_REQUESTS = false; // sync long requests to daemon (e.g. refresh, update pool) // TODO: performance suffers by syncing daemon requests, but otherwise we sometimes get sporadic errors? @@ -520,4 +521,8 @@ public static boolean isReadTimeout(Exception e) { public static boolean isUnresponsive(Exception e) { return isConnectionRefused(e) || isReadTimeout(e); } + + public static boolean isNotEnoughSigners(Exception e) { + return e != null && e.getMessage().contains("Not enough signers"); + } } diff --git a/core/src/main/java/haveno/core/trade/Trade.java b/core/src/main/java/haveno/core/trade/Trade.java index 4b3aed5bd4c..ca3df853580 100644 --- a/core/src/main/java/haveno/core/trade/Trade.java +++ b/core/src/main/java/haveno/core/trade/Trade.java @@ -1190,7 +1190,7 @@ public MoneroTxWallet createPayoutTx() { } catch (IllegalArgumentException | IllegalStateException e) { throw e; } catch (Exception e) { - log.warn("Failed to create payout tx, tradeId={}, attempt={}/{}, error={}", i + 1, TradeProtocol.MAX_ATTEMPTS, getShortId(), e.getMessage()); + log.warn("Failed to create payout tx, tradeId={}, attempt={}/{}, error={}", getShortId(), i + 1, TradeProtocol.MAX_ATTEMPTS, e.getMessage()); handleWalletError(e, sourceConnection); if (i == TradeProtocol.MAX_ATTEMPTS - 1) throw e; HavenoUtils.waitFor(TradeProtocol.REPROCESS_DELAY_MS); // wait before retrying @@ -1250,7 +1250,7 @@ public MoneroTxWallet createDisputePayoutTx(MoneroTxConfig txConfig) { throw e; } catch (Exception e) { if (e.getMessage().contains("not possible")) throw new IllegalArgumentException("Loser payout is too small to cover the mining fee"); - log.warn("Failed to create dispute payout tx, tradeId={}, attempt={}/{}, error={}", i + 1, TradeProtocol.MAX_ATTEMPTS, getShortId(), e.getMessage()); + log.warn("Failed to create dispute payout tx, tradeId={}, attempt={}/{}, error={}", getShortId(), i + 1, TradeProtocol.MAX_ATTEMPTS, e.getMessage()); handleWalletError(e, sourceConnection); if (i == TradeProtocol.MAX_ATTEMPTS - 1) throw e; HavenoUtils.waitFor(TradeProtocol.REPROCESS_DELAY_MS); // wait before retrying @@ -1279,7 +1279,7 @@ public void processPayoutTx(String payoutTxHex, boolean sign, boolean publish) { } catch (IllegalArgumentException | IllegalStateException e) { throw e; } catch (Exception e) { - log.warn("Failed to process payout tx, attempt={}/{}, tradeId={}, error={}", i + 1, TradeProtocol.MAX_ATTEMPTS, getShortId(), e.getMessage()); + log.warn("Failed to process payout tx, tradeId={}, attempt={}/{}, error={}", getShortId(), i + 1, TradeProtocol.MAX_ATTEMPTS, e.getMessage()); handleWalletError(e, sourceConnection); if (i == TradeProtocol.MAX_ATTEMPTS - 1) throw e; HavenoUtils.waitFor(TradeProtocol.REPROCESS_DELAY_MS); // wait before retrying @@ -1381,6 +1381,8 @@ private void doProcessPayoutTx(String payoutTxHex, boolean sign, boolean publish wallet.submitMultisigTxHex(payoutTxHex); setPayoutStatePublished(); } catch (Exception e) { + if (isPayoutPublished()) throw new IllegalStateException("Payout tx already published for " + getClass().getSimpleName() + " " + getShortId()); + if (HavenoUtils.isNotEnoughSigners(e)) throw new IllegalArgumentException(e); throw new RuntimeException("Failed to submit payout tx for " + getClass().getSimpleName() + " " + getId(), e); } } diff --git a/core/src/main/java/haveno/core/trade/protocol/tasks/MaybeSendSignContractRequest.java b/core/src/main/java/haveno/core/trade/protocol/tasks/MaybeSendSignContractRequest.java index 6dcef7f00b5..69d1620aeaf 100644 --- a/core/src/main/java/haveno/core/trade/protocol/tasks/MaybeSendSignContractRequest.java +++ b/core/src/main/java/haveno/core/trade/protocol/tasks/MaybeSendSignContractRequest.java @@ -104,7 +104,7 @@ protected void run() { try { depositTx = trade.getXmrWalletService().createDepositTx(trade, reserveExactAmount, subaddressIndex); } catch (Exception e) { - log.warn("Error creating deposit tx, attempt={}/{}, tradeId={}, error={}", i + 1, TradeProtocol.MAX_ATTEMPTS, trade.getShortId(), e.getMessage()); + log.warn("Error creating deposit tx, tradeId={}, attempt={}/{}, error={}", trade.getShortId(), i + 1, TradeProtocol.MAX_ATTEMPTS, e.getMessage()); trade.getXmrWalletService().handleWalletError(e, sourceConnection); if (isTimedOut()) throw new RuntimeException("Trade protocol has timed out while creating deposit tx, tradeId=" + trade.getShortId()); if (i == TradeProtocol.MAX_ATTEMPTS - 1) throw e; diff --git a/core/src/main/java/haveno/core/trade/protocol/tasks/TakerReserveTradeFunds.java b/core/src/main/java/haveno/core/trade/protocol/tasks/TakerReserveTradeFunds.java index 0ac0a540b56..5ab91343aa5 100644 --- a/core/src/main/java/haveno/core/trade/protocol/tasks/TakerReserveTradeFunds.java +++ b/core/src/main/java/haveno/core/trade/protocol/tasks/TakerReserveTradeFunds.java @@ -70,7 +70,7 @@ protected void run() { try { reserveTx = model.getXmrWalletService().createReserveTx(penaltyFee, takerFee, sendAmount, securityDeposit, returnAddress, false, null); } catch (Exception e) { - log.warn("Error creating reserve tx, attempt={}/{}, tradeId={}, error={}", i + 1, TradeProtocol.MAX_ATTEMPTS, trade.getShortId(), e.getMessage()); + log.warn("Error creating reserve tx, tradeId={}, attempt={}/{}, error={}", trade.getShortId(), i + 1, TradeProtocol.MAX_ATTEMPTS, e.getMessage()); trade.getXmrWalletService().handleWalletError(e, sourceConnection); if (isTimedOut()) throw new RuntimeException("Trade protocol has timed out while creating reserve tx, tradeId=" + trade.getShortId()); if (i == TradeProtocol.MAX_ATTEMPTS - 1) throw e; diff --git a/core/src/main/java/haveno/core/xmr/wallet/XmrWalletService.java b/core/src/main/java/haveno/core/xmr/wallet/XmrWalletService.java index ea0d77fb06c..325bdc13134 100644 --- a/core/src/main/java/haveno/core/xmr/wallet/XmrWalletService.java +++ b/core/src/main/java/haveno/core/xmr/wallet/XmrWalletService.java @@ -136,7 +136,6 @@ public class XmrWalletService extends XmrWalletBase { private final WalletsSetup walletsSetup; private final File walletDir; - private final File xmrWalletFile; private final int rpcBindPort; private final boolean useNativeXmrWallet; protected final CopyOnWriteArraySet balanceListeners = new CopyOnWriteArraySet<>(); @@ -151,6 +150,7 @@ public class XmrWalletService extends XmrWalletBase { private TaskLooper pollLooper; private boolean pollInProgress; private Long pollPeriodMs; + private long lastLogDaemonNotSyncedTimestamp; private long lastLogPollErrorTimestamp; private long lastPollTxsTimestamp; private final Object pollLock = new Object(); @@ -180,7 +180,6 @@ public class XmrWalletService extends XmrWalletBase { this.walletDir = walletDir; this.rpcBindPort = rpcBindPort; this.useNativeXmrWallet = useNativeXmrWallet; - this.xmrWalletFile = new File(walletDir, MONERO_WALLET_NAME); HavenoUtils.xmrWalletService = this; HavenoUtils.xmrConnectionService = xmrConnectionService; this.xmrConnectionService = xmrConnectionService; // TODO: super's is null unless set here from injection @@ -1326,7 +1325,7 @@ private void doMaybeInitMainWallet(boolean sync, int numAttempts) { if (wallet == null) { MoneroDaemonRpc daemon = xmrConnectionService.getDaemon(); log.info("Initializing main wallet with monerod=" + (daemon == null ? "null" : daemon.getRpcConnection().getUri())); - if (MoneroUtils.walletExists(xmrWalletFile.getPath())) { + if (walletExists(MONERO_WALLET_NAME)) { wallet = openWallet(MONERO_WALLET_NAME, rpcBindPort, isProxyApplied(wasWalletSynced)); } else if (Boolean.TRUE.equals(xmrConnectionService.isConnected())) { wallet = createWallet(MONERO_WALLET_NAME, rpcBindPort); @@ -1474,11 +1473,54 @@ private MoneroWalletFull openWalletFull(MoneroWalletConfig config, boolean apply MoneroRpcConnection connection = new MoneroRpcConnection(xmrConnectionService.getConnection()); if (!applyProxyUri) connection.setProxyUri(null); - // open wallet + // try opening wallet config.setNetworkType(getMoneroNetworkType()); config.setServer(connection); log.info("Opening full wallet " + config.getPath() + " with monerod=" + connection.getUri() + ", proxyUri=" + connection.getProxyUri()); - walletFull = MoneroWalletFull.openWallet(config); + try { + walletFull = MoneroWalletFull.openWallet(config); + } catch (Exception e) { + log.warn("Failed to open full wallet '{}', attempting to use backup cache, error={}", config.getPath(), e.getMessage()); + boolean retrySuccessful = false; + try { + + // rename wallet cache to backup + String cachePath = walletDir.toString() + File.separator + MONERO_WALLET_NAME; + File originalCacheFile = new File(cachePath); + if (originalCacheFile.exists()) originalCacheFile.renameTo(new File(cachePath + ".backup")); + + // copy latest wallet cache backup to main folder + File backupCacheFile = FileUtil.getLatestBackupFile(walletDir, MONERO_WALLET_NAME); + if (backupCacheFile != null) FileUtil.copyFile(backupCacheFile, new File(cachePath)); + + // retry opening wallet without original cache + try { + walletFull = MoneroWalletFull.openWallet(config); + log.info("Successfully opened full wallet using backup cache"); + retrySuccessful = true; + } catch (Exception e2) { + // ignore + } + + // handle success or failure + if (retrySuccessful) { + originalCacheFile.delete(); // delete original wallet cache backup + } else { + + // restore original wallet cache + log.warn("Failed to open full wallet using backup cache, restoring original cache"); + File cacheFile = new File(cachePath); + if (cacheFile.exists()) cacheFile.delete(); + File originalCacheBackup = new File(cachePath + ".backup"); + if (originalCacheBackup.exists()) originalCacheBackup.renameTo(new File(cachePath)); + + // throw exception + throw e; + } + } catch (Exception e2) { + throw e; // throw original exception + } + } if (walletFull.getDaemonConnection() != null) walletFull.getDaemonConnection().setPrintStackTrace(PRINT_RPC_STACK_TRACE); log.info("Done opening full wallet " + config.getPath()); return walletFull; @@ -1517,7 +1559,7 @@ private MoneroWalletRpc createWalletRpc(MoneroWalletConfig config, Integer port) } catch (Exception e) { e.printStackTrace(); if (walletRpc != null) forceCloseWallet(walletRpc, config.getPath()); - throw new IllegalStateException("Could not create wallet '" + config.getPath() + "'. Please close Haveno, stop all monero-wallet-rpc processes, and restart Haveno."); + throw new IllegalStateException("Could not create wallet '" + config.getPath() + "'. Please close Haveno, stop all monero-wallet-rpc processes in your task manager, and restart Haveno."); } } @@ -1536,17 +1578,60 @@ private MoneroWalletRpc openWalletRpc(MoneroWalletConfig config, Integer port, b MoneroRpcConnection connection = new MoneroRpcConnection(xmrConnectionService.getConnection()); if (!applyProxyUri) connection.setProxyUri(null); - // open wallet + // try opening wallet log.info("Opening RPC wallet " + config.getPath() + " with monerod=" + connection.getUri() + ", proxyUri=" + connection.getProxyUri()); config.setServer(connection); - walletRpc.openWallet(config); + try { + walletRpc.openWallet(config); + } catch (Exception e) { + log.warn("Failed to open RPC wallet '{}', attempting to use backup cache, error={}", config.getPath(), e.getMessage()); + boolean retrySuccessful = false; + try { + + // rename wallet cache to backup + String cachePath = walletDir.toString() + File.separator + MONERO_WALLET_NAME; + File originalCacheFile = new File(cachePath); + if (originalCacheFile.exists()) originalCacheFile.renameTo(new File(cachePath + ".backup")); + + // copy latest wallet cache backup to main folder + File backupCacheFile = FileUtil.getLatestBackupFile(walletDir, MONERO_WALLET_NAME); + if (backupCacheFile != null) FileUtil.copyFile(backupCacheFile, new File(cachePath)); + + // retry opening wallet without original cache + try { + walletRpc.openWallet(config); + log.info("Successfully opened RPC wallet using backup cache"); + retrySuccessful = true; + } catch (Exception e2) { + // ignore + } + + // handle success or failure + if (retrySuccessful) { + originalCacheFile.delete(); // delete original wallet cache backup + } else { + + // restore original wallet cache + log.warn("Failed to open RPC wallet using backup cache, restoring original cache"); + File cacheFile = new File(cachePath); + if (cacheFile.exists()) cacheFile.delete(); + File originalCacheBackup = new File(cachePath + ".backup"); + if (originalCacheBackup.exists()) originalCacheBackup.renameTo(new File(cachePath)); + + // throw exception + throw e; + } + } catch (Exception e2) { + throw e; // throw original exception + } + } if (walletRpc.getDaemonConnection() != null) walletRpc.getDaemonConnection().setPrintStackTrace(PRINT_RPC_STACK_TRACE); log.info("Done opening RPC wallet " + config.getPath()); return walletRpc; } catch (Exception e) { e.printStackTrace(); if (walletRpc != null) forceCloseWallet(walletRpc, config.getPath()); - throw new IllegalStateException("Could not open wallet '" + config.getPath() + "'. Please close Haveno, stop all monero-wallet-rpc processes, and restart Haveno.\n\nError message: " + e.getMessage()); + throw new IllegalStateException("Could not open wallet '" + config.getPath() + "'. Please close Haveno, stop all monero-wallet-rpc processes in your task manager, and restart Haveno.\n\nError message: " + e.getMessage()); } } @@ -1767,9 +1852,15 @@ private void doPollWallet(boolean updateTxs) { return; } if (!xmrConnectionService.isSyncedWithinTolerance()) { - log.warn("Monero daemon is not synced within tolerance, height={}, targetHeight={}", xmrConnectionService.chainHeightProperty().get(), xmrConnectionService.getTargetHeight()); + + // throttle warnings + if (System.currentTimeMillis() - lastLogDaemonNotSyncedTimestamp > HavenoUtils.LOG_DAEMON_NOT_SYNCED_WARN_PERIOD_MS) { + log.warn("Monero daemon is not synced within tolerance, height={}, targetHeight={}, monerod={}", xmrConnectionService.chainHeightProperty().get(), xmrConnectionService.getTargetHeight(), xmrConnectionService.getConnection() == null ? null : xmrConnectionService.getConnection().getUri()); + lastLogDaemonNotSyncedTimestamp = System.currentTimeMillis(); + } return; } + // sync wallet if behind daemon if (walletHeight.get() < xmrConnectionService.getTargetHeight()) { synchronized (walletLock) { // avoid long sync from blocking other operations diff --git a/desktop/package/macosx/Info.plist b/desktop/package/macosx/Info.plist index 29c4932bc1c..5a058b31a22 100644 --- a/desktop/package/macosx/Info.plist +++ b/desktop/package/macosx/Info.plist @@ -5,10 +5,10 @@ CFBundleVersion - 1.0.10 + 1.0.11 CFBundleShortVersionString - 1.0.10 + 1.0.11 CFBundleExecutable Haveno diff --git a/desktop/src/main/java/haveno/desktop/main/funds/withdrawal/WithdrawalView.java b/desktop/src/main/java/haveno/desktop/main/funds/withdrawal/WithdrawalView.java index 0fa45624ae3..cc5a7cd1d4b 100644 --- a/desktop/src/main/java/haveno/desktop/main/funds/withdrawal/WithdrawalView.java +++ b/desktop/src/main/java/haveno/desktop/main/funds/withdrawal/WithdrawalView.java @@ -242,8 +242,12 @@ private void onWithdraw() { if (GUIUtil.isReadyForTxBroadcastOrShowPopup(xmrWalletService)) { try { + // collect tx fields to local variables + String withdrawToAddress = withdrawToTextField.getText(); + boolean sendMax = this.sendMax; + BigInteger amount = this.amount; + // validate address - final String withdrawToAddress = withdrawToTextField.getText(); if (!MoneroUtils.isValidAddress(withdrawToAddress, XmrWalletService.getMoneroNetworkType())) { throw new IllegalArgumentException(Res.get("validation.xmr.invalidAddress")); } @@ -298,7 +302,7 @@ private void popupConfirmationMessage(MoneroTxWallet tx) { BigInteger receiverAmount = tx.getOutgoingTransfer().getDestinations().get(0).getAmount(); BigInteger fee = tx.getFee(); String messageText = Res.get("shared.sendFundsDetailsWithFee", - HavenoUtils.formatXmr(amount, true), + HavenoUtils.formatXmr(receiverAmount.add(fee), true), withdrawToAddress, HavenoUtils.formatXmr(fee, true), HavenoUtils.formatXmr(receiverAmount, true)); diff --git a/docs/create-mainnet.md b/docs/create-mainnet.md index 32606a0eff5..38c211d4cd5 100644 --- a/docs/create-mainnet.md +++ b/docs/create-mainnet.md @@ -57,6 +57,12 @@ For example, change "Haveno" to "HavenoX", which will use this application folde - macOS: ~/Library/Application Support/HavenoX/ - Windows: ~\AppData\Roaming\HavenoX\ +## Change the P2P network version + +To avoid interference with other networks, change `P2P_NETWORK_VERSION` in [Version.java](https://github.com/haveno-dex/haveno/blob/a7e90395d24ec3d33262dd5d09c5faec61651a51/common/src/main/java/haveno/common/app/Version.java#L83). + +For example, change it to `"B"`. + ## Start the seed nodes Rebuild for the previous changes to the source code to take effect: `make skip-tests`. diff --git a/docs/deployment-guide.md b/docs/deployment-guide.md index ce538319ced..2f7590d0ecc 100644 --- a/docs/deployment-guide.md +++ b/docs/deployment-guide.md @@ -219,6 +219,12 @@ For example, change "Haveno" to "HavenoX", which will use this application folde - macOS: ~/Library/Application Support/HavenoX/ - Windows: ~\AppData\Roaming\HavenoX\ +## Change the P2P network version + +To avoid interference with other networks, change `P2P_NETWORK_VERSION` in [Version.java](https://github.com/haveno-dex/haveno/blob/a7e90395d24ec3d33262dd5d09c5faec61651a51/common/src/main/java/haveno/common/app/Version.java#L83). + +For example, change it to `"B"`. + ## Set the network's release date Optionally set the network's approximate release date by setting `RELEASE_DATE` in HavenoUtils.java. diff --git a/docs/installing.md b/docs/installing.md index f779d8ef0b8..dcf9c81df58 100644 --- a/docs/installing.md +++ b/docs/installing.md @@ -1,18 +1,19 @@ -# Set up environment +# Build and run Haveno -These are the steps needed to build Haveno and test it on our test network or locally. +These are the steps needed to build and run Haveno. You can test it locally or on our test network using the official Haveno repository. -## Install dependencies - -On Linux and macOS, install Java JDK 21: +> [!note] +> Trying to use Haveno on mainnet? +> +> The official Haveno repository does not operate or endorse any mainnet network. +> +> Find a third party network and use their installer or build their repository. Alternatively [create your own mainnet network](create-mainnet.md). -``` -curl -s "https://get.sdkman.io" | bash -sdk install java 21.0.2.fx-librca -``` +## Install dependencies -On Windows, install MSYS2 and Java JDK 21: +On Ubuntu: `sudo apt install make wget git` +On Windows, first install MSYS2: 1. Install [MSYS2](https://www.msys2.org/). 2. Start MSYS2 MINGW64 or MSYS MINGW32 depending on your system. Use MSYS2 for all commands throughout this document. 4. Update pacman: `pacman -Syy` @@ -21,12 +22,17 @@ On Windows, install MSYS2 and Java JDK 21: 64-bit: `pacman -S mingw-w64-x86_64-toolchain make mingw-w64-x86_64-cmake git` 32-bit: `pacman -S mingw-w64-i686-toolchain make mingw-w64-i686-cmake git` - 6. `curl -s "https://get.sdkman.io" | bash` - 7. `sdk install java 21.0.2.fx-librca` + +On all platforms, install Java JDK 21: + +``` +curl -s "https://get.sdkman.io" | bash +sdk install java 21.0.2.fx-librca +``` ## Build Haveno -If it's the first time you are building Haveno, run the following commands to download the repository, the needed dependencies, and build the latest release: +If it's the first time you are building Haveno, run the following commands to download the repository, the needed dependencies, and build the latest release. If using a third party network, replace the repository URL with theirs: ``` git clone https://github.com/haveno-dex/haveno.git @@ -45,15 +51,23 @@ git pull make clean && make ``` -Make sure to delete the folder with the local settings, as there are breaking changes between releases: +## Run Haveno + +> [!note] +> When you run Haveno, your application folder will be installed to: +> * Linux: `~/.local/share/Haveno/` +> * macOS: `~/Library/Application\ Support/Haveno/` +> * Windows: `~\AppData\Roaming\Haveno\` -On **Linux**: remove everything inside `~/.local/share/haveno-*/xmr_stagenet/`, except the `wallet` folder. +### Mainnet -On **Mac**: remove everything inside `~/Library/Application\ Support/haveno-*/xmr_stagenet/`, except the `wallet` folder. +If you are building a third party repository which supports mainnet, you can start Haveno with: -On **Windows**: remove everything inside `~\AppData\Roaming\haveno-*/xmr_stagenet/`, except the `wallet` folder. +``` +make haveno-desktop-mainnet +``` -## Join the public test network +### Join the public test network If you want to try Haveno in a live setup, launch a Haveno instance that will connect to other peers on our public test environment, which runs on Monero's stagenet (you won't need to download the blockchain locally). You'll be able to make test trades with other users and have a preview of Haveno's trade protocol in action. Note that development is very much ongoing. Things are slow and might break. @@ -67,11 +81,11 @@ Steps: 6. Now if you are taking a trade you'll be asked to confirm you have sent the payment outside Haveno. Confirm in the app and wait for the confirmation of received payment from the other trader. 7. Once the other trader confirms, deposits are sent back to the owners and the trade is complete. -# Run a local test network +### Run a local test network If you are a developer who wants to test Haveno in a more controlled way, follow the next steps to build a local test environment. -## Run a local XMR testnet +#### Run a local XMR testnet 1. In a new terminal window run `make monerod1-local` 1. In a new terminal window run `make monerod2-local` @@ -79,7 +93,7 @@ If you are a developer who wants to test Haveno in a more controlled way, follow `start_mining 9tsUiG9bwcU7oTbAdBwBk2PzxFtysge5qcEsHEpetmEKgerHQa1fDqH7a4FiquZmms7yM22jdifVAD7jAb2e63GSJMuhY75 1` -## Deploy +#### Deploy If you are a *screen* user, simply run `make deploy`. This command will open all needed Haveno instances (seednode, user1, user2, arbitrator) using *screen*. @@ -93,7 +107,7 @@ If you don't use *screen*, open 4 terminal windows and run in each one of them: If this is the first time launching the arbitrator desktop application, register the arbitrator after the interface opens. Go to the *Account* tab and press `cmd+r`. Confirm the registration of the arbitrator. -## Fund your wallets +#### Fund your wallets When running user1 and user2, you'll see a Monero address prompted in the terminal. Send test XMR to the addresses of both user1 and user2 to be able to initiate a trade. @@ -101,6 +115,6 @@ You can fund the two wallets by mining some test XMR coins to those addresses. T monerod will start mining local testnet coins on your device using one thread. Replace `ADDRESS` with the address of user1 first, and then user2's. Run `stop_mining` to stop mining. -## Start testing +#### Start testing You are all set. Now that everything is running and your wallets are funded, you can create test trades between user1 and user2. Remember to mine a few blocks after opening and accepting the test trade so the transaction will be confirmed. diff --git a/docs/user-guide.md b/docs/user-guide.md index a7072ca6d85..8ea7db1bacb 100644 --- a/docs/user-guide.md +++ b/docs/user-guide.md @@ -2,10 +2,24 @@ This document is a guide for Haveno users. -# Running a Local Monero Node +## Running a Local Monero Node For the best experience using Haveno, it is highly recommended to run your own local Monero node to improve security and responsiveness. By default, Haveno will automatically connect to a local node if it is detected. Additionally, Haveno will automatically start and connect to your local Monero node if it was last used and is currently offline. -Otherwise, Haveno will connect to a pre-configured remote node, unless manually configured otherwise. \ No newline at end of file +Otherwise, Haveno will connect to a pre-configured remote node, unless manually configured otherwise. + +## UI Scaling For High DPI Displays + +If the UI is too small on your display, you can force UI scaling to a value of your choice using one of the following approaches. The examples below scale the UI to 200%, you can replace the '2' with a value of your choice, e.g. '1.5' for 150%. + +### Edit The Application Shortcut (KDE Plasma) + +1) Open the properties of your shortcut to haveno +2) Click on Program +3) Add `JAVA_TOOL_OPTIONS=-Dglass.gtk.uiScale=2` to the environment variables + +### Launching From The Command Line + +Prepend `JAVA_TOOL_OPTIONS=-Dglass.gtk.uiScale=2` to the command you use to launch haveno (e.g. `JAVA_TOOL_OPTIONS=-Dglass.gtk.uiScale=2 haveno-desktop`). diff --git a/gradle/verification-metadata.xml b/gradle/verification-metadata.xml index 5e2bf6c536d..dc626c93e1e 100644 --- a/gradle/verification-metadata.xml +++ b/gradle/verification-metadata.xml @@ -173,14 +173,6 @@ - - - - - - - - diff --git a/media/donate_bitcoin.png b/media/donate_bitcoin.png new file mode 100644 index 00000000000..ac25c2c2c8a Binary files /dev/null and b/media/donate_bitcoin.png differ diff --git a/media/donate_monero.png b/media/donate_monero.png new file mode 100644 index 00000000000..35b3e21d8fc Binary files /dev/null and b/media/donate_monero.png differ diff --git a/media/qrbtc.png b/media/qrbtc.png deleted file mode 100644 index 66e90ebd7d1..00000000000 Binary files a/media/qrbtc.png and /dev/null differ diff --git a/media/qrhaveno.png b/media/qrhaveno.png deleted file mode 100644 index 347b04a597a..00000000000 Binary files a/media/qrhaveno.png and /dev/null differ diff --git a/seednode/src/main/java/haveno/seednode/SeedNodeMain.java b/seednode/src/main/java/haveno/seednode/SeedNodeMain.java index dedb7d1cbaf..5659ab2e0bd 100644 --- a/seednode/src/main/java/haveno/seednode/SeedNodeMain.java +++ b/seednode/src/main/java/haveno/seednode/SeedNodeMain.java @@ -41,7 +41,7 @@ @Slf4j public class SeedNodeMain extends ExecutableForAppWithP2p { private static final long CHECK_CONNECTION_LOSS_SEC = 30; - private static final String VERSION = "1.0.10"; + private static final String VERSION = "1.0.11"; private SeedNode seedNode; private Timer checkConnectionLossTime; diff --git a/seednode/torrc b/seednode/torrc index d90b0111111..d6e2c55b55d 100644 --- a/seednode/torrc +++ b/seednode/torrc @@ -124,8 +124,8 @@ HiddenServiceEnableIntroDoSDefense 1 ## Proof of Work (PoW) before establishing Rendezvous Circuits ## The lower the queue and burst rates, the higher the puzzle effort tends to be for users. HiddenServicePoWDefensesEnabled 1 -HiddenServicePoWQueueRate 200 # (Default: 250) -HiddenServicePoWQueueBurst 1000 # (Default: 2500) +HiddenServicePoWQueueRate 50 # (Default: 250) +HiddenServicePoWQueueBurst 250 # (Default: 2500) ## Stream limits in the established Rendezvous Circuits ## The maximum number of simultaneous streams (connections) per rendezvous circuit. The max value allowed is 65535. (0 = unlimited) @@ -143,8 +143,8 @@ HiddenServiceEnableIntroDoSDefense 1 #HiddenServiceNumIntroductionPoints 3 # (Default: 3) HiddenServicePoWDefensesEnabled 1 -HiddenServicePoWQueueRate 200 # (Default: 250) -HiddenServicePoWQueueBurst 1000 # (Default: 2500) +HiddenServicePoWQueueRate 50 # (Default: 250) +HiddenServicePoWQueueBurst 250 # (Default: 2500) HiddenServiceMaxStreams 25 #HiddenServiceMaxStreamsCloseCircuit 1