mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 19:24:34 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			314 lines
		
	
	
	
		
			7.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			314 lines
		
	
	
	
		
			7.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """A POP3 client class.
 | |
| 
 | |
| Based on the J. Myers POP3 draft, Jan. 96
 | |
| 
 | |
| Author: David Ascher <david_ascher@brown.edu>
 | |
|         [heavily stealing from nntplib.py]
 | |
| Updated: Piers Lauder <piers@cs.su.oz.au> [Jul '97]
 | |
| """
 | |
| 
 | |
| # Example (see the test function at the end of this file)
 | |
| 
 | |
| TESTSERVER = "localhost"
 | |
| TESTACCOUNT = "test"
 | |
| TESTPASSWORD = "_passwd_"
 | |
| 
 | |
| # Imports
 | |
| 
 | |
| import regex, socket, string
 | |
| 
 | |
| # Exception raised when an error or invalid response is received:
 | |
| 
 | |
| class error_proto(Exception): pass
 | |
| 
 | |
| # Standard Port
 | |
| POP3_PORT = 110
 | |
| 
 | |
| # Line terminators (we always output CRLF, but accept any of CRLF, LFCR, LF)
 | |
| CR = '\r'
 | |
| LF = '\n'
 | |
| CRLF = CR+LF
 | |
| 
 | |
| 
 | |
| class POP3:
 | |
| 
 | |
| 	"""This class supports both the minimal and optional command sets.
 | |
| 	Arguments can be strings or integers (where appropriate)
 | |
| 	(e.g.: retr(1) and retr('1') both work equally well.
 | |
| 
 | |
| 	Minimal Command Set:
 | |
| 		USER name		user(name)
 | |
| 		PASS string		pass_(string)
 | |
| 		STAT			stat()
 | |
| 		LIST [msg]		list(msg = None)
 | |
| 		RETR msg		retr(msg)
 | |
| 		DELE msg		dele(msg)
 | |
| 		NOOP			noop()
 | |
| 		RSET			rset()
 | |
| 		QUIT			quit()
 | |
| 
 | |
| 	Optional Commands (some servers support these):
 | |
| 		RPOP name		rpop(name)
 | |
| 		APOP name digest	apop(name, digest)
 | |
| 		TOP msg n		top(msg, n)
 | |
| 		UIDL [msg]		uidl(msg = None)
 | |
| 
 | |
| 	Raises one exception: 'error_proto'.
 | |
| 
 | |
| 	Instantiate with:
 | |
| 		POP3(hostname, port=110)
 | |
| 
 | |
| 	NB:	the POP protocol locks the mailbox from user
 | |
| 		authorisation until QUIT, so be sure to get in, suck
 | |
| 		the messages, and quit, each time you access the
 | |
| 		mailbox.
 | |
| 
 | |
| 		POP is a line-based protocol, which means large mail
 | |
| 		messages consume lots of python cycles reading them
 | |
| 		line-by-line.
 | |
| 
 | |
| 		If it's available on your mail server, use IMAP4
 | |
| 		instead, it doesn't suffer from the two problems
 | |
| 		above.
 | |
| 	"""
 | |
| 
 | |
| 
 | |
| 	def __init__(self, host, port = POP3_PORT):
 | |
| 		self.host = host
 | |
| 		self.port = port
 | |
| 		self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
 | |
| 		self.sock.connect(self.host, self.port)
 | |
| 		self.file = self.sock.makefile('rb')
 | |
| 		self._debugging = 0
 | |
| 		self.welcome = self._getresp()
 | |
| 
 | |
| 
 | |
| 	def _putline(self, line):
 | |
| 		#if self._debugging > 1: print '*put*', `line`
 | |
| 		self.sock.send('%s%s' % (line, CRLF))
 | |
| 
 | |
| 
 | |
| 	# Internal: send one command to the server (through _putline())
 | |
| 
 | |
| 	def _putcmd(self, line):
 | |
| 		#if self._debugging: print '*cmd*', `line`
 | |
| 		self._putline(line)
 | |
| 
 | |
| 
 | |
| 	# Internal: return one line from the server, stripping CRLF.
 | |
| 	# This is where all the CPU time of this module is consumed.
 | |
| 	# Raise error_proto('-ERR EOF') if the connection is closed.
 | |
| 
 | |
