FreedomBox/plinth/modules/searx/privileged.py
James Valleroy 0e698eb4b4
apache: Use a Uwsgi native socket systemd unit for each app
[Sunil]:

- Drop Uwsgi component entirely. After the changes, it mostly looks like Daemon
component minus some features. One change that Uwsgi component does is when
component is disabled, it also stops and disables the .service unit. Stopping
the service is useful and we can add this to Daemon component.

- Use /run instead of /var/run/ as 1) /var/run is a symlink to /run 2) /run/
path is what is listed in uwsgi-app@.socket unit file.

- Implement upgrade for apps from older version. Disable and mask uwsgi init.d
script. Enable the daemon component if the webserver component is enabled.

- Update manifest files to deal with .socket units instead of 'uwsgi' service.
Backup the /var/lib/private directories as that is actual directory to backup
with DynamicUser=yes.

- For bepasty load the configuration as a systemd provided credential since
DynamicUser=yes.

- Remove the /var/lib/private directories during uninstall.

- Don't create user/group for bepasty as it is not needed with DynamicUser=yes.

Tests:

- Radicale

  - Functional tests pass

  - Freshly install radicale.

  - Web interface works.

  - Create and edit calendars

  - Path of the storage directory is in /var/lib/private/radicale (after
  accessing web interface)

  - Permissions on the storage folder and files inside are set to nobody:nobody.

  - Uninstall removes the /var/lib/private/radicale directory.

  - Create a calender and backup the app. Uninstall the app. Re-install the app.
  The calendar is not available. After restoring the backup, the calendar is
  available.

  - Install radicale without patch and create a calendar. Apply patches and
  start plinth.service. Setup is run. UWSGI is disabled and masked. Service is
  running. Old calender is visible.

  - Install radicale without patch. Disable and apply patches and start
  plinth.service. Setup is run. UWSGI is disabled and masked. Service is not
  running. Enabling the service works.

  - After upgrade, data storage path got migrated to /var/lib/private/radicale.
  Old data is accessible.

  - After upgrade the directory is still owned by radicale:radicale.

  - Freshly install radicale with patch and restore an old backup. The data is
  available in the web interface and data was migrated to
  /var/lib/private/radicale.

- Bepasty

  - Functional tests pass

  - Freshly install bepasy.

  - Enabling and disabling rapidly works.

  - Uploading files works.

  - Path of the storage directory is /var/lib/private/bepasty.

  - Permissions on the storage folder are as expect 755 but on the parent are
  700.

  - Permissions on the stored files are 644 and owned by nobody:nobody.

  - Uninstall removes the /var/lib/private/bepasty directory.

  - Upload a picture and backup the app. Uninstall the app. Re-install the app.
  The uploaded file is not available. After restoring the backup, the uploaded
  file is available.

  - Install bepasty without patch and upload a file. Apply patches and start
  plinth.service. Setup is run. UWSGI is disabled and masked. Service is
  running. Old uploaded picture is visible.

  - Install bepasty without patch. Disable app. Apply patches and start
  plinth.service. Setup is run. UWSGI is disabled and masked. Service is not
  running. Enabling the service works.

  - After upgrade, data storage path got migrated to /var/lib/private/bepasty.
  Old data is accessible.

  - After upgrade the directory is still owned by bepasty:bepasty.

  - Freshly install bepasty with patch and restore an old backup. The uploaded
  file is available in the web interface and data was migrated to
  /var/lib/private/bepasty.

Signed-off-by: James Valleroy <jvalleroy@mailbox.org>
Signed-off-by: Sunil Mohan Adapa <sunil@medhas.org>
Reviewed-by: James Valleroy <jvalleroy@mailbox.org>
2026-03-21 07:45:51 -07:00

164 lines
4.7 KiB
Python

