add install/uninstall functionality

This commit is contained in:
2022-01-11 20:58:01 +01:00
parent eceec36681
commit d761fef998
3 changed files with 193 additions and 10 deletions

168
pyquarantine/_install.py Executable file
View File

@@ -0,0 +1,168 @@
#!/usr/bin/env python3
# pyinotifyd 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.
#
# pyinotifyd 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 pyinotifyd. If not, see <http://www.gnu.org/licenses/>.
#
import filecmp
import logging
import os
import shutil
import sys
SYSTEMD_PATH = "/lib/systemd/system"
OPENRC = "/sbin/openrc"
def _systemd_files(pkg_dir, name):
return [
(f"{pkg_dir}/misc/systemd/{name}-milter.service",
f"{SYSTEMD_PATH}/{name}-milter.service", True)]
def _openrc_files(pkg_dir, name):
return [
(f"{pkg_dir}/misc/openrc/{name}-milter.initd", f"/etc/init.d/{name}-milter", True),
(f"{pkg_dir}/misc/openrc/{name}-milter.confd", f"/etc/conf.d/{name}-milter", False)]
def _config_files(pkg_dir, name):
return [
(f"{pkg_dir}/misc/{name}.conf.default", f"/etc/{name}/{name}.conf", False),
(f"{pkg_dir}/misc/templates/removed.png", f"/etc/{name}/templates/removed.png", False),
(f"{pkg_dir}/misc/templates/disclaimer_html.template", f"/etc/{name}/templates/disclaimer_html.template", False),
(f"{pkg_dir}/misc/templates/disclaimer_text.template", f"/etc/{name}/templates/disclaimer_text.template", False),
(f"{pkg_dir}/misc/templates/notification.template", f"/etc/{name}/templates/notification.template", False)]
def _install_files(files):
for src, dst, force in files:
if os.path.exists(dst):
if os.path.isdir(dst):
logging.error(
" => unable to copy file, destination path is a directory")
continue
elif not force:
logging.info(f" => file {dst} already exists")
continue
try:
logging.info(f" => install file {dst}")
shutil.copy2(src, dst)
except Exception as e:
logging.error(f" => unable to install file {dst}: {e}")
def _uninstall_files(files):
for src, dst, force in files:
if not os.path.isfile(dst):
continue
if not force and not filecmp.cmp(src, dst, shallow=True):
logging.warning(
f" => keep modified file {dst}, "
f"you have to remove it manually")
continue
try:
logging.info(f" => uninstall file {dst}")
os.remove(dst)
except Exception as e:
logging.error(f" => unable to uninstall file {dst}: {e}")
def _create_dir(path):
if os.path.isdir(path):
logging.info(f" => directory {path} already exists")
else:
try:
logging.info(f" => create directory {path}")
os.mkdir(path)
except Exception as e:
logging.error(f" => unable to create directory {path}: {e}")
return False
return True
def _delete_dir(path):
if os.path.isdir(path):
if not os.listdir(path):
try:
logging.info(f" => delete directory {path}")
os.rmdir(path)
except Exception as e:
logging.error(f" => unable to delete directory {path}: {e}")
else:
logging.warning(f" => keep non-empty directory {path}")
def _check_root():
if os.getuid() != 0:
logging.error("you need to have root privileges, please try again")
return False
return True
def _check_systemd():
systemd = os.path.isdir(SYSTEMD_PATH)
if systemd:
logging.info("systemd detected")
return systemd
def _check_openrc():
openrc = os.path.isfile(OPENRC) and os.access(OPENRC, os.X_OK)
if openrc:
logging.info("openrc detected")
return openrc
def install(name):
if not _check_root():
sys.exit(2)
pkg_dir = os.path.dirname(__file__)
if _check_systemd():
_install_files(_systemd_files(pkg_dir, name))
if _check_openrc():
_install_files(_openrc_files(pkg_dir, name))
if not _create_dir(f"/etc/{name}"):
logging.error(" => unable to create config dir, giving up ...")
sys.exit(3)
_install_files(_config_files(pkg_dir, name))
logging.info(f"{name} successfully installed")
def uninstall(name):
if not _check_root():
sys.exit(2)
pkg_dir = os.path.dirname(__file__)
_uninstall_files(_systemd_files(pkg_dir, name))
_uninstall_files(_openrc_files(pkg_dir, name))
_uninstall_files(_config_files(pkg_dir, name))
_delete_dir(f"/etc/{name}")
logging.info(f"{name} successfully uninstalled")

View File

@@ -20,6 +20,7 @@ import logging
import logging.handlers import logging.handlers
import sys import sys
from pyquarantine._install import install, uninstall
from pyquarantine import mailer from pyquarantine import mailer
from pyquarantine import QuarantineMilter from pyquarantine import QuarantineMilter
from pyquarantine import __version__ as version from pyquarantine import __version__ as version
@@ -35,37 +36,51 @@ def main():
description="pyquarantine-milter daemon", description="pyquarantine-milter daemon",
formatter_class=lambda prog: argparse.HelpFormatter( formatter_class=lambda prog: argparse.HelpFormatter(
prog, max_help_position=45, width=140)) prog, max_help_position=45, width=140))
parser.add_argument( parser.add_argument(
"-c", "--config", help="Config file to read.", "-c", "--config", help="Config file to read.",
default="/etc/pyquarantine/pyquarantine.conf") default="/etc/pyquarantine/pyquarantine.conf")
parser.add_argument( parser.add_argument(
"-s", "-s",
"--socket", "--socket",
help="Socket used to communicate with the MTA.", help="Socket used to communicate with the MTA.",
default="") default="")
parser.add_argument( parser.add_argument(
"-d", "-d",
"--debug", "--debug",
help="Log debugging messages.", help="Log debugging messages.",
action="store_true") action="store_true")
parser.add_argument( exclusive = parser.add_mutually_exclusive_group()
exclusive.add_argument(
"-v", "--version",
help="Print version.",
action="version",
version=f"%(prog)s {version} (python {python_version})")
exclusive.add_argument(
"-i",
"--install",
help="install service files and config",
action="store_true")
exclusive.add_argument(
"-u",
"--uninstall",
help="uninstall service files and unmodified config",
action="store_true")
exclusive.add_argument(
"-t", "-t",
"--test", "--test",
help="Check configuration.", help="Check configuration.",
action="store_true") action="store_true")
parser.add_argument(
"-v", "--version",
help="Print version.",
action="version",
version=f"%(prog)s {version} (python {python_version})")
args = parser.parse_args() args = parser.parse_args()
name = "pyquarantine"
if args.install:
sys.exit(install(name))
if args.uninstall:
sys.exit(uninstall(name))
root_logger = logging.getLogger() root_logger = logging.getLogger()
root_logger.setLevel(logging.DEBUG) root_logger.setLevel(logging.DEBUG)