Compare commits

...

2 Commits

Author SHA1 Message Date
Stuart Gathman 16dea6e187 Release 0.7.1 2005-05-31 18:09:06 +00:00
Stuart Gathman 802dc01c84 Release 0.7.0 2005-05-31 18:08:20 +00:00
15 changed files with 828 additions and 159 deletions
+1
View File
@@ -8,6 +8,7 @@ include testsample.py
include testmime.py include testmime.py
include testbms.py include testbms.py
include testdspam.py include testdspam.py
include rejects.py
include bms.py include bms.py
include spf.py include spf.py
include spfquery.py include spfquery.py
+9 -8
View File
@@ -8,15 +8,12 @@ import milter
import thread import thread
from milter import ACCEPT,CONTINUE,REJECT,DISCARD,TEMPFAIL, \ from milter import ACCEPT,CONTINUE,REJECT,DISCARD,TEMPFAIL, \
set_flags, setdbg, \ set_flags, setdbg, setbacklog, settimeout, \
ADDHDRS, CHGBODY, ADDRCPT, DELRCPT, CHGHDRS, \ ADDHDRS, CHGBODY, ADDRCPT, DELRCPT, CHGHDRS, \
V1_ACTS, V2_ACTS, CURR_ACTS V1_ACTS, V2_ACTS, CURR_ACTS
try: try: from milter import QUARANTINE
from milter import QUARANTINE except: pass
except:
#print 'No QUARANTINE support'
pass
_seq_lock = thread.allocate_lock() _seq_lock = thread.allocate_lock()
_seq = 0 _seq = 0
@@ -100,8 +97,10 @@ class Milter:
def getsymval(self,sym): def getsymval(self,sym):
return self.__ctx.getsymval(sym) return self.__ctx.getsymval(sym)
def setreply(self,rcode,xcode,msg): # If sendmail does not support setmlreply, then only the
return self.__ctx.setreply(rcode,xcode,msg) # first msg line is used.
def setreply(self,rcode,xcode=None,msg=None,*ml):
return self.__ctx.setreply(rcode,xcode,msg,*ml)
# Milter methods which can only be called from eom callback. # Milter methods which can only be called from eom callback.
def addheader(self,field,value): def addheader(self,field,value):
@@ -119,6 +118,8 @@ class Milter:
def replacebody(self,body): def replacebody(self,body):
return self.__ctx.replacebody(body) return self.__ctx.replacebody(body)
# When quarantined, a message goes into the mailq as if to be delivered,
# but delivery is deferred until the message is unquarantined.
def quarantine(self,reason): def quarantine(self,reason):
return self.__ctx.quarantine(reason) return self.__ctx.quarantine(reason)
+10
View File
@@ -1,5 +1,15 @@
Here is a history of user visible changes to Python milter. Here is a history of user visible changes to Python milter.
0.7.1 Handle modifying mislabeled multipart messages without an exception
Support setbacklog, setmlreply
Allow multi-recipient CBV
Return TEMPFAIL for SPF softfail
0.7.0 SPF check hello name
Move pythonsock to /var/run/milter
Move milter.cfg to /etc/mail/pymilter.cfg
Check M$ style XML CID records by converting to SPF
Recognize, but never match ip6 - until we properly support it.
Option to reject when no PTR and no SPF
0.6.9 Reject invalid SRS immediately for benefit of callback verifiers 0.6.9 Reject invalid SRS immediately for benefit of callback verifiers
Fix include bug in spf.py Fix include bug in spf.py
Fix check_header bug Fix check_header bug
+6 -2
View File
@@ -1,10 +1,14 @@
Move milter,Milter,mime,spf modules to pymilter
milter package will have bms.py application
spf.py has no recursion bound on CNAME lookup
Support SMTP AUTH and disable SPF checks when connection is authorized.
Web admin interface Web admin interface
RHBL RHSBL
Check valid domains allowed by internal senders to detect PCs infected Check valid domains allowed by internal senders to detect PCs infected
with spam trojans. with spam trojans.
Do CBV (callback verification) for mail with no published SPF record. Do CBV (callback verification) for mail with no published SPF record.
message log for automated stats and blacklisting message log for automated stats and blacklisting
adapt init script to work on RH9
Skip dspam when SPF pass? Skip dspam when SPF pass?
Report 551 with rcpt on SPF fail? Report 551 with rcpt on SPF fail?
check spam keywords with character classes, e.g. check spam keywords with character classes, e.g.
+133 -29
View File
@@ -1,6 +1,48 @@
#!/usr/bin/env python #!/usr/bin/env python
# A simple milter. # A simple milter.
# $Log$ # $Log$
# Revision 1.117 2004/08/23 02:27:53 stuart
# Allow multi rcpt CBV. Add some multiline replies.
#
# Revision 1.116 2004/08/20 22:27:52 stuart
# Generate TEMPFAIL for SPF softfail.
#
# Revision 1.115 2004/08/19 20:55:49 stuart
# Always show reversed SRS path.
# Check if encodings are an ASCII superset. Some messages were encoded as
# BIG5 and getting rejected even though chars were all in ascii subset.
#
# Revision 1.114 2004/07/27 00:40:12 stuart
# Make reject on no PTR optional.
#
# Revision 1.113 2004/07/23 23:11:14 stuart
# Log known malformed messages differently than general processing exceptions.
#
# Revision 1.112 2004/07/21 19:18:33 stuart
# Punt on UnicodeDecodeError when decoding headers.
# Accept a pass with default SPF for missing reverse IP.
#
# Revision 1.111 2004/07/18 13:13:31 stuart
# Reject invalid SRS only for SRS domain (which is the only one we
# know the key for).
# Reject senders that have neither reverse IP nor SPF.
#
# Revision 1.110 2004/06/12 03:13:18 stuart
# Block bounces only for SRS domain. Also treat mail from
# postmaster or mailer-daemon as DSN for SRS/SES checking purposes.
#
# Revision 1.109 2004/05/01 02:56:55 stuart
# Let multiple screeners share work.
#
# Revision 1.108 2004/04/29 20:36:23 stuart
# Require HELO name
#
# Revision 1.107 2004/04/24 22:55:29 stuart
# Move some files to make the RPM more standard.
#
# Revision 1.106 2004/04/21 18:29:08 stuart
# Validate hello name with SPF.
#
# Revision 1.105 2004/04/20 15:16:00 stuart # Revision 1.105 2004/04/20 15:16:00 stuart
# Release 0.6.9 # Release 0.6.9
# #
@@ -242,14 +284,16 @@ dspam_users = {}
dspam_userdir = None dspam_userdir = None
dspam_exempt = {} dspam_exempt = {}
dspam_whitelist = {} dspam_whitelist = {}
dspam_screener = None dspam_screener = ()
dspam_internal = True # True if internal mail should be dspammed dspam_internal = True # True if internal mail should be dspammed
dspam_reject = () dspam_reject = ()
dspam_sizelimit = 180000 dspam_sizelimit = 180000
srs = None srs = None
srs_reject_spoofed = False srs_reject_spoofed = False
srs_fwdomain = None
spf_reject_neutral = () spf_reject_neutral = ()
spf_best_guess = False spf_best_guess = False
spf_reject_noptr = False
timeout = 600 timeout = 600
class MilterConfigParser(ConfigParser.ConfigParser): class MilterConfigParser(ConfigParser.ConfigParser):
@@ -300,7 +344,7 @@ class MilterConfigParser(ConfigParser.ConfigParser):
def read_config(list): def read_config(list):
cp = MilterConfigParser({ cp = MilterConfigParser({
'tempdir': "/var/log/milter/save", 'tempdir': "/var/log/milter/save",
'socket': "/var/log/milter/pythonsock", 'socket': "/var/run/milter/pythonsock",
'timeout': '600', 'timeout': '600',
'scan_html': 'no', 'scan_html': 'no',
'scan_rfc822': 'yes', 'scan_rfc822': 'yes',
@@ -310,6 +354,7 @@ def read_config(list):
'maxage': '8', 'maxage': '8',
'hashlength': '8', 'hashlength': '8',
'reject_spoofed': 'no', 'reject_spoofed': 'no',
'reject_noptr': 'no',
'best_guess': 'no' 'best_guess': 'no'
}) })
cp.read(list) cp.read(list)
@@ -355,13 +400,13 @@ def read_config(list):
global dspam_dict, dspam_users, dspam_userdir, dspam_exempt global dspam_dict, dspam_users, dspam_userdir, dspam_exempt
global dspam_screener,dspam_whitelist,dspam_reject,dspam_sizelimit global dspam_screener,dspam_whitelist,dspam_reject,dspam_sizelimit
global spf_reject_neutral,spf_best_guess,SRS global spf_reject_neutral,spf_best_guess,SRS,spf_reject_noptr
dspam_dict = cp.getdefault('dspam','dspam_dict') dspam_dict = cp.getdefault('dspam','dspam_dict')
dspam_exempt = cp.getaddrset('dspam','dspam_exempt') dspam_exempt = cp.getaddrset('dspam','dspam_exempt')
dspam_whitelist = cp.getaddrset('dspam','dspam_whitelist') dspam_whitelist = cp.getaddrset('dspam','dspam_whitelist')
dspam_users = cp.getaddrdict('dspam','dspam_users') dspam_users = cp.getaddrdict('dspam','dspam_users')
dspam_userdir = cp.getdefault('dspam','dspam_userdir') dspam_userdir = cp.getdefault('dspam','dspam_userdir')
dspam_screener = cp.getdefault('dspam','dspam_screener') dspam_screener = cp.getlist('dspam','dspam_screener')
dspam_reject = cp.getlist('dspam','dspam_reject') dspam_reject = cp.getlist('dspam','dspam_reject')
if cp.has_option('dspam','dspam_sizelimit'): if cp.has_option('dspam','dspam_sizelimit'):
dspam_sizelimit = cp.getint('dspam','dspam_sizelimit') dspam_sizelimit = cp.getint('dspam','dspam_sizelimit')
@@ -370,11 +415,12 @@ def read_config(list):
spf.DELEGATE = cp.getdefault('spf','delegate') spf.DELEGATE = cp.getdefault('spf','delegate')
spf_reject_neutral = cp.getlist('spf','reject_neutral') spf_reject_neutral = cp.getlist('spf','reject_neutral')
spf_best_guess = cp.getboolean('spf','best_guess') spf_best_guess = cp.getboolean('spf','best_guess')
spf_reject_noptr = cp.getboolean('spf','reject_noptr')
srs_config = cp.getdefault('srs','config') srs_config = cp.getdefault('srs','config')
if srs_config: cp.read([srs_config]) if srs_config: cp.read([srs_config])
srs_secret = cp.getdefault('srs','secret') srs_secret = cp.getdefault('srs','secret')
if SRS and srs_secret: if SRS and srs_secret:
global srs,srs_reject_spoofed global srs,srs_reject_spoofed,srs_fwdomain
database = cp.getdefault('srs','database') database = cp.getdefault('srs','database')
srs_reject_spoofed = cp.getboolean('srs','reject_spoofed') srs_reject_spoofed = cp.getboolean('srs','reject_spoofed')
maxage = cp.getint('srs','maxage') maxage = cp.getint('srs','maxage')
@@ -387,7 +433,7 @@ def read_config(list):
else: else:
srs = SRS.Guarded.Guarded(secret=srs_secret, srs = SRS.Guarded.Guarded(secret=srs_secret,
maxage=maxage,hashlength=hashlength,separator=separator) maxage=maxage,hashlength=hashlength,separator=separator)
srs_fwdomain = cp.getdefault('srs','fwdomain')
def parse_addr(t): def parse_addr(t):
if t.startswith('<') and t.endswith('>'): t = t[1:-1] if t.startswith('<') and t.endswith('>'): t = t[1:-1]
@@ -400,7 +446,10 @@ def parse_header(val):
u = [] u = []
for s,enc in h: for s,enc in h:
if enc: if enc:
try:
u.append(unicode(s,enc)) u.append(unicode(s,enc))
except LookupError:
u.append(unicode(s))
else: else:
u.append(unicode(s)) u.append(unicode(s))
u = ''.join(u) u = ''.join(u)
@@ -408,11 +457,14 @@ def parse_header(val):
try: try:
return u.encode(enc) return u.encode(enc)
except UnicodeError: continue except UnicodeError: continue
except LookupError: except UnicodeDecodeError: pass
except LookupError: pass
return val return val
class bmsMilter(Milter.Milter): class bmsMilter(Milter.Milter):
"Milter to replace attachments poisonous to Windows with a WARNING message." """Milter to replace attachments poisonous to Windows with a WARNING message,
check SPF, and other anti-forgery features, and implement wiretapping
and smart alias redirection."""
def log(self,*msg): def log(self,*msg):
print "%s [%d]" % (time.strftime('%Y%b%d %H:%M:%S'),self.id), print "%s [%d]" % (time.strftime('%Y%b%d %H:%M:%S'),self.id),
@@ -448,6 +500,7 @@ class bmsMilter(Milter.Milter):
self.log('%s: %s' % (name,val)) self.log('%s: %s' % (name,val))
def connect(self,hostname,unused,hostaddr): def connect(self,hostname,unused,hostaddr):
self.missing_ptr = hostname.startswith('[') and hostname.endswith(']')
self.internal_connection = False self.internal_connection = False
self.trusted_relay = False self.trusted_relay = False
self.receiver = self.getsymval('j') self.receiver = self.getsymval('j')
@@ -475,6 +528,8 @@ class bmsMilter(Milter.Milter):
if self.trusted_relay: if self.trusted_relay:
connecttype += ' TRUSTED' connecttype += ' TRUSTED'
self.log("connect from %s at %s %s" % (hostname,hostaddr,connecttype)) self.log("connect from %s at %s %s" % (hostname,hostaddr,connecttype))
self.hello_name = None
self.connecthost = hostname
return Milter.CONTINUE return Milter.CONTINUE
def hello(self,hostname): def hello(self,hostname):
@@ -531,6 +586,10 @@ class bmsMilter(Milter.Milter):
self.dspam = False self.dspam = False
else: else:
self.rejectvirus = False self.rejectvirus = False
if not self.hello_name:
self.log("REJECT: missing HELO")
self.setreply('550','5.7.1',"It's polite to say HELO first.")
return Milter.REJECT
if not (self.internal_connection or self.trusted_relay) \ if not (self.internal_connection or self.trusted_relay) \
and self.connectip and spf: and self.connectip and spf:
return self.check_spf() return self.check_spf()
@@ -543,20 +602,56 @@ class bmsMilter(Milter.Milter):
q.set_default_explanation('SPF fail: see http://spf.pobox.com/why.html') q.set_default_explanation('SPF fail: see http://spf.pobox.com/why.html')
res,code,txt = q.check() res,code,txt = q.check()
receiver = self.receiver receiver = self.receiver
if res == 'none' and spf_best_guess: if res == 'none':
if self.mailfrom != '<>':
# check hello name via spf
hres,hcode,htxt = spf.check(self.connectip,'',self.hello_name)
if hres in ('deny','fail','neutral','softfail'):
self.log('REJECT: hello SPF: %s 550 %s' % (hres,htxt))
self.setreply('550','5.7.1',htxt,
"The hostname given in your MTA's HELO response is not listed",
"as a legitimate MTA in the SPF records for your domain.",
"If you get this bounce, the message was not in fact a forgery,",
"and you should notify your email administrator of the problem."
)
return Milter.REJECT
if spf_best_guess:
#self.log('SPF: no record published, guessing') #self.log('SPF: no record published, guessing')
q.set_default_explanation('SPF guess: see http://spf.pobox.com/why.html') q.set_default_explanation(
'SPF guess: see http://spf.pobox.com/why.html')
# best_guess should not result in fail # best_guess should not result in fail
res,code,txt = q.best_guess() res,code,txt = q.best_guess()
receiver += ': guessing' receiver += ': guessing'
if self.missing_ptr and res in ('neutral', 'none') and spf_reject_noptr:
self.log('REJECT: no PTR or SPF')
self.setreply('550','5.7.1',
'You must have a reverse lookup or publish SPF: http://spf.pobox.com'
)
return Milter.REJECT
if res in ('deny', 'fail'): if res in ('deny', 'fail'):
self.log('REJECT: SPF %s %i %s' % (res,code,txt)) self.log('REJECT: SPF %s %i %s' % (res,code,txt))
self.setreply(str(code),'5.7.1',txt) self.setreply(str(code),'5.7.1',txt)
return Milter.REJECT return Milter.REJECT
if res == 'softfail':
self.log('TEMPFAIL: SPF %s 450 %s' % (res,txt))
self.setreply('450','4.3.0',
'SPF softfail: will keep trying until your SPF record is fixed.',
'If you get this Delivery Status Notice, your email was probably',
'legitimate. Your administrator has published SPF records in a',
'testing mode. The SPF record reported your email as a forgery,',
'which is a mistake if you are reading this. Please notify your',
'administrator of the problem.'
)
return Milter.TEMPFAIL
if res == 'neutral' and q.o in spf_reject_neutral: if res == 'neutral' and q.o in spf_reject_neutral:
self.log('REJECT: SPF neutral for',q.s) self.log('REJECT: SPF neutral for',q.s)
self.setreply('550','5.7.1', self.setreply('550','5.7.1',
'mail from %s must pass SPF: http://spf.pobox.com/why.html' % q.o 'mail from %s must pass SPF: http://spf.pobox.com/why.html' % q.o,
'The %s domain is one that spammers love to forge. Due to' % q.o,
'the volume of forged mail, we can only accept mail that',
'the SPF record for %s explicitly designates as legitimate.' % q.o,
'Sending your email through the recommended outgoing SMTP',
'servers for %s should accomplish this.' % q.o
) )
return Milter.REJECT return Milter.REJECT
if res == 'error': if res == 'error':
@@ -576,29 +671,31 @@ class bmsMilter(Milter.Milter):
self.log("rcpt to",to,str) self.log("rcpt to",to,str)
t = parse_addr(to.lower()) t = parse_addr(to.lower())
if len(t) == 2: if len(t) == 2:
if self.mailfrom == '<>': user,domain = t
if self.mailfrom == '<>' or self.canon_from.startswith('postmaster@') \
or self.canon_from.startswith('mailer-daemon@'):
if self.recipients: if self.recipients:
self.log('REJECT: Multiple bounce recipients') self.data_allowed = False
self.setreply('550','5.7.1','Multiple bounce recipients') if srs and domain == srs_fwdomain:
return Milter.REJECT
if srs and not (self.internal_connection or self.trusted_relay):
oldaddr = '@'.join(parse_addr(to)) oldaddr = '@'.join(parse_addr(to))
try: try:
newaddr = srs.reverse(oldaddr) newaddr = srs.reverse(oldaddr)
# Currently, a sendmail map reverses SRS. We just log it here.
self.log("srs rcpt:",newaddr) self.log("srs rcpt:",newaddr)
except: except:
if not (self.internal_connection or self.trusted_relay):
if srsre.match(oldaddr): if srsre.match(oldaddr):
self.log("REJECT: srs spoofed:",oldaddr) self.log("REJECT: srs spoofed:",oldaddr)
self.setreply('550','5.7.1','Invalid SRS signature') self.setreply('550','5.7.1','Invalid SRS signature')
return Milter.REJECT return Milter.REJECT
self.data_allowed = not srs_reject_spoofed self.data_allowed = not srs_reject_spoofed
# non DSN mail to SRS address will bounce due to invalid local part
self.recipients.append('@'.join(t)) self.recipients.append('@'.join(t))
user,domain = t
users = check_user.get(domain) users = check_user.get(domain)
if self.discard: if self.discard:
self.del_recipient(to) self.del_recipient(to)
if users and not user in users: if users and not user in users:
self.log('REJECT: RCPT TO:',to,str) self.log('REJECT: RCPT TO:',to)
return Milter.REJECT return Milter.REJECT
if user in block_forward.get(domain,()): if user in block_forward.get(domain,()):
self.forward = False self.forward = False
@@ -685,8 +782,12 @@ class bmsMilter(Milter.Milter):
def header(self,name,hval): def header(self,name,hval):
if not self.data_allowed: if not self.data_allowed:
if len(self.recipients) > 1:
self.log('REJECT: Multiple bounce recipients')
self.setreply('550','5.7.1','Multiple bounce recipients')
else:
self.log('REJECT: bounce with no SRS encoding') self.log('REJECT: bounce with no SRS encoding')
self.setreply('550','5.7.1',"spoofed reply address") self.setreply('550','5.7.1',"I did not send you that message.")
return Milter.REJECT return Milter.REJECT
lname = name.lower() lname = name.lower()
# decode near ascii text to unobfuscate # decode near ascii text to unobfuscate
@@ -694,8 +795,8 @@ class bmsMilter(Milter.Milter):
if not self.internal_connection: if not self.internal_connection:
# even if we wanted the Taiwanese spam, we can't read Chinese # even if we wanted the Taiwanese spam, we can't read Chinese
if block_chinese and lname == 'subject': if block_chinese and lname == 'subject':
if hval.startswith('=?big5') or hval.startswith('=?ISO-2022-JP'): if val.startswith('=?big5') or val.startswith('=?ISO-2022-JP'):
self.log('REJECT: %s: %s' % (name,hval)) self.log('REJECT: %s: %s' % (name,val))
self.setreply('550','5.7.1',"We don't understand chinese") self.setreply('550','5.7.1',"We don't understand chinese")
return Milter.REJECT return Milter.REJECT
rc = self.check_header(name,val) rc = self.check_header(name,val)
@@ -832,21 +933,22 @@ class bmsMilter(Milter.Milter):
print x print x
# screen if no recipients are dspam_users # screen if no recipients are dspam_users
if not modified and dspam_screener and not self.internal_connection \ if not modified and dspam_screener and not self.internal_connection \
and (self.dspam or self.reject_spam): and self.dspam:
self.fp.seek(0) self.fp.seek(0)
txt = self.fp.read() txt = self.fp.read()
if len(txt) > dspam_sizelimit: if len(txt) > dspam_sizelimit:
self.log("Large message:",len(txt)) self.log("Large message:",len(txt))
return False return False
if not ds.check_spam(dspam_screener,txt,self.recipients, screener = dspam_screener[self.id % len(dspam_screener)]
if not ds.check_spam(screener,txt,self.recipients,
classify=True,quarantine=not self.reject_spam): classify=True,quarantine=not self.reject_spam):
self.fp = None self.fp = None
if self.reject_spam: if self.reject_spam:
self.log("DSPAM:",dspam_screener, self.log("DSPAM:",screener,
'REJECT: X-DSpam-Score: %f' % ds.probability) 'REJECT: X-DSpam-Score: %f' % ds.probability)
self.setreply('550','5.7.1','Your Message looks spammy') self.setreply('550','5.7.1','Your Message looks spammy')
return True return True
self.log("DSPAM:",dspam_screener,"SCREENED") self.log("DSPAM:",screener,"SCREENED")
return modified return modified
def eom(self): def eom(self):
@@ -881,16 +983,18 @@ class bmsMilter(Milter.Milter):
fname = tempfile.mktemp(".fail") # save message that caused crash fname = tempfile.mktemp(".fail") # save message that caused crash
os.rename(self.tempname,fname) os.rename(self.tempname,fname)
self.tempname = None self.tempname = None
self.log("FAIL: %s" % fname) # log filename
if exc_type == email.Errors.BoundaryError: if exc_type == email.Errors.BoundaryError:
self.log("MALFORMED: %s" % fname) # log filename
self.setreply('554','5.7.7', self.setreply('554','5.7.7',
'Boundary error in your message, are you a spammer?') 'Boundary error in your message, are you a spammer?')
return Milter.REJECT return Milter.REJECT
if exc_type == email.Errors.HeaderParseError: if exc_type == email.Errors.HeaderParseError:
self.log("MALFORMED: %s" % fname) # log filename
self.setreply('554','5.7.7', self.setreply('554','5.7.7',
'Header parse error in your message, are you a spammer?') 'Header parse error in your message, are you a spammer?')
return Milter.REJECT return Milter.REJECT
# let default exception handler print traceback and return 451 code # let default exception handler print traceback and return 451 code
self.log("FAIL: %s" % fname) # log filename
raise raise
if rc == Milter.REJECT: return rc; if rc == Milter.REJECT: return rc;
if rc == Milter.DISCARD: return rc; if rc == Milter.DISCARD: return rc;
@@ -967,13 +1071,13 @@ def main():
if srs or len(discard_users) > 0 or smart_alias or dspam_userdir: if srs or len(discard_users) > 0 or smart_alias or dspam_userdir:
flags = flags + Milter.DELRCPT flags = flags + Milter.DELRCPT
Milter.set_flags(flags) Milter.set_flags(flags)
print "bms milter startup" print "%s bms milter startup" % time.strftime('%Y%b%d %H:%M:%S')
sys.stdout.flush() sys.stdout.flush()
Milter.runmilter("pythonfilter",socketname,timeout) Milter.runmilter("pythonfilter",socketname,timeout)
print "bms milter shutdown" print "%s bms milter shutdown" % time.strftime('%Y%b%d %H:%M:%S')
if __name__ == "__main__": if __name__ == "__main__":
read_config(["milter.cfg"]) read_config(["/etc/mail/pymilter.cfg","milter.cfg"])
if dspam_dict: if dspam_dict:
import dspam # low level spam check import dspam # low level spam check
if dspam_userdir: if dspam_userdir:
+153
View File
@@ -0,0 +1,153 @@
#!/usr/bin/python2.3
# Convert a MS Caller-ID entry (XML) to a SPF entry
#
# (c) 2004 by Ernesto Baschny
# (c) 2004 Python version by Stuart Gathman
#
# Date: 2004-02-25
# Version: 1.0
#
# Usage:
# ./cid2spf.pl "<ep xmlns='http://ms.net/1'>...</ep>"
#
# Note that the 'include' directives will also have to be checked and
# "translated". Future versions of this script might be able to get a
# domain name as an argument and "crawl" the DNS for the necessary
# information.
#
# A complete reverse translation (SPF -> CID) might be impossible, since
# there are no way to handle:
# - PTR and EXISTS mechanism
# - MX mechanism with an different domain as argument
# - macros
#
# References:
# http://www.microsoft.com/mscorp/twc/privacy/spam_callerid.mspx
# http://spf.pobox.com/
#
# Known bugs:
# - Currently it won't handle the exclusions provided in the A and R
# tags (prefix '!'). They will show up "as-is" in the SPF record
# - I really haven't read the MS-CID specs in-depth, so there are probably
# other bugs too :)
#
# Ernesto Baschny <ernst@baschny.de>
#
import xml.sax
import spf
# -------------------------------------------------------------------------
class CIDParser(xml.sax.ContentHandler):
"Convert a MS Caller-ID entry (XML) to a SPF entry"
def __init__(self,q=None):
self.spf = []
self.action = '-all'
self.has_servers = None
self.spf_entry = None
if q:
self.spf_query = q
else:
self.spf_query = spf.query(i='127.0.0.1', s='localhost', h='unknown')
def startElement(self,tag,attr):
if tag == 'm':
if self.has_servers != None and not self.has_servers:
raise ValueError(
"Declared <noMailServers\> and later <m>, this CID entry is not valid."
)
self.has_servers = True
elif tag == 'noMailServers':
if self.has_servers:
raise ValueError(
"Declared <m> and later <noMailServers\>, this CID entry is not valid."
)
self.has_servers = False
elif tag == 'ep':
if attr.has_key('testing') and attr.getValue('testing') == 'true':
# A CID with 'testing' found:
# From the MS-specs:
# "Documents in which such attribute is present with a true
# value SHOULD be entirely ignored (one should act as if the
# document were absent)"
# From the SPF-specs:
# "Neutral (?): The SPF client MUST proceed as if a domain did
# not publish SPF data."
# So we set SPF action to "neutral":
self.action = '?all'
elif tag == 'mx':
# The empty MX-tag, same as SPF's MX-mechanism
self.spf.append('mx')
self.tag = tag
def characters(self,text):
tag = self.tag
# Remove starting and trailing spaces from text:
text = text.strip()
if tag == 'a' or tag == 'r':
# The A and R tags from MS-CID are both handled by the
# ipv4/6-mechanisms from SPF:
if text.find(':') < 0:
mechanism = 'ip4'
else:
mechanism = 'ip6'
self.spf.append(mechanism + ':' + text)
elif tag == 'indirect':
# MS-CID's indirect is "sort of" the include from SPF:
# Not really true, because the <indirect> tag from MS-CID also
# provides a fallback in case the included domain doesn't provide
# _ep-records: The inbound MX-servers of the included domains
# are added to the list of allowed outgoing mailservers for the
# domain that declared the _ep-record with the <indirect> tag.
# In SPF you would use the 'mx:domain' to handle this, but this
# wouldn't depend on referred domain having or not SPF-records.
cid_xml = self.cid_txt(text)
if cid_xml:
p = CIDParser()
xml.sax.parseString(cid_xml,p)
if p.has_servers != False:
self.spf += p.spf
else:
self.spf.append('mx:' + text)
def cid_txt(self,domain):
q = self.spf_query
domain='_ep.' + domain
a = q.dns_txt(domain)
if not a: return None
if a[0].lower().startswith('<ep ') and a[-1].lower().endswith('</ep>'):
return ''.join(a)
return None
def endElement(self,tag):
if tag == 'ep':
# This is the end... assemble what we've got
spf_entry = ['v=spf1']
if self.has_servers != False:
spf_entry += self.spf
spf_entry.append(self.action)
self.spf_entry = ' '.join(spf_entry)
def spf_txt(self,cid_xml):
if not cid_xml.startswith('<'):
cid_xml = self.cid_txt(cid_xml)
if not cid_xml: return None
# Parse the beast. Any XML-problem will be reported by xlm.sax
self.spf_entry = None
xml.sax.parseString(cid_xml,self)
return self.spf_entry
if __name__ == '__main__':
import sys
if len(sys.argv) < 2:
print >>sys.stderr, \
"""Usage: %s "<ep xmlns='http://ms.net/1'>...</ep>" """ % sys.argv[0]
sys.exit(1)
cid_xml = sys.argv[1]
p = CIDParser()
print p.spf_txt(cid_xml)
+21
View File
@@ -134,5 +134,26 @@ is a milter declaration for sendmail.cf with all timeouts specified:
Xpythonfilter, S=local:/var/log/milter/pythonsock, F=T, T=C:5m;S:20s;R:60s;E:5m Xpythonfilter, S=local:/var/log/milter/pythonsock, F=T, T=C:5m;S:20s;R:60s;E:5m
</pre> </pre>
<a name="spf">
<li> Q. So how do I use the SPF support? The sample.py milter doesn't seem
to use it.
<p> A. The bms.py milter supports spf. The RedHat RPMs will set almost
everything up for you. For other systems:
<ol type=i>
<li> Arrange to run bms.py in the background (as a service perhaps) and
redirect output and errors to a logfile. For instance, on AIX you'll want
to use SRC (System Resource Controller).
<li> Copy milter.cfg to the directory you run bms.py in, and edit it. The
comments should explain the options.
<li> Start bms.py in the background as arranged.
<li> Add Xpythonfilter to sendmail.cf or add an INPUT_MAIL_FILTER to
sendmail.mc. Regen sendmail.cf if you use sendmail.mc and restart
sendmail.
<li> Arrange to rotate log files and remove old defang files in
<code>tempdir</code>. The RedHat RPM uses <code>logrotate</code> for
logfiles and a simple cron script using <code>find</code> to clean
<code>tempdir</code>.
</ol>
</ol> </ol>
</html> </html>
+38 -19
View File
@@ -1,63 +1,79 @@
# features intended to filter or block incoming mail # features intended to filter or block incoming mail
[milter] [milter]
;socket=/var/log/milter/pythonsock # the socket used to communicate with sendmail. Must match sendmail.cf
;socket=/var/run/milter/pythonsock
# where to save original copies of defanged and failed messages
tempdir = /var/log/milter/save tempdir = /var/log/milter/save
# how long to wait for a response from sendmail before giving up
;timeout=600 ;timeout=600
# do virus scanning on attached messages also
scan_rfc822 = 1 scan_rfc822 = 1
# can be CPU intensive # Comment out scripts in HTML attachments. Can be CPU intensive.
scan_html = 0 scan_html = 0
# reject asian fonts because we can't read them # reject messages with asian fonts because we can't read them
block_chinese = 1 block_chinese = 1
# users who hate forwarded mail # list users who hate forwarded mail
;block_forward = egghead@mycorp.com, busybee@mycorp.com ;block_forward = egghead@mycorp.com, busybee@mycorp.com
log_headers = 0 log_headers = 0
# Reject mail for domains mentioned unless user is mentioned here also # Reject mail for domains mentioned unless user is mentioned here also
;check_user = joe@mycorp.com, mary@mycorp.com, file:bigcorp.com ;check_user = joe@mycorp.com, mary@mycorp.com, file:bigcorp.com
# porn words are case insensitive # reject mail with these case insensitive strings in the subject
porn_words = penis, breast, pussy, horse cock, porn, xenical, diet pill, d1ck, porn_words = penis, breast, pussy, horse cock, porn, xenical, diet pill, d1ck,
vi*gra, vi-a-gra, viag, tits, p0rn, hunza, horny, sexy, c0ck, vi*gra, vi-a-gra, viag, tits, p0rn, hunza, horny, sexy, c0ck,
p-e-n-i-s, hydrocodone, vicodin, xanax, vicod1n, x@nax, diazepam, p-e-n-i-s, hydrocodone, vicodin, xanax, vicod1n, x@nax, diazepam,
v1@gra, xan@x, cialis, ci@lis, frëe, xãnax, valíum, vãlium, via-gra, v1@gra, xan@x, cialis, ci@lis, frëe, xãnax, valíum, vãlium, via-gra,
x@n3x, vicod3n, penís, v|c0d1n, phentermine, en1arge, dip1oma, v1codin x@n3x, vicod3n, penís, c0d1n, phentermine, en1arge, dip1oma, v1codin
# spam words are case sensitive # reject mail with these case sensitive strings in the subject
spam_words = $$$, !!!, XXX, FREE, HGH spam_words = $$$, !!!, XXX, FREE, HGH
# connection ips and hostnames are matched against this glob style list # connection ips and hostnames are matched against this glob style list
# to recognize internal senders # to recognize internal senders
;internal_connect = 192.168.*.* ;internal_connect = 192.168.*.*
# mail that is not an internal_connect and claims to be from an # mail that is not an internal_connect and claims to be from an
# internal domain is rejected. # internal domain is rejected. You should enable SPF instead if you can.
# SPF is much more comprehensive and flexible.
;internal_domains = mycorp.com ;internal_domains = mycorp.com
# connections from a trusted relay can trust the first Received header # connections from a trusted relay can trust the first Received header
# SPF checks are bypassed for internal connections and trusted relays.
;trusted_relay = 1.2.3.4, 66.12.34.56 ;trusted_relay = 1.2.3.4, 66.12.34.56
# reject external senders with hello names no legit external sender would use # reject external senders with hello names no legit external sender would use
# SPF will do this also, but listing your own domain and mailserver here
# will save some DNS lookups when rejecting certain viruses.
;hello_blacklist = mycorp.com, 66.12.34.56 ;hello_blacklist = mycorp.com, 66.12.34.56
# See http://bmsi.com/python/pysrs.html for details
[srs] [srs]
config=/etc/mail/pysrs.cfg config=/etc/mail/pysrs.cfg
# SRS options can be set here also, but must match the sendmail plugin
;secret="shhhh!" ;secret="shhhh!"
;maxage=21 ;maxage=21
;hashlength=4 ;hashlength=4
;database=/var/log/milter/srsdata ;database=/var/log/milter/srsdata
;fwdomain = mydomain.com ;fwdomain = mydomain.com
# turn this on after a grace period # turn this on after a grace period to reject spoofed DSNs
reject_spoofed = 0 reject_spoofed = 0
# See http://spf.pobox.com for more info on SPF.
[spf] [spf]
# namespace where SPF records can be supplied for domains without one # namespace where SPF records can be supplied for domains without one
# records are search for under _spf.domain.com # records are searched for under _spf.domain.com
;delegate = domain.com ;delegate = domain.com
# domains where a neutral SPF result should cause mail to be rejected # domains where a neutral SPF result should cause mail to be rejected
;reject_neutral = aol.com ;reject_neutral = aol.com
# use a default (v=spf1 a/24 mx/24 ptr) when no SPF records are published # use a default (v=spf1 a/24 mx/24 ptr) when no SPF records are published
;best_guess = 0 ;best_guess = 0
# reject senders that have neither PTR nor SPF records
;reject_noptr = 0
# features intended to clean up outgoing mail # features intended to clean up outgoing mail
[scrub] [scrub]
# domains that stupidly block visible private nodes # domains that block visible private nodes
;hide_path = jcpenney.com ;hide_path = jcpenney.com
# block, don't just replace with warning, viruses from these domains # reject, don't just replace with warning, viruses from these domains
;reject_virus_from = mycorp.com ;reject_virus_from = mycorp.com
# features intended for spying on users and coworkers # features intended for spying on users and coworkers
@@ -86,16 +102,18 @@ blind = 1
;walter1 = cust@othercorp.com,walter@bigcorp.com,boss@bigcorp.com, ;walter1 = cust@othercorp.com,walter@bigcorp.com,boss@bigcorp.com,
; walter@bigcorp.com ; walter@bigcorp.com
# See http://bmsi.com/python/dspam.html
[dspam] [dspam]
# Select a well moderated dspam dictionary to reject spammy headers # Select a well moderated dspam dictionary to reject spammy headers.
# dspam-python must be installed to use: http://bmsi.com/python/dspam.html # To filter on the entire message, use the full setup below.
# only EXTERNAL messages are dspam filtered # only EXTERNAL messages are dspam filtered
;dspam_dict=/var/lib/dspam/moderator.dict ;dspam_dict=/var/lib/dspam/moderator.dict
# Opt-opt recipients from dspam screening and header triage # Opt-opt recipients from dspam screening and header triage
;dspam_exempt=getitall@mycorp.com ;dspam_exempt=getitall@mycorp.com
# Do not scan mail (ostensibly) from these senders # Do not scan mail (ostensibly) from these senders
;dspam_whitelist=getitall@sender.com ;dspam_whitelist=getitall@sender.com
# Reject spam to these domains, perhaps because we are a backup MX server # Reject spam to these domains instead of quarantining it.
;dspam_reject=othercorp.com ;dspam_reject=othercorp.com
# directory for dspam user quarantine, signature db, and dictionaries # directory for dspam user quarantine, signature db, and dictionaries
@@ -113,8 +131,9 @@ blind = 1
;spam=spam@foocorp.com ;spam=spam@foocorp.com
# address to forward false positives to. milter will process and not deliver # address to forward false positives to. milter will process and not deliver
;falsepositive=ham@foocorp.com ;falsepositive=ham@foocorp.com
# the dspam_screener is used to screen mail for all recipients who are # the dspam_screener is a list of dspam users who screen mail for all
# not dspam_users. Spam goes to the screeners quarantine, and the original # recipients who are not dspam_users. Spam goes to the screeners quarantine,
# recipients saved so that false positives can be properly delivered. # and the original recipients are saved so that false positives can be properly
# delivered.
;dspam_screener=david,goliath
# The dspam CGI can also be used: logins must match dspam users # The dspam CGI can also be used: logins must match dspam users
+59 -21
View File
@@ -13,8 +13,8 @@ ALT="Viewable With Any Browser" BORDER="0"></A>
usemap="#banner_4" alt="Your vote?"> usemap="#banner_4" alt="Your vote?">
<map name="banner_4"> <map name="banner_4">
<area shape="rect" coords="330,25,426,59" <area shape="rect" coords="330,25,426,59"
href="http://www.sepschool.org/survey/" alt="I Disagree"> href="http://education-survey.org/" alt="I Disagree">
<area shape="rect" coords="234,28,304,57" href="http://sepschool.org/" alt="I Agree"> <area shape="rect" coords="234,28,304,57" href="http://www.honestEd.com/" alt="I Agree">
</map> </map>
</P> </P>
@@ -24,12 +24,14 @@ ALT="Viewable With Any Browser" BORDER="0"></A>
Stuart D. Gathman</a><br> Stuart D. Gathman</a><br>
This web page is written by Stuart D. Gathman<br>and<br>sponsored by This web page is written by Stuart D. Gathman<br>and<br>sponsored by
<a href="http://www.bmsi.com">Business Management Systems, Inc.</a> <br> <a href="http://www.bmsi.com">Business Management Systems, Inc.</a> <br>
Last updated Apr 21, 2004</h4> Last updated Aug 06, 2004</h4>
See the <a href="faq.html">FAQ</a> | <a href="#download">Download now</a> | See the <a href="faq.html">FAQ</a> | <a href="#download">Download now</a> |
<a href="/mailman/listinfo/pymilter">Subscribe to mailing list</a> <a href="/mailman/listinfo/pymilter">Subscribe to mailing list</a> |
<a href="#overview">Overview</a>
<p> <p>
<img src="python55.gif" align=left alt="A Python"> <a href="//www.python.org">
<img src="python55.gif" align=left alt="A Python"></a>
<a href="//www.sendmail.org/">Sendmail</a> introduced a <a href="//www.sendmail.org/">Sendmail</a> introduced a
<a href="http://www.milter.org/milter_api/api.html"> new API</a> beginning with version 8.10 - <a href="http://www.milter.org/milter_api/api.html"> new API</a> beginning with version 8.10 -
libmilter. The milter module for <a href="//www.python.org">Python</a> libmilter. The milter module for <a href="//www.python.org">Python</a>
@@ -40,11 +42,18 @@ Version 8.12 seems to be more robust, and includes new privilege
separation features to enhance security. separation features to enhance security.
I recommend upgrading. I recommend upgrading.
<h2> <a name=dspam>Bayesian Filtering</a> </h2> <h2> Recent Changes </h2>
The RPM for release 0.7.0 moves the config file and socket locations to
I have selected the <a href="http://www.nuclearelephant.com/projects/dspam/"> /etc/mail and /var/run/milter respectively. We now parse Microsoft CID records
dspam bayes filter project</a> and <a href="dspam.html"> - but only hotmail.com uses them. They seem to have a patent on the brilliant
packaged it for python</a>. idea of examining the mail headers to see who the message is from.
We aren't doing that here, so not to worry - but I am not a lawyer, so if you
are worried, change spf.py around line 626 to return None instead of
calling CIDParser(). There is a new option to reject mail with no PTR
and no SPF.
<p>
<a href="http://spf.pobox.com">
<img src="SPF.gif" align=left alt="SPF logo"></a>
Release 0.6.6 adds support for <a href="http://spf.pobox.com/">SPF</a>, Release 0.6.6 adds support for <a href="http://spf.pobox.com/">SPF</a>,
a protocol to prevent forging of the envelope from address. a protocol to prevent forging of the envelope from address.
SPF support requires <a href="http://pydns.sourceforge.net/">pydns</a>. SPF support requires <a href="http://pydns.sourceforge.net/">pydns</a>.
@@ -52,15 +61,15 @@ The included spf.py module is an updated version of the original 1.6
version at <a href="http://www.wayforward.net/spf/">wayforward.net</a>. version at <a href="http://www.wayforward.net/spf/">wayforward.net</a>.
The updated version tracks the draft RFC and test suite. The updated version tracks the draft RFC and test suite.
<p> <p>
Release 0.6.0 offers a simple application of dspam I call "header triage", The FAQ addresses <a href="faq.html#spf">how to get started with SPF</a>.
which rejects messages with spammy headers. Since sendmail has to
read the entire message anyway once we start reading headers, it
would probably be better to scan the whole message - except that
we replace dangerous attachments elsewhere in the milter - which screws up the
body statistics for messages with dangerous attachments.
<p> <p>
Release 0.6.1 adds a full milter based dspam application. Release 0.6.1 adds a full milter based dspam application.
<p> <p>
I have selected the <a href="http://www.nuclearelephant.com/projects/dspam/">
dspam bayes filter project</a> and <a href="dspam.html">
packaged it for python</a>.
Release 0.6.0 offers a simple application of dspam I call "header triage",
which rejects messages with spammy headers.
To use header triage, you must have <a href="dspam.html">DSPAM</a> installed, To use header triage, you must have <a href="dspam.html">DSPAM</a> installed,
and select a dictionary that is well moderated by someone who gets and select a dictionary that is well moderated by someone who gets
lots of spam. That dictionary can be used to block spam that is lots of spam. That dictionary can be used to block spam that is
@@ -109,7 +118,7 @@ entries. Be sure to handle both Bcc and file copies, and designating what
mail should be copied. How should "outgoing" be defined? Implementing it is mail should be copied. How should "outgoing" be defined? Implementing it is
easy once the configuration is designed. easy once the configuration is designed.
<h3>Overview</h3> <h3><a name=overview>Overview</a></h3>
This package provides a robust toolkit for Python <a This package provides a robust toolkit for Python <a
href="#milter">milters</a>, and the beginnings of a general purpose mail href="#milter">milters</a>, and the beginnings of a general purpose mail
@@ -141,21 +150,50 @@ methods that
do nothing, and also provides wrappers for the libmilter methods to mutate do nothing, and also provides wrappers for the libmilter methods to mutate
the message. the message.
<p> <p>
The 'spf' module provides an implementation of <a href="http://spf.pobox.com">
SPF</a> useful for detecting email forgery.
<p>
The 'mime' module provides a wrapper for the Python email package that
fixes some bugs, and simplifies modifying selected parts of a MIME message.
<p>
Finally, the bms.py application is both a sample of how to use the Finally, the bms.py application is both a sample of how to use the
Milter module, and the beginnings of a general purpose SPAM filtering, Milter and spf modules, and the beginnings of a general purpose SPAM filtering,
wiretapping, and Win32 virus protection milter. wiretapping, SPF checking, and Win32 virus protecting milter. It can
make use of the <a href="pysrs.html">pysrs</a> package when available for
SRS/SES checking and the <a href="dspam.html">pydspam</a> package for Bayesian
content filtering. SPF checking
requires <a href="http://pydns.sourceforge.net/">
pydns</a>. Configuration documentation is currently included as comments
in the <a href="milter.cfg">sample config file</a> for the bms.py milter.
<h3><a name=download>Downloading</a></h3> <h3><a name=download>Downloading</a></h3>
The latest stable release is <a href="#stable">0.6.9</a>. A stable The latest stable release is <a href="#stable">0.7.0</a>. A stable
release is one which has been installed (and working correctly) on release is one which has been installed (and working correctly) on
production systems long enough to convince me that it is stable. As production systems long enough to convince me that it is stable. As
the package gains more features and complexity, stable will mean no the package gains more features and complexity, stable will mean no
bug reports from outside users either. bug reports from outside users either.
<p> <p>
The latest version is 0.6.9-1. See the <a href=NEWS>Change Log</a>. The latest version is 0.7.0-1. See the <a href=NEWS>Change Log</a>.
<p> <p>
<a name="stable"><b>Stable</b></a> <a name="stable"><b>Stable</b></a>
<a href="http://bmsi.com/python/milter-0.7.0.tar.gz">
milter-0.7.0.tar.gz</a> Move config file and default socket location.
Parse M$ CID records.
<br>
<a href="http://bmsi.com/linux/rh72/milter-0.7.0-1.i386.rpm">
milter-0.7.0-1.i386.rpm</a> Binary RPM for Redhat 7.x, now requires
sendmail-8.12 and <a href="http://www.python.org/2.3.3/rpms.html">
python2.3</a>.
<br>
<a href="http://bmsi.com/linux/rh9/milter-0.7.0-1rh9.i386.rpm">
milter-0.7.0-1rh9.i386.rpm</a> Binary RPM for Redhat 9, requires
sendmail-8.12 and <a href="http://www.python.org/2.3.3/rpms.html">
python2.3</a>.
<br>
<a href="http://bmsi.com/linux/rh9/milter-0.7.0-1.src.rpm">
milter-0.7.0-1.src.rpm</a> Source RPM for Redhat 9,7.x.
<p>
<a href="http://bmsi.com/python/milter-0.6.9.tar.gz"> <a href="http://bmsi.com/python/milter-0.6.9.tar.gz">
milter-0.6.9.tar.gz</a> Add SPF test suite driver, and validate milter-0.6.9.tar.gz</a> Add SPF test suite driver, and validate
spf.py against test suite. Add best_guess and get_header to spf.py. spf.py against test suite. Add best_guess and get_header to spf.py.
+22 -5
View File
@@ -1,5 +1,5 @@
%define name milter %define name milter
%define version 0.6.9 %define version 0.7.1
%define release 1 %define release 1
# Redhat 7.x and earlier (multiple ps lines per thread) # Redhat 7.x and earlier (multiple ps lines per thread)
%define sysvinit milter.rc7 %define sysvinit milter.rc7
@@ -24,8 +24,8 @@ Prefix: %{_prefix}
Vendor: Stuart D. Gathman <stuart@bmsi.com> Vendor: Stuart D. Gathman <stuart@bmsi.com>
Packager: Stuart D. Gathman <stuart@bmsi.com> Packager: Stuart D. Gathman <stuart@bmsi.com>
Url: http://www.bmsi.com/python/milter.html Url: http://www.bmsi.com/python/milter.html
Requires: %{python} >= 2.2.2, sendmail >= 8.12 Requires: %{python} >= 2.2.2, sendmail >= 8.12.10
BuildRequires: %{python}-devel >= 2.2.2, sendmail-devel >= 8.12 BuildRequires: %{python}-devel >= 2.2.2, sendmail-devel >= 8.12.10
%description %description
This is a python extension module to enable python scripts to This is a python extension module to enable python scripts to
@@ -43,8 +43,10 @@ env CFLAGS="$RPM_OPT_FLAGS" %{python} setup.py build
rm -rf $RPM_BUILD_ROOT rm -rf $RPM_BUILD_ROOT
%{python} setup.py install --root=$RPM_BUILD_ROOT --record=INSTALLED_FILES %{python} setup.py install --root=$RPM_BUILD_ROOT --record=INSTALLED_FILES
mkdir -p $RPM_BUILD_ROOT/var/log/milter mkdir -p $RPM_BUILD_ROOT/var/log/milter
mkdir -p $RPM_BUILD_ROOT/etc/mail
mkdir $RPM_BUILD_ROOT/var/log/milter/save mkdir $RPM_BUILD_ROOT/var/log/milter/save
cp bms.py milter.cfg $RPM_BUILD_ROOT/var/log/milter cp bms.py $RPM_BUILD_ROOT/var/log/milter
cp milter.cfg $RPM_BUILD_ROOT/etc/mail/pymilter.cfg
# logfile rotation # logfile rotation
mkdir -p $RPM_BUILD_ROOT/etc/logrotate.d mkdir -p $RPM_BUILD_ROOT/etc/logrotate.d
@@ -103,6 +105,9 @@ mkssys -s milter -p /var/log/milter/start.sh -u 25 -S -n 15 -f 9 -G mail || :
if [ $1 = 0 ]; then if [ $1 = 0 ]; then
rmssys -s milter || : rmssys -s milter || :
fi fi
%else
%post
echo "pythonsock has moved to /var/run/milter, update /etc/mail/sendmail.cf"
%endif %endif
%clean %clean
@@ -124,9 +129,21 @@ rm -rf $RPM_BUILD_ROOT
%dir /var/log/milter/save %dir /var/log/milter/save
%config /var/log/milter/start.sh %config /var/log/milter/start.sh
%config /var/log/milter/bms.py %config /var/log/milter/bms.py
%config /var/log/milter/milter.cfg %config(noreplace) /etc/mail/pymilter.cfg
%changelog %changelog
* Sun Aug 22 2004 Stuart Gathman <stuart@bmsi.com> 0.7.1-1
- Handle modifying mislabeled multipart messages without an exception
- Support setbacklog, setmlreply
- allow multi-recipient CBV
- return TEMPFAIL for SPF softfail
* Fri Jul 23 2004 Stuart Gathman <stuart@bmsi.com> 0.7.0-1
- SPF check hello name
- Move pythonsock to /var/run/milter
- Move milter.cfg to /etc/mail/pymilter.cfg
- Check M$ style XML CID records by converting to SPF
- Recognize, but never match ip6 until we properly support it.
- Option to reject when no PTR and no SPF
* Fri Apr 09 2004 Stuart Gathman <stuart@bmsi.com> 0.6.9-1 * Fri Apr 09 2004 Stuart Gathman <stuart@bmsi.com> 0.6.9-1
- Validate spf.py against test suite, and add Received-SPF support to spf.py - Validate spf.py against test suite, and add Received-SPF support to spf.py
- Support best_guess for SPF - Support best_guess for SPF
+104 -23
View File
@@ -33,6 +33,18 @@ $ python setup.py help
libraries=["milter","smutil","resolv"] libraries=["milter","smutil","resolv"]
* $Log$ * $Log$
* Revision 2.31 2004/08/23 02:24:36 stuart
* Support setbacklog
*
* Revision 2.30 2004/08/21 20:29:53 stuart
* Support option of 11 lines max for mlreply.
*
* Revision 2.29 2004/08/21 04:14:29 stuart
* mlreply support
*
* Revision 2.28 2004/08/21 02:45:21 stuart
* Don't leak int constants if module unloaded.
*
* Revision 2.27 2004/04/06 03:19:59 stuart * Revision 2.27 2004/04/06 03:19:59 stuart
* Release 0.6.8 * Release 0.6.8
* *
@@ -127,11 +139,20 @@ $ python setup.py help
* *
*/ */
#ifndef MAX_ML_REPLY
#define MAX_ML_REPLY 32
#endif
#if MAX_ML_REPLY != 1 && MAX_ML_REPLY != 32 && MAX_ML_REPLY != 11
#error MAX_ML_REPLY must be 1 or 11 or 32
#endif
#define _FFR_MULTILINE (MAX_ML_REPLY > 1)
#include <pthread.h> #include <pthread.h>
#include <netinet/in.h> #include <netinet/in.h>
#include <Python.h> #include <Python.h>
#include <libmilter/mfapi.h> #include <libmilter/mfapi.h>
/* See if we have IPv4 and/or IPv6 support in this OS and in /* See if we have IPv4 and/or IPv6 support in this OS and in
* libmilter. We need to make several macro tests because some OS's * libmilter. We need to make several macro tests because some OS's
* may define some if IPv6 is only partially supported, and we may * may define some if IPv6 is only partially supported, and we may
@@ -746,6 +767,18 @@ milter_setdbg(PyObject *self, PyObject *args) {
return _generic_return(smfi_setdbg(val), "cannot set debug value"); return _generic_return(smfi_setdbg(val), "cannot set debug value");
} }
static char milter_setbacklog__doc__[] =
"setbacklog(int) -> None\n\
Set the TCP connection queue size for the milter socket.";
static PyObject *
milter_setbacklog(PyObject *self, PyObject *args) {
int val;
if (!PyArg_ParseTuple(args, "i:setbacklog", &val)) return NULL;
return _generic_return(smfi_setbacklog(val), "cannot set backlog");
}
static char milter_settimeout__doc__[] = static char milter_settimeout__doc__[] =
"settimeout(int) -> None\n\ "settimeout(int) -> None\n\
Set the time (in seconds) that sendmail will wait before\n\ Set the time (in seconds) that sendmail will wait before\n\
@@ -820,13 +853,54 @@ static PyObject *
milter_setreply(PyObject *self, PyObject *args) { milter_setreply(PyObject *self, PyObject *args) {
char *rcode; char *rcode;
char *xcode; char *xcode;
char *message; char *message[MAX_ML_REPLY];
char fmt[MAX_ML_REPLY + 16];
SMFICTX *ctx; SMFICTX *ctx;
if (!PyArg_ParseTuple(args, "szz:setreply", &rcode, &xcode, &message)) int i;
strcpy(fmt,"sz|");
for (i = 0; i < MAX_ML_REPLY; ++i) {
message[i] = 0;
fmt[i+3] = 's';
}
strcpy(fmt+i+3,":setreply");
if (!PyArg_ParseTuple(args, fmt,
&rcode, &xcode, message
#if MAX_ML_REPLY > 1
,message+1,message+2,message+3,message+4,message+5,message+6,
message+7,message+8,message+9,message+10
#if MAX_ML_REPLY > 11
,message+11,message+12,message+13,message+14,message+15,
message+16,message+17,message+18,message+19,message+20,
message+21,message+22,message+23,message+24,message+25,
message+26,message+27,message+28,message+29,message+30,
message+31
#endif
#endif
))
return NULL; return NULL;
ctx = _find_context(self); ctx = _find_context(self);
if (ctx == NULL) return NULL; if (ctx == NULL) return NULL;
return _generic_return(smfi_setreply(ctx, rcode, xcode, message), #if MAX_ML_REPLY > 1
/*
* C varargs might be convenient for some things, but they sure are a pain
* when the number of args is not known at compile time.
*/
if (message[0] && message[1])
return _generic_return(smfi_setmlreply(ctx, rcode, xcode,
message[0],
message[1],message[2],message[3],message[4],message[5],
message[6],message[7],message[8],message[9],message[10],
#if MAX_ML_REPLY > 11
message[11],message[12],message[13],message[14],message[15],
message[16],message[17],message[18],message[19],message[20],
message[21],message[22],message[23],message[24],message[25],
message[26],message[27],message[28],message[29],message[30],
message[31],
#endif
(char *)0
), "cannot set reply");
#endif
return _generic_return(smfi_setreply(ctx, rcode, xcode, message[0]),
"cannot set reply"); "cannot set reply");
} }
@@ -986,7 +1060,7 @@ milter_getpriv(PyObject *self, PyObject *args) {
return o; return o;
} }
#if _FFR_QUARANTINE #ifdef SMFIF_QUARANTINE
static char milter_quarantine__doc__[] = static char milter_quarantine__doc__[] =
"quarantine(string) -> None\n\ "quarantine(string) -> None\n\
Place the message in quarantine. A string with a description of the reason\n\ Place the message in quarantine. A string with a description of the reason\n\
@@ -1035,7 +1109,7 @@ static PyMethodDef context_methods[] = {
{ "replacebody", milter_replacebody, METH_VARARGS, milter_replacebody__doc__}, { "replacebody", milter_replacebody, METH_VARARGS, milter_replacebody__doc__},
{ "setpriv", milter_setpriv, METH_VARARGS, milter_setpriv__doc__}, { "setpriv", milter_setpriv, METH_VARARGS, milter_setpriv__doc__},
{ "getpriv", milter_getpriv, METH_VARARGS, milter_getpriv__doc__}, { "getpriv", milter_getpriv, METH_VARARGS, milter_getpriv__doc__},
#if _FFR_QUARANTINE #ifdef SMFIF_QUARANTINE
{ "quarantine", milter_quarantine, METH_VARARGS, milter_quarantine__doc__}, { "quarantine", milter_quarantine, METH_VARARGS, milter_quarantine__doc__},
#endif #endif
#if _FFR_SMFI_PROGRESS #if _FFR_SMFI_PROGRESS
@@ -1081,6 +1155,7 @@ static PyMethodDef milter_methods[] = {
{ "main", milter_main, METH_VARARGS, milter_main__doc__}, { "main", milter_main, METH_VARARGS, milter_main__doc__},
{ "setdbg", milter_setdbg, METH_VARARGS, milter_setdbg__doc__}, { "setdbg", milter_setdbg, METH_VARARGS, milter_setdbg__doc__},
{ "settimeout", milter_settimeout, METH_VARARGS, milter_settimeout__doc__}, { "settimeout", milter_settimeout, METH_VARARGS, milter_settimeout__doc__},
{ "setbacklog", milter_setbacklog, METH_VARARGS, milter_setbacklog__doc__},
{ "setconn", milter_setconn, METH_VARARGS, milter_setconn__doc__}, { "setconn", milter_setconn, METH_VARARGS, milter_setconn__doc__},
{ "stop", milter_stop, METH_VARARGS, milter_stop__doc__}, { "stop", milter_stop, METH_VARARGS, milter_stop__doc__},
{ NULL, NULL } { NULL, NULL }
@@ -1116,6 +1191,12 @@ allowing one to write email filters directly in Python.\n\
Libmilter is currently marked FFR, and needs to be explicitly installed.\n\ Libmilter is currently marked FFR, and needs to be explicitly installed.\n\
See <sendmailsource>/libmilter/README for details on setting it up.\n"; See <sendmailsource>/libmilter/README for details on setting it up.\n";
static void setitem(PyObject *d,const char *name,long val) {
PyObject *v = PyInt_FromLong(val);
PyDict_SetItemString(d,name,v);
Py_DECREF(v);
}
void void
initmilter(void) { initmilter(void) {
PyObject *m, *d; PyObject *m, *d;
@@ -1125,24 +1206,24 @@ initmilter(void) {
d = PyModule_GetDict(m); d = PyModule_GetDict(m);
MilterError = PyErr_NewException("milter.error", NULL, NULL); MilterError = PyErr_NewException("milter.error", NULL, NULL);
PyDict_SetItemString(d,"error", MilterError); PyDict_SetItemString(d,"error", MilterError);
PyDict_SetItemString(d,"SUCCESS", PyInt_FromLong((long) MI_SUCCESS)); setitem(d,"SUCCESS", MI_SUCCESS);
PyDict_SetItemString(d,"FAILURE", PyInt_FromLong((long) MI_FAILURE)); setitem(d,"FAILURE", MI_FAILURE);
PyDict_SetItemString(d,"VERSION", PyInt_FromLong((long) SMFI_VERSION)); setitem(d,"VERSION", SMFI_VERSION);
PyDict_SetItemString(d,"ADDHDRS", PyInt_FromLong((long) SMFIF_ADDHDRS)); setitem(d,"ADDHDRS", SMFIF_ADDHDRS);
PyDict_SetItemString(d,"CHGBODY", PyInt_FromLong((long) SMFIF_CHGBODY)); setitem(d,"CHGBODY", SMFIF_CHGBODY);
PyDict_SetItemString(d,"MODBODY", PyInt_FromLong((long) SMFIF_MODBODY)); setitem(d,"MODBODY", SMFIF_MODBODY);
PyDict_SetItemString(d,"ADDRCPT", PyInt_FromLong((long) SMFIF_ADDRCPT)); setitem(d,"ADDRCPT", SMFIF_ADDRCPT);
PyDict_SetItemString(d,"DELRCPT", PyInt_FromLong((long) SMFIF_DELRCPT)); setitem(d,"DELRCPT", SMFIF_DELRCPT);
PyDict_SetItemString(d,"CHGHDRS", PyInt_FromLong((long) SMFIF_CHGHDRS)); setitem(d,"CHGHDRS", SMFIF_CHGHDRS);
PyDict_SetItemString(d,"V1_ACTS", PyInt_FromLong((long) SMFI_V1_ACTS)); setitem(d,"V1_ACTS", SMFI_V1_ACTS);
PyDict_SetItemString(d,"V2_ACTS", PyInt_FromLong((long) SMFI_V2_ACTS)); setitem(d,"V2_ACTS", SMFI_V2_ACTS);
PyDict_SetItemString(d,"CURR_ACTS", PyInt_FromLong((long) SMFI_CURR_ACTS)); setitem(d,"CURR_ACTS", SMFI_CURR_ACTS);
#ifdef SMFIF_QUARANTINE #ifdef SMFIF_QUARANTINE
PyDict_SetItemString(d,"QUARANTINE",PyInt_FromLong((long)SMFIF_QUARANTINE)); setitem(d,"QUARANTINE",SMFIF_QUARANTINE);
#endif #endif
PyDict_SetItemString(d,"CONTINUE", PyInt_FromLong((long) SMFIS_CONTINUE)); setitem(d,"CONTINUE", SMFIS_CONTINUE);
PyDict_SetItemString(d,"REJECT", PyInt_FromLong((long) SMFIS_REJECT)); setitem(d,"REJECT", SMFIS_REJECT);
PyDict_SetItemString(d,"DISCARD", PyInt_FromLong((long) SMFIS_DISCARD)); setitem(d,"DISCARD", SMFIS_DISCARD);
PyDict_SetItemString(d,"ACCEPT", PyInt_FromLong((long) SMFIS_ACCEPT)); setitem(d,"ACCEPT", SMFIS_ACCEPT);
PyDict_SetItemString(d,"TEMPFAIL", PyInt_FromLong((long) SMFIS_TEMPFAIL)); setitem(d,"TEMPFAIL", SMFIS_TEMPFAIL);
} }
+57 -31
View File
@@ -1,4 +1,13 @@
# $Log$ # $Log$
# Revision 1.54 2004/08/18 01:59:46 stuart
# Handle mislabeled multipart messages
#
# Revision 1.53 2004/04/24 22:53:20 stuart
# Rename some local variables to avoid shadowing builtins
#
# Revision 1.52 2004/04/24 22:47:13 stuart
# Convert header values to str
#
# Revision 1.51 2004/03/25 03:19:10 stuart # Revision 1.51 2004/03/25 03:19:10 stuart
# Correctly defang rfc822 attachments when boundary specified with # Correctly defang rfc822 attachments when boundary specified with
# content-type message/rfc822. # content-type message/rfc822.
@@ -52,6 +61,18 @@ except: from email.Parser import nlcre as NLCRE
from email import Errors from email import Errors
class MimeGenerator(Generator):
def _dispatch(self, msg):
# Get the Content-Type: for the message, then try to dispatch to
# self._handle_<maintype>_<subtype>(). If there's no handler for the
# full MIME type, then dispatch to self._handle_<maintype>(). If
# that's missing too, then dispatch to self._writeBody().
main = msg.get_content_maintype()
if msg.is_multipart() and main.lower() != 'multipart':
self._handle_multipart(msg)
else:
Generator._dispatch(self,msg)
class MimeParser(Parser): class MimeParser(Parser):
# This is a copy of _parsebody from email.Parser, with a fix # This is a copy of _parsebody from email.Parser, with a fix
@@ -192,19 +213,19 @@ class MimeParser(Parser):
text = firstbodyline + '\n' + text text = firstbodyline + '\n' + text
container.set_payload(text) container.set_payload(text)
def unquote(str): def unquote(s):
"""Remove quotes from a string.""" """Remove quotes from a string."""
if len(str) > 1: if len(s) > 1:
if str.startswith('"'): if s.startswith('"'):
if str.endswith('"'): if s.endswith('"'):
str = str[1:-1] s = s[1:-1]
else: # remove garbage after trailing quote else: # remove garbage after trailing quote
try: str = str[1:str[1:].index('"')+1] try: s = s[1:s[1:].index('"')+1]
except: return str except: return s
return str.replace('\\\\', '\\').replace('\\"', '"') return s.replace('\\\\', '\\').replace('\\"', '"')
if str.startswith('<') and str.endswith('>'): if s.startswith('<') and s.endswith('>'):
return str[1:-1] return s[1:-1]
return str return s
from types import TupleType from types import TupleType
@@ -216,21 +237,21 @@ def _unquotevalue(value):
email.Message._unquotevalue = _unquotevalue email.Message._unquotevalue = _unquotevalue
def _parseparam(str): def _parseparam(s):
plist = [] plist = []
while str[:1] == ';': while s[:1] == ';':
str = str[1:] s = s[1:]
end = str.find(';') end = s.find(';')
while end > 0 and (str.count('"',0,end) & 1): while end > 0 and (s.count('"',0,end) & 1):
end = str.find(';',end + 1) end = s.find(';',end + 1)
if end < 0: end = len(str) if end < 0: end = len(s)
f = str[:end] f = s[:end]
if '=' in f: if '=' in f:
i = f.index('=') i = f.index('=')
f = f[:i].strip().lower() + \ f = f[:i].strip().lower() + \
'=' + f[i+1:].strip() '=' + f[i+1:].strip()
plist.append(f.strip()) plist.append(f.strip())
str = str[end:] s = s[end:]
return plist return plist
# Enhance email.Message # Enhance email.Message
@@ -343,16 +364,22 @@ class MimeMessage(Message):
def dump(self,file,unixfrom=False): def dump(self,file,unixfrom=False):
"Write this message (and all subparts) to a file" "Write this message (and all subparts) to a file"
g = Generator(file) g = MimeGenerator(file)
g.flatten(self,unixfrom=unixfrom) g.flatten(self,unixfrom=unixfrom)
def as_string(self, unixfrom=False):
"Return the entire formatted message as a string."
fp = StringIO.StringIO()
self.dump(fp,unixfrom=unixfrom)
return fp.getvalue()
def getencoding(self): def getencoding(self):
return self.get('content-transfer-encoding',None) return self.get('content-transfer-encoding',None)
# Decode body to stream according to transfer encoding, return encoding name # Decode body to stream according to transfer encoding, return encoding name
def decode(self,filter): def decode(self,filt):
try: try:
filter.write(self.get_payload(decode=True)) filt.write(self.get_payload(decode=True))
except: except:
pass pass
return self.getencoding() return self.getencoding()
@@ -363,7 +390,7 @@ class MimeMessage(Message):
def __setitem__(self, name, value): def __setitem__(self, name, value):
rc = Message.__setitem__(self,name,value) rc = Message.__setitem__(self,name,value)
self.modified = True self.modified = True
if self.headerchange: self.headerchange(self,name,value) if self.headerchange: self.headerchange(self,name,str(value))
return rc return rc
def __delitem__(self, name): def __delitem__(self, name):
@@ -423,7 +450,7 @@ See your administrator.
def check_name(msg,savname=None,ckname=check_ext): def check_name(msg,savname=None,ckname=check_ext):
"Replace attachment with a warning if its name is suspicious." "Replace attachment with a warning if its name is suspicious."
for (key,name) in msg.getnames(): for key,name in msg.getnames():
badname = ckname(name) badname = ckname(name)
if badname: if badname:
hostname = socket.gethostname() hostname = socket.gethostname()
@@ -571,7 +598,6 @@ class HTMLScriptFilter(SGMLFilter):
def handle_comment(self,comment): def handle_comment(self,comment):
if not self.ignoring: SGMLFilter.handle_comment(self,comment) if not self.ignoring: SGMLFilter.handle_comment(self,comment)
def check_html(msg,savname=None): def check_html(msg,savname=None):
"Remove scripts from HTML attachments." "Remove scripts from HTML attachments."
msgtype = msg.get_content_type().lower() msgtype = msg.get_content_type().lower()
@@ -582,14 +608,14 @@ def check_html(msg,savname=None):
msgtype = 'text/html' msgtype = 'text/html'
if msgtype == 'text/html': if msgtype == 'text/html':
out = StringIO.StringIO() out = StringIO.StringIO()
filter = HTMLScriptFilter(out) htmlfilter = HTMLScriptFilter(out)
try: try:
filter.write(msg.get_payload(decode=True)) htmlfilter.write(msg.get_payload(decode=True))
filter.close() htmlfilter.close()
#except sgmllib.SGMLParseError: #except sgmllib.SGMLParseError:
except: except:
#mimetools.copyliteral(msg.get_payload(),open('debug.out','w') #mimetools.copyliteral(msg.get_payload(),open('debug.out','w')
filter.close() htmlfilter.close()
hostname = socket.gethostname() hostname = socket.gethostname()
msg.set_payload( msg.set_payload(
"An HTML attachment could not be parsed. The original is saved as '%s:%s'" "An HTML attachment could not be parsed. The original is saved as '%s:%s'"
@@ -600,7 +626,7 @@ def check_html(msg,savname=None):
name = "WARNING.TXT" name = "WARNING.TXT"
msg["Content-Type"] = "text/plain; name="+name msg["Content-Type"] = "text/plain; name="+name
return Milter.CONTINUE return Milter.CONTINUE
if filter.modified: if htmlfilter.modified:
msg.set_payload(out) # remove embedded scripts msg.set_payload(out) # remove embedded scripts
del msg["content-transfer-encoding"] del msg["content-transfer-encoding"]
email.Encoders.encode_quopri(msg) email.Encoders.encode_quopri(msg)
+38
View File
@@ -0,0 +1,38 @@
# Analyze milter log to find abusers
fp = open('/var/log/milter/milter.log','r')
subdict = {}
ipdict = {}
spamcnt = {}
for line in fp:
a = line.split(None,4)
if len(a) < 4: continue
dt,tm,id,op = a[:4]
if op == 'Subject:':
if len(a) > 4: subdict[id] = a[4].rstrip()
elif op == 'connect':
ipdict[id] = a[4].rstrip()
elif op in ('eom','dspam'):
if id in subdict: del subdict[id]
if id in ipdict: del ipdict[id]
elif op in ('REJECT:','DSPAM:','SPAM:','abort'):
if id in subdict:
if id in ipdict:
ip = ipdict[id]
del ipdict[id]
f,host,raw = ip.split(None,2)
if host in spamcnt:
spamcnt[host] += 1
else:
spamcnt[host] = 1
else: ip = ''
print dt,tm,op,a[4].rstrip(),subdict[id]
del subdict[id]
else:
print line.rstrip()
print len(subdict),'leftover entries'
spamlist = filter(lambda x: x[1] > 1,spamcnt.items())
spamlist.sort(lambda x,y: x[1] - y[1])
for ip,cnt in spamlist:
print cnt,ip
+5 -2
View File
@@ -12,7 +12,7 @@ if sys.version < '2.2.3':
DistributionMetadata.classifiers = None DistributionMetadata.classifiers = None
DistributionMetadata.download_url = None DistributionMetadata.download_url = None
setup(name = "milter", version = "0.6.9", setup(name = "milter", version = "0.7.1",
description="Python interface to sendmail milter API", description="Python interface to sendmail milter API",
long_description="""\ long_description="""\
This is a python extension module to enable python scripts to This is a python extension module to enable python scripts to
@@ -28,7 +28,10 @@ querying SPF records.
url="http://www.bmsi.com/python/milter.html", url="http://www.bmsi.com/python/milter.html",
py_modules=["Milter","mime","spf"], py_modules=["Milter","mime","spf"],
ext_modules=[ ext_modules=[
Extension("milter", ["miltermodule.c"],libraries=libs), Extension("milter", ["miltermodule.c"],
libraries=libs,
define_macros = [ ('MAX_ML_REPLY',32) ]
),
], ],
keywords = ['sendmail','milter'], keywords = ['sendmail','milter'],
classifiers = [ classifiers = [
+155 -2
View File
@@ -45,6 +45,18 @@ For news, bugfixes, etc. visit the home page for this implementation at
# Terrence is not responding to email. # Terrence is not responding to email.
# #
# $Log$ # $Log$
# Revision 1.14 2004/08/23 02:28:24 stuart
# Remove Perl usage message.
#
# Revision 1.13 2004/07/23 19:23:12 stuart
# Always fail to match on ip6, until we support it properly.
#
# Revision 1.12 2004/07/23 18:48:15 stuart
# Fold CID parsing into spf
#
# Revision 1.11 2004/07/21 21:32:01 stuart
# Handle CID records (Microsoft XML format).
#
# Revision 1.10 2004/04/19 22:12:11 stuart # Revision 1.10 2004/04/19 22:12:11 stuart
# Release 0.6.9 # Release 0.6.9
# #
@@ -97,6 +109,135 @@ import struct # for pack() and unpack()
import time # for time() import time # for time()
import DNS # http://pydns.sourceforge.net import DNS # http://pydns.sourceforge.net
import xml.sax
# -------------------------------------------------------------------------
# Convert a MS Caller-ID entry (XML) to a SPF entry
#
# (c) 2004 by Ernesto Baschny
# (c) 2004 Python version by Stuart Gathman
#
# Date: 2004-02-25
#
# A complete reverse translation (SPF -> CID) might be impossible, since
# there are no way to handle:
# - PTR and EXISTS mechanism
# - MX mechanism with an different domain as argument
# - macros
#
# References:
# http://www.microsoft.com/mscorp/twc/privacy/spam_callerid.mspx
# http://spf.pobox.com/
#
# Known bugs:
# - Currently it won't handle the exclusions provided in the A and R
# tags (prefix '!'). They will show up "as-is" in the SPF record
# - I really haven't read the MS-CID specs in-depth, so there are probably
# other bugs too :)
#
# Ernesto Baschny <ernst@baschny.de>
#
class CIDParser(xml.sax.ContentHandler):
"Convert a MS Caller-ID entry (XML) to a SPF entry."
def __init__(self,q=None):
self.spf = []
self.action = '-all'
self.has_servers = None
self.spf_entry = None
if q:
self.spf_query = q
else:
self.spf_query = query(i='127.0.0.1', s='localhost', h='unknown')
def startElement(self,tag,attr):
if tag == 'm':
if self.has_servers != None and not self.has_servers:
raise ValueError(
"Declared <noMailServers\> and later <m>, this CID entry is not valid."
)
self.has_servers = True
elif tag == 'noMailServers':
if self.has_servers:
raise ValueError(
"Declared <m> and later <noMailServers\>, this CID entry is not valid."
)
self.has_servers = False
elif tag == 'ep':
if attr.has_key('testing') and attr.getValue('testing') == 'true':
# A CID with 'testing' found:
# From the MS-specs:
# "Documents in which such attribute is present with a true
# value SHOULD be entirely ignored (one should act as if the
# document were absent)"
# From the SPF-specs:
# "Neutral (?): The SPF client MUST proceed as if a domain did
# not publish SPF data."
# So we set SPF action to "neutral":
self.action = '?all'
elif tag == 'mx':
# The empty MX-tag, same as SPF's MX-mechanism
self.spf.append('mx')
self.tag = tag
def characters(self,text):
tag = self.tag
# Remove starting and trailing spaces from text:
text = text.strip()
if tag == 'a' or tag == 'r':
# The A and R tags from MS-CID are both handled by the
# ipv4/6-mechanisms from SPF:
if text.find(':') < 0:
mechanism = 'ip4'
else:
mechanism = 'ip6'
self.spf.append(mechanism + ':' + text)
elif tag == 'indirect':
# MS-CID's indirect is "sort of" the include from SPF:
# Not really true, because the <indirect> tag from MS-CID also
# provides a fallback in case the included domain doesn't provide
# _ep-records: The inbound MX-servers of the included domains
# are added to the list of allowed outgoing mailservers for the
# domain that declared the _ep-record with the <indirect> tag.
# In SPF you would use the 'mx:domain' to handle this, but this
# wouldn't depend on referred domain having or not SPF-records.
cid_xml = self.cid_txt(text)
if cid_xml:
p = CIDParser()
xml.sax.parseString(cid_xml,p)
if p.has_servers != False:
self.spf += p.spf
else:
self.spf.append('mx:' + text)
def cid_txt(self,domain):
q = self.spf_query
domain='_ep.' + domain
a = q.dns_txt(domain)
if not a: return None
if a[0].lower().startswith('<ep ') and a[-1].lower().endswith('</ep>'):
return ''.join(a)
return None
def endElement(self,tag):
if tag == 'ep':
# This is the end... assemble what we've got
spf_entry = ['v=spf1']
if self.has_servers != False:
spf_entry += self.spf
spf_entry.append(self.action)
self.spf_entry = ' '.join(spf_entry)
def spf_txt(self,cid_xml):
if not cid_xml.startswith('<'):
cid_xml = self.cid_txt(cid_xml)
if not cid_xml: return None
# Parse the beast. Any XML-problem will be reported by xlm.sax
self.spf_entry = None
xml.sax.parseString(cid_xml,self)
return self.spf_entry
# 32-bit IPv4 address mask # 32-bit IPv4 address mask
MASK = 0xFFFFFFFFL MASK = 0xFFFFFFFFL
@@ -330,9 +471,14 @@ class query(object):
cidrlength): cidrlength):
break break
elif m in ('ip4', 'ipv4') and arg != self.d: elif m in ('ip4', 'ipv4', 'ip') and arg != self.d:
if cidrmatch(self.i, [arg], cidrlength): if cidrmatch(self.i, [arg], cidrlength):
break break
elif m == 'ip6':
# Until we support IPV6, we should never
# get an IPv6 connection. So this mech
# will never match.
pass
elif m in ('ptr', 'prt'): elif m in ('ptr', 'prt'):
if domainmatch(self.validated_ptrs(self.i), if domainmatch(self.validated_ptrs(self.i),
@@ -465,17 +611,24 @@ class query(object):
is found. is found.
""" """
a = [t for t in self.dns_txt(domain) if t.startswith('v=spf1')] a = [t for t in self.dns_txt(domain) if t.startswith('v=spf1')]
if not a and DELEGATE: if not a:
if DELEGATE:
a = [t a = [t
for t in self.dns_txt(domain+'._spf.'+DELEGATE) for t in self.dns_txt(domain+'._spf.'+DELEGATE)
if t.startswith('v=spf1') if t.startswith('v=spf1')
] ]
if not a:
# No SPF record: convert and return CID if present
p = CIDParser(q=self)
return p.spf_txt(domain)
if len(a) == 1: if len(a) == 1:
return a[0] return a[0]
else: else:
return None return None
def dns_txt(self, domainname): def dns_txt(self, domainname):
"Get a list of TXT records for a domain name."
if domainname: if domainname:
return [t for a in self.dns(domainname, 'TXT') for t in a] return [t for a in self.dns(domainname, 'TXT') for t in a]
return [] return []