| 	def _getline(self):
 | |
| 		line = self.file.readline()
 | |
| 		#if self._debugging > 1: print '*get*', `line`
 | |
| 		if not line: raise error_proto('-ERR EOF')
 | |
| 		octets = len(line)
 | |
| 		# server can send any combination of CR & LF
 | |
| 		# however, 'readline()' returns lines ending in LF
 | |
| 		# so only possibilities are ...LF, ...CRLF, CR...LF
 | |
| 		if line[-2:] == CRLF:
 | |
| 			return line[:-2], octets
 | |
| 		if line[0] == CR:
 | |
| 			return line[1:-1], octets
 | |
| 		return line[:-1], octets
 | |
| 
 | |
| 
 | |
| 	# Internal: get a response from the server.
 | |
| 	# Raise 'error_proto' if the response doesn't start with '+'.
 | |
| 
 | |
| 	def _getresp(self):
 | |
| 		resp, o = self._getline()
 | |
| 		#if self._debugging > 1: print '*resp*', `resp`
 | |
| 		c = resp[:1]
 | |
| 		if c != '+':
 | |
| 			raise error_proto(resp)
 | |
| 		return resp
 | |
| 
 | |
| 
 | |
| 	# Internal: get a response plus following text from the server.
 | |
| 
 | |
| 	def _getlongresp(self):
 | |
| 		resp = self._getresp()
 | |
| 		list = []; octets = 0
 | |
| 		line, o = self._getline()
 | |
| 		while line != '.':
 | |
| 			octets = octets + o
 | |
| 			list.append(line)
 | |
| 			line, o = self._getline()
 | |
| 		return resp, list, octets
 | |
| 
 | |
| 
 | |
| 	# Internal: send a command and get the response
 | |
| 
 | |
| 	def _shortcmd(self, line):
 | |
| 		self._putcmd(line)
 | |
| 		return self._getresp()
 | |
| 
 | |
| 
 | |
| 	# Internal: send a command and get the response plus following text
 | |
| 
 | |
| 	def _longcmd(self, line):
 | |
| 		self._putcmd(line)
 | |
| 		return self._getlongresp()
 | |
| 
 | |
| 
 | |
| 	# These can be useful:
 | |
| 
 | |
| 	def getwelcome(self): 
 | |
| 		return self.welcome
 | |
| 
 | |
| 
 | |
| 	def set_debuglevel(self, level):
 | |
| 		self._debugging = level
 | |
| 
 | |
| 
 | |
| 	# Here are all the POP commands:
 | |
| 
 | |
| 	def user(self, user):
 | |
| 		"""Send user name, return response
 | |
| 		
 | |
| 		(should indicate password required).
 | |
| 		"""
 | |
| 		return self._shortcmd('USER %s' % user)
 | |
| 
 | |
| 
 | |
| 	def pass_(self, pswd):
 | |
| 		"""Send password, return response
 | |
| 		
 | |
| 		(response includes message count, mailbox size).
 | |
| 
 | |
| 		NB: mailbox is locked by server from here to 'quit()'
 | |
| 		"""
 | |
| 		return self._shortcmd('PASS %s' % pswd)
 | |
| 
 | |
| 
 | |
| 	def stat(self):
 | |
| 		"""Get mailbox status.
 | |
| 		
 | |
| 		Result is tuple of 2 ints (message count, mailbox size)
 | |
| 		"""
 | |
| 		retval = self._shortcmd('STAT')
 | |
| 		rets = string.split(retval)
 | |
| 		#if self._debugging: print '*stat*', `rets`
 | |
| 		numMessages = string.atoi(rets[1])
 | |
| 		sizeMessages = string.atoi(rets[2])
 | |
| 		return (numMessages, sizeMessages)
 | |
| 
 | |
| 
 | |
| 	def list(self, which=None):
 | |
| 		"""Request listing, return result.
 | |
| 		Result is in form ['response', ['mesg_num octets', ...]].
 | |
| 
 | |
| 		Unsure what the optional 'msg' arg does.
 | |
| 		"""
 | |
| 		if which:
 | |
| 			return self._longcmd('LIST %s' % which)
 | |
| 		return self._longcmd('LIST')
 | |
| 
 | |
| 
 | |
| 	def retr(self, which):
 | |
