# SPDX-License-Identifier: AGPL-3.0-or-later """FreedomBox app for configuring Tor.""" import logging from django.utils.translation import gettext_noop from django.views.generic.edit import FormView from plinth import app as app_module from plinth import operation as operation_module from plinth.modules import tor from plinth.views import AppView from . import privileged from . import utils as tor_utils from .forms import TorForm config_process = None logger = logging.getLogger(__name__) class TorAppView(AppView): """Show Tor app main page.""" app_id = 'tor' template_name = 'tor.html' form_class = TorForm prefix = 'tor' status = None def get_initial(self): """Return the values to fill in the form.""" if not self.status: self.status = tor_utils.get_status() initial = super().get_initial() initial.update(self.status) return initial def get_context_data(self, *args, **kwargs): """Add additional context data for template.""" if not self.status: self.status = tor_utils.get_status() context = super().get_context_data(*args, **kwargs) context['status'] = self.status return context def form_valid(self, form): """Configure tor app on successful form submission.""" operation_module.manager.new('tor-configuration', self.app_id, gettext_noop('Updating configuration'), _apply_changes, [form.initial, form.cleaned_data], show_notification=False) # Skip check for 'Settings unchanged' message by calling grandparent return super(FormView, self).form_valid(form) def _apply_changes(old_status, new_status): """Try to apply changes and handle errors.""" logger.info('tor: applying configuration changes') exception_to_update = None message = None try: __apply_changes(old_status, new_status) except Exception as exception: exception_to_update = exception message = gettext_noop('Error configuring app: {error}').format( error=exception) else: message = gettext_noop('Configuration updated.') logger.info('tor: configuration changes completed') operation = operation_module.Operation.get_operation() operation.on_update(message, exception_to_update) def __apply_changes(old_status, new_status): """Apply the changes.""" arguments = {} app = app_module.App.get('tor') is_enabled = app.is_enabled() if old_status['relay_enabled'] != new_status['relay_enabled']: arguments['relay'] = new_status['relay_enabled'] if old_status['bridge_relay_enabled'] != \ new_status['bridge_relay_enabled']: arguments['bridge_relay'] = new_status['bridge_relay_enabled'] if old_status['hs_enabled'] != new_status['hs_enabled']: arguments['hidden_service'] = new_status['hs_enabled'] if old_status['use_upstream_bridges'] != \ new_status['use_upstream_bridges']: arguments['use_upstream_bridges'] = new_status['use_upstream_bridges'] if old_status['upstream_bridges'] != new_status['upstream_bridges']: arguments['upstream_bridges'] = new_status['upstream_bridges'] if arguments: privileged.configure(**arguments) if is_enabled: privileged.restart() status = tor_utils.get_status() tor.update_hidden_service_domain(status)