diff options
| author | Barry Warsaw | 2010-08-08 10:49:55 -0400 |
|---|---|---|
| committer | Barry Warsaw | 2010-08-08 10:49:55 -0400 |
| commit | 79cc1bc34c9386058a9f0734ab9ad7fad3b637b5 (patch) | |
| tree | 0252af62b19a0edbd311693b303b12bc1f2e9868 /src/mailman/bouncers/postfix.py | |
| parent | 83f78ec26541ab0c7b91794ab6b3bc1d8285f9a9 (diff) | |
| download | mailman-79cc1bc34c9386058a9f0734ab9ad7fad3b637b5.tar.gz mailman-79cc1bc34c9386058a9f0734ab9ad7fad3b637b5.tar.zst mailman-79cc1bc34c9386058a9f0734ab9ad7fad3b637b5.zip | |
Diffstat (limited to 'src/mailman/bouncers/postfix.py')
| -rw-r--r-- | src/mailman/bouncers/postfix.py | 109 |
1 files changed, 109 insertions, 0 deletions
diff --git a/src/mailman/bouncers/postfix.py b/src/mailman/bouncers/postfix.py new file mode 100644 index 000000000..c178f48c0 --- /dev/null +++ b/src/mailman/bouncers/postfix.py @@ -0,0 +1,109 @@ +# Copyright (C) 1998-2010 by the Free Software Foundation, Inc. +# +# This file is part of GNU Mailman. +# +# GNU Mailman 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. +# +# GNU Mailman 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 +# GNU Mailman. If not, see <http://www.gnu.org/licenses/>. + +"""Parse bounce messages generated by Postfix. + +This also matches something called 'Keftamail' which looks just like Postfix +bounces with the word Postfix scratched out and the word 'Keftamail' written +in in crayon. + +It also matches something claiming to be 'The BNS Postfix program', and +'SMTP_Gateway'. Everybody's gotta be different, huh? +""" + +from __future__ import absolute_import, unicode_literals + +__metaclass__ = type +__all__ = [ + 'Postfix', + ] + + +import re + +from cStringIO import StringIO +from flufl.enum import Enum +from zope.interface import implements + +from mailman.interfaces.bounce import IBounceDetector + + +# Are these heuristics correct or guaranteed? +pcre = re.compile(r'[ \t]*the\s*(bns)?\s*(postfix|keftamail|smtp_gateway)', + re.IGNORECASE) +rcre = re.compile(r'failure reason:$', re.IGNORECASE) +acre = re.compile(r'<(?P<addr>[^>]*)>:') + +REPORT_TYPES = ('multipart/mixed', 'multipart/report') + + +class ParseState(Enum): + start = 0 + salutation_found = 1 + + + +def flatten(msg, leaves): + # Give us all the leaf (non-multipart) subparts. + if msg.is_multipart(): + for part in msg.get_payload(): + flatten(part, leaves) + else: + leaves.append(msg) + + + +def findaddr(msg): + addresses = set() + body = StringIO(msg.get_payload()) + state = ParseState.start + for line in body: + # Preserve leading whitespace. + line = line.rstrip() + # Yes, use match() to match at beginning of string. + if state is ParseState.start and ( + pcre.match(line) or rcre.match(line)): + # Then... + state = ParseState.salutation_found + elif state is ParseState.salutation_found and line: + mo = acre.search(line) + if mo: + addresses.add(mo.group('addr')) + # Probably a continuation line. + return addresses + + + +class Postfix: + """Parse bounce messages generated by Postfix.""" + + implements(IBounceDetector) + + def process(self, msg): + """See `IBounceDetector`.""" + if msg.get_content_type() not in REPORT_TYPES: + return None + # We're looking for the plain/text subpart with a Content-Description: + # of 'notification'. + leaves = [] + flatten(msg, leaves) + for subpart in leaves: + content_type = subpart.get_content_type() + content_desc = subpart.get('content-description', '').lower() + if content_type == 'text/plain' and content_desc == 'notification': + return set(findaddr(subpart)) + return None |
