| 
									
										
										
										
											2000-02-04 15:10:34 +00:00
										 |  |  | """Various tools used by MIME-reading or MIME-writing programs.""" | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | import rfc822 | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | import string | 
					
						
							|  |  |  | import tempfile | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Message(rfc822.Message): | 
					
						
							| 
									
										
										
										
											2000-02-04 15:10:34 +00:00
										 |  |  | 	"""A derived class of rfc822.Message that knows about MIME headers and
 | 
					
						
							|  |  |  | 	contains some hooks for decoding encoded and multipart messages."""
 | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1995-08-07 20:13:56 +00:00
										 |  |  | 	def __init__(self, fp, seekable = 1): | 
					
						
							|  |  |  | 		rfc822.Message.__init__(self, fp, seekable) | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 		self.encodingheader = \ | 
					
						
							| 
									
										
										
										
											1995-08-29 19:25:11 +00:00
										 |  |  | 			self.getheader('content-transfer-encoding') | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 		self.typeheader = \ | 
					
						
							|  |  |  | 			self.getheader('content-type') | 
					
						
							|  |  |  | 		self.parsetype() | 
					
						
							|  |  |  | 		self.parseplist() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def parsetype(self): | 
					
						
							|  |  |  | 		str = self.typeheader | 
					
						
							|  |  |  | 		if str == None: | 
					
						
							|  |  |  | 			str = 'text/plain' | 
					
						
							|  |  |  | 		if ';' in str: | 
					
						
							|  |  |  | 			i = string.index(str, ';') | 
					
						
							|  |  |  | 			self.plisttext = str[i:] | 
					
						
							|  |  |  | 			str = str[:i] | 
					
						
							|  |  |  | 		else: | 
					
						
							|  |  |  | 			self.plisttext = '' | 
					
						
							|  |  |  | 		fields = string.splitfields(str, '/') | 
					
						
							|  |  |  | 		for i in range(len(fields)): | 
					
						
							|  |  |  | 			fields[i] = string.lower(string.strip(fields[i])) | 
					
						
							|  |  |  | 		self.type = string.joinfields(fields, '/') | 
					
						
							|  |  |  | 		self.maintype = fields[0] | 
					
						
							|  |  |  | 		self.subtype = string.joinfields(fields[1:], '/') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def parseplist(self): | 
					
						
							|  |  |  | 		str = self.plisttext | 
					
						
							|  |  |  | 		self.plist = [] | 
					
						
							|  |  |  | 		while str[:1] == ';': | 
					
						
							|  |  |  | 			str = str[1:] | 
					
						
							|  |  |  | 			if ';' in str: | 
					
						
							|  |  |  | 				# XXX Should parse quotes! | 
					
						
							|  |  |  | 				end = string.index(str, ';') | 
					
						
							|  |  |  | 			else: | 
					
						
							|  |  |  | 				end = len(str) | 
					
						
							|  |  |  | 			f = str[:end] | 
					
						
							|  |  |  | 			if '=' in f: | 
					
						
							|  |  |  | 				i = string.index(f, '=') | 
					
						
							|  |  |  | 				f = string.lower(string.strip(f[:i])) + \ | 
					
						
							|  |  |  | 					'=' + string.strip(f[i+1:]) | 
					
						
							|  |  |  | 			self.plist.append(string.strip(f)) | 
					
						
							| 
									
										
										
										
											1996-01-25 18:07:08 +00:00
										 |  |  | 			str = str[end:] | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	def getplist(self): | 
					
						
							|  |  |  | 		return self.plist | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def getparam(self, name): | 
					
						
							|  |  |  | 		name = string.lower(name) + '=' | 
					
						
							|  |  |  | 		n = len(name) | 
					
						
							|  |  |  | 		for p in self.plist: | 
					
						
							|  |  |  | 			if p[:n] == name: | 
					
						
							|  |  |  | 				return rfc822.unquote(p[n:]) | 
					
						
							|  |  |  | 		return None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-10-04 20:14:02 +00:00
										 |  |  | 	def getparamnames(self): | 
					
						
							|  |  |  | 		result = [] | 
					
						
							|  |  |  | 		for p in self.plist: | 
					
						
							|  |  |  | 			i = string.find(p, '=') | 
					
						
							|  |  |  | 			if i >= 0: | 
					
						
							|  |  |  | 				result.append(string.lower(p[:i])) | 
					
						
							|  |  |  | 		return result | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 	def getencoding(self): | 
					
						
							|  |  |  | 		if self.encodingheader == None: | 
					
						
							|  |  |  | 			return '7bit' | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | 		return string.lower(self.encodingheader) | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	def gettype(self): | 
					
						
							|  |  |  | 		return self.type | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def getmaintype(self): | 
					
						
							|  |  |  | 		return self.maintype | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	def getsubtype(self): | 
					
						
							|  |  |  | 		return self.subtype | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Utility functions | 
					
						
							|  |  |  | # ----------------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _prefix = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def choose_boundary(): | 
					
						
							| 
									
										
										
										
											2000-02-04 15:10:34 +00:00
										 |  |  | 	"""Return a random string usable as a multipart boundary.
 | 
					
						
							|  |  |  | 	The method used is so that it is *very* unlikely that the same | 
					
						
							|  |  |  | 	string of characters will every occur again in the Universe, | 
					
						
							|  |  |  | 	so the caller needn't check the data it is packing for the | 
					
						
							|  |  |  | 	occurrence of the boundary. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	The boundary contains dots so you have to quote it in the header."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-05-28 22:59:58 +00:00
										 |  |  | 	global _prefix | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 	import time | 
					
						
							| 
									
										
										
										
											1998-05-20 17:05:52 +00:00
										 |  |  | 	import random | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 	if _prefix == None: | 
					
						
							|  |  |  | 		import socket | 
					
						
							|  |  |  | 		import os | 
					
						
							|  |  |  | 		hostid = socket.gethostbyname(socket.gethostname()) | 
					
						
							| 
									
										
										
										
											1996-08-26 16:40:20 +00:00
										 |  |  | 		try: | 
					
						
							|  |  |  | 		    uid = `os.getuid()` | 
					
						
							|  |  |  | 		except: | 
					
						
							|  |  |  | 		    uid = '1' | 
					
						
							|  |  |  | 		try: | 
					
						
							|  |  |  | 		    pid = `os.getpid()` | 
					
						
							|  |  |  | 		except: | 
					
						
							|  |  |  | 		    pid = '1' | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 		_prefix = hostid + '.' + uid + '.' + pid | 
					
						
							| 
									
										
										
										
											1998-04-11 03:06:02 +00:00
										 |  |  | 	timestamp = '%.3f' % time.time() | 
					
						
							| 
									
										
										
										
											1998-05-20 17:05:52 +00:00
										 |  |  | 	seed = `random.randint(0, 32767)` | 
					
						
							| 
									
										
										
										
											1992-07-13 14:28:59 +00:00
										 |  |  | 	return _prefix + '.' + timestamp + '.' + seed | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Subroutines for decoding some common content-transfer-types | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def decode(input, output, encoding): | 
					
						
							| 
									
										
										
										
											2000-02-04 15:10:34 +00:00
										 |  |  | 	"""Decode common content-transfer-encodings (base64, quopri, uuencode).""" | 
					
						
							| 
									
										
										
										
											1997-07-11 16:33:26 +00:00
										 |  |  | 	if encoding == 'base64': | 
					
						
							|  |  |  | 		import base64 | 
					
						
							|  |  |  | 		return base64.decode(input, output) | 
					
						
							|  |  |  | 	if encoding == 'quoted-printable': | 
					
						
							|  |  |  | 		import quopri | 
					
						
							|  |  |  | 		return quopri.decode(input, output) | 
					
						
							| 
									
										
										
										
											1997-12-10 18:54:36 +00:00
										 |  |  | 	if encoding in ('uuencode', 'x-uuencode', 'uue', 'x-uue'): | 
					
						
							| 
									
										
										
										
											1997-07-11 16:33:26 +00:00
										 |  |  | 		import uu | 
					
						
							|  |  |  | 		return uu.decode(input, output) | 
					
						
							| 
									
										
										
										
											2000-04-04 20:53:07 +00:00
										 |  |  | 	if encoding in ('7bit', '8bit'): | 
					
						
							|  |  |  | 		output.write(input.read()) | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | 	if decodetab.has_key(encoding): | 
					
						
							|  |  |  | 		pipethrough(input, decodetab[encoding], output) | 
					
						
							|  |  |  | 	else: | 
					
						
							|  |  |  | 		raise ValueError, \ | 
					
						
							|  |  |  | 		      'unknown Content-Transfer-Encoding: %s' % encoding | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def encode(input, output, encoding): | 
					
						
							| 
									
										
										
										
											2000-02-04 15:10:34 +00:00
										 |  |  | 	"""Encode common content-transfer-encodings (base64, quopri, uuencode).""" | 
					
						
							| 
									
										
										
										
											1997-07-11 16:33:26 +00:00
										 |  |  | 	if encoding == 'base64': | 
					
						
							|  |  |  | 		import base64 | 
					
						
							|  |  |  | 		return base64.encode(input, output) | 
					
						
							|  |  |  | 	if encoding == 'quoted-printable': | 
					
						
							|  |  |  | 		import quopri | 
					
						
							|  |  |  | 		return quopri.encode(input, output, 0) | 
					
						
							| 
									
										
										
										
											1997-12-10 18:54:36 +00:00
										 |  |  | 	if encoding in ('uuencode', 'x-uuencode', 'uue', 'x-uue'): | 
					
						
							| 
									
										
										
										
											1997-07-11 16:33:26 +00:00
										 |  |  | 		import uu | 
					
						
							|  |  |  | 		return uu.encode(input, output) | 
					
						
							| 
									
										
										
										
											2000-04-04 20:53:07 +00:00
										 |  |  | 	if encoding in ('7bit', '8bit'): | 
					
						
							|  |  |  | 		output.write(input.read()) | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | 	if encodetab.has_key(encoding): | 
					
						
							|  |  |  | 		pipethrough(input, encodetab[encoding], output) | 
					
						
							|  |  |  | 	else: | 
					
						
							|  |  |  | 		raise ValueError, \ | 
					
						
							|  |  |  | 		      'unknown Content-Transfer-Encoding: %s' % encoding | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-07-11 16:33:26 +00:00
										 |  |  | # The following is no longer used for standard encodings | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # XXX This requires that uudecode and mmencode are in $PATH | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | uudecode_pipe = '''(
 | 
					
						
							|  |  |  | TEMP=/tmp/@uu.$$ | 
					
						
							|  |  |  | sed "s%^begin [0-7][0-7]* .*%begin 600 $TEMP%" | uudecode | 
					
						
							|  |  |  | cat $TEMP | 
					
						
							|  |  |  | rm $TEMP | 
					
						
							|  |  |  | )'''
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | decodetab = { | 
					
						
							|  |  |  | 	'uuencode':		uudecode_pipe, | 
					
						
							|  |  |  | 	'x-uuencode':		uudecode_pipe, | 
					
						
							| 
									
										
										
										
											1997-12-10 18:54:36 +00:00
										 |  |  | 	'uue':			uudecode_pipe, | 
					
						
							|  |  |  | 	'x-uue':		uudecode_pipe, | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | 	'quoted-printable':	'mmencode -u -q', | 
					
						
							|  |  |  | 	'base64':		'mmencode -u -b', | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | encodetab = { | 
					
						
							|  |  |  | 	'x-uuencode':		'uuencode tempfile', | 
					
						
							|  |  |  | 	'uuencode':		'uuencode tempfile', | 
					
						
							| 
									
										
										
										
											1997-12-10 18:54:36 +00:00
										 |  |  | 	'x-uue':		'uuencode tempfile', | 
					
						
							|  |  |  | 	'uue':			'uuencode tempfile', | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | 	'quoted-printable':	'mmencode -q', | 
					
						
							|  |  |  | 	'base64':		'mmencode -b', | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def pipeto(input, command): | 
					
						
							|  |  |  | 	pipe = os.popen(command, 'w') | 
					
						
							|  |  |  | 	copyliteral(input, pipe) | 
					
						
							|  |  |  | 	pipe.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def pipethrough(input, command, output): | 
					
						
							|  |  |  | 	tempname = tempfile.mktemp() | 
					
						
							|  |  |  | 	try: | 
					
						
							|  |  |  | 		temp = open(tempname, 'w') | 
					
						
							|  |  |  | 	except IOError: | 
					
						
							|  |  |  | 		print '*** Cannot create temp file', `tempname` | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	copyliteral(input, temp) | 
					
						
							|  |  |  | 	temp.close() | 
					
						
							|  |  |  | 	pipe = os.popen(command + ' <' + tempname, 'r') | 
					
						
							|  |  |  | 	copybinary(pipe, output) | 
					
						
							|  |  |  | 	pipe.close() | 
					
						
							|  |  |  | 	os.unlink(tempname) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def copyliteral(input, output): | 
					
						
							|  |  |  | 	while 1: | 
					
						
							|  |  |  | 		line = input.readline() | 
					
						
							|  |  |  | 		if not line: break | 
					
						
							|  |  |  | 		output.write(line) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def copybinary(input, output): | 
					
						
							|  |  |  | 	BUFSIZE = 8192 | 
					
						
							|  |  |  | 	while 1: | 
					
						
							|  |  |  | 		line = input.read(BUFSIZE) | 
					
						
							|  |  |  | 		if not line: break | 
					
						
							|  |  |  | 		output.write(line) |