# SPDX-License-Identifier: AGPL-3.0-or-later
"""Configure searx."""
import gzip
import os
import pathlib
import secrets
import shutil
import augeas
import yaml
from plinth import action_utils
from plinth.actions import privileged
from plinth.modules.searx.manifest import PUBLIC_ACCESS_SETTING_FILE
from plinth.utils import gunzip
SETTINGS_FILE = '/etc/searx/settings.yml'
UWSGI_FILE = '/etc/uwsgi/apps-available/searx.ini'
def _copy_uwsgi_configuration():
"""Copy example uwsgi configuration
Copy the example uwsgi configuration shipped with Searx documentation to
the appropriate uwsgi directory.
"""
example_config = ('/usr/share/doc/searx/examples/'
'uwsgi/apps-available/searx.ini')
if not os.path.exists(UWSGI_FILE):
shutil.copy(example_config, os.path.dirname(UWSGI_FILE))
def _update_uwsgi_configuration():
"""Fix uwsgi configuration.
uwsgi 2.0.15-debian crashes when trying to autoload.
"""
aug = augeas.Augeas(flags=augeas.Augeas.NO_LOAD +
augeas.Augeas.NO_MODL_AUTOLOAD)
aug.set('/augeas/load/inifile/lens', 'Puppet.lns')
aug.set('/augeas/load/inifile/incl[last() + 1]', UWSGI_FILE)
aug.load()
aug.set('/files/etc/uwsgi/apps-available/searx.ini/uwsgi/autoload',
'false')
aug.set('/files/etc/uwsgi/apps-available/searx.ini/uwsgi/chmod-socket',
'660')
aug.save()
def _generate_secret_key(settings):
"""Generate a secret key for the Searx installation."""
secret_key = secrets.token_hex(64)
settings['server']['secret_key'] = secret_key
def _set_title(settings):
"""Set the page title to '{box_name} Web Search'."""
title = 'FreedomBox Web Search'
settings['general']['instance_name'] = title
def _set_timeout(settings):
"""Set timeout to 20 seconds."""
settings['outgoing']['request_timeout'] = 20.0
def _set_safe_search(settings):
"""Set safe search to Moderate."""
settings['search']['safe_search'] = 1
@privileged
def set_safe_search(filter_: int):
"""Set safe search filter for search results."""
settings = read_settings()
settings['search']['safe_search'] = filter_
write_settings(settings)
@privileged
def get_safe_search() -> int:
"""Return the value of the safe search setting."""
if os.path.exists(SETTINGS_FILE):
settings = read_settings()
return int(settings['search']['safe_search'])
else:
return 0
def read_settings():
"""Load settings as dictionary from YAML config file."""
with open(SETTINGS_FILE, 'rb') as settings_file:
return yaml.safe_load(settings_file)
def write_settings(settings):
"""Write settings from dictionary to YAML config file."""
with open(SETTINGS_FILE, 'w', encoding='utf-8') as settings_file:
yaml.dump(settings, settings_file)
def _get_example_settings_file():
searx_doc_dir = pathlib.Path('/usr/share/doc/searx/examples/')
if (searx_doc_dir / 'settings.yml').exists():
return searx_doc_dir / 'settings.yml'
return searx_doc_dir / 'settings.yml.gz'
def _update_search_engines(settings):
"""Update settings with the latest supported search engines."""
example_settings_file = _get_example_settings_file()
open_func = gzip.open if example_settings_file.suffix == '.gz' else open
with open_func(example_settings_file, 'rb') as example_settings:
settings['engines'] = yaml.safe_load(example_settings)['engines']
@privileged
def setup():
"""Post installation actions for Searx."""
_copy_uwsgi_configuration()
_update_uwsgi_configuration()
if not os.path.exists(SETTINGS_FILE):
example_settings_file = _get_example_settings_file()
if example_settings_file.suffix == '.gz':
gunzip(str(example_settings_file), SETTINGS_FILE)
else:
pathlib.Path(SETTINGS_FILE).parent.mkdir(mode=0o755)
shutil.copy(example_settings_file, SETTINGS_FILE)
settings = read_settings()
_generate_secret_key(settings)
_set_title(settings)
_set_timeout(settings)
_set_safe_search(settings)
_update_search_engines(settings)
write_settings(settings)
# Service is started again by socket.
action_utils.service_stop('uwsgi-app@searx.service')
@privileged
def enable_public_access():
"""Enable public access to the SearX application."""
open(PUBLIC_ACCESS_SETTING_FILE, 'w', encoding='utf-8').close()
@privileged
def disable_public_access():
"""Disable public access to the SearX application."""
if os.path.exists(PUBLIC_ACCESS_SETTING_FILE):
os.remove(PUBLIC_ACCESS_SETTING_FILE)
@privileged
def uninstall():
"""Remove configuration uWSGI file."""
shutil.rmtree('/etc/searx', ignore_errors=True)
pathlib.Path(UWSGI_FILE).unlink(missing_ok=True)