| 		"""Retrieve whole message number 'which'.
 | |
| 
 | |
| 		Result is in form ['response', ['line', ...], octets].
 | |
| 		"""
 | |
| 		return self._longcmd('RETR %s' % which)
 | |
| 
 | |
| 
 | |
| 	def dele(self, which):
 | |
| 		"""Delete message number 'which'.
 | |
| 
 | |
| 		Result is 'response'.
 | |
| 		"""
 | |
| 		return self._shortcmd('DELE %s' % which)
 | |
| 
 | |
| 
 | |
| 	def noop(self):
 | |
| 		"""Does nothing.
 | |
| 		
 | |
| 		One supposes the response indicates the server is alive.
 | |
| 		"""
 | |
| 		return self._shortcmd('NOOP')
 | |
| 
 | |
| 
 | |
| 	def rset(self):
 | |
| 		"""Not sure what this does."""
 | |
| 		return self._shortcmd('RSET')
 | |
| 
 | |
| 
 | |
| 	def quit(self):
 | |
| 		"""Signoff: commit changes on server, unlock mailbox, close connection."""
 | |
| 		try:
 | |
| 			resp = self._shortcmd('QUIT')
 | |
| 		except error_proto, val:
 | |
| 			resp = val
 | |
| 		self.file.close()
 | |
| 		self.sock.close()
 | |
| 		del self.file, self.sock
 | |
| 		return resp
 | |
| 
 | |
| 	#__del__ = quit
 | |
| 
 | |
| 
 | |
| 	# optional commands:
 | |
| 
 | |
| 	def rpop(self, user):
 | |
| 		"""Not sure what this does."""
 | |
| 		return self._shortcmd('RPOP %s' % user)
 | |
| 
 | |
| 
 | |
| 	timestamp = regex.compile('\+OK.*\(<[^>]+>\)')
 | |
| 
 | |
| 	def apop(self, user, secret):
 | |
| 		"""Authorisation
 | |
| 		
 | |
| 		- only possible if server has supplied a timestamp in initial greeting.
 | |
| 
 | |
| 		Args:
 | |
| 			user	- mailbox user;
 | |
| 			secret	- secret shared between client and server.
 | |
| 
 | |
| 		NB: mailbox is locked by server from here to 'quit()'
 | |
| 		"""
 | |
| 		if self.timestamp.match(self.welcome) <= 0:
 | |
| 			raise error_proto('-ERR APOP not supported by server')
 | |
| 		import md5
 | |
| 		digest = md5.new(self.timestamp.group(1)+secret).digest()
 | |
| 		digest = string.join(map(lambda x:'%02x'%ord(x), digest), '')
 | |
| 		return self._shortcmd('APOP %s %s' % (user, digest))
 | |
| 
 | |
| 
 | |
| 	def top(self, which, howmuch):
 | |
| 		"""Retrieve message header of message number 'which'
 | |
| 		and first 'howmuch' lines of message body.
 | |
| 
 | |
| 		Result is in form ['response', ['line', ...], octets].
 | |
| 		"""
 | |
| 		return self._longcmd('TOP %s %s' % (which, howmuch))
 | |
| 
 | |
| 
 | |
| 	def uidl(self, which=None):
 | |
| 		"""Return message digest (unique id) list.
 | |
| 
 | |
| 		If 'which', result contains unique id for that message,
 | |
| 		otherwise result is list ['response', ['mesgnum uid', ...], octets]
 | |
| 		"""
 | |
| 		if which:
 | |
| 			return self._shortcmd('UIDL %s' % which)
 | |
| 		return self._longcmd('UIDL')
 | |
| 
 | |
| 				
 | |
| if __name__ == "__main__":
 | |
| 	a = POP3(TESTSERVER)
 | |
| 	print a.getwelcome()
 | |
| 	a.user(TESTACCOUNT)
 | |
| 	a.pass_(TESTPASSWORD)
 | |
| 	a.list()
 | |
| 	(numMsgs, totalSize) = a.stat()
 | |
| 	for i in range(1, numMsgs + 1):
 | |
| 		(header, msg, octets) = a.retr(i)
 | |
| 		print "Message ", `i`, ':'
 | |
| 		for line in msg:
 | |
| 			print '   ' + line
 | |
| 		print '-----------------------'
 | |
| 	a.quit()
 | 
