378 lines
12 KiB
Python
378 lines
12 KiB
Python
# PyMod-Milter 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.
|
|
#
|
|
# PyMod-Milter 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 PyMod-Milter. If not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
|
|
__all__ = [
|
|
"actions",
|
|
"conditions",
|
|
"run",
|
|
"CustomLogger",
|
|
"Rule",
|
|
"ModifyMilter"]
|
|
|
|
__version__ = "1.1.4"
|
|
|
|
from pymodmilter import _runtime_patches
|
|
|
|
import Milter
|
|
import logging
|
|
|
|
from Milter.utils import parse_addr
|
|
|
|
from collections import defaultdict
|
|
|
|
from email.header import Header
|
|
from email.message import MIMEPart
|
|
from email.parser import BytesFeedParser
|
|
from email.policy import default as default_policy, SMTP
|
|
|
|
from pymodmilter.actions import Action
|
|
from pymodmilter.conditions import Conditions
|
|
|
|
|
|
class CustomLogger(logging.LoggerAdapter):
|
|
def process(self, msg, kwargs):
|
|
if "name" in self.extra:
|
|
msg = "{}: {}".format(self.extra["name"], msg)
|
|
|
|
if "qid" in self.extra:
|
|
msg = "{}: {}".format(self.extra["qid"], msg)
|
|
|
|
if self.logger.getEffectiveLevel() != logging.DEBUG:
|
|
msg = msg.replace("\n", "").replace("\r", "")
|
|
|
|
return msg, kwargs
|
|
|
|
|
|
class Rule:
|
|
"""
|
|
Rule to implement multiple actions on emails.
|
|
"""
|
|
|
|
def __init__(self, milter_cfg, cfg):
|
|
logger = logging.getLogger(cfg["name"])
|
|
self.logger = CustomLogger(logger, {"name": cfg["name"]})
|
|
self.logger.setLevel(cfg["loglevel"])
|
|
|
|
if cfg["conditions"] is None:
|
|
self.conditions = None
|
|
else:
|
|
self.conditions = Conditions(milter_cfg, cfg["conditions"])
|
|
|
|
self._need_body = False
|
|
|
|
self.actions = []
|
|
for action_cfg in cfg["actions"]:
|
|
action = Action(milter_cfg, action_cfg)
|
|
self.actions.append(action)
|
|
if action.need_body():
|
|
self._need_body = True
|
|
|
|
self.pretend = cfg["pretend"]
|
|
|
|
def need_body(self):
|
|
"""Return True if this rule needs the message body."""
|
|
return self._need_body
|
|
|
|
def ignores(self, host=None, envfrom=None, envto=None):
|
|
args = {}
|
|
|
|
if host is not None:
|
|
args["host"] = host
|
|
|
|
if envfrom is not None:
|
|
args["envfrom"] = envfrom
|
|
|
|
if envto is not None:
|
|
args["envto"] = envto
|
|
|
|
if self.conditions is None or self.conditions.match(args):
|
|
for action in self.actions:
|
|
if action.conditions is None or action.conditions.match(args):
|
|
return False
|
|
|
|
return True
|
|
|
|
def execute(self, milter, pretend=None):
|
|
"""Execute all actions of this rule."""
|
|
if pretend is None:
|
|
pretend = self.pretend
|
|
|
|
for action in self.actions:
|
|
milter_action = action.execute(milter, pretend=pretend)
|
|
if milter_action is not None:
|
|
return milter_action
|
|
|
|
|
|
class MilterMessage(MIMEPart):
|
|
def replace_header(self, _name, _value, idx=None):
|
|
_name = _name.lower()
|
|
counter = 0
|
|
for i, (k, v) in zip(range(len(self._headers)), self._headers):
|
|
if k.lower() == _name:
|
|
counter += 1
|
|
if not idx or counter == idx:
|
|
self._headers[i] = self.policy.header_store_parse(
|
|
k, _value)
|
|
break
|
|
|
|
else:
|
|
raise KeyError(_name)
|
|
|
|
def remove_header(self, name, idx=None):
|
|
name = name.lower()
|
|
newheaders = []
|
|
counter = 0
|
|
for k, v in self._headers:
|
|
if k.lower() == name:
|
|
counter += 1
|
|
if counter != idx:
|
|
newheaders.append((k, v))
|
|
else:
|
|
newheaders.append((k, v))
|
|
|
|
self._headers = newheaders
|
|
|
|
|
|
def replace_illegal_chars(string):
|
|
"""Replace illegal characters in header values."""
|
|
return string.replace(
|
|
"\x00", "").replace(
|
|
"\r", "").replace(
|
|
"\n", "")
|
|
|
|
|
|
class ModifyMilter(Milter.Base):
|
|
"""ModifyMilter based on Milter.Base to implement milter communication"""
|
|
|
|
_rules = []
|
|
_loglevel = logging.INFO
|
|
|
|
@staticmethod
|
|
def set_config(cfg):
|
|
ModifyMilter._loglevel = cfg["loglevel"]
|
|
for rule_cfg in cfg["rules"]:
|
|
ModifyMilter._rules.append(
|
|
Rule(cfg, rule_cfg))
|
|
|
|
def __init__(self):
|
|
self.logger = logging.getLogger(__name__)
|
|
self.logger.setLevel(ModifyMilter._loglevel)
|
|
|
|
# save rules, it must not change during runtime
|
|
self.rules = ModifyMilter._rules.copy()
|
|
|
|
self.msg = None
|
|
self._replace_body = False
|
|
|
|
def addheader(self, field, value, idx=-1):
|
|
value = replace_illegal_chars(Header(s=value).encode())
|
|
self.logger.debug(f"milter: addheader: {field}: {value}")
|
|
super().addheader(field, value, idx)
|
|
|
|
def chgheader(self, field, value, idx=1):
|
|
value = replace_illegal_chars(Header(s=value).encode())
|
|
if value:
|
|
self.logger.debug(f"milter: chgheader: {field}[{idx}]: {value}")
|
|
else:
|
|
self.logger.debug(f"milter: delheader: {field}[{idx}]")
|
|
super().chgheader(field, idx, value)
|
|
|
|
def update_headers(self, old_headers):
|
|
if self.msg.is_multipart() and not self.msg["MIME-Version"]:
|
|
self.msg.add_header("MIME-Version", "1.0")
|
|
|
|
# serialize the message object so it updates its internal strucure
|
|
self.msg.as_bytes()
|
|
|
|
old_headers = [(f, f.lower(), v) for f, v in old_headers]
|
|
headers = [(f, f.lower(), v) for f, v in self.msg.items()]
|
|
|
|
idx = defaultdict(int)
|
|
for field, field_lower, value in old_headers:
|
|
idx[field_lower] += 1
|
|
if (field, field_lower, value) not in headers:
|
|
self.chgheader(field, "", idx=idx[field_lower])
|
|
idx[field] -= 1
|
|
|
|
for field, value in self.msg.items():
|
|
field_lower = field.lower()
|
|
if (field, field_lower, value) not in old_headers:
|
|
self.addheader(field, value)
|
|
|
|
def replacebody(self):
|
|
self._replace_body = True
|
|
|
|
def connect(self, IPname, family, hostaddr):
|
|
try:
|
|
if hostaddr is None:
|
|
self.logger.error("unable to proceed, host address is None")
|
|
return Milter.TEMPFAIL
|
|
|
|
self.logger.debug(
|
|
f"accepted milter connection from {hostaddr[0]} "
|
|
f"port {hostaddr[1]}")
|
|
|
|
# remove rules which ignore this host
|
|
for rule in self.rules.copy():
|
|
if rule.ignores(host=hostaddr[0]):
|
|
self.rules.remove(rule)
|
|
|
|
if not self.rules:
|
|
self.logger.debug(
|
|
f"host {hostaddr[0]} is ignored by all rules, "
|
|
f"skip further processing")
|
|
return Milter.ACCEPT
|
|
except Exception as e:
|
|
self.logger.exception(
|
|
f"an exception occured in connect method: {e}")
|
|
return Milter.TEMPFAIL
|
|
|
|
return Milter.CONTINUE
|
|
|
|
def envfrom(self, mailfrom, *str):
|
|
try:
|
|
mailfrom = "@".join(parse_addr(mailfrom)).lower()
|
|
for rule in self.rules.copy():
|
|
if rule.ignores(envfrom=mailfrom):
|
|
self.rules.remove(rule)
|
|
|
|
if not self.rules:
|
|
self.logger.debug(
|
|
f"envelope-from address {mailfrom} is ignored by "
|
|
f"all rules, skip further processing")
|
|
return Milter.ACCEPT
|
|
|
|
self.recipients = set()
|
|
except Exception as e:
|
|
self.logger.exception(
|
|
f"an exception occured in envfrom method: {e}")
|
|
return Milter.TEMPFAIL
|
|
|
|
return Milter.CONTINUE
|
|
|
|
def envrcpt(self, to, *str):
|
|
try:
|
|
self.recipients.add("@".join(parse_addr(to)).lower())
|
|
except Exception as e:
|
|
self.logger.exception(
|
|
f"an exception occured in envrcpt method: {e}")
|
|
return Milter.TEMPFAIL
|
|
|
|
return Milter.CONTINUE
|
|
|
|
def data(self):
|
|
try:
|
|
for rule in self.rules.copy():
|
|
if rule.ignores(envto=[*self.recipients]):
|
|
self.rules.remove(rule)
|
|
|
|
if not self.rules:
|
|
self.logger.debug(
|
|
"envelope-to addresses are ignored by all rules, "
|
|
"skip further processing")
|
|
return Milter.ACCEPT
|
|
|
|
self.qid = self.getsymval('i')
|
|
self.logger = CustomLogger(self.logger, {"qid": self.qid})
|
|
self.logger.debug("received queue-id from MTA")
|
|
|
|
self.fields = None
|
|
self.fields_bytes = None
|
|
self.body_data = None
|
|
|
|
self._fp = BytesFeedParser(
|
|
_factory=MilterMessage, policy=default_policy)
|
|
self._keep_body = False
|
|
for rule in self.rules:
|
|
if rule.need_body():
|
|
self._keep_body = True
|
|
break
|
|
|
|
except Exception as e:
|
|
self.logger.exception(
|
|
f"an exception occured in data method: {e}")
|
|
return Milter.TEMPFAIL
|
|
|
|
return Milter.CONTINUE
|
|
|
|
def header(self, field, value):
|
|
try:
|
|
# remove surrogates
|
|
field = field.encode("ascii", errors="replace")
|
|
value = value.encode("ascii", errors="replace")
|
|
|
|
self._fp.feed(field + b": " + value + b"\r\n")
|
|
except Exception as e:
|
|
self.logger.exception(
|
|
f"an exception occured in header method: {e}")
|
|
return Milter.TEMPFAIL
|
|
|
|
return Milter.CONTINUE
|
|
|
|
def eoh(self):
|
|
try:
|
|
self._fp.feed(b"\r\n")
|
|
except Exception as e:
|
|
self.logger.exception(
|
|
f"an exception occured in eoh method: {e}")
|
|
return Milter.TEMPFAIL
|
|
|
|
return Milter.CONTINUE
|
|
|
|
def body(self, chunk):
|
|
try:
|
|
if self._keep_body:
|
|
self._fp.feed(chunk)
|
|
except Exception as e:
|
|
self.logger.exception(
|
|
f"an exception occured in body method: {e}")
|
|
return Milter.TEMPFAIL
|
|
|
|
return Milter.CONTINUE
|
|
|
|
def eom(self):
|
|
try:
|
|
self.msg = self._fp.close()
|
|
milter_action = None
|
|
for rule in self.rules:
|
|
milter_action = rule.execute(self)
|
|
|
|
if milter_action is not None:
|
|
break
|
|
|
|
if self._replace_body:
|
|
data = self.msg.as_bytes(policy=SMTP)
|
|
body_pos = data.find(b"\r\n\r\n") + 4
|
|
self.logger.debug("milter: replacebody")
|
|
super().replacebody(data[body_pos:])
|
|
del data
|
|
|
|
if milter_action is not None:
|
|
if milter_action["action"] == "reject":
|
|
self.setreply("554", "5.7.0", milter_action["reason"])
|
|
return Milter.REJECT
|
|
|
|
if milter_action["action"] == "accept":
|
|
return Milter.ACCEPT
|
|
|
|
if milter_action["action"] == "discard":
|
|
return Milter.DISCARD
|
|
|
|
except Exception as e:
|
|
self.logger.exception(
|
|
f"an exception occured in eom method: {e}")
|
|
return Milter.TEMPFAIL
|
|
|
|
return Milter.ACCEPT
|