D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
opt
/
imunify360
/
venv
/
lib
/
python3.11
/
site-packages
/
im360
/
plugins
/
Filename :
waf_rules_configurator.py
back
Copy
""" A plugin responsible for periodically launching AppVersionDetector and optionally limiting ModSecurity rulesets for the sites that use various CMS. Currently it sets up and maintains a cron job to achieve this. """ import os from contextlib import suppress from logging import getLogger from pathlib import Path from random import randint from typing import Optional from defence360agent.contracts.config import SystemConfig from defence360agent.contracts.messages import MessageType from defence360agent.contracts.plugins import MessageSink, expect from defence360agent.utils.cronjob import CronJob from im360.contracts.config import Modsec as Config from defence360agent.subsys import web_server from defence360agent.subsys.persistent_state import load_state, save_state from im360.subsys.panels.hosting_panel import HostingPanel from im360.subsys.waf_rules_configurator import ( is_webserver_supported, try_restore_config_from_backup, ) logger = getLogger(__name__) UPDATE_COMPONENTS_SCRIPT = ( "/opt/imunify360/venv/share/imunify360/scripts/" "update_components_versions.py" ) WAF_CONFIGURATOR_CRON_PATH = "/etc/cron.d/waf_configurator" class WAFRuleSetConfigurator(MessageSink): def __init__(self): self._cron_job = self._read_cron() async def create_sink(self, loop): self._app_specific_ruleset = load_state("WAFRuleSetConfigurator").get( "app_specific_ruleset" ) if ( self._app_specific_ruleset is None or self._app_specific_ruleset != Config.APP_SPECIFIC_RULESET ): self._app_specific_ruleset = Config.APP_SPECIFIC_RULESET await self._update_cron_job(Config.APP_SPECIFIC_RULESET) await try_restore_config_from_backup() async def shutdown(self): save_state( "WAFRuleSetConfigurator", {"app_specific_ruleset": self._app_specific_ruleset}, ) @staticmethod def _read_cron() -> Optional[str]: with suppress(FileNotFoundError): return CronJob.from_str( Path(WAF_CONFIGURATOR_CRON_PATH).read_text() ) return None @staticmethod def _write_cron(cron_job: CronJob): path = Path(WAF_CONFIGURATOR_CRON_PATH) path.touch(mode=0o600) path.write_text(str(cron_job)) @staticmethod def _get_cron_job(update_modsec_rulesets) -> CronJob: shell_cmd = ( "/usr/libexec/report-command-error " + UPDATE_COMPONENTS_SCRIPT + (" --update-modsec-rulesets" if update_modsec_rulesets else "") + " > /dev/null 2>&1" ) hour, minute = 4, randint(0, 59) return CronJob(minute=minute, hour=hour, cmd=shell_cmd) async def _update_cron_job(self, enabled): cron_job = self._get_cron_job( update_modsec_rulesets=enabled and await is_webserver_supported() ) if self._cron_job and cron_job.cmd == self._cron_job.cmd: return logger.info("Updating AppVersionDetector version cron") self._write_cron(cron_job) self._cron_job = cron_job async def _truncate_conf(self): """ If app-specific httpd config exists and is not empty, truncate it """ try: config_path = HostingPanel().get_app_specific_waf_config() st = os.stat(config_path) except (FileNotFoundError, NotImplementedError): pass else: if st.st_size: open(config_path, "w").close() await web_server.graceful_restart() logger.info("App specific ruleset config truncated") @expect(MessageType.ConfigUpdate) async def update_cron_job(self, message): if isinstance(message["conf"], SystemConfig): enabled = Config.APP_SPECIFIC_RULESET if enabled != self._app_specific_ruleset: self._app_specific_ruleset = enabled await self._update_cron_job(enabled) if not enabled: await self._truncate_conf()