Source code for plinth.config
"""Components for managing configuration files."""
import logging
import pathlib
from django.utils.text import format_lazy
from django.utils.translation import gettext_lazy as _
from plinth.privileged import config as privileged
from . import app as app_module
logger = logging.getLogger(__name__)
[docs]class DropinConfigs(app_module.FollowerComponent):
"""Component to manage config files dropped into /etc.
When configuring a daemon, it is often simpler to ship a configuration file
into the daemon's configuration directory. However, if the user modifies
this configuration file and freedombox ships a new version of this
configuration file, then a conflict arises between user's changes and
changes in the new version of configuration file shipped by freedombox.
This leads to freedombox package getting marked by unattended-upgrades as
not automatically upgradable. Dpkg's solution of resolving the conflicts is
to present the option to the user which is also not acceptable.
Further, if a package is purged from the system, sometimes the
configuration directories are fully removed by deb's scripts. This removes
files installed by freedombox package. Dpkg treats these files as if user
has explictly removed them and may lead to a configuration conflict
described above.
The approach freedombox takes to address these issues is using this
component. Files are shipped into /usr/share/freedombox/etc/ instead of
/etc/ (keeping the subpath unchanged). Then when an app is enabled, a
symlink or copy is created from the /usr/share/freedombox/etc/ into /etc/.
This way, user's understand the configuration file is not meant to be
edited. Even if they do, next upgrade of freedombox package will silently
overwrite those changes without causing merge conflicts. Also when purging
a package removes entire configuration directory, only symlinks/copies are
lost. They will recreated when the app is reinstalled/enabled.
"""
ROOT = '/' # To make writing tests easier
DROPIN_CONFIG_ROOT = '/usr/share/freedombox/'
[docs] def __init__(self, component_id, etc_paths=None, copy_only=False):
"""Initialize the drop-in configuration component.
component_id should be a unique ID across all components of an app and
across all components.
etc_paths is a list of all drop-in configuration files as absolute
paths in /etc/ which need to managed by this component. For each of the
paths, it is expected that the actual configuration file exists in
/usr/share/freedombox/etc/. A link to the file or copy of the file is
created in /etc/ when app is enabled and the link or file is removed
when app is disabled. For example, if etc_paths contains
/etc/apache/conf-enabled/myapp.conf then
/usr/share/freedombox/etc/apache/conf-enabled/myapp.conf must be
shipped and former path will be link to or be a copy of the latter when
app is enabled.
"""
super().__init__(component_id)
self.etc_paths = etc_paths or []
self.copy_only = copy_only
[docs] def setup(self, old_version):
"""Create symlinks or copies of files during app update.
During the transition from shipped configs to the symlink/copy
approach, files in /etc will be removed during .deb upgrade. This
method ensures that symlinks or copies are properly recreated.
"""
if self.app_id and self.app.is_enabled():
self.enable()
[docs] def enable(self):
"""Create a symlink or copy in /etc/ of the configuration file."""
for path in self.etc_paths:
etc_path = self._get_etc_path(path)
target = self._get_target_path(path)
if etc_path.exists() or etc_path.is_symlink():
if (not self.copy_only and etc_path.is_symlink()
and etc_path.readlink() == target):
continue
if (self.copy_only and etc_path.is_file()
and etc_path.read_text() == target.read_text()):
continue
logger.warning('Removing dropin configuration: %s', path)
privileged.dropin_unlink(self.app_id, path)
privileged.dropin_link(self.app_id, path, self.copy_only)
[docs] def disable(self):
"""Remove the links/copies in /etc/ of the configuration files."""
for path in self.etc_paths:
privileged.dropin_unlink(self.app_id, path, missing_ok=True)
[docs] def diagnose(self):
"""Check all links/copies and return generate diagnostic results."""
results = []
for path in self.etc_paths:
etc_path = self._get_etc_path(path)
target = self._get_target_path(path)
if self.copy_only:
result = (etc_path.is_file()
and etc_path.read_text() == target.read_text())
else:
result = (etc_path.is_symlink()
and etc_path.readlink() == target)
result_string = 'passed' if result else 'failed'
template = _('Static configuration {etc_path} is setup properly')
test_name = format_lazy(template, etc_path=str(etc_path))
results.append([test_name, result_string])
return results
@staticmethod
def _get_target_path(path):
"""Return Path object for a target path."""
target = pathlib.Path(DropinConfigs.ROOT)
target /= DropinConfigs.DROPIN_CONFIG_ROOT.lstrip('/')
return target / path.lstrip('/')
@staticmethod
def _get_etc_path(path):
"""Return Path object for etc path."""
return pathlib.Path(DropinConfigs.ROOT) / path.lstrip('/')