#!/usr/bin/python3 # -*- mode: python -*- # # This file is part of FreedomBox. # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero 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 Affero General Public License for more details. # # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see . # """ Configuration helper for Apache web server. """ import argparse import glob import re import subprocess from plinth import action_utils def parse_arguments(): """Return parsed command line arguments as dictionary""" parser = argparse.ArgumentParser() subparsers = parser.add_subparsers(dest='subcommand', help='Sub command') subparser = subparsers.add_parser('setup', help='Setup for Apache') subparser.add_argument( '--old-version', type=int, required=True, help='Earlier version of the app that is already setup.') subparsers.required = True return parser.parse_args() def _get_sort_key_of_version(version): """Return the sort key for a given version string. Simple implementation hoping that PHP Apache module version numbers will be simple. """ parts = [] for part in version.split('.'): try: parts.append(int(part)) except ValueError: parts.append(part) return parts def _sort_versions(versions): """Return a list of sorted version strings.""" return sorted(versions, key=_get_sort_key_of_version, reverse=True) def _enable_latest_php(webserver): """Disable all older PHP versions and enable the latest one. Idempotent and harmless if all or no PHP modules are identified. Problematic if only some modules are found. """ paths = glob.glob('/etc/apache2/mods-available/php*.conf') versions = [] for path in paths: match = re.search(r'\/php(.*)\.conf$', path) if match: versions.append(match[1]) versions = _sort_versions(versions) for version in versions[1:]: webserver.disable('php' + version, kind='module') if versions: webserver.enable('php' + versions[0], kind='module') def subcommand_setup(arguments): """Setup Apache configuration.""" # Regenerate the snakeoil self-signed SSL certificate. This is so that # FreedomBox images don't all have the same certificate. When FreedomBox # package is installed via apt, don't regenerate. When upgrading to newer # version of Apache FreedomBox app and setting up for the first time don't # regenerate. if action_utils.is_disk_image() and arguments.old_version == 0: subprocess.run([ 'make-ssl-cert', 'generate-default-snakeoil', '--force-overwrite' ], check=True) with action_utils.WebserverChange() as webserver: # set the prefork worker model webserver.disable('mpm_event', kind='module') webserver.disable('mpm_worker', kind='module') webserver.enable('mpm_prefork', kind='module') # enable miscellaneous modules. webserver.enable('proxy', kind='module') webserver.enable('proxy_http', kind='module') webserver.enable('rewrite', kind='module') # enable GnuTLS webserver.disable('ssl', kind='module') webserver.enable('gnutls', kind='module') # enable mod_alias for RedirectMatch webserver.enable('alias', kind='module') # enable mod_headers for HSTS webserver.enable('headers', kind='module') # enable some critical modules to avoid restart while installing # FreedomBox applications. webserver.enable('cgi', kind='module') webserver.enable('authnz_ldap', kind='module') # Workaround for bug https://bugs.debian.org/893481 . Ideally, don't # explicitly enable module php and rely on the package # libapache2-mod-php installing the current version of the package and # enabling it. This ensures that when PHP version changes, the code is # not broken. _enable_latest_php(webserver) # enable users to share files uploaded to ~/public_html webserver.enable('userdir', kind='module') # setup freedombox site webserver.enable('freedombox', kind='config') # enable serving Debian javascript libraries webserver.enable('javascript-common', kind='config') # default sites webserver.enable('000-default', kind='site') webserver.disable('default-ssl', kind='site') webserver.enable('default-tls', kind='site') webserver.enable('plinth', kind='site') webserver.enable('plinth-ssl', kind='site') def main(): """Parse arguments and perform all duties""" arguments = parse_arguments() subcommand = arguments.subcommand.replace('-', '_') subcommand_method = globals()['subcommand_' + subcommand] subcommand_method(arguments) if __name__ == '__main__': main()