gpg-lacre/gpg-mailgate.py

139 lines
4.4 KiB
Python
Raw Normal View History

2010-01-20 21:53:25 +01:00
#!/usr/bin/python
#
# gpg-mailgate
#
# This file is part of the gpg-mailgate source code.
#
# gpg-mailgate is free software: you can redistribute it and/or modify
2013-10-02 20:27:28 +02:00
# 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.
#
# gpg-mailgate source code 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
2013-10-02 20:27:28 +02:00
# GNU General Public License for more details.
#
2013-10-02 20:27:28 +02:00
# You should have received a copy of the GNU General Public License
# along with gpg-mailgate source code. If not, see <http://www.gnu.org/licenses/>.
#
2010-01-20 21:53:25 +01:00
from ConfigParser import RawConfigParser
from email.mime.base import MIMEBase
2010-01-20 21:53:25 +01:00
import email
2013-04-14 13:13:52 +02:00
import email.message
import re
2010-01-20 21:53:25 +01:00
import GnuPG
import smtplib
import sys
def appendLog(msg):
if cfg.has_key('logging') and cfg['logging'].has_key('file'):
log = open(cfg['logging']['file'], 'a')
log.write(msg + "\n")
log.close()
2010-01-20 21:53:25 +01:00
# Read configuration from /etc/gpg-mailgate.conf
_cfg = RawConfigParser()
_cfg.read('/etc/gpg-mailgate.conf')
cfg = dict()
for sect in _cfg.sections():
cfg[sect] = dict()
for (name, value) in _cfg.items(sect):
cfg[sect][name] = value
# Read e-mail from stdin
raw = sys.stdin.read()
raw_message = email.message_from_string( raw )
from_addr = raw_message['From']
to_addrs = sys.argv[1:]
2010-01-20 21:53:25 +01:00
def send_msg( message, recipients = None ):
if recipients == None:
recipients = to_addrs
appendLog("Sending email to: <%s>" % '> <'.join( recipients ))
2010-01-20 21:53:25 +01:00
relay = (cfg['relay']['host'], int(cfg['relay']['port']))
smtp = smtplib.SMTP(relay[0], relay[1])
smtp.sendmail( from_addr, recipients, message.as_string() )
def encrypt_payload( payload, gpg_to_cmdline ):
raw_payload = payload.get_payload(decode=True)
2013-09-28 16:28:00 +02:00
if "-----BEGIN PGP MESSAGE-----" in raw_payload and "-----END PGP MESSAGE-----" in raw_payload:
return payload
gpg = GnuPG.GPGEncryptor( cfg['gpg']['keyhome'], gpg_to_cmdline, payload.get_content_charset() )
gpg.update( raw_payload )
payload.set_payload( gpg.encrypt() )
isAttachment = payload.get_param( 'attachment', None, 'Content-Disposition' ) is not None
if isAttachment:
filename = payload.get_filename()
if filename:
pgpFilename = filename + ".pgp"
if payload.get('Content-Disposition') is not None:
payload.set_param( 'filename', pgpFilename, 'Content-Disposition' )
if payload.get('Content-Type') is not None:
if payload.get_param( 'name' ) is not None:
payload.set_param( 'name', pgpFilename )
if payload.get('Content-Transfer-Encoding') is not None:
payload.replace_header( 'Content-Transfer-Encoding', "7bit" )
return payload
def encrypt_all_payloads( message, gpg_to_cmdline ):
2013-04-04 20:07:32 +02:00
encrypted_payloads = list()
if type( message.get_payload() ) == str:
return encrypt_payload( message, gpg_to_cmdline ).get_payload()
for payload in message.get_payload():
2013-04-04 20:07:32 +02:00
if( type( payload.get_payload() ) == list ):
encrypted_payloads.extend( encrypt_all_payloads( payload, gpg_to_cmdline ) )
2013-04-04 20:07:32 +02:00
else:
2013-08-11 17:06:27 +02:00
encrypted_payloads.append( encrypt_payload( payload, gpg_to_cmdline ) )
return encrypted_payloads
2013-04-04 20:07:32 +02:00
2010-01-21 15:50:51 +01:00
def get_msg( message ):
if not message.is_multipart():
return message.get_payload()
return '\n\n'.join( [str(m) for m in message.get_payload()] )
2010-01-21 15:50:51 +01:00
keys = GnuPG.public_keys( cfg['gpg']['keyhome'] )
2010-01-20 21:53:25 +01:00
gpg_to = list()
2010-01-20 22:09:34 +01:00
ungpg_to = list()
2010-01-20 21:53:25 +01:00
for to in to_addrs:
if to in keys and not ( cfg['default'].has_key('keymap_only') and cfg['default']['keymap_only'] == 'yes' ):
gpg_to.append( (to, to) )
elif cfg.has_key('keymap') and cfg['keymap'].has_key(to):
gpg_to.append( (to, cfg['keymap'][to]) )
2010-01-20 22:09:34 +01:00
else:
ungpg_to.append(to)
2010-01-20 21:53:25 +01:00
if gpg_to == list():
if cfg['default'].has_key('add_header') and cfg['default']['add_header'] == 'yes':
raw_message['X-GPG-Mailgate'] = 'Not encrypted, public key not found'
send_msg( raw_message )
exit()
2010-01-20 21:53:25 +01:00
2010-01-20 22:09:34 +01:00
if ungpg_to != list():
send_msg( raw_message, ungpg_to )
appendLog("Encrypting email to: %s" % ' '.join( map(lambda x: x[0], gpg_to) ))
2010-01-21 15:50:51 +01:00
2010-01-20 21:53:25 +01:00
if cfg['default'].has_key('add_header') and cfg['default']['add_header'] == 'yes':
2010-01-21 15:50:51 +01:00
raw_message['X-GPG-Mailgate'] = 'Encrypted by GPG Mailgate'
2010-01-20 21:53:25 +01:00
2010-01-20 22:09:34 +01:00
gpg_to_cmdline = list()
gpg_to_smtp = list()
for rcpt in gpg_to:
gpg_to_smtp.append(rcpt[0])
gpg_to_cmdline.extend(rcpt[1].split(','))
encrypted_payloads = encrypt_all_payloads( raw_message, gpg_to_cmdline )
raw_message.set_payload( encrypted_payloads )
2010-01-20 22:09:34 +01:00
send_msg( raw_message, gpg_to_smtp )