diff options
Diffstat (limited to 'Lib/poplib.py')
-rw-r--r-- | Lib/poplib.py | 430 |
1 files changed, 215 insertions, 215 deletions
diff --git a/Lib/poplib.py b/Lib/poplib.py index 5a3813c..2f55f74 100644 --- a/Lib/poplib.py +++ b/Lib/poplib.py @@ -32,290 +32,290 @@ 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'. + """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) + Instantiate with: + POP3(hostname, port=110) - NB: the POP protocol locks the mailbox from user - authorization until QUIT, so be sure to get in, suck - the messages, and quit, each time you access the - mailbox. + NB: the POP protocol locks the mailbox from user + authorization 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. + 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. - """ + 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 __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)) + 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()) + # Internal: send one command to the server (through _putline()) - def _putcmd(self, line): - #if self._debugging: print '*cmd*', `line` - self._putline(line) + 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. + # 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 + 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 '+'. + # 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 + 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. + # 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 != '.': - if line[:2] == '..': - o = o-1 - line = line[1:] - octets = octets + o - list.append(line) - line, o = self._getline() - return resp, list, octets + def _getlongresp(self): + resp = self._getresp() + list = []; octets = 0 + line, o = self._getline() + while line != '.': + if line[:2] == '..': + o = o-1 + line = line[1:] + octets = octets + o + list.append(line) + line, o = self._getline() + return resp, list, octets - # Internal: send a command and get the response + # Internal: send a command and get the response - def _shortcmd(self, line): - self._putcmd(line) - return self._getresp() + def _shortcmd(self, line): + self._putcmd(line) + return self._getresp() - # Internal: send a command and get the response plus following text + # Internal: send a command and get the response plus following text - def _longcmd(self, line): - self._putcmd(line) - return self._getlongresp() + def _longcmd(self, line): + self._putcmd(line) + return self._getlongresp() - # These can be useful: + # These can be useful: - def getwelcome(self): - return self.welcome + def getwelcome(self): + return self.welcome - def set_debuglevel(self, level): - self._debugging = level + def set_debuglevel(self, level): + self._debugging = level - # Here are all the POP commands: + # 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 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 pass_(self, pswd): + """Send password, return response + (response includes message count, mailbox size). - 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) + NB: mailbox is locked by server from here to 'quit()' + """ + return self._shortcmd('PASS %s' % pswd) - def list(self, which=None): - """Request listing, return result. + def stat(self): + """Get mailbox status. - Result without a message number argument is in form - ['response', ['mesg_num octets', ...]]. + 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) - Result when a message number argument is given is a - single response: the "scan listing" for that message. - """ - if which: - return self._shortcmd('LIST %s' % which) - return self._longcmd('LIST') + def list(self, which=None): + """Request listing, return result. - def retr(self, which): - """Retrieve whole message number 'which'. + Result without a message number argument is in form + ['response', ['mesg_num octets', ...]]. - Result is in form ['response', ['line', ...], octets]. - """ - return self._longcmd('RETR %s' % which) + Result when a message number argument is given is a + single response: the "scan listing" for that message. + """ + if which: + return self._shortcmd('LIST %s' % which) + return self._longcmd('LIST') - def dele(self, which): - """Delete message number 'which'. + def retr(self, which): + """Retrieve whole message number 'which'. - Result is 'response'. - """ - return self._shortcmd('DELE %s' % which) + Result is in form ['response', ['line', ...], octets]. + """ + return self._longcmd('RETR %s' % which) - def noop(self): - """Does nothing. - - One supposes the response indicates the server is alive. - """ - return self._shortcmd('NOOP') + def dele(self, which): + """Delete message number 'which'. + Result is 'response'. + """ + return self._shortcmd('DELE %s' % which) - def rset(self): - """Not sure what this does.""" - return self._shortcmd('RSET') + def noop(self): + """Does nothing. - 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 + One supposes the response indicates the server is alive. + """ + return self._shortcmd('NOOP') - #__del__ = quit + def rset(self): + """Not sure what this does.""" + return self._shortcmd('RSET') - # optional commands: - def rpop(self, user): - """Not sure what this does.""" - return self._shortcmd('RPOP %s' % user) + 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 - timestamp = regex.compile('\+OK.*\(<[^>]+>\)') - def apop(self, user, secret): - """Authorisation - - - only possible if server has supplied a timestamp in initial greeting. + # optional commands: - Args: - user - mailbox user; - secret - secret shared between client and server. + def rpop(self, user): + """Not sure what this does.""" + return self._shortcmd('RPOP %s' % user) - 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)) + timestamp = regex.compile('\+OK.*\(<[^>]+>\)') - def top(self, which, howmuch): - """Retrieve message header of message number 'which' - and first 'howmuch' lines of message body. + def apop(self, user, secret): + """Authorisation - Result is in form ['response', ['line', ...], octets]. - """ - return self._longcmd('TOP %s %s' % (which, howmuch)) + - only possible if server has supplied a timestamp in initial greeting. + Args: + user - mailbox user; + secret - secret shared between client and server. - def uidl(self, which=None): - """Return message digest (unique id) list. + 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 + in the form 'response mesgnum uid', otherwise result is + the list ['response', ['mesgnum uid', ...], octets] + """ + if which: + return self._shortcmd('UIDL %s' % which) + return self._longcmd('UIDL') - If 'which', result contains unique id for that message - in the form 'response mesgnum uid', otherwise result is - the 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() + 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() |