2020-05-18 18:00:13 +02:00
|
|
|
"""
|
|
|
|
This module setting file is for settings
|
|
|
|
"""
|
2019-10-03 17:34:27 +02:00
|
|
|
import ConfigParser
|
2018-11-13 16:03:38 +01:00
|
|
|
import os
|
|
|
|
import sys
|
2020-06-09 12:17:45 +02:00
|
|
|
import tempfile
|
2018-11-13 16:03:38 +01:00
|
|
|
|
2022-08-29 23:35:22 +02:00
|
|
|
import six
|
2018-11-20 15:53:25 +01:00
|
|
|
from PyQt4 import QtCore, QtGui
|
2018-11-13 16:03:38 +01:00
|
|
|
|
|
|
|
import debug
|
|
|
|
import defaults
|
|
|
|
import namecoin
|
|
|
|
import openclpow
|
|
|
|
import paths
|
|
|
|
import queues
|
|
|
|
import state
|
|
|
|
import widgets
|
2022-02-16 21:16:41 +01:00
|
|
|
from bmconfigparser import config as config_obj
|
2018-11-13 16:03:38 +01:00
|
|
|
from helper_sql import sqlExecute, sqlStoredProcedure
|
2019-10-03 17:34:27 +02:00
|
|
|
from helper_startup import start_proxyconfig
|
2020-09-16 17:41:51 +02:00
|
|
|
from network import knownnodes, AnnounceThread
|
2018-11-13 16:03:38 +01:00
|
|
|
from network.asyncore_pollchoose import set_rates
|
|
|
|
from tr import _translate
|
|
|
|
|
|
|
|
|
2022-02-16 21:16:41 +01:00
|
|
|
def getSOCKSProxyType(config):
|
2019-12-07 20:33:30 +01:00
|
|
|
"""Get user socksproxytype setting from *config*"""
|
|
|
|
try:
|
|
|
|
result = ConfigParser.SafeConfigParser.get(
|
2022-02-16 21:16:41 +01:00
|
|
|
config, 'bitmessagesettings', 'socksproxytype')
|
2019-12-07 20:33:30 +01:00
|
|
|
except (ConfigParser.NoSectionError, ConfigParser.NoOptionError):
|
2020-06-09 12:17:45 +02:00
|
|
|
return None
|
2019-12-07 20:33:30 +01:00
|
|
|
else:
|
|
|
|
if result.lower() in ('', 'none', 'false'):
|
|
|
|
result = None
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2018-11-13 16:03:38 +01:00
|
|
|
class SettingsDialog(QtGui.QDialog):
|
|
|
|
"""The "Settings" dialog"""
|
|
|
|
def __init__(self, parent=None, firstrun=False):
|
|
|
|
super(SettingsDialog, self).__init__(parent)
|
|
|
|
widgets.load('settings.ui', self)
|
|
|
|
|
|
|
|
self.parent = parent
|
|
|
|
self.firstrun = firstrun
|
2022-02-16 21:16:41 +01:00
|
|
|
self.config = config_obj
|
2018-11-20 15:53:25 +01:00
|
|
|
self.net_restart_needed = False
|
|
|
|
self.timer = QtCore.QTimer()
|
2018-11-13 16:03:38 +01:00
|
|
|
|
2020-10-31 09:11:30 +01:00
|
|
|
if self.config.safeGetBoolean('bitmessagesettings', 'dontconnect'):
|
|
|
|
self.firstrun = False
|
2019-10-03 17:34:27 +02:00
|
|
|
try:
|
|
|
|
import pkg_resources
|
|
|
|
except ImportError:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
# Append proxy types defined in plugins
|
2021-07-02 17:31:43 +02:00
|
|
|
# FIXME: this should be a function in mod:`plugin`
|
2019-10-03 17:34:27 +02:00
|
|
|
for ep in pkg_resources.iter_entry_points(
|
|
|
|
'bitmessage.proxyconfig'):
|
2021-07-02 17:31:43 +02:00
|
|
|
try:
|
|
|
|
ep.load()
|
|
|
|
except Exception: # it should add only functional plugins
|
|
|
|
# many possible exceptions, which are don't matter
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
self.comboBoxProxyType.addItem(ep.name)
|
2019-10-03 17:34:27 +02:00
|
|
|
|
2018-05-28 15:35:30 +02:00
|
|
|
self.lineEditMaxOutboundConnections.setValidator(
|
|
|
|
QtGui.QIntValidator(0, 8, self.lineEditMaxOutboundConnections))
|
2018-11-13 16:03:38 +01:00
|
|
|
|
|
|
|
self.adjust_from_config(self.config)
|
|
|
|
if firstrun:
|
|
|
|
# switch to "Network Settings" tab if user selected
|
|
|
|
# "Let me configure special network settings first" on first run
|
|
|
|
self.tabWidgetSettings.setCurrentIndex(
|
|
|
|
self.tabWidgetSettings.indexOf(self.tabNetworkSettings)
|
|
|
|
)
|
|
|
|
QtGui.QWidget.resize(self, QtGui.QWidget.sizeHint(self))
|
|
|
|
|
2022-02-16 21:16:41 +01:00
|
|
|
def adjust_from_config(self, config):
|
2018-11-13 16:03:38 +01:00
|
|
|
"""Adjust all widgets state according to config settings"""
|
|
|
|
# pylint: disable=too-many-branches,too-many-statements
|
2019-12-05 17:17:35 +01:00
|
|
|
if not self.parent.tray.isSystemTrayAvailable():
|
|
|
|
self.groupBoxTray.setEnabled(False)
|
|
|
|
self.groupBoxTray.setTitle(_translate(
|
|
|
|
"MainWindow", "Tray (not available in your system)"))
|
|
|
|
for setting in (
|
|
|
|
'minimizetotray', 'trayonclose', 'startintray'):
|
2022-02-16 21:16:41 +01:00
|
|
|
config.set('bitmessagesettings', setting, 'false')
|
2019-12-05 17:17:35 +01:00
|
|
|
else:
|
|
|
|
self.checkBoxMinimizeToTray.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getboolean('bitmessagesettings', 'minimizetotray'))
|
2019-12-05 17:17:35 +01:00
|
|
|
self.checkBoxTrayOnClose.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.safeGetBoolean('bitmessagesettings', 'trayonclose'))
|
2019-12-05 17:17:35 +01:00
|
|
|
self.checkBoxStartInTray.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getboolean('bitmessagesettings', 'startintray'))
|
2019-12-05 17:17:35 +01:00
|
|
|
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxHideTrayConnectionNotifications.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getboolean(
|
2019-12-05 17:17:35 +01:00
|
|
|
'bitmessagesettings', 'hidetrayconnectionnotifications'))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxShowTrayNotifications.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getboolean('bitmessagesettings', 'showtraynotifications'))
|
2019-12-05 17:17:35 +01:00
|
|
|
|
|
|
|
self.checkBoxStartOnLogon.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getboolean('bitmessagesettings', 'startonlogon'))
|
2019-12-05 17:17:35 +01:00
|
|
|
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxWillinglySendToMobile.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.safeGetBoolean(
|
2019-12-05 17:17:35 +01:00
|
|
|
'bitmessagesettings', 'willinglysendtomobile'))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxUseIdenticons.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.safeGetBoolean('bitmessagesettings', 'useidenticons'))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxReplyBelow.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.safeGetBoolean('bitmessagesettings', 'replybelow'))
|
2018-11-13 16:03:38 +01:00
|
|
|
|
|
|
|
if state.appdata == paths.lookupExeFolder():
|
|
|
|
self.checkBoxPortableMode.setChecked(True)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
tempfile.NamedTemporaryFile(
|
|
|
|
dir=paths.lookupExeFolder(), delete=True
|
|
|
|
).close() # should autodelete
|
2020-05-18 18:00:13 +02:00
|
|
|
except Exception:
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxPortableMode.setDisabled(True)
|
|
|
|
|
|
|
|
if 'darwin' in sys.platform:
|
|
|
|
self.checkBoxMinimizeToTray.setDisabled(True)
|
|
|
|
self.checkBoxMinimizeToTray.setText(_translate(
|
2019-12-05 17:17:35 +01:00
|
|
|
"MainWindow",
|
|
|
|
"Minimize-to-tray not yet supported on your OS."))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxShowTrayNotifications.setDisabled(True)
|
|
|
|
self.checkBoxShowTrayNotifications.setText(_translate(
|
2019-12-05 17:17:35 +01:00
|
|
|
"MainWindow",
|
|
|
|
"Tray notifications not yet supported on your OS."))
|
2019-05-06 17:30:21 +02:00
|
|
|
|
|
|
|
if 'win' not in sys.platform and not self.parent.desktop:
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxStartOnLogon.setDisabled(True)
|
|
|
|
self.checkBoxStartOnLogon.setText(_translate(
|
|
|
|
"MainWindow", "Start-on-login not yet supported on your OS."))
|
2019-05-06 17:30:21 +02:00
|
|
|
|
2018-11-13 16:03:38 +01:00
|
|
|
# On the Network settings tab:
|
|
|
|
self.lineEditTCPPort.setText(str(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'port')))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxUPnP.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.safeGetBoolean('bitmessagesettings', 'upnp'))
|
2020-09-16 17:41:51 +02:00
|
|
|
self.checkBoxUDP.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.safeGetBoolean('bitmessagesettings', 'udp'))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxAuthentication.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getboolean('bitmessagesettings', 'socksauthentication'))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.checkBoxSocksListen.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getboolean('bitmessagesettings', 'sockslisten'))
|
2019-10-08 22:08:42 +02:00
|
|
|
self.checkBoxOnionOnly.setChecked(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.safeGetBoolean('bitmessagesettings', 'onionservicesonly'))
|
2018-11-13 16:03:38 +01:00
|
|
|
|
2022-02-16 21:16:41 +01:00
|
|
|
self._proxy_type = getSOCKSProxyType(config)
|
2019-10-03 17:34:27 +02:00
|
|
|
self.comboBoxProxyType.setCurrentIndex(
|
2019-12-07 20:33:30 +01:00
|
|
|
0 if not self._proxy_type
|
2019-10-03 17:34:27 +02:00
|
|
|
else self.comboBoxProxyType.findText(self._proxy_type))
|
|
|
|
self.comboBoxProxyTypeChanged(self.comboBoxProxyType.currentIndex())
|
2018-11-13 16:03:38 +01:00
|
|
|
|
2022-08-29 23:35:22 +02:00
|
|
|
if self._proxy_type:
|
|
|
|
for node, info in six.iteritems(
|
|
|
|
knownnodes.knownNodes.get(
|
|
|
|
min(state.streamsInWhichIAmParticipating), [])
|
|
|
|
):
|
|
|
|
if (
|
|
|
|
node.host.endswith('.onion') and len(node.host) > 22
|
|
|
|
and not info.get('self')
|
|
|
|
):
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
if self.checkBoxOnionOnly.isChecked():
|
|
|
|
self.checkBoxOnionOnly.setText(
|
|
|
|
self.checkBoxOnionOnly.text() + ", " + _translate(
|
|
|
|
"MainWindow", "may cause connection problems!"))
|
|
|
|
self.checkBoxOnionOnly.setStyleSheet(
|
|
|
|
"QCheckBox { color : red; }")
|
|
|
|
else:
|
|
|
|
self.checkBoxOnionOnly.setEnabled(False)
|
|
|
|
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditSocksHostname.setText(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'sockshostname'))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditSocksPort.setText(str(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'socksport')))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditSocksUsername.setText(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'socksusername'))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditSocksPassword.setText(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'sockspassword'))
|
2018-11-13 16:03:38 +01:00
|
|
|
|
|
|
|
self.lineEditMaxDownloadRate.setText(str(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'maxdownloadrate')))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditMaxUploadRate.setText(str(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'maxuploadrate')))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditMaxOutboundConnections.setText(str(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'maxoutboundconnections')))
|
2018-11-13 16:03:38 +01:00
|
|
|
|
|
|
|
# Demanded difficulty tab
|
|
|
|
self.lineEditTotalDifficulty.setText(str((float(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getint(
|
2018-11-13 16:03:38 +01:00
|
|
|
'bitmessagesettings', 'defaultnoncetrialsperbyte')
|
|
|
|
) / defaults.networkDefaultProofOfWorkNonceTrialsPerByte)))
|
|
|
|
self.lineEditSmallMessageDifficulty.setText(str((float(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getint(
|
2018-11-13 16:03:38 +01:00
|
|
|
'bitmessagesettings', 'defaultpayloadlengthextrabytes')
|
|
|
|
) / defaults.networkDefaultPayloadLengthExtraBytes)))
|
|
|
|
|
|
|
|
# Max acceptable difficulty tab
|
|
|
|
self.lineEditMaxAcceptableTotalDifficulty.setText(str((float(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getint(
|
2018-11-13 16:03:38 +01:00
|
|
|
'bitmessagesettings', 'maxacceptablenoncetrialsperbyte')
|
|
|
|
) / defaults.networkDefaultProofOfWorkNonceTrialsPerByte)))
|
|
|
|
self.lineEditMaxAcceptableSmallMessageDifficulty.setText(str((float(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.getint(
|
2018-11-13 16:03:38 +01:00
|
|
|
'bitmessagesettings', 'maxacceptablepayloadlengthextrabytes')
|
|
|
|
) / defaults.networkDefaultPayloadLengthExtraBytes)))
|
|
|
|
|
|
|
|
# OpenCL
|
|
|
|
self.comboBoxOpenCL.setEnabled(openclpow.openclAvailable())
|
|
|
|
self.comboBoxOpenCL.clear()
|
|
|
|
self.comboBoxOpenCL.addItem("None")
|
|
|
|
self.comboBoxOpenCL.addItems(openclpow.vendors)
|
|
|
|
self.comboBoxOpenCL.setCurrentIndex(0)
|
|
|
|
for i in range(self.comboBoxOpenCL.count()):
|
2022-02-16 21:16:41 +01:00
|
|
|
if self.comboBoxOpenCL.itemText(i) == config.safeGet(
|
2018-11-13 16:03:38 +01:00
|
|
|
'bitmessagesettings', 'opencl'):
|
|
|
|
self.comboBoxOpenCL.setCurrentIndex(i)
|
|
|
|
break
|
|
|
|
|
|
|
|
# Namecoin integration tab
|
2022-02-16 21:16:41 +01:00
|
|
|
nmctype = config.get('bitmessagesettings', 'namecoinrpctype')
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditNamecoinHost.setText(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'namecoinrpchost'))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditNamecoinPort.setText(str(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'namecoinrpcport')))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditNamecoinUser.setText(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'namecoinrpcuser'))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditNamecoinPassword.setText(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'namecoinrpcpassword'))
|
2018-11-13 16:03:38 +01:00
|
|
|
|
|
|
|
if nmctype == "namecoind":
|
|
|
|
self.radioButtonNamecoinNamecoind.setChecked(True)
|
|
|
|
elif nmctype == "nmcontrol":
|
|
|
|
self.radioButtonNamecoinNmcontrol.setChecked(True)
|
|
|
|
self.lineEditNamecoinUser.setEnabled(False)
|
|
|
|
self.labelNamecoinUser.setEnabled(False)
|
|
|
|
self.lineEditNamecoinPassword.setEnabled(False)
|
|
|
|
self.labelNamecoinPassword.setEnabled(False)
|
|
|
|
else:
|
|
|
|
assert False
|
|
|
|
|
|
|
|
# Message Resend tab
|
|
|
|
self.lineEditDays.setText(str(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'stopresendingafterxdays')))
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditMonths.setText(str(
|
2022-02-16 21:16:41 +01:00
|
|
|
config.get('bitmessagesettings', 'stopresendingafterxmonths')))
|
2018-11-13 16:03:38 +01:00
|
|
|
|
|
|
|
def comboBoxProxyTypeChanged(self, comboBoxIndex):
|
|
|
|
"""A callback for currentIndexChanged event of comboBoxProxyType"""
|
|
|
|
if comboBoxIndex == 0:
|
|
|
|
self.lineEditSocksHostname.setEnabled(False)
|
|
|
|
self.lineEditSocksPort.setEnabled(False)
|
|
|
|
self.lineEditSocksUsername.setEnabled(False)
|
|
|
|
self.lineEditSocksPassword.setEnabled(False)
|
|
|
|
self.checkBoxAuthentication.setEnabled(False)
|
|
|
|
self.checkBoxSocksListen.setEnabled(False)
|
2019-10-08 22:08:42 +02:00
|
|
|
self.checkBoxOnionOnly.setEnabled(False)
|
2019-10-03 17:34:27 +02:00
|
|
|
else:
|
2018-11-13 16:03:38 +01:00
|
|
|
self.lineEditSocksHostname.setEnabled(True)
|
|
|
|
self.lineEditSocksPort.setEnabled(True)
|
|
|
|
self.checkBoxAuthentication.setEnabled(True)
|
|
|
|
self.checkBoxSocksListen.setEnabled(True)
|
2019-10-08 22:08:42 +02:00
|
|
|
self.checkBoxOnionOnly.setEnabled(True)
|
2018-11-13 16:03:38 +01:00
|
|
|
if self.checkBoxAuthentication.isChecked():
|
|
|
|
self.lineEditSocksUsername.setEnabled(True)
|
|
|
|
self.lineEditSocksPassword.setEnabled(True)
|
|
|
|
|
|
|
|
def getNamecoinType(self):
|
|
|
|
"""
|
|
|
|
Check status of namecoin integration radio buttons
|
|
|
|
and translate it to a string as in the options.
|
|
|
|
"""
|
|
|
|
if self.radioButtonNamecoinNamecoind.isChecked():
|
|
|
|
return "namecoind"
|
|
|
|
if self.radioButtonNamecoinNmcontrol.isChecked():
|
|
|
|
return "nmcontrol"
|
|
|
|
assert False
|
|
|
|
|
|
|
|
# Namecoin connection type was changed.
|
|
|
|
def namecoinTypeChanged(self, checked): # pylint: disable=unused-argument
|
|
|
|
"""A callback for toggled event of radioButtonNamecoinNamecoind"""
|
|
|
|
nmctype = self.getNamecoinType()
|
|
|
|
assert nmctype == "namecoind" or nmctype == "nmcontrol"
|
|
|
|
|
|
|
|
isNamecoind = (nmctype == "namecoind")
|
|
|
|
self.lineEditNamecoinUser.setEnabled(isNamecoind)
|
|
|
|
self.labelNamecoinUser.setEnabled(isNamecoind)
|
|
|
|
self.lineEditNamecoinPassword.setEnabled(isNamecoind)
|
|
|
|
self.labelNamecoinPassword.setEnabled(isNamecoind)
|
|
|
|
|
|
|
|
if isNamecoind:
|
|
|
|
self.lineEditNamecoinPort.setText(defaults.namecoinDefaultRpcPort)
|
|
|
|
else:
|
|
|
|
self.lineEditNamecoinPort.setText("9000")
|
|
|
|
|
|
|
|
def click_pushButtonNamecoinTest(self):
|
|
|
|
"""Test the namecoin settings specified in the settings dialog."""
|
|
|
|
self.labelNamecoinTestResult.setText(
|
|
|
|
_translate("MainWindow", "Testing..."))
|
|
|
|
nc = namecoin.namecoinConnection({
|
|
|
|
'type': self.getNamecoinType(),
|
|
|
|
'host': str(self.lineEditNamecoinHost.text().toUtf8()),
|
|
|
|
'port': str(self.lineEditNamecoinPort.text().toUtf8()),
|
|
|
|
'user': str(self.lineEditNamecoinUser.text().toUtf8()),
|
|
|
|
'password': str(self.lineEditNamecoinPassword.text().toUtf8())
|
|
|
|
})
|
|
|
|
status, text = nc.test()
|
|
|
|
self.labelNamecoinTestResult.setText(text)
|
|
|
|
if status == 'success':
|
|
|
|
self.parent.namecoin = nc
|
|
|
|
|
|
|
|
def accept(self):
|
|
|
|
"""A callback for accepted event of buttonBox (OK button pressed)"""
|
|
|
|
# pylint: disable=too-many-branches,too-many-statements
|
|
|
|
super(SettingsDialog, self).accept()
|
|
|
|
if self.firstrun:
|
|
|
|
self.config.remove_option('bitmessagesettings', 'dontconnect')
|
|
|
|
self.config.set('bitmessagesettings', 'startonlogon', str(
|
|
|
|
self.checkBoxStartOnLogon.isChecked()))
|
|
|
|
self.config.set('bitmessagesettings', 'minimizetotray', str(
|
|
|
|
self.checkBoxMinimizeToTray.isChecked()))
|
|
|
|
self.config.set('bitmessagesettings', 'trayonclose', str(
|
|
|
|
self.checkBoxTrayOnClose.isChecked()))
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'hidetrayconnectionnotifications',
|
|
|
|
str(self.checkBoxHideTrayConnectionNotifications.isChecked()))
|
|
|
|
self.config.set('bitmessagesettings', 'showtraynotifications', str(
|
|
|
|
self.checkBoxShowTrayNotifications.isChecked()))
|
|
|
|
self.config.set('bitmessagesettings', 'startintray', str(
|
|
|
|
self.checkBoxStartInTray.isChecked()))
|
|
|
|
self.config.set('bitmessagesettings', 'willinglysendtomobile', str(
|
|
|
|
self.checkBoxWillinglySendToMobile.isChecked()))
|
|
|
|
self.config.set('bitmessagesettings', 'useidenticons', str(
|
|
|
|
self.checkBoxUseIdenticons.isChecked()))
|
|
|
|
self.config.set('bitmessagesettings', 'replybelow', str(
|
|
|
|
self.checkBoxReplyBelow.isChecked()))
|
|
|
|
|
|
|
|
lang = str(self.languageComboBox.itemData(
|
|
|
|
self.languageComboBox.currentIndex()).toString())
|
|
|
|
self.config.set('bitmessagesettings', 'userlocale', lang)
|
|
|
|
self.parent.change_translation()
|
|
|
|
|
|
|
|
if int(self.config.get('bitmessagesettings', 'port')) != int(
|
|
|
|
self.lineEditTCPPort.text()):
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'port', str(self.lineEditTCPPort.text()))
|
2020-09-16 17:41:51 +02:00
|
|
|
if not self.config.safeGetBoolean(
|
|
|
|
'bitmessagesettings', 'dontconnect'):
|
2018-11-20 15:53:25 +01:00
|
|
|
self.net_restart_needed = True
|
|
|
|
|
2018-11-13 16:03:38 +01:00
|
|
|
if self.checkBoxUPnP.isChecked() != self.config.safeGetBoolean(
|
|
|
|
'bitmessagesettings', 'upnp'):
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'upnp',
|
|
|
|
str(self.checkBoxUPnP.isChecked()))
|
|
|
|
if self.checkBoxUPnP.isChecked():
|
|
|
|
import upnp
|
|
|
|
upnpThread = upnp.uPnPThread()
|
|
|
|
upnpThread.start()
|
|
|
|
|
2020-09-16 17:41:51 +02:00
|
|
|
udp_enabled = self.checkBoxUDP.isChecked()
|
|
|
|
if udp_enabled != self.config.safeGetBoolean(
|
|
|
|
'bitmessagesettings', 'udp'):
|
|
|
|
self.config.set('bitmessagesettings', 'udp', str(udp_enabled))
|
|
|
|
if udp_enabled:
|
|
|
|
announceThread = AnnounceThread()
|
|
|
|
announceThread.daemon = True
|
|
|
|
announceThread.start()
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
state.announceThread.stopThread()
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
|
2019-10-03 17:34:27 +02:00
|
|
|
proxytype_index = self.comboBoxProxyType.currentIndex()
|
|
|
|
if proxytype_index == 0:
|
2019-11-05 17:54:04 +01:00
|
|
|
if self._proxy_type and state.statusIconColor != 'red':
|
2019-10-03 17:34:27 +02:00
|
|
|
self.net_restart_needed = True
|
2020-09-16 17:41:51 +02:00
|
|
|
elif state.statusIconColor == 'red' and self.config.safeGetBoolean(
|
|
|
|
'bitmessagesettings', 'dontconnect'):
|
2020-09-05 12:47:52 +02:00
|
|
|
self.net_restart_needed = False
|
2019-10-03 17:34:27 +02:00
|
|
|
elif self.comboBoxProxyType.currentText() != self._proxy_type:
|
2018-11-20 15:53:25 +01:00
|
|
|
self.net_restart_needed = True
|
2018-11-13 16:03:38 +01:00
|
|
|
self.parent.statusbar.clearMessage()
|
2019-07-30 10:16:48 +02:00
|
|
|
|
2018-11-20 15:53:25 +01:00
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'socksproxytype',
|
2019-10-03 17:34:27 +02:00
|
|
|
'none' if self.comboBoxProxyType.currentIndex() == 0
|
|
|
|
else str(self.comboBoxProxyType.currentText())
|
2018-11-20 15:53:25 +01:00
|
|
|
)
|
2019-10-03 17:34:27 +02:00
|
|
|
if proxytype_index > 2: # last literal proxytype in ui
|
|
|
|
start_proxyconfig()
|
|
|
|
|
2018-11-13 16:03:38 +01:00
|
|
|
self.config.set('bitmessagesettings', 'socksauthentication', str(
|
|
|
|
self.checkBoxAuthentication.isChecked()))
|
|
|
|
self.config.set('bitmessagesettings', 'sockshostname', str(
|
|
|
|
self.lineEditSocksHostname.text()))
|
|
|
|
self.config.set('bitmessagesettings', 'socksport', str(
|
|
|
|
self.lineEditSocksPort.text()))
|
|
|
|
self.config.set('bitmessagesettings', 'socksusername', str(
|
|
|
|
self.lineEditSocksUsername.text()))
|
|
|
|
self.config.set('bitmessagesettings', 'sockspassword', str(
|
|
|
|
self.lineEditSocksPassword.text()))
|
|
|
|
self.config.set('bitmessagesettings', 'sockslisten', str(
|
|
|
|
self.checkBoxSocksListen.isChecked()))
|
2020-09-16 17:41:51 +02:00
|
|
|
if (
|
|
|
|
self.checkBoxOnionOnly.isChecked()
|
|
|
|
and not self.config.safeGetBoolean(
|
|
|
|
'bitmessagesettings', 'onionservicesonly')
|
|
|
|
):
|
2019-10-08 22:08:42 +02:00
|
|
|
self.net_restart_needed = True
|
|
|
|
self.config.set('bitmessagesettings', 'onionservicesonly', str(
|
|
|
|
self.checkBoxOnionOnly.isChecked()))
|
2018-11-13 16:03:38 +01:00
|
|
|
try:
|
|
|
|
# Rounding to integers just for aesthetics
|
|
|
|
self.config.set('bitmessagesettings', 'maxdownloadrate', str(
|
|
|
|
int(float(self.lineEditMaxDownloadRate.text()))))
|
|
|
|
self.config.set('bitmessagesettings', 'maxuploadrate', str(
|
|
|
|
int(float(self.lineEditMaxUploadRate.text()))))
|
|
|
|
except ValueError:
|
|
|
|
QtGui.QMessageBox.about(
|
|
|
|
self, _translate("MainWindow", "Number needed"),
|
|
|
|
_translate(
|
|
|
|
"MainWindow",
|
|
|
|
"Your maximum download and upload rate must be numbers."
|
|
|
|
" Ignoring what you typed.")
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
set_rates(
|
|
|
|
self.config.safeGetInt('bitmessagesettings', 'maxdownloadrate'),
|
|
|
|
self.config.safeGetInt('bitmessagesettings', 'maxuploadrate'))
|
|
|
|
|
|
|
|
self.config.set('bitmessagesettings', 'maxoutboundconnections', str(
|
|
|
|
int(float(self.lineEditMaxOutboundConnections.text()))))
|
|
|
|
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'namecoinrpctype', self.getNamecoinType())
|
|
|
|
self.config.set('bitmessagesettings', 'namecoinrpchost', str(
|
|
|
|
self.lineEditNamecoinHost.text()))
|
|
|
|
self.config.set('bitmessagesettings', 'namecoinrpcport', str(
|
|
|
|
self.lineEditNamecoinPort.text()))
|
|
|
|
self.config.set('bitmessagesettings', 'namecoinrpcuser', str(
|
|
|
|
self.lineEditNamecoinUser.text()))
|
|
|
|
self.config.set('bitmessagesettings', 'namecoinrpcpassword', str(
|
|
|
|
self.lineEditNamecoinPassword.text()))
|
|
|
|
self.parent.resetNamecoinConnection()
|
|
|
|
|
|
|
|
# Demanded difficulty tab
|
|
|
|
if float(self.lineEditTotalDifficulty.text()) >= 1:
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'defaultnoncetrialsperbyte',
|
|
|
|
str(int(
|
2020-05-18 18:00:13 +02:00
|
|
|
float(self.lineEditTotalDifficulty.text())
|
|
|
|
* defaults.networkDefaultProofOfWorkNonceTrialsPerByte)))
|
2018-11-13 16:03:38 +01:00
|
|
|
if float(self.lineEditSmallMessageDifficulty.text()) >= 1:
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'defaultpayloadlengthextrabytes',
|
|
|
|
str(int(
|
2020-05-18 18:00:13 +02:00
|
|
|
float(self.lineEditSmallMessageDifficulty.text())
|
|
|
|
* defaults.networkDefaultPayloadLengthExtraBytes)))
|
2018-11-13 16:03:38 +01:00
|
|
|
|
|
|
|
if self.comboBoxOpenCL.currentText().toUtf8() != self.config.safeGet(
|
|
|
|
'bitmessagesettings', 'opencl'):
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'opencl',
|
|
|
|
str(self.comboBoxOpenCL.currentText()))
|
|
|
|
queues.workerQueue.put(('resetPoW', ''))
|
|
|
|
|
|
|
|
acceptableDifficultyChanged = False
|
|
|
|
|
|
|
|
if (
|
2020-09-16 17:41:51 +02:00
|
|
|
float(self.lineEditMaxAcceptableTotalDifficulty.text()) >= 1
|
|
|
|
or float(self.lineEditMaxAcceptableTotalDifficulty.text()) == 0
|
2018-11-13 16:03:38 +01:00
|
|
|
):
|
|
|
|
if self.config.get(
|
2020-06-09 12:17:45 +02:00
|
|
|
'bitmessagesettings', 'maxacceptablenoncetrialsperbyte'
|
2018-11-13 16:03:38 +01:00
|
|
|
) != str(int(
|
2020-05-18 18:00:13 +02:00
|
|
|
float(self.lineEditMaxAcceptableTotalDifficulty.text())
|
2020-06-09 12:17:45 +02:00
|
|
|
* defaults.networkDefaultProofOfWorkNonceTrialsPerByte)):
|
2018-11-13 16:03:38 +01:00
|
|
|
# the user changed the max acceptable total difficulty
|
|
|
|
acceptableDifficultyChanged = True
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'maxacceptablenoncetrialsperbyte',
|
|
|
|
str(int(
|
2020-05-18 18:00:13 +02:00
|
|
|
float(self.lineEditMaxAcceptableTotalDifficulty.text())
|
|
|
|
* defaults.networkDefaultProofOfWorkNonceTrialsPerByte))
|
2018-11-13 16:03:38 +01:00
|
|
|
)
|
|
|
|
if (
|
2020-09-16 17:41:51 +02:00
|
|
|
float(self.lineEditMaxAcceptableSmallMessageDifficulty.text()) >= 1
|
|
|
|
or float(self.lineEditMaxAcceptableSmallMessageDifficulty.text()) == 0
|
2018-11-13 16:03:38 +01:00
|
|
|
):
|
|
|
|
if self.config.get(
|
2020-06-09 12:17:45 +02:00
|
|
|
'bitmessagesettings', 'maxacceptablepayloadlengthextrabytes'
|
2018-11-13 16:03:38 +01:00
|
|
|
) != str(int(
|
2020-05-18 18:00:13 +02:00
|
|
|
float(self.lineEditMaxAcceptableSmallMessageDifficulty.text())
|
2020-06-09 12:17:45 +02:00
|
|
|
* defaults.networkDefaultPayloadLengthExtraBytes)):
|
2018-11-13 16:03:38 +01:00
|
|
|
# the user changed the max acceptable small message difficulty
|
|
|
|
acceptableDifficultyChanged = True
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'maxacceptablepayloadlengthextrabytes',
|
|
|
|
str(int(
|
2020-05-18 18:00:13 +02:00
|
|
|
float(self.lineEditMaxAcceptableSmallMessageDifficulty.text())
|
|
|
|
* defaults.networkDefaultPayloadLengthExtraBytes))
|
2018-11-13 16:03:38 +01:00
|
|
|
)
|
|
|
|
if acceptableDifficultyChanged:
|
|
|
|
# It might now be possible to send msgs which were previously
|
|
|
|
# marked as toodifficult. Let us change them to 'msgqueued'.
|
|
|
|
# The singleWorker will try to send them and will again mark
|
|
|
|
# them as toodifficult if the receiver's required difficulty
|
|
|
|
# is still higher than we are willing to do.
|
|
|
|
sqlExecute(
|
|
|
|
"UPDATE sent SET status='msgqueued'"
|
|
|
|
" WHERE status='toodifficult'")
|
|
|
|
queues.workerQueue.put(('sendmessage', ''))
|
|
|
|
|
2020-04-28 17:31:44 +02:00
|
|
|
stopResendingDefaults = False
|
|
|
|
|
2018-11-13 16:03:38 +01:00
|
|
|
# UI setting to stop trying to send messages after X days/months
|
|
|
|
# I'm open to changing this UI to something else if someone has a better idea.
|
|
|
|
if self.lineEditDays.text() == '' and self.lineEditMonths.text() == '':
|
|
|
|
# We need to handle this special case. Bitmessage has its
|
|
|
|
# default behavior. The input is blank/blank
|
|
|
|
self.config.set('bitmessagesettings', 'stopresendingafterxdays', '')
|
|
|
|
self.config.set('bitmessagesettings', 'stopresendingafterxmonths', '')
|
2019-11-05 17:54:04 +01:00
|
|
|
state.maximumLengthOfTimeToBotherResendingMessages = float('inf')
|
2020-04-28 17:31:44 +02:00
|
|
|
stopResendingDefaults = True
|
2018-11-13 16:03:38 +01:00
|
|
|
|
|
|
|
try:
|
|
|
|
days = float(self.lineEditDays.text())
|
|
|
|
except ValueError:
|
|
|
|
self.lineEditDays.setText("0")
|
|
|
|
days = 0.0
|
|
|
|
try:
|
|
|
|
months = float(self.lineEditMonths.text())
|
|
|
|
except ValueError:
|
|
|
|
self.lineEditMonths.setText("0")
|
|
|
|
months = 0.0
|
|
|
|
|
2020-04-28 17:31:44 +02:00
|
|
|
if days >= 0 and months >= 0 and not stopResendingDefaults:
|
2019-11-05 17:54:04 +01:00
|
|
|
state.maximumLengthOfTimeToBotherResendingMessages = \
|
2018-11-13 16:03:38 +01:00
|
|
|
days * 24 * 60 * 60 + months * 60 * 60 * 24 * 365 / 12
|
2019-11-05 17:54:04 +01:00
|
|
|
if state.maximumLengthOfTimeToBotherResendingMessages < 432000:
|
2018-11-13 16:03:38 +01:00
|
|
|
# If the time period is less than 5 hours, we give
|
|
|
|
# zero values to all fields. No message will be sent again.
|
|
|
|
QtGui.QMessageBox.about(
|
|
|
|
self,
|
|
|
|
_translate("MainWindow", "Will not resend ever"),
|
|
|
|
_translate(
|
|
|
|
"MainWindow",
|
|
|
|
"Note that the time limit you entered is less"
|
|
|
|
" than the amount of time Bitmessage waits for"
|
|
|
|
" the first resend attempt therefore your"
|
|
|
|
" messages will never be resent.")
|
|
|
|
)
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'stopresendingafterxdays', '0')
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'stopresendingafterxmonths', '0')
|
2019-11-05 17:54:04 +01:00
|
|
|
state.maximumLengthOfTimeToBotherResendingMessages = 0.0
|
2018-11-13 16:03:38 +01:00
|
|
|
else:
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'stopresendingafterxdays', str(days))
|
|
|
|
self.config.set(
|
|
|
|
'bitmessagesettings', 'stopresendingafterxmonths',
|
|
|
|
str(months))
|
|
|
|
|
|
|
|
self.config.save()
|
|
|
|
|
2018-11-20 15:53:25 +01:00
|
|
|
if self.net_restart_needed:
|
|
|
|
self.net_restart_needed = False
|
2019-09-05 17:31:16 +02:00
|
|
|
self.config.setTemp('bitmessagesettings', 'dontconnect', 'true')
|
2018-11-20 15:53:25 +01:00
|
|
|
self.timer.singleShot(
|
|
|
|
5000, lambda:
|
2019-09-05 17:31:16 +02:00
|
|
|
self.config.setTemp(
|
|
|
|
'bitmessagesettings', 'dontconnect', 'false')
|
2018-11-20 15:53:25 +01:00
|
|
|
)
|
|
|
|
|
2018-11-13 16:03:38 +01:00
|
|
|
self.parent.updateStartOnLogon()
|
|
|
|
|
|
|
|
if (
|
2020-09-16 17:41:51 +02:00
|
|
|
state.appdata != paths.lookupExeFolder()
|
|
|
|
and self.checkBoxPortableMode.isChecked()
|
2018-11-13 16:03:38 +01:00
|
|
|
):
|
|
|
|
# If we are NOT using portable mode now but the user selected
|
|
|
|
# that we should...
|
|
|
|
# Write the keys.dat file to disk in the new location
|
|
|
|
sqlStoredProcedure('movemessagstoprog')
|
|
|
|
with open(paths.lookupExeFolder() + 'keys.dat', 'wb') as configfile:
|
|
|
|
self.config.write(configfile)
|
|
|
|
# Write the knownnodes.dat file to disk in the new location
|
|
|
|
knownnodes.saveKnownNodes(paths.lookupExeFolder())
|
|
|
|
os.remove(state.appdata + 'keys.dat')
|
|
|
|
os.remove(state.appdata + 'knownnodes.dat')
|
|
|
|
previousAppdataLocation = state.appdata
|
|
|
|
state.appdata = paths.lookupExeFolder()
|
|
|
|
debug.resetLogging()
|
|
|
|
try:
|
|
|
|
os.remove(previousAppdataLocation + 'debug.log')
|
|
|
|
os.remove(previousAppdataLocation + 'debug.log.1')
|
2020-05-18 18:00:13 +02:00
|
|
|
except Exception:
|
2018-11-13 16:03:38 +01:00
|
|
|
pass
|
|
|
|
|
|
|
|
if (
|
2020-09-16 17:41:51 +02:00
|
|
|
state.appdata == paths.lookupExeFolder()
|
|
|
|
and not self.checkBoxPortableMode.isChecked()
|
2018-11-13 16:03:38 +01:00
|
|
|
):
|
|
|
|
# If we ARE using portable mode now but the user selected
|
|
|
|
# that we shouldn't...
|
|
|
|
state.appdata = paths.lookupAppdataFolder()
|
|
|
|
if not os.path.exists(state.appdata):
|
|
|
|
os.makedirs(state.appdata)
|
|
|
|
sqlStoredProcedure('movemessagstoappdata')
|
|
|
|
# Write the keys.dat file to disk in the new location
|
|
|
|
self.config.save()
|
|
|
|
# Write the knownnodes.dat file to disk in the new location
|
|
|
|
knownnodes.saveKnownNodes(state.appdata)
|
|
|
|
os.remove(paths.lookupExeFolder() + 'keys.dat')
|
|
|
|
os.remove(paths.lookupExeFolder() + 'knownnodes.dat')
|
|
|
|
debug.resetLogging()
|
|
|
|
try:
|
|
|
|
os.remove(paths.lookupExeFolder() + 'debug.log')
|
|
|
|
os.remove(paths.lookupExeFolder() + 'debug.log.1')
|
2020-05-18 18:00:13 +02:00
|
|
|
except Exception:
|
2018-11-13 16:03:38 +01:00
|
|
|
pass
|