2016-12-28 18:44:41 -08:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
"""
|
|
|
|
OnionShare | https://onionshare.org/
|
|
|
|
|
2018-04-24 10:07:59 -07:00
|
|
|
Copyright (C) 2014-2018 Micah Lee <micah@micahflee.com>
|
2016-12-28 18:44:41 -08:00
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
"""
|
|
|
|
from PyQt5 import QtCore, QtWidgets, QtGui
|
2018-01-14 20:12:24 +11:00
|
|
|
import sys, platform, datetime, re
|
2016-12-28 18:44:41 -08:00
|
|
|
|
2017-05-16 11:05:48 -07:00
|
|
|
from onionshare import strings, common
|
2016-12-28 19:52:21 -08:00
|
|
|
from onionshare.settings import Settings
|
2017-04-08 18:10:17 -07:00
|
|
|
from onionshare.onion import *
|
2016-12-29 09:58:13 -08:00
|
|
|
|
|
|
|
from .alert import Alert
|
2017-04-15 18:04:05 -07:00
|
|
|
from .update_checker import *
|
2017-05-16 16:50:33 -07:00
|
|
|
from .tor_connection_dialog import TorConnectionDialog
|
2016-12-28 18:44:41 -08:00
|
|
|
|
|
|
|
class SettingsDialog(QtWidgets.QDialog):
|
|
|
|
"""
|
|
|
|
Settings dialog.
|
|
|
|
"""
|
2017-05-22 17:08:05 -07:00
|
|
|
settings_saved = QtCore.pyqtSignal()
|
|
|
|
|
2018-04-22 17:38:28 -07:00
|
|
|
def __init__(self, common, onion, qtapp, config=False, local_only=False):
|
2017-04-13 22:56:47 -07:00
|
|
|
super(SettingsDialog, self).__init__()
|
2018-03-08 10:18:31 -08:00
|
|
|
|
|
|
|
self.common = common
|
|
|
|
|
|
|
|
self.common.log('SettingsDialog', '__init__')
|
2017-05-16 11:31:52 -07:00
|
|
|
|
2017-05-14 19:54:12 -07:00
|
|
|
self.onion = onion
|
2017-04-13 22:56:47 -07:00
|
|
|
self.qtapp = qtapp
|
2017-06-01 17:35:27 +10:00
|
|
|
self.config = config
|
2018-03-07 16:13:22 +11:00
|
|
|
self.local_only = local_only
|
2016-12-28 18:44:41 -08:00
|
|
|
|
|
|
|
self.setModal(True)
|
|
|
|
self.setWindowTitle(strings._('gui_settings_window_title', True))
|
2018-03-08 10:18:31 -08:00
|
|
|
self.setWindowIcon(QtGui.QIcon(self.common.get_resource_path('images/logo.png')))
|
2016-12-28 18:44:41 -08:00
|
|
|
|
2018-02-26 14:02:15 +11:00
|
|
|
self.system = platform.system()
|
2017-04-15 15:24:08 -07:00
|
|
|
|
2017-04-08 13:42:07 -07:00
|
|
|
# Sharing options
|
|
|
|
|
|
|
|
# Close after first download
|
|
|
|
self.close_after_first_download_checkbox = QtWidgets.QCheckBox()
|
|
|
|
self.close_after_first_download_checkbox.setCheckState(QtCore.Qt.Checked)
|
|
|
|
self.close_after_first_download_checkbox.setText(strings._("gui_settings_close_after_first_download_option", True))
|
|
|
|
|
2018-02-07 09:55:55 -08:00
|
|
|
# Whether or not to use a shutdown timer
|
|
|
|
self.shutdown_timeout_checkbox = QtWidgets.QCheckBox()
|
|
|
|
self.shutdown_timeout_checkbox.setCheckState(QtCore.Qt.Checked)
|
|
|
|
self.shutdown_timeout_checkbox.setText(strings._("gui_settings_shutdown_timeout_checkbox", True))
|
|
|
|
|
2017-12-07 12:45:29 +11:00
|
|
|
# Whether or not to save the Onion private key for reuse
|
|
|
|
self.save_private_key_checkbox = QtWidgets.QCheckBox()
|
|
|
|
self.save_private_key_checkbox.setCheckState(QtCore.Qt.Unchecked)
|
|
|
|
self.save_private_key_checkbox.setText(strings._("gui_save_private_key_checkbox", True))
|
|
|
|
|
2017-04-08 13:42:07 -07:00
|
|
|
# Sharing options layout
|
|
|
|
sharing_group_layout = QtWidgets.QVBoxLayout()
|
|
|
|
sharing_group_layout.addWidget(self.close_after_first_download_checkbox)
|
2018-02-07 09:55:55 -08:00
|
|
|
sharing_group_layout.addWidget(self.shutdown_timeout_checkbox)
|
2017-12-07 12:45:29 +11:00
|
|
|
sharing_group_layout.addWidget(self.save_private_key_checkbox)
|
2017-04-08 13:42:07 -07:00
|
|
|
sharing_group = QtWidgets.QGroupBox(strings._("gui_settings_sharing_label", True))
|
|
|
|
sharing_group.setLayout(sharing_group_layout)
|
|
|
|
|
|
|
|
# Stealth options
|
|
|
|
|
|
|
|
# Stealth
|
|
|
|
stealth_details = QtWidgets.QLabel(strings._("gui_settings_stealth_option_details", True))
|
|
|
|
stealth_details.setWordWrap(True)
|
2017-12-27 20:08:38 +11:00
|
|
|
stealth_details.setTextInteractionFlags(QtCore.Qt.TextBrowserInteraction)
|
|
|
|
stealth_details.setOpenExternalLinks(True)
|
2018-02-08 16:02:00 +11:00
|
|
|
stealth_details.setMinimumSize(stealth_details.sizeHint())
|
2017-04-08 13:42:07 -07:00
|
|
|
self.stealth_checkbox = QtWidgets.QCheckBox()
|
|
|
|
self.stealth_checkbox.setCheckState(QtCore.Qt.Unchecked)
|
|
|
|
self.stealth_checkbox.setText(strings._("gui_settings_stealth_option", True))
|
|
|
|
|
2017-12-09 06:49:34 +11:00
|
|
|
hidservauth_details = QtWidgets.QLabel(strings._('gui_settings_stealth_hidservauth_string', True))
|
|
|
|
hidservauth_details.setWordWrap(True)
|
2018-02-08 16:02:00 +11:00
|
|
|
hidservauth_details.setMinimumSize(hidservauth_details.sizeHint())
|
2017-12-09 06:49:34 +11:00
|
|
|
hidservauth_details.hide()
|
|
|
|
|
|
|
|
self.hidservauth_copy_button = QtWidgets.QPushButton(strings._('gui_copy_hidservauth', True))
|
|
|
|
self.hidservauth_copy_button.clicked.connect(self.hidservauth_copy_button_clicked)
|
|
|
|
self.hidservauth_copy_button.hide()
|
|
|
|
|
2017-04-08 13:42:07 -07:00
|
|
|
# Stealth options layout
|
|
|
|
stealth_group_layout = QtWidgets.QVBoxLayout()
|
|
|
|
stealth_group_layout.addWidget(stealth_details)
|
|
|
|
stealth_group_layout.addWidget(self.stealth_checkbox)
|
2017-12-09 06:49:34 +11:00
|
|
|
stealth_group_layout.addWidget(hidservauth_details)
|
|
|
|
stealth_group_layout.addWidget(self.hidservauth_copy_button)
|
2017-04-08 13:42:07 -07:00
|
|
|
stealth_group = QtWidgets.QGroupBox(strings._("gui_settings_stealth_label", True))
|
|
|
|
stealth_group.setLayout(stealth_group_layout)
|
|
|
|
|
2017-04-15 15:24:08 -07:00
|
|
|
# Automatic updates options
|
|
|
|
|
|
|
|
# Autoupdate
|
|
|
|
self.autoupdate_checkbox = QtWidgets.QCheckBox()
|
|
|
|
self.autoupdate_checkbox.setCheckState(QtCore.Qt.Unchecked)
|
|
|
|
self.autoupdate_checkbox.setText(strings._("gui_settings_autoupdate_option", True))
|
|
|
|
|
|
|
|
# Last update time
|
|
|
|
self.autoupdate_timestamp = QtWidgets.QLabel()
|
|
|
|
|
|
|
|
# Check for updates button
|
|
|
|
self.check_for_updates_button = QtWidgets.QPushButton(strings._('gui_settings_autoupdate_check_button', True))
|
|
|
|
self.check_for_updates_button.clicked.connect(self.check_for_updates)
|
2018-01-03 11:16:50 +11:00
|
|
|
# We can't check for updates if not connected to Tor
|
|
|
|
if not self.onion.connected_to_tor:
|
|
|
|
self.check_for_updates_button.setEnabled(False)
|
2017-04-15 15:24:08 -07:00
|
|
|
|
|
|
|
# Autoupdate options layout
|
|
|
|
autoupdate_group_layout = QtWidgets.QVBoxLayout()
|
|
|
|
autoupdate_group_layout.addWidget(self.autoupdate_checkbox)
|
|
|
|
autoupdate_group_layout.addWidget(self.autoupdate_timestamp)
|
|
|
|
autoupdate_group_layout.addWidget(self.check_for_updates_button)
|
|
|
|
autoupdate_group = QtWidgets.QGroupBox(strings._("gui_settings_autoupdate_label", True))
|
|
|
|
autoupdate_group.setLayout(autoupdate_group_layout)
|
|
|
|
|
|
|
|
# Autoupdate is only available for Windows and Mac (Linux updates using package manager)
|
2018-02-26 14:02:15 +11:00
|
|
|
if self.system != 'Windows' and self.system != 'Darwin':
|
2017-04-15 15:24:08 -07:00
|
|
|
autoupdate_group.hide()
|
|
|
|
|
2016-12-28 18:44:41 -08:00
|
|
|
# Connection type: either automatic, control port, or socket file
|
|
|
|
|
2017-04-08 17:48:58 -07:00
|
|
|
# Bundled Tor
|
|
|
|
self.connection_type_bundled_radio = QtWidgets.QRadioButton(strings._('gui_settings_connection_type_bundled_option', True))
|
|
|
|
self.connection_type_bundled_radio.toggled.connect(self.connection_type_bundled_toggled)
|
|
|
|
|
2017-04-13 22:22:34 -07:00
|
|
|
# Bundled Tor doesn't work on dev mode in Windows or Mac
|
2018-02-26 14:02:15 +11:00
|
|
|
if (self.system == 'Windows' or self.system == 'Darwin') and getattr(sys, 'onionshare_dev_mode', False):
|
2017-04-08 17:48:58 -07:00
|
|
|
self.connection_type_bundled_radio.setEnabled(False)
|
|
|
|
|
2017-12-12 08:43:12 +11:00
|
|
|
# Bridge options for bundled tor
|
|
|
|
|
|
|
|
# No bridges option radio
|
|
|
|
self.tor_bridges_no_bridges_radio = QtWidgets.QRadioButton(strings._('gui_settings_tor_bridges_no_bridges_radio_option', True))
|
|
|
|
self.tor_bridges_no_bridges_radio.toggled.connect(self.tor_bridges_no_bridges_radio_toggled)
|
|
|
|
|
2018-01-17 16:30:12 +11:00
|
|
|
# obfs4 option radio
|
2018-01-15 12:49:29 +11:00
|
|
|
# if the obfs4proxy binary is missing, we can't use obfs4 transports
|
2018-03-08 10:18:31 -08:00
|
|
|
(self.tor_path, self.tor_geo_ip_file_path, self.tor_geo_ipv6_file_path, self.obfs4proxy_file_path) = self.common.get_tor_paths()
|
2018-01-17 16:30:12 +11:00
|
|
|
if not os.path.isfile(self.obfs4proxy_file_path):
|
|
|
|
self.tor_bridges_use_obfs4_radio = QtWidgets.QRadioButton(strings._('gui_settings_tor_bridges_obfs4_radio_option_no_obfs4proxy', True))
|
|
|
|
self.tor_bridges_use_obfs4_radio.setEnabled(False)
|
|
|
|
else:
|
|
|
|
self.tor_bridges_use_obfs4_radio = QtWidgets.QRadioButton(strings._('gui_settings_tor_bridges_obfs4_radio_option', True))
|
|
|
|
self.tor_bridges_use_obfs4_radio.toggled.connect(self.tor_bridges_use_obfs4_radio_toggled)
|
2017-12-12 08:43:12 +11:00
|
|
|
|
2018-02-15 23:19:53 +00:00
|
|
|
# meek_lite-amazon option radio
|
|
|
|
# if the obfs4proxy binary is missing, we can't use meek_lite-amazon transports
|
2018-03-08 10:18:31 -08:00
|
|
|
(self.tor_path, self.tor_geo_ip_file_path, self.tor_geo_ipv6_file_path, self.obfs4proxy_file_path) = self.common.get_tor_paths()
|
2018-02-15 23:19:53 +00:00
|
|
|
if not os.path.isfile(self.obfs4proxy_file_path):
|
|
|
|
self.tor_bridges_use_meek_lite_amazon_radio = QtWidgets.QRadioButton(strings._('gui_settings_tor_bridges_meek_lite_amazon_radio_option_no_obfs4proxy', True))
|
2018-02-24 14:09:42 -08:00
|
|
|
self.tor_bridges_use_meek_lite_amazon_radio.setEnabled(False)
|
2018-02-15 23:19:53 +00:00
|
|
|
else:
|
|
|
|
self.tor_bridges_use_meek_lite_amazon_radio = QtWidgets.QRadioButton(strings._('gui_settings_tor_bridges_meek_lite_amazon_radio_option', True))
|
|
|
|
self.tor_bridges_use_meek_lite_amazon_radio.toggled.connect(self.tor_bridges_use_meek_lite_amazon_radio_toggled)
|
|
|
|
|
|
|
|
# meek_lite-azure option radio
|
|
|
|
# if the obfs4proxy binary is missing, we can't use meek_lite-azure transports
|
2018-03-08 10:18:31 -08:00
|
|
|
(self.tor_path, self.tor_geo_ip_file_path, self.tor_geo_ipv6_file_path, self.obfs4proxy_file_path) = self.common.get_tor_paths()
|
2018-02-15 23:19:53 +00:00
|
|
|
if not os.path.isfile(self.obfs4proxy_file_path):
|
|
|
|
self.tor_bridges_use_meek_lite_azure_radio = QtWidgets.QRadioButton(strings._('gui_settings_tor_bridges_meek_lite_azure_radio_option_no_obfs4proxy', True))
|
2018-02-24 14:09:42 -08:00
|
|
|
self.tor_bridges_use_meek_lite_azure_radio.setEnabled(False)
|
2018-02-15 23:19:53 +00:00
|
|
|
else:
|
|
|
|
self.tor_bridges_use_meek_lite_azure_radio = QtWidgets.QRadioButton(strings._('gui_settings_tor_bridges_meek_lite_azure_radio_option', True))
|
|
|
|
self.tor_bridges_use_meek_lite_azure_radio.toggled.connect(self.tor_bridges_use_meek_lite_azure_radio_toggled)
|
|
|
|
|
2018-02-26 13:48:28 +11:00
|
|
|
# meek_lite currently not supported on the version of obfs4proxy bundled with TorBrowser
|
2018-02-26 14:02:15 +11:00
|
|
|
if self.system == 'Windows' or self.system == 'Darwin':
|
2018-02-26 13:48:28 +11:00
|
|
|
self.tor_bridges_use_meek_lite_amazon_radio.hide()
|
|
|
|
self.tor_bridges_use_meek_lite_azure_radio.hide()
|
|
|
|
|
2017-12-12 08:43:12 +11:00
|
|
|
# Custom bridges radio and textbox
|
|
|
|
self.tor_bridges_use_custom_radio = QtWidgets.QRadioButton(strings._('gui_settings_tor_bridges_custom_radio_option', True))
|
|
|
|
self.tor_bridges_use_custom_radio.toggled.connect(self.tor_bridges_use_custom_radio_toggled)
|
|
|
|
|
|
|
|
self.tor_bridges_use_custom_label = QtWidgets.QLabel(strings._('gui_settings_tor_bridges_custom_label', True))
|
2018-01-14 18:57:52 +11:00
|
|
|
self.tor_bridges_use_custom_label.setTextInteractionFlags(QtCore.Qt.TextBrowserInteraction)
|
|
|
|
self.tor_bridges_use_custom_label.setOpenExternalLinks(True)
|
2017-12-12 08:43:12 +11:00
|
|
|
self.tor_bridges_use_custom_textbox = QtWidgets.QPlainTextEdit()
|
2018-01-15 10:49:17 +11:00
|
|
|
self.tor_bridges_use_custom_textbox.setMaximumHeight(200)
|
2017-12-12 08:43:12 +11:00
|
|
|
self.tor_bridges_use_custom_textbox.setPlaceholderText('[address:port] [identifier]')
|
|
|
|
|
|
|
|
tor_bridges_use_custom_textbox_options_layout = QtWidgets.QVBoxLayout()
|
|
|
|
tor_bridges_use_custom_textbox_options_layout.addWidget(self.tor_bridges_use_custom_label)
|
|
|
|
tor_bridges_use_custom_textbox_options_layout.addWidget(self.tor_bridges_use_custom_textbox)
|
|
|
|
|
|
|
|
self.tor_bridges_use_custom_textbox_options = QtWidgets.QWidget()
|
|
|
|
self.tor_bridges_use_custom_textbox_options.setLayout(tor_bridges_use_custom_textbox_options_layout)
|
|
|
|
self.tor_bridges_use_custom_textbox_options.hide()
|
|
|
|
|
|
|
|
# Bridges layout/widget
|
|
|
|
bridges_layout = QtWidgets.QVBoxLayout()
|
|
|
|
bridges_layout.addWidget(self.tor_bridges_no_bridges_radio)
|
|
|
|
bridges_layout.addWidget(self.tor_bridges_use_obfs4_radio)
|
2018-02-15 23:19:53 +00:00
|
|
|
bridges_layout.addWidget(self.tor_bridges_use_meek_lite_amazon_radio)
|
|
|
|
bridges_layout.addWidget(self.tor_bridges_use_meek_lite_azure_radio)
|
2017-12-12 08:43:12 +11:00
|
|
|
bridges_layout.addWidget(self.tor_bridges_use_custom_radio)
|
|
|
|
bridges_layout.addWidget(self.tor_bridges_use_custom_textbox_options)
|
|
|
|
|
|
|
|
self.bridges = QtWidgets.QWidget()
|
|
|
|
self.bridges.setLayout(bridges_layout)
|
|
|
|
|
2016-12-28 18:44:41 -08:00
|
|
|
# Automatic
|
|
|
|
self.connection_type_automatic_radio = QtWidgets.QRadioButton(strings._('gui_settings_connection_type_automatic_option', True))
|
|
|
|
self.connection_type_automatic_radio.toggled.connect(self.connection_type_automatic_toggled)
|
|
|
|
|
|
|
|
# Control port
|
|
|
|
self.connection_type_control_port_radio = QtWidgets.QRadioButton(strings._('gui_settings_connection_type_control_port_option', True))
|
|
|
|
self.connection_type_control_port_radio.toggled.connect(self.connection_type_control_port_toggled)
|
|
|
|
|
|
|
|
connection_type_control_port_extras_label = QtWidgets.QLabel(strings._('gui_settings_control_port_label', True))
|
2016-12-28 19:52:21 -08:00
|
|
|
self.connection_type_control_port_extras_address = QtWidgets.QLineEdit()
|
|
|
|
self.connection_type_control_port_extras_port = QtWidgets.QLineEdit()
|
2016-12-28 18:44:41 -08:00
|
|
|
connection_type_control_port_extras_layout = QtWidgets.QHBoxLayout()
|
|
|
|
connection_type_control_port_extras_layout.addWidget(connection_type_control_port_extras_label)
|
|
|
|
connection_type_control_port_extras_layout.addWidget(self.connection_type_control_port_extras_address)
|
|
|
|
connection_type_control_port_extras_layout.addWidget(self.connection_type_control_port_extras_port)
|
|
|
|
|
|
|
|
self.connection_type_control_port_extras = QtWidgets.QWidget()
|
|
|
|
self.connection_type_control_port_extras.setLayout(connection_type_control_port_extras_layout)
|
|
|
|
self.connection_type_control_port_extras.hide()
|
|
|
|
|
|
|
|
# Socket file
|
|
|
|
self.connection_type_socket_file_radio = QtWidgets.QRadioButton(strings._('gui_settings_connection_type_socket_file_option', True))
|
|
|
|
self.connection_type_socket_file_radio.toggled.connect(self.connection_type_socket_file_toggled)
|
|
|
|
|
|
|
|
connection_type_socket_file_extras_label = QtWidgets.QLabel(strings._('gui_settings_socket_file_label', True))
|
2016-12-28 19:52:21 -08:00
|
|
|
self.connection_type_socket_file_extras_path = QtWidgets.QLineEdit()
|
2016-12-28 18:44:41 -08:00
|
|
|
connection_type_socket_file_extras_layout = QtWidgets.QHBoxLayout()
|
|
|
|
connection_type_socket_file_extras_layout.addWidget(connection_type_socket_file_extras_label)
|
|
|
|
connection_type_socket_file_extras_layout.addWidget(self.connection_type_socket_file_extras_path)
|
|
|
|
|
|
|
|
self.connection_type_socket_file_extras = QtWidgets.QWidget()
|
|
|
|
self.connection_type_socket_file_extras.setLayout(connection_type_socket_file_extras_layout)
|
|
|
|
self.connection_type_socket_file_extras.hide()
|
|
|
|
|
2017-04-15 16:33:41 -07:00
|
|
|
# Tor SOCKS address and port
|
|
|
|
gui_settings_socks_label = QtWidgets.QLabel(strings._('gui_settings_socks_label', True))
|
|
|
|
self.connection_type_socks_address = QtWidgets.QLineEdit()
|
|
|
|
self.connection_type_socks_port = QtWidgets.QLineEdit()
|
|
|
|
connection_type_socks_layout = QtWidgets.QHBoxLayout()
|
|
|
|
connection_type_socks_layout.addWidget(gui_settings_socks_label)
|
|
|
|
connection_type_socks_layout.addWidget(self.connection_type_socks_address)
|
|
|
|
connection_type_socks_layout.addWidget(self.connection_type_socks_port)
|
|
|
|
|
|
|
|
self.connection_type_socks = QtWidgets.QWidget()
|
|
|
|
self.connection_type_socks.setLayout(connection_type_socks_layout)
|
|
|
|
self.connection_type_socks.hide()
|
|
|
|
|
2016-12-28 18:44:41 -08:00
|
|
|
# Authentication options
|
|
|
|
|
|
|
|
# No authentication
|
|
|
|
self.authenticate_no_auth_radio = QtWidgets.QRadioButton(strings._('gui_settings_authenticate_no_auth_option', True))
|
|
|
|
self.authenticate_no_auth_radio.toggled.connect(self.authenticate_no_auth_toggled)
|
|
|
|
|
|
|
|
# Password
|
|
|
|
self.authenticate_password_radio = QtWidgets.QRadioButton(strings._('gui_settings_authenticate_password_option', True))
|
|
|
|
self.authenticate_password_radio.toggled.connect(self.authenticate_password_toggled)
|
|
|
|
|
|
|
|
authenticate_password_extras_label = QtWidgets.QLabel(strings._('gui_settings_password_label', True))
|
|
|
|
self.authenticate_password_extras_password = QtWidgets.QLineEdit('')
|
|
|
|
authenticate_password_extras_layout = QtWidgets.QHBoxLayout()
|
|
|
|
authenticate_password_extras_layout.addWidget(authenticate_password_extras_label)
|
|
|
|
authenticate_password_extras_layout.addWidget(self.authenticate_password_extras_password)
|
|
|
|
|
|
|
|
self.authenticate_password_extras = QtWidgets.QWidget()
|
|
|
|
self.authenticate_password_extras.setLayout(authenticate_password_extras_layout)
|
|
|
|
self.authenticate_password_extras.hide()
|
|
|
|
|
|
|
|
# Authentication options layout
|
|
|
|
authenticate_group_layout = QtWidgets.QVBoxLayout()
|
|
|
|
authenticate_group_layout.addWidget(self.authenticate_no_auth_radio)
|
|
|
|
authenticate_group_layout.addWidget(self.authenticate_password_radio)
|
|
|
|
authenticate_group_layout.addWidget(self.authenticate_password_extras)
|
|
|
|
self.authenticate_group = QtWidgets.QGroupBox(strings._("gui_settings_authenticate_label", True))
|
|
|
|
self.authenticate_group.setLayout(authenticate_group_layout)
|
|
|
|
|
2017-05-18 17:10:47 +10:00
|
|
|
# Put the radios into their own group so they are exclusive
|
|
|
|
connection_type_radio_group_layout = QtWidgets.QVBoxLayout()
|
|
|
|
connection_type_radio_group_layout.addWidget(self.connection_type_bundled_radio)
|
|
|
|
connection_type_radio_group_layout.addWidget(self.connection_type_automatic_radio)
|
|
|
|
connection_type_radio_group_layout.addWidget(self.connection_type_control_port_radio)
|
|
|
|
connection_type_radio_group_layout.addWidget(self.connection_type_socket_file_radio)
|
|
|
|
connection_type_radio_group = QtWidgets.QGroupBox(strings._("gui_settings_connection_type_label", True))
|
|
|
|
connection_type_radio_group.setLayout(connection_type_radio_group_layout)
|
|
|
|
|
2017-12-11 14:53:13 +11:00
|
|
|
# The Bridges options are not exclusive (enabling Bridges offers obfs4 or custom bridges)
|
|
|
|
connection_type_bridges_radio_group_layout = QtWidgets.QVBoxLayout()
|
2017-12-12 08:43:12 +11:00
|
|
|
connection_type_bridges_radio_group_layout.addWidget(self.bridges)
|
|
|
|
self.connection_type_bridges_radio_group = QtWidgets.QGroupBox(strings._("gui_settings_tor_bridges", True))
|
|
|
|
self.connection_type_bridges_radio_group.setLayout(connection_type_bridges_radio_group_layout)
|
|
|
|
self.connection_type_bridges_radio_group.hide()
|
2017-12-11 14:53:13 +11:00
|
|
|
|
2018-04-22 17:20:58 -07:00
|
|
|
# Test tor settings button
|
|
|
|
self.connection_type_test_button = QtWidgets.QPushButton(strings._('gui_settings_connection_type_test_button', True))
|
|
|
|
self.connection_type_test_button.clicked.connect(self.test_tor_clicked)
|
|
|
|
connection_type_test_button_layout = QtWidgets.QHBoxLayout()
|
|
|
|
connection_type_test_button_layout.addWidget(self.connection_type_test_button)
|
|
|
|
connection_type_test_button_layout.addStretch()
|
|
|
|
|
2018-04-22 17:15:15 -07:00
|
|
|
# Connection type layout
|
|
|
|
connection_type_layout = QtWidgets.QVBoxLayout()
|
|
|
|
connection_type_layout.addWidget(self.connection_type_control_port_extras)
|
|
|
|
connection_type_layout.addWidget(self.connection_type_socket_file_extras)
|
|
|
|
connection_type_layout.addWidget(self.connection_type_socks)
|
|
|
|
connection_type_layout.addWidget(self.authenticate_group)
|
|
|
|
connection_type_layout.addWidget(self.connection_type_bridges_radio_group)
|
2018-04-22 17:20:58 -07:00
|
|
|
connection_type_layout.addLayout(connection_type_test_button_layout)
|
2018-04-22 17:15:15 -07:00
|
|
|
|
2016-12-28 18:44:41 -08:00
|
|
|
# Buttons
|
2017-04-13 22:56:47 -07:00
|
|
|
self.save_button = QtWidgets.QPushButton(strings._('gui_settings_button_save', True))
|
|
|
|
self.save_button.clicked.connect(self.save_clicked)
|
|
|
|
self.cancel_button = QtWidgets.QPushButton(strings._('gui_settings_button_cancel', True))
|
|
|
|
self.cancel_button.clicked.connect(self.cancel_clicked)
|
2018-03-08 10:18:31 -08:00
|
|
|
version_label = QtWidgets.QLabel('OnionShare {0:s}'.format(self.common.version))
|
2018-02-06 19:05:02 -08:00
|
|
|
version_label.setStyleSheet('color: #666666')
|
2017-05-20 13:56:20 +10:00
|
|
|
self.help_button = QtWidgets.QPushButton(strings._('gui_settings_button_help', True))
|
|
|
|
self.help_button.clicked.connect(self.help_clicked)
|
2016-12-28 18:44:41 -08:00
|
|
|
buttons_layout = QtWidgets.QHBoxLayout()
|
2018-02-06 19:05:02 -08:00
|
|
|
buttons_layout.addWidget(version_label)
|
2017-05-20 14:04:52 +10:00
|
|
|
buttons_layout.addWidget(self.help_button)
|
2017-05-16 17:29:02 -07:00
|
|
|
buttons_layout.addStretch()
|
2017-04-13 22:56:47 -07:00
|
|
|
buttons_layout.addWidget(self.save_button)
|
|
|
|
buttons_layout.addWidget(self.cancel_button)
|
|
|
|
|
2017-04-15 15:24:08 -07:00
|
|
|
# Tor network connection status
|
|
|
|
self.tor_status = QtWidgets.QLabel()
|
|
|
|
self.tor_status.setStyleSheet('background-color: #ffffff; color: #000000; padding: 10px')
|
|
|
|
self.tor_status.hide()
|
|
|
|
|
2016-12-28 18:44:41 -08:00
|
|
|
# Layout
|
2017-05-16 17:02:00 -07:00
|
|
|
left_col_layout = QtWidgets.QVBoxLayout()
|
|
|
|
left_col_layout.addWidget(sharing_group)
|
|
|
|
left_col_layout.addWidget(stealth_group)
|
|
|
|
left_col_layout.addWidget(autoupdate_group)
|
|
|
|
left_col_layout.addStretch()
|
|
|
|
|
|
|
|
right_col_layout = QtWidgets.QVBoxLayout()
|
2017-05-18 17:10:47 +10:00
|
|
|
right_col_layout.addWidget(connection_type_radio_group)
|
2018-04-22 17:15:15 -07:00
|
|
|
right_col_layout.addLayout(connection_type_layout)
|
2017-05-16 17:02:00 -07:00
|
|
|
right_col_layout.addWidget(self.tor_status)
|
|
|
|
right_col_layout.addStretch()
|
|
|
|
|
|
|
|
col_layout = QtWidgets.QHBoxLayout()
|
|
|
|
col_layout.addLayout(left_col_layout)
|
|
|
|
col_layout.addLayout(right_col_layout)
|
|
|
|
|
2016-12-28 18:44:41 -08:00
|
|
|
layout = QtWidgets.QVBoxLayout()
|
2017-05-16 17:02:00 -07:00
|
|
|
layout.addLayout(col_layout)
|
2016-12-28 18:44:41 -08:00
|
|
|
layout.addLayout(buttons_layout)
|
2017-05-16 17:02:00 -07:00
|
|
|
|
2016-12-28 18:44:41 -08:00
|
|
|
self.setLayout(layout)
|
2017-04-15 15:24:08 -07:00
|
|
|
self.cancel_button.setFocus()
|
2016-12-28 19:52:21 -08:00
|
|
|
|
|
|
|
# Load settings, and fill them in
|
2018-03-08 10:18:31 -08:00
|
|
|
self.old_settings = Settings(self.common, self.config)
|
2017-05-16 16:50:33 -07:00
|
|
|
self.old_settings.load()
|
2016-12-29 08:02:32 -08:00
|
|
|
|
2017-05-16 16:50:33 -07:00
|
|
|
close_after_first_download = self.old_settings.get('close_after_first_download')
|
2017-04-08 13:42:07 -07:00
|
|
|
if close_after_first_download:
|
|
|
|
self.close_after_first_download_checkbox.setCheckState(QtCore.Qt.Checked)
|
|
|
|
else:
|
|
|
|
self.close_after_first_download_checkbox.setCheckState(QtCore.Qt.Unchecked)
|
|
|
|
|
2018-02-07 09:55:55 -08:00
|
|
|
shutdown_timeout = self.old_settings.get('shutdown_timeout')
|
|
|
|
if shutdown_timeout:
|
|
|
|
self.shutdown_timeout_checkbox.setCheckState(QtCore.Qt.Checked)
|
|
|
|
else:
|
|
|
|
self.shutdown_timeout_checkbox.setCheckState(QtCore.Qt.Unchecked)
|
|
|
|
|
2018-01-13 20:58:24 +11:00
|
|
|
save_private_key = self.old_settings.get('save_private_key')
|
2017-12-07 12:45:29 +11:00
|
|
|
if save_private_key:
|
|
|
|
self.save_private_key_checkbox.setCheckState(QtCore.Qt.Checked)
|
|
|
|
else:
|
|
|
|
self.save_private_key_checkbox.setCheckState(QtCore.Qt.Unchecked)
|
|
|
|
|
2017-05-16 16:50:33 -07:00
|
|
|
use_stealth = self.old_settings.get('use_stealth')
|
2017-04-08 13:42:07 -07:00
|
|
|
if use_stealth:
|
|
|
|
self.stealth_checkbox.setCheckState(QtCore.Qt.Checked)
|
2017-12-09 06:49:34 +11:00
|
|
|
if save_private_key:
|
|
|
|
hidservauth_details.show()
|
|
|
|
self.hidservauth_copy_button.show()
|
2017-04-08 13:42:07 -07:00
|
|
|
else:
|
|
|
|
self.stealth_checkbox.setCheckState(QtCore.Qt.Unchecked)
|
|
|
|
|
2017-05-16 16:50:33 -07:00
|
|
|
use_autoupdate = self.old_settings.get('use_autoupdate')
|
2017-04-15 15:24:08 -07:00
|
|
|
if use_autoupdate:
|
|
|
|
self.autoupdate_checkbox.setCheckState(QtCore.Qt.Checked)
|
|
|
|
else:
|
|
|
|
self.autoupdate_checkbox.setCheckState(QtCore.Qt.Unchecked)
|
|
|
|
|
2017-05-16 16:50:33 -07:00
|
|
|
autoupdate_timestamp = self.old_settings.get('autoupdate_timestamp')
|
2017-04-15 18:04:05 -07:00
|
|
|
self._update_autoupdate_timestamp(autoupdate_timestamp)
|
2017-04-15 15:24:08 -07:00
|
|
|
|
2017-05-16 16:50:33 -07:00
|
|
|
connection_type = self.old_settings.get('connection_type')
|
2017-04-08 17:48:58 -07:00
|
|
|
if connection_type == 'bundled':
|
|
|
|
if self.connection_type_bundled_radio.isEnabled():
|
|
|
|
self.connection_type_bundled_radio.setChecked(True)
|
|
|
|
else:
|
|
|
|
# If bundled tor is disabled, fallback to automatic
|
|
|
|
self.connection_type_automatic_radio.setChecked(True)
|
|
|
|
elif connection_type == 'automatic':
|
2016-12-28 19:52:21 -08:00
|
|
|
self.connection_type_automatic_radio.setChecked(True)
|
|
|
|
elif connection_type == 'control_port':
|
2016-12-28 20:03:32 -08:00
|
|
|
self.connection_type_control_port_radio.setChecked(True)
|
2016-12-28 19:52:21 -08:00
|
|
|
elif connection_type == 'socket_file':
|
|
|
|
self.connection_type_socket_file_radio.setChecked(True)
|
2017-05-16 16:50:33 -07:00
|
|
|
self.connection_type_control_port_extras_address.setText(self.old_settings.get('control_port_address'))
|
|
|
|
self.connection_type_control_port_extras_port.setText(str(self.old_settings.get('control_port_port')))
|
|
|
|
self.connection_type_socket_file_extras_path.setText(self.old_settings.get('socket_file_path'))
|
|
|
|
self.connection_type_socks_address.setText(self.old_settings.get('socks_address'))
|
|
|
|
self.connection_type_socks_port.setText(str(self.old_settings.get('socks_port')))
|
|
|
|
auth_type = self.old_settings.get('auth_type')
|
2016-12-28 19:52:21 -08:00
|
|
|
if auth_type == 'no_auth':
|
|
|
|
self.authenticate_no_auth_radio.setChecked(True)
|
|
|
|
elif auth_type == 'password':
|
|
|
|
self.authenticate_password_radio.setChecked(True)
|
2017-05-16 16:50:33 -07:00
|
|
|
self.authenticate_password_extras_password.setText(self.old_settings.get('auth_password'))
|
2016-12-28 18:44:41 -08:00
|
|
|
|
2017-12-11 14:53:13 +11:00
|
|
|
if self.old_settings.get('no_bridges'):
|
|
|
|
self.tor_bridges_no_bridges_radio.setChecked(True)
|
|
|
|
self.tor_bridges_use_obfs4_radio.setChecked(False)
|
2018-02-15 23:19:53 +00:00
|
|
|
self.tor_bridges_use_meek_lite_amazon_radio.setChecked(False)
|
|
|
|
self.tor_bridges_use_meek_lite_azure_radio.setChecked(False)
|
2017-12-11 14:53:13 +11:00
|
|
|
self.tor_bridges_use_custom_radio.setChecked(False)
|
|
|
|
else:
|
|
|
|
self.tor_bridges_no_bridges_radio.setChecked(False)
|
|
|
|
self.tor_bridges_use_obfs4_radio.setChecked(self.old_settings.get('tor_bridges_use_obfs4'))
|
2018-02-15 23:19:53 +00:00
|
|
|
self.tor_bridges_use_meek_lite_amazon_radio.setChecked(self.old_settings.get('tor_bridges_use_meek_lite_amazon'))
|
|
|
|
self.tor_bridges_use_meek_lite_azure_radio.setChecked(self.old_settings.get('tor_bridges_use_meek_lite_azure'))
|
|
|
|
|
2017-12-11 14:53:13 +11:00
|
|
|
if self.old_settings.get('tor_bridges_use_custom_bridges'):
|
|
|
|
self.tor_bridges_use_custom_radio.setChecked(True)
|
2017-12-11 17:58:53 +11:00
|
|
|
# Remove the 'Bridge' lines at the start of each bridge.
|
|
|
|
# They are added automatically to provide compatibility with
|
|
|
|
# copying/pasting bridges provided from https://bridges.torproject.org
|
|
|
|
new_bridges = []
|
|
|
|
bridges = self.old_settings.get('tor_bridges_use_custom_bridges').split('Bridge ')
|
|
|
|
for bridge in bridges:
|
|
|
|
new_bridges.append(bridge)
|
|
|
|
new_bridges = ''.join(new_bridges)
|
|
|
|
self.tor_bridges_use_custom_textbox.setPlainText(new_bridges)
|
2017-12-11 14:53:13 +11:00
|
|
|
|
2017-04-08 17:48:58 -07:00
|
|
|
def connection_type_bundled_toggled(self, checked):
|
|
|
|
"""
|
|
|
|
Connection type bundled was toggled. If checked, hide authentication fields.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'connection_type_bundled_toggled')
|
2017-04-08 17:48:58 -07:00
|
|
|
if checked:
|
|
|
|
self.authenticate_group.hide()
|
2017-04-15 16:33:41 -07:00
|
|
|
self.connection_type_socks.hide()
|
2017-12-12 08:43:12 +11:00
|
|
|
self.connection_type_bridges_radio_group.show()
|
2017-04-08 17:48:58 -07:00
|
|
|
|
2017-12-11 14:53:13 +11:00
|
|
|
def tor_bridges_no_bridges_radio_toggled(self, checked):
|
|
|
|
"""
|
|
|
|
'No bridges' option was toggled. If checked, enable other bridge options.
|
|
|
|
"""
|
|
|
|
if checked:
|
2017-12-11 16:48:26 +11:00
|
|
|
self.tor_bridges_use_custom_textbox_options.hide()
|
2017-12-11 14:53:13 +11:00
|
|
|
|
|
|
|
def tor_bridges_use_obfs4_radio_toggled(self, checked):
|
|
|
|
"""
|
|
|
|
obfs4 bridges option was toggled. If checked, disable custom bridge options.
|
|
|
|
"""
|
|
|
|
if checked:
|
2017-12-11 16:48:26 +11:00
|
|
|
self.tor_bridges_use_custom_textbox_options.hide()
|
2017-12-11 14:53:13 +11:00
|
|
|
|
2018-02-15 23:19:53 +00:00
|
|
|
def tor_bridges_use_meek_lite_amazon_radio_toggled(self, checked):
|
|
|
|
"""
|
|
|
|
meek_lite-amazon bridges option was toggled. If checked, disable custom bridge options.
|
|
|
|
"""
|
|
|
|
if checked:
|
|
|
|
self.tor_bridges_use_custom_textbox_options.hide()
|
2018-03-05 13:52:50 +11:00
|
|
|
# Alert the user about meek's costliness if it looks like they're turning it on
|
|
|
|
if not self.old_settings.get('tor_bridges_use_meek_lite_amazon'):
|
|
|
|
Alert(strings._('gui_settings_meek_lite_expensive_warning', True), QtWidgets.QMessageBox.Warning)
|
2018-02-15 23:19:53 +00:00
|
|
|
|
|
|
|
def tor_bridges_use_meek_lite_azure_radio_toggled(self, checked):
|
|
|
|
"""
|
2018-02-24 14:09:42 -08:00
|
|
|
meek_lite_azure bridges option was toggled. If checked, disable custom bridge options.
|
2018-02-15 23:19:53 +00:00
|
|
|
"""
|
|
|
|
if checked:
|
|
|
|
self.tor_bridges_use_custom_textbox_options.hide()
|
2018-03-05 13:52:50 +11:00
|
|
|
# Alert the user about meek's costliness if it looks like they're turning it on
|
|
|
|
if not self.old_settings.get('tor_bridges_use_meek_lite_azure'):
|
|
|
|
Alert(strings._('gui_settings_meek_lite_expensive_warning', True), QtWidgets.QMessageBox.Warning)
|
2018-02-15 23:19:53 +00:00
|
|
|
|
2017-12-11 14:53:13 +11:00
|
|
|
def tor_bridges_use_custom_radio_toggled(self, checked):
|
|
|
|
"""
|
|
|
|
Custom bridges option was toggled. If checked, show custom bridge options.
|
|
|
|
"""
|
|
|
|
if checked:
|
2017-12-11 16:48:26 +11:00
|
|
|
self.tor_bridges_use_custom_textbox_options.show()
|
2017-04-08 17:48:58 -07:00
|
|
|
|
2016-12-28 18:44:41 -08:00
|
|
|
def connection_type_automatic_toggled(self, checked):
|
|
|
|
"""
|
2017-04-08 17:48:58 -07:00
|
|
|
Connection type automatic was toggled. If checked, hide authentication fields.
|
2016-12-28 18:44:41 -08:00
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'connection_type_automatic_toggled')
|
2016-12-28 18:44:41 -08:00
|
|
|
if checked:
|
2017-04-08 13:42:07 -07:00
|
|
|
self.authenticate_group.hide()
|
2017-04-15 16:33:41 -07:00
|
|
|
self.connection_type_socks.hide()
|
2017-12-12 08:43:12 +11:00
|
|
|
self.connection_type_bridges_radio_group.hide()
|
2016-12-28 18:44:41 -08:00
|
|
|
|
|
|
|
def connection_type_control_port_toggled(self, checked):
|
|
|
|
"""
|
|
|
|
Connection type control port was toggled. If checked, show extra fields
|
|
|
|
for Tor control address and port. If unchecked, hide those extra fields.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'connection_type_control_port_toggled')
|
2016-12-28 18:44:41 -08:00
|
|
|
if checked:
|
2017-04-08 17:48:58 -07:00
|
|
|
self.authenticate_group.show()
|
2016-12-28 18:44:41 -08:00
|
|
|
self.connection_type_control_port_extras.show()
|
2017-04-15 16:33:41 -07:00
|
|
|
self.connection_type_socks.show()
|
2017-12-12 08:43:12 +11:00
|
|
|
self.connection_type_bridges_radio_group.hide()
|
2016-12-28 18:44:41 -08:00
|
|
|
else:
|
|
|
|
self.connection_type_control_port_extras.hide()
|
|
|
|
|
|
|
|
|
|
|
|
def connection_type_socket_file_toggled(self, checked):
|
|
|
|
"""
|
|
|
|
Connection type socket file was toggled. If checked, show extra fields
|
|
|
|
for socket file. If unchecked, hide those extra fields.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'connection_type_socket_file_toggled')
|
2016-12-28 18:44:41 -08:00
|
|
|
if checked:
|
2017-04-08 17:48:58 -07:00
|
|
|
self.authenticate_group.show()
|
2016-12-28 18:44:41 -08:00
|
|
|
self.connection_type_socket_file_extras.show()
|
2017-04-15 16:33:41 -07:00
|
|
|
self.connection_type_socks.show()
|
2017-12-12 08:43:12 +11:00
|
|
|
self.connection_type_bridges_radio_group.hide()
|
2016-12-28 18:44:41 -08:00
|
|
|
else:
|
|
|
|
self.connection_type_socket_file_extras.hide()
|
|
|
|
|
|
|
|
def authenticate_no_auth_toggled(self, checked):
|
|
|
|
"""
|
|
|
|
Authentication option no authentication was toggled.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'authenticate_no_auth_toggled')
|
2016-12-28 18:44:41 -08:00
|
|
|
|
|
|
|
def authenticate_password_toggled(self, checked):
|
|
|
|
"""
|
|
|
|
Authentication option password was toggled. If checked, show extra fields
|
|
|
|
for password auth. If unchecked, hide those extra fields.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'authenticate_password_toggled')
|
2016-12-28 18:44:41 -08:00
|
|
|
if checked:
|
|
|
|
self.authenticate_password_extras.show()
|
|
|
|
else:
|
|
|
|
self.authenticate_password_extras.hide()
|
|
|
|
|
2017-12-09 06:49:34 +11:00
|
|
|
def hidservauth_copy_button_clicked(self):
|
|
|
|
"""
|
|
|
|
Toggle the 'Copy HidServAuth' button
|
|
|
|
to copy the saved HidServAuth to clipboard.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'hidservauth_copy_button_clicked', 'HidServAuth was copied to clipboard')
|
2017-12-09 06:49:34 +11:00
|
|
|
clipboard = self.qtapp.clipboard()
|
|
|
|
clipboard.setText(self.old_settings.get('hidservauth_string'))
|
|
|
|
|
2017-04-15 13:05:11 -07:00
|
|
|
def test_tor_clicked(self):
|
2016-12-28 18:44:41 -08:00
|
|
|
"""
|
2017-04-15 15:24:08 -07:00
|
|
|
Test Tor Settings button clicked. With the given settings, see if we can
|
2016-12-28 18:44:41 -08:00
|
|
|
successfully connect and authenticate to Tor.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'test_tor_clicked')
|
2016-12-29 08:02:32 -08:00
|
|
|
settings = self.settings_from_fields()
|
2016-12-29 09:58:13 -08:00
|
|
|
|
2017-04-13 22:22:34 -07:00
|
|
|
try:
|
2017-04-13 23:08:25 -07:00
|
|
|
# Show Tor connection status if connection type is bundled tor
|
2017-04-13 22:56:47 -07:00
|
|
|
if settings.get('connection_type') == 'bundled':
|
2017-04-15 18:04:05 -07:00
|
|
|
self.tor_status.show()
|
|
|
|
self._disable_buttons()
|
2017-05-14 17:35:35 -07:00
|
|
|
|
|
|
|
def tor_status_update_func(progress, summary):
|
|
|
|
self._tor_status_update(progress, summary)
|
2017-05-14 18:36:31 -07:00
|
|
|
return True
|
2017-04-13 22:56:47 -07:00
|
|
|
else:
|
2017-05-14 17:35:35 -07:00
|
|
|
tor_status_update_func = None
|
2017-04-13 22:56:47 -07:00
|
|
|
|
2018-03-13 03:28:47 -07:00
|
|
|
onion = Onion(self.common)
|
|
|
|
onion.connect(custom_settings=settings, config=self.config, tor_status_update_func=tor_status_update_func)
|
2017-04-08 19:00:31 -07:00
|
|
|
|
2017-04-13 22:22:34 -07:00
|
|
|
# If an exception hasn't been raised yet, the Tor settings work
|
2018-03-08 10:18:31 -08:00
|
|
|
Alert(self.common, strings._('settings_test_success', True).format(onion.tor_version, onion.supports_ephemeral, onion.supports_stealth))
|
2016-12-29 09:58:13 -08:00
|
|
|
|
2017-04-14 10:00:56 -07:00
|
|
|
# Clean up
|
|
|
|
onion.cleanup()
|
|
|
|
|
|
|
|
except (TorErrorInvalidSetting, TorErrorAutomatic, TorErrorSocketPort, TorErrorSocketFile, TorErrorMissingPassword, TorErrorUnreadableCookieFile, TorErrorAuthError, TorErrorProtocolError, BundledTorNotSupported, BundledTorTimeout) as e:
|
2018-03-08 10:18:31 -08:00
|
|
|
Alert(self.common, e.args[0], QtWidgets.QMessageBox.Warning)
|
2017-04-14 10:00:56 -07:00
|
|
|
if settings.get('connection_type') == 'bundled':
|
2017-04-15 18:04:05 -07:00
|
|
|
self.tor_status.hide()
|
|
|
|
self._enable_buttons()
|
2016-12-28 18:44:41 -08:00
|
|
|
|
2017-04-15 15:24:08 -07:00
|
|
|
def check_for_updates(self):
|
|
|
|
"""
|
|
|
|
Check for Updates button clicked. Manually force an update check.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'check_for_updates')
|
2017-05-14 19:54:12 -07:00
|
|
|
# Disable buttons
|
|
|
|
self._disable_buttons()
|
|
|
|
self.qtapp.processEvents()
|
2017-04-15 18:04:05 -07:00
|
|
|
|
2018-01-23 16:32:14 +11:00
|
|
|
def update_timestamp():
|
|
|
|
# Update the last checked label
|
2018-03-08 10:18:31 -08:00
|
|
|
settings = Settings(self.common, self.config)
|
2018-01-23 16:32:14 +11:00
|
|
|
settings.load()
|
|
|
|
autoupdate_timestamp = settings.get('autoupdate_timestamp')
|
|
|
|
self._update_autoupdate_timestamp(autoupdate_timestamp)
|
|
|
|
|
2018-01-23 16:51:13 +11:00
|
|
|
def close_forced_update_thread():
|
2018-01-23 16:32:14 +11:00
|
|
|
forced_update_thread.quit()
|
|
|
|
# Enable buttons
|
|
|
|
self._enable_buttons()
|
|
|
|
# Update timestamp
|
|
|
|
update_timestamp()
|
|
|
|
|
2017-04-15 18:04:05 -07:00
|
|
|
# Check for updates
|
2017-04-15 18:55:41 -07:00
|
|
|
def update_available(update_url, installed_version, latest_version):
|
2018-03-08 10:18:31 -08:00
|
|
|
Alert(self.common, strings._("update_available", True).format(update_url, installed_version, latest_version))
|
2018-01-23 16:51:13 +11:00
|
|
|
close_forced_update_thread()
|
|
|
|
|
2017-04-15 18:55:41 -07:00
|
|
|
def update_not_available():
|
2018-03-08 10:18:31 -08:00
|
|
|
Alert(self.common, strings._('update_not_available', True))
|
2018-01-23 16:51:13 +11:00
|
|
|
close_forced_update_thread()
|
2017-04-15 18:55:41 -07:00
|
|
|
|
2018-01-23 16:51:13 +11:00
|
|
|
def update_error():
|
2018-03-08 10:18:31 -08:00
|
|
|
Alert(self.common, strings._('update_error_check_error', True), QtWidgets.QMessageBox.Warning)
|
2018-01-23 16:51:13 +11:00
|
|
|
close_forced_update_thread()
|
2017-04-15 18:04:05 -07:00
|
|
|
|
2018-01-23 16:51:13 +11:00
|
|
|
def update_invalid_version():
|
2018-03-08 10:18:31 -08:00
|
|
|
Alert(self.common, strings._('update_error_invalid_latest_version', True).format(e.latest_version), QtWidgets.QMessageBox.Warning)
|
2018-01-23 16:51:13 +11:00
|
|
|
close_forced_update_thread()
|
2018-01-23 16:32:14 +11:00
|
|
|
|
2018-03-08 10:18:31 -08:00
|
|
|
forced_update_thread = UpdateThread(self.common, self.onion, self.config, force=True)
|
2018-01-23 16:32:14 +11:00
|
|
|
forced_update_thread.update_available.connect(update_available)
|
|
|
|
forced_update_thread.update_not_available.connect(update_not_available)
|
2018-01-23 16:51:13 +11:00
|
|
|
forced_update_thread.update_error.connect(update_error)
|
|
|
|
forced_update_thread.update_invalid_version.connect(update_invalid_version)
|
2018-01-23 16:32:14 +11:00
|
|
|
forced_update_thread.start()
|
2017-04-15 15:24:08 -07:00
|
|
|
|
2016-12-28 18:44:41 -08:00
|
|
|
def save_clicked(self):
|
|
|
|
"""
|
|
|
|
Save button clicked. Save current settings to disk.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'save_clicked')
|
2017-05-16 11:31:52 -07:00
|
|
|
|
2016-12-29 08:02:32 -08:00
|
|
|
settings = self.settings_from_fields()
|
2018-02-25 19:42:38 +11:00
|
|
|
if settings:
|
|
|
|
settings.save()
|
|
|
|
|
|
|
|
# If Tor isn't connected, or if Tor settings have changed, Reinitialize
|
|
|
|
# the Onion object
|
|
|
|
reboot_onion = False
|
2018-03-07 16:13:22 +11:00
|
|
|
if not self.local_only:
|
|
|
|
if self.onion.is_authenticated():
|
2018-04-22 17:46:14 -07:00
|
|
|
self.common.log('SettingsDialog', 'save_clicked', 'Connected to Tor')
|
2018-03-07 16:13:22 +11:00
|
|
|
def changed(s1, s2, keys):
|
|
|
|
"""
|
|
|
|
Compare the Settings objects s1 and s2 and return true if any values
|
|
|
|
have changed for the given keys.
|
|
|
|
"""
|
|
|
|
for key in keys:
|
|
|
|
if s1.get(key) != s2.get(key):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
if changed(settings, self.old_settings, [
|
|
|
|
'connection_type', 'control_port_address',
|
|
|
|
'control_port_port', 'socks_address', 'socks_port',
|
|
|
|
'socket_file_path', 'auth_type', 'auth_password',
|
|
|
|
'no_bridges', 'tor_bridges_use_obfs4',
|
|
|
|
'tor_bridges_use_meek_lite_amazon', 'tor_bridges_use_meek_lite_azure',
|
|
|
|
'tor_bridges_use_custom_bridges']):
|
|
|
|
|
|
|
|
reboot_onion = True
|
|
|
|
|
|
|
|
else:
|
2018-04-22 17:46:14 -07:00
|
|
|
self.common.log('SettingsDialog', 'save_clicked', 'Not connected to Tor')
|
2018-03-07 16:13:22 +11:00
|
|
|
# Tor isn't connected, so try connecting
|
2018-02-25 19:42:38 +11:00
|
|
|
reboot_onion = True
|
2017-05-16 16:50:33 -07:00
|
|
|
|
2018-03-07 16:13:22 +11:00
|
|
|
# Do we need to reinitialize Tor?
|
|
|
|
if reboot_onion:
|
|
|
|
# Reinitialize the Onion object
|
2018-04-22 17:46:14 -07:00
|
|
|
self.common.log('SettingsDialog', 'save_clicked', 'rebooting the Onion')
|
2018-03-07 16:13:22 +11:00
|
|
|
self.onion.cleanup()
|
2017-05-16 16:50:33 -07:00
|
|
|
|
2018-03-07 16:13:22 +11:00
|
|
|
tor_con = TorConnectionDialog(self.qtapp, settings, self.onion)
|
|
|
|
tor_con.start()
|
2017-05-16 16:50:33 -07:00
|
|
|
|
2018-04-22 17:46:14 -07:00
|
|
|
self.common.log('SettingsDialog', 'save_clicked', 'Onion done rebooting, connected to Tor: {}'.format(self.onion.connected_to_tor))
|
2017-05-16 16:50:33 -07:00
|
|
|
|
2018-03-07 16:13:22 +11:00
|
|
|
if self.onion.is_authenticated() and not tor_con.wasCanceled():
|
|
|
|
self.settings_saved.emit()
|
|
|
|
self.close()
|
2017-05-16 16:50:33 -07:00
|
|
|
|
2018-03-07 16:13:22 +11:00
|
|
|
else:
|
2018-02-25 19:42:38 +11:00
|
|
|
self.settings_saved.emit()
|
|
|
|
self.close()
|
|
|
|
else:
|
2017-05-22 17:08:05 -07:00
|
|
|
self.settings_saved.emit()
|
2017-05-16 16:50:33 -07:00
|
|
|
self.close()
|
|
|
|
|
2016-12-29 08:02:32 -08:00
|
|
|
def cancel_clicked(self):
|
|
|
|
"""
|
|
|
|
Cancel button clicked.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'cancel_clicked')
|
2018-04-23 21:16:10 -07:00
|
|
|
if not self.local_only and self.onion.is_authenticated():
|
2018-03-08 10:18:31 -08:00
|
|
|
Alert(self.common, strings._('gui_tor_connection_canceled', True), QtWidgets.QMessageBox.Warning)
|
2017-12-04 18:43:40 +11:00
|
|
|
sys.exit()
|
|
|
|
else:
|
|
|
|
self.close()
|
2016-12-29 08:02:32 -08:00
|
|
|
|
2017-05-20 13:56:20 +10:00
|
|
|
def help_clicked(self):
|
|
|
|
"""
|
|
|
|
Help button clicked.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'help_clicked')
|
2017-05-20 13:56:20 +10:00
|
|
|
help_site = 'https://github.com/micahflee/onionshare/wiki'
|
2017-05-23 08:30:24 +10:00
|
|
|
QtGui.QDesktopServices.openUrl(QtCore.QUrl(help_site))
|
2017-05-20 13:56:20 +10:00
|
|
|
|
2016-12-29 08:02:32 -08:00
|
|
|
def settings_from_fields(self):
|
|
|
|
"""
|
|
|
|
Return a Settings object that's full of values from the settings dialog.
|
|
|
|
"""
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'settings_from_fields')
|
|
|
|
settings = Settings(self.common, self.config)
|
2017-04-15 18:04:05 -07:00
|
|
|
settings.load() # To get the last update timestamp
|
2016-12-29 08:02:32 -08:00
|
|
|
|
2017-04-08 13:42:07 -07:00
|
|
|
settings.set('close_after_first_download', self.close_after_first_download_checkbox.isChecked())
|
2018-02-07 09:55:55 -08:00
|
|
|
settings.set('shutdown_timeout', self.shutdown_timeout_checkbox.isChecked())
|
2017-12-07 12:45:29 +11:00
|
|
|
if self.save_private_key_checkbox.isChecked():
|
2018-01-13 20:58:24 +11:00
|
|
|
settings.set('save_private_key', True)
|
2017-12-09 08:11:04 +11:00
|
|
|
settings.set('private_key', self.old_settings.get('private_key'))
|
2018-01-15 10:01:34 +11:00
|
|
|
settings.set('slug', self.old_settings.get('slug'))
|
2017-12-09 08:11:04 +11:00
|
|
|
settings.set('hidservauth_string', self.old_settings.get('hidservauth_string'))
|
2017-12-07 12:45:29 +11:00
|
|
|
else:
|
2018-01-13 20:58:24 +11:00
|
|
|
settings.set('save_private_key', False)
|
2017-12-07 12:45:29 +11:00
|
|
|
settings.set('private_key', '')
|
2018-01-15 10:01:34 +11:00
|
|
|
settings.set('slug', '')
|
2017-12-09 06:49:34 +11:00
|
|
|
# Also unset the HidServAuth if we are removing our reusable private key
|
|
|
|
settings.set('hidservauth_string', '')
|
2017-04-08 13:42:07 -07:00
|
|
|
settings.set('use_stealth', self.stealth_checkbox.isChecked())
|
2017-12-09 06:49:34 +11:00
|
|
|
# Always unset the HidServAuth if Stealth mode is unset
|
|
|
|
if not self.stealth_checkbox.isChecked():
|
|
|
|
settings.set('hidservauth_string', '')
|
2017-04-08 13:42:07 -07:00
|
|
|
|
2017-04-08 17:48:58 -07:00
|
|
|
if self.connection_type_bundled_radio.isChecked():
|
|
|
|
settings.set('connection_type', 'bundled')
|
2016-12-28 20:03:32 -08:00
|
|
|
if self.connection_type_automatic_radio.isChecked():
|
2016-12-29 08:02:32 -08:00
|
|
|
settings.set('connection_type', 'automatic')
|
2016-12-28 20:03:32 -08:00
|
|
|
if self.connection_type_control_port_radio.isChecked():
|
2016-12-29 08:02:32 -08:00
|
|
|
settings.set('connection_type', 'control_port')
|
2016-12-28 20:03:32 -08:00
|
|
|
if self.connection_type_socket_file_radio.isChecked():
|
2016-12-29 08:02:32 -08:00
|
|
|
settings.set('connection_type', 'socket_file')
|
2016-12-28 20:03:32 -08:00
|
|
|
|
2018-01-18 07:54:19 +11:00
|
|
|
if self.autoupdate_checkbox.isChecked():
|
|
|
|
settings.set('use_autoupdate', True)
|
|
|
|
else:
|
|
|
|
settings.set('use_autoupdate', False)
|
|
|
|
|
2016-12-29 08:02:32 -08:00
|
|
|
settings.set('control_port_address', self.connection_type_control_port_extras_address.text())
|
2017-04-15 16:33:41 -07:00
|
|
|
settings.set('control_port_port', self.connection_type_control_port_extras_port.text())
|
2016-12-29 08:02:32 -08:00
|
|
|
settings.set('socket_file_path', self.connection_type_socket_file_extras_path.text())
|
2016-12-28 20:03:32 -08:00
|
|
|
|
2017-04-15 16:33:41 -07:00
|
|
|
settings.set('socks_address', self.connection_type_socks_address.text())
|
|
|
|
settings.set('socks_port', self.connection_type_socks_port.text())
|
|
|
|
|
2016-12-28 20:03:32 -08:00
|
|
|
if self.authenticate_no_auth_radio.isChecked():
|
2016-12-29 08:02:32 -08:00
|
|
|
settings.set('auth_type', 'no_auth')
|
2016-12-28 20:03:32 -08:00
|
|
|
if self.authenticate_password_radio.isChecked():
|
2016-12-29 08:02:32 -08:00
|
|
|
settings.set('auth_type', 'password')
|
2016-12-28 20:03:32 -08:00
|
|
|
|
2016-12-29 08:02:32 -08:00
|
|
|
settings.set('auth_password', self.authenticate_password_extras_password.text())
|
2016-12-28 18:44:41 -08:00
|
|
|
|
2017-12-11 14:53:13 +11:00
|
|
|
# Whether we use bridges
|
|
|
|
if self.tor_bridges_no_bridges_radio.isChecked():
|
|
|
|
settings.set('no_bridges', True)
|
|
|
|
settings.set('tor_bridges_use_obfs4', False)
|
2018-02-15 23:19:53 +00:00
|
|
|
settings.set('tor_bridges_use_meek_lite_amazon', False)
|
|
|
|
settings.set('tor_bridges_use_meek_lite_azure', False)
|
2017-12-11 14:53:13 +11:00
|
|
|
settings.set('tor_bridges_use_custom_bridges', '')
|
|
|
|
if self.tor_bridges_use_obfs4_radio.isChecked():
|
|
|
|
settings.set('no_bridges', False)
|
|
|
|
settings.set('tor_bridges_use_obfs4', True)
|
2018-02-15 23:19:53 +00:00
|
|
|
settings.set('tor_bridges_use_meek_lite_amazon', False)
|
|
|
|
settings.set('tor_bridges_use_meek_lite_azure', False)
|
|
|
|
settings.set('tor_bridges_use_custom_bridges', '')
|
|
|
|
if self.tor_bridges_use_meek_lite_amazon_radio.isChecked():
|
|
|
|
settings.set('no_bridges', False)
|
|
|
|
settings.set('tor_bridges_use_obfs4', False)
|
|
|
|
settings.set('tor_bridges_use_meek_lite_amazon', True)
|
|
|
|
settings.set('tor_bridges_use_meek_lite_azure', False)
|
|
|
|
settings.set('tor_bridges_use_custom_bridges', '')
|
|
|
|
if self.tor_bridges_use_meek_lite_azure_radio.isChecked():
|
|
|
|
settings.set('no_bridges', False)
|
|
|
|
settings.set('tor_bridges_use_obfs4', False)
|
|
|
|
settings.set('tor_bridges_use_meek_lite_amazon', False)
|
|
|
|
settings.set('tor_bridges_use_meek_lite_azure', True)
|
2017-12-11 14:53:13 +11:00
|
|
|
settings.set('tor_bridges_use_custom_bridges', '')
|
|
|
|
if self.tor_bridges_use_custom_radio.isChecked():
|
|
|
|
settings.set('no_bridges', False)
|
|
|
|
settings.set('tor_bridges_use_obfs4', False)
|
2018-02-15 23:19:53 +00:00
|
|
|
settings.set('tor_bridges_use_meek_lite_amazon', False)
|
|
|
|
settings.set('tor_bridges_use_meek_lite_azure', False)
|
|
|
|
|
2017-12-11 17:58:53 +11:00
|
|
|
# Insert a 'Bridge' line at the start of each bridge.
|
|
|
|
# This makes it easier to copy/paste a set of bridges
|
|
|
|
# provided from https://bridges.torproject.org
|
|
|
|
new_bridges = []
|
|
|
|
bridges = self.tor_bridges_use_custom_textbox.toPlainText().split('\n')
|
2018-01-14 20:12:24 +11:00
|
|
|
bridges_valid = False
|
2017-12-11 17:58:53 +11:00
|
|
|
for bridge in bridges:
|
|
|
|
if bridge != '':
|
2018-01-14 20:12:24 +11:00
|
|
|
# Check the syntax of the custom bridge to make sure it looks legitimate
|
2018-02-20 03:25:43 +00:00
|
|
|
ipv4_pattern = re.compile("(obfs4\s+)?(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5]):([0-9]+)(\s+)([A-Z0-9]+)(.+)$")
|
|
|
|
ipv6_pattern = re.compile("(obfs4\s+)?\[(([0-9a-fA-F]{1,4}:){7,7}[0-9a-fA-F]{1,4}|([0-9a-fA-F]{1,4}:){1,7}:|([0-9a-fA-F]{1,4}:){1,6}:[0-9a-fA-F]{1,4}|([0-9a-fA-F]{1,4}:){1,5}(:[0-9a-fA-F]{1,4}){1,2}|([0-9a-fA-F]{1,4}:){1,4}(:[0-9a-fA-F]{1,4}){1,3}|([0-9a-fA-F]{1,4}:){1,3}(:[0-9a-fA-F]{1,4}){1,4}|([0-9a-fA-F]{1,4}:){1,2}(:[0-9a-fA-F]{1,4}){1,5}|[0-9a-fA-F]{1,4}:((:[0-9a-fA-F]{1,4}){1,6})|:((:[0-9a-fA-F]{1,4}){1,7}|:)|fe80:(:[0-9a-fA-F]{0,4}){0,4}%[0-9a-zA-Z]{1,}|::(ffff(:0{1,4}){0,1}:){0,1}((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\.){3,3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])|([0-9a-fA-F]{1,4}:){1,4}:((25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9])\.){3,3}(25[0-5]|(2[0-4]|1{0,1}[0-9]){0,1}[0-9]))\]:[0-9]+\s+[A-Z0-9]+(.+)$")
|
2018-02-20 01:24:44 +00:00
|
|
|
meek_lite_pattern = re.compile("(meek_lite)(\s)+([0-9]+\.[0-9]+\.[0-9]+\.[0-9]+:[0-9]+)(\s)+([0-9A-Z]+)(\s)+url=(.+)(\s)+front=(.+)")
|
|
|
|
if ipv4_pattern.match(bridge) or \
|
2018-02-26 13:48:28 +11:00
|
|
|
ipv6_pattern.match(bridge):
|
|
|
|
new_bridges.append(''.join(['Bridge ', bridge, '\n']))
|
|
|
|
bridges_valid = True
|
2018-02-26 14:02:15 +11:00
|
|
|
if self.system != 'Windows' and self.system != 'Darwin' and meek_lite_pattern.match(bridge):
|
2018-01-14 20:12:24 +11:00
|
|
|
new_bridges.append(''.join(['Bridge ', bridge, '\n']))
|
|
|
|
bridges_valid = True
|
2018-02-26 13:48:28 +11:00
|
|
|
|
2018-01-14 20:12:24 +11:00
|
|
|
if bridges_valid:
|
|
|
|
new_bridges = ''.join(new_bridges)
|
|
|
|
settings.set('tor_bridges_use_custom_bridges', new_bridges)
|
|
|
|
else:
|
2018-03-08 10:18:31 -08:00
|
|
|
Alert(self.common, strings._('gui_settings_tor_bridges_invalid', True))
|
2018-01-14 20:12:24 +11:00
|
|
|
settings.set('no_bridges', True)
|
2018-02-25 19:42:38 +11:00
|
|
|
return False
|
2017-12-11 14:53:13 +11:00
|
|
|
|
2016-12-29 08:02:32 -08:00
|
|
|
return settings
|
2017-04-15 18:04:05 -07:00
|
|
|
|
2017-05-16 16:50:33 -07:00
|
|
|
def closeEvent(self, e):
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', 'closeEvent')
|
2017-05-16 16:50:33 -07:00
|
|
|
|
|
|
|
# On close, if Tor isn't connected, then quit OnionShare altogether
|
2018-03-07 16:13:22 +11:00
|
|
|
if not self.local_only:
|
|
|
|
if not self.onion.is_authenticated():
|
2018-04-22 17:38:28 -07:00
|
|
|
self.common.log('SettingsDialog', 'closeEvent', 'Closing while not connected to Tor')
|
2017-05-16 16:50:33 -07:00
|
|
|
|
2018-03-07 16:13:22 +11:00
|
|
|
# Wait 1ms for the event loop to finish, then quit
|
|
|
|
QtCore.QTimer.singleShot(1, self.qtapp.quit)
|
2017-05-16 16:50:33 -07:00
|
|
|
|
2017-04-15 18:04:05 -07:00
|
|
|
def _update_autoupdate_timestamp(self, autoupdate_timestamp):
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', '_update_autoupdate_timestamp')
|
2017-05-16 11:31:52 -07:00
|
|
|
|
2017-04-15 18:04:05 -07:00
|
|
|
if autoupdate_timestamp:
|
|
|
|
dt = datetime.datetime.fromtimestamp(autoupdate_timestamp)
|
|
|
|
last_checked = dt.strftime('%B %d, %Y %H:%M')
|
|
|
|
else:
|
|
|
|
last_checked = strings._('gui_settings_autoupdate_timestamp_never', True)
|
|
|
|
self.autoupdate_timestamp.setText(strings._('gui_settings_autoupdate_timestamp', True).format(last_checked))
|
|
|
|
|
2017-05-14 18:36:31 -07:00
|
|
|
def _tor_status_update(self, progress, summary):
|
2017-05-17 12:09:56 -07:00
|
|
|
self.tor_status.setText('<strong>{}</strong><br>{}% {}'.format(strings._('connecting_to_tor', True), progress, summary))
|
2017-04-15 18:04:05 -07:00
|
|
|
self.qtapp.processEvents()
|
2017-05-14 18:36:31 -07:00
|
|
|
if 'Done' in summary:
|
2017-04-15 18:04:05 -07:00
|
|
|
self.tor_status.hide()
|
|
|
|
self._enable_buttons()
|
|
|
|
|
|
|
|
def _disable_buttons(self):
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', '_disable_buttons')
|
2017-05-16 11:31:52 -07:00
|
|
|
|
2017-04-15 18:04:05 -07:00
|
|
|
self.check_for_updates_button.setEnabled(False)
|
|
|
|
self.connection_type_test_button.setEnabled(False)
|
|
|
|
self.save_button.setEnabled(False)
|
|
|
|
self.cancel_button.setEnabled(False)
|
|
|
|
|
|
|
|
def _enable_buttons(self):
|
2018-03-08 10:18:31 -08:00
|
|
|
self.common.log('SettingsDialog', '_enable_buttons')
|
2018-01-04 08:43:43 +11:00
|
|
|
# We can't check for updates if we're still not connected to Tor
|
|
|
|
if not self.onion.connected_to_tor:
|
|
|
|
self.check_for_updates_button.setEnabled(False)
|
|
|
|
else:
|
|
|
|
self.check_for_updates_button.setEnabled(True)
|
2017-04-15 18:04:05 -07:00
|
|
|
self.connection_type_test_button.setEnabled(True)
|
|
|
|
self.save_button.setEnabled(True)
|
|
|
|
self.cancel_button.setEnabled(True)
|