diff options
-rw-r--r-- | Lib/imaplib.py | 216 |
1 files changed, 175 insertions, 41 deletions
diff --git a/Lib/imaplib.py b/Lib/imaplib.py index caea5bf..8bab8d8 100644 --- a/Lib/imaplib.py +++ b/Lib/imaplib.py @@ -4,6 +4,8 @@ Based on RFC 2060. Author: Piers Lauder <piers@cs.su.oz.au> December 1997. +Authentication code contributed by Donn Cave <donn@u.washington.edu> June 1998. + Public class: IMAP4 Public variable: Debug Public functions: Internaldate2tuple @@ -11,8 +13,12 @@ Public functions: Internaldate2tuple ParseFlags Time2Internaldate """ +# +# $Header$ +# +__version__ = "$Revision$" -import re, socket, string, time, random +import binascii, re, socket, string, time, random # Globals @@ -41,6 +47,7 @@ Commands = { 'LOGOUT': ('NONAUTH', 'AUTH', 'SELECTED', 'LOGOUT'), 'LSUB': ('AUTH', 'SELECTED'), 'NOOP': ('NONAUTH', 'AUTH', 'SELECTED', 'LOGOUT'), + 'PARTIAL': ('SELECTED',), 'RENAME': ('AUTH', 'SELECTED'), 'SEARCH': ('SELECTED',), 'SELECT': ('AUTH', 'SELECTED'), @@ -53,7 +60,7 @@ Commands = { # Patterns to match server responses -Continuation = re.compile(r'\+ (?P<data>.*)') +Continuation = re.compile(r'\+( (?P<data>.*))?') Flags = re.compile(r'.*FLAGS \((?P<flags>[^\)]*)\)') InternalDate = re.compile(r'.*INTERNALDATE "' r'(?P<day>[ 123][0-9])-(?P<mon>[A-Z][a-z][a-z])-(?P<year>[0-9][0-9][0-9][0-9])' @@ -62,7 +69,7 @@ InternalDate = re.compile(r'.*INTERNALDATE "' r'"') Literal = re.compile(r'(?P<data>.*) {(?P<size>\d+)}$') Response_code = re.compile(r'\[(?P<type>[A-Z-]+)( (?P<data>[^\]]*))?\]') -Untagged_response = re.compile(r'\* (?P<type>[A-Z-]+) (?P<data>.*)') +Untagged_response = re.compile(r'\* (?P<type>[A-Z-]+)( (?P<data>.*))?') Untagged_status = re.compile(r'\* (?P<data>\d+) (?P<type>[A-Z-]+)( (?P<data2>.*))?') @@ -81,8 +88,9 @@ class IMAP4: All arguments to commands are converted to strings, except for the last argument to APPEND which is passed as an IMAP4 - literal. If necessary (the string isn't enclosed with either - parentheses or double quotes) each converted string is quoted. + literal. If necessary (the string contains white-space and + isn't enclosed with either parentheses or double quotes) each + string is quoted. Each command returns a tuple: (type, [data, ...]) where 'type' is usually 'OK' or 'NO', and 'data' is either the text from the @@ -91,6 +99,11 @@ class IMAP4: Errors raise the exception class <instance>.error("<reason>"). IMAP4 server errors raise <instance>.abort("<reason>"), which is a sub-class of 'error'. + + Note: to use this module, you must read the RFCs pertaining + to the IMAP4 protocol, as the semantics of the arguments to + each IMAP4 command are left to the invoker, not to mention + the results. """ class error(Exception): pass # Logical errors - debug required @@ -110,9 +123,7 @@ class IMAP4: # Open socket to server. - self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) - self.sock.connect(self.host, self.port) - self.file = self.sock.makefile('r') + self.open(host, port) # Create unique tag for this session, # and compile tagged response matcher. @@ -156,6 +167,13 @@ class IMAP4: raise self.error('server not IMAP4 compliant') + def open(self, host, port): + """Setup 'self.sock' and 'self.file'.""" + self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) + self.sock.connect(self.host, self.port) + self.file = self.sock.makefile('r') + + def __getattr__(self, attr): """Allow UPPERCASE variants of all following IMAP4 commands.""" if Commands.has_key(attr): @@ -173,7 +191,8 @@ class IMAP4: """ name = 'APPEND' if flags: - flags = '(%s)' % flags + if (flags[0],flags[-1]) != ('(',')'): + flags = '(%s)' % flags else: flags = None if date_time: @@ -184,12 +203,32 @@ class IMAP4: return self._simple_command(name, mailbox, flags, date_time) - def authenticate(self, func): + def authenticate(self, mechanism, authobject): """Authenticate command - requires response processing. - UNIMPLEMENTED + 'mechanism' specifies which authentication mechanism is to + be used - it must appear in <instance>.capabilities in the + form AUTH=<mechanism>. + + 'authobject' must be a callable object: + + data = authobject(response) + + It will be called to process server continuation responses. + It should return data that will be encoded and sent to server. + It should return None if the client abort response '*' should + be sent instead. """ - raise self.error('UNIMPLEMENTED') + mech = string.upper(mechanism) + cap = 'AUTH=%s' % mech + if not cap in self.capabilities: + raise self.error("Server doesn't allow %s authentication." % mech) + self.literal = _Authenticator(authobject).process + typ, dat = self._simple_command('AUTHENTICATE', mech) + if typ != 'OK': + raise self.error(dat) + self.state = 'AUTH' + return typ, dat def check(self): @@ -324,18 +363,32 @@ class IMAP4: (typ, data) = <instance>.noop() """ + if __debug__ and self.debug >= 3: + print '\tuntagged responses: %s' % `self.untagged_responses` return self._simple_command('NOOP') + def partial(self, message_num, message_part, start, length): + """Fetch truncated part of a message. + + (typ, [data, ...]) = <instance>.partial(message_num, message_part, start, length) + + 'data' is tuple of message part envelope and data. + """ + name = 'PARTIAL' + typ, dat = self._simple_command(name, message_num, message_part, start, length) + return self._untagged_response(typ, 'FETCH') + + def recent(self): - """Return most recent 'RECENT' response if it exists, + """Return most recent 'RECENT' responses if any exist, else prompt server for an update using the 'NOOP' command, and flush all untagged responses. (typ, [data]) = <instance>.recent() 'data' is None if no new messages, - else value of RECENT response. + else list of RECENT responses, most recent last. """ name = 'RECENT' typ, dat = self._untagged_response('OK', name) @@ -361,7 +414,7 @@ class IMAP4: (code, [data]) = <instance>.response(code) """ - return self._untagged_response(code, code) + return self._untagged_response(code, string.upper(code)) def search(self, charset, criteria): @@ -403,6 +456,14 @@ class IMAP4: return typ, self.untagged_responses.get('EXISTS', [None]) + def socket(self): + """Return socket instance used to connect to IMAP4 server. + + socket = <instance>.socket() + """ + return self.sock + + def status(self, mailbox, names): """Request named status conditions for mailbox. @@ -440,8 +501,14 @@ class IMAP4: Returns response appropriate to 'command'. """ + command = string.upper(command) + if not Commands.has_key(command): + raise self.error("Unknown IMAP4 UID command: %s" % command) + if self.state not in Commands[command]: + raise self.error('command %s illegal in state %s' + % (command, self.state)) name = 'UID' - typ, dat = apply(self._simple_command, ('UID', command) + args) + typ, dat = apply(self._simple_command, (name, command) + args) if command == 'SEARCH': name = 'SEARCH' else: @@ -476,13 +543,13 @@ class IMAP4: def _append_untagged(self, typ, dat): - if self.untagged_responses.has_key(typ): - self.untagged_responses[typ].append(dat) + ur = self.untagged_responses + if ur.has_key(typ): + ur[typ].append(dat) else: - self.untagged_responses[typ] = [dat] - + ur[typ] = [dat] if __debug__ and self.debug >= 5: - print '\tuntagged_responses[%s] += %.20s..' % (typ, `dat`) + print '\tuntagged_responses[%s] %s += %s' % (typ, len(`ur[typ]`), _trunc(20, `dat`)) def _command(self, name, *args): @@ -492,6 +559,9 @@ class IMAP4: raise self.error( 'command %s illegal in state %s' % (name, self.state)) + if self.untagged_responses.has_key('OK'): + del self.untagged_responses['OK'] + tag = self._new_tag() data = '%s %s' % (tag, name) for d in args: @@ -508,7 +578,11 @@ class IMAP4: literal = self.literal if literal is not None: self.literal = None - data = '%s {%s}' % (data, len(literal)) + if type(literal) is type(self._command): + literator = literal + else: + literator = None + data = '%s {%s}' % (data, len(literal)) try: self.sock.send('%s%s' % (data, CRLF)) @@ -521,22 +595,29 @@ class IMAP4: if literal is None: return tag - # Wait for continuation response + while 1: + # Wait for continuation response - while self._get_response(): - if self.tagged_commands[tag]: # BAD/NO? - return tag + while self._get_response(): + if self.tagged_commands[tag]: # BAD/NO? + return tag - # Send literal + # Send literal - if __debug__ and self.debug >= 4: - print '\twrite literal size %s' % len(literal) + if literator: + literal = literator(self.continuation_response) - try: - self.sock.send(literal) - self.sock.send(CRLF) - except socket.error, val: - raise self.abort('socket error: %s' % val) + if __debug__ and self.debug >= 4: + print '\twrite literal size %s' % len(literal) + + try: + self.sock.send(literal) + self.sock.send(CRLF) + except socket.error, val: + raise self.abort('socket error: %s' % val) + + if not literator: + break return tag @@ -590,10 +671,11 @@ class IMAP4: self.continuation_response = self.mo.group('data') return None # NB: indicates continuation - raise self.abort('unexpected response: %s' % resp) + raise self.abort("unexpected response: '%s'" % resp) typ = self.mo.group('type') dat = self.mo.group('data') + if dat is None: dat = '' # Null untagged response if dat2: dat = dat + ' ' + dat2 # Is there a literal to come? @@ -679,12 +761,56 @@ class IMAP4: return typ, [None] data = self.untagged_responses[name] if __debug__ and self.debug >= 5: - print '\tuntagged_responses[%s] => %.20s..' % (name, `data`) + print '\tuntagged_responses[%s] => %s' % (name, _trunc(20, `data`)) del self.untagged_responses[name] return typ, data +class _Authenticator: + + """Private class to provide en/decoding + for base64-based authentication conversation. + """ + + def __init__(self, mechinst): + self.mech = mechinst # Callable object to provide/process data + + def process(self, data): + ret = self.mech(self.decode(data)) + if ret is None: + return '*' # Abort conversation + return self.encode(ret) + + def encode(self, inp): + # + # Invoke binascii.b2a_base64 iteratively with + # short even length buffers, strip the trailing + # line feed from the result and append. "Even" + # means a number that factors to both 6 and 8, + # so when it gets to the end of the 8-bit input + # there's no partial 6-bit output. + # + oup = '' + while inp: + if len(inp) > 48: + t = inp[:48] + inp = inp[48:] + else: + t = inp + inp = '' + e = binascii.b2a_base64(t) + if e: + oup = oup + e[:-1] + return oup + + def decode(self, inp): + if not inp: + return '' + return binascii.a2b_base64(inp) + + + Mon2num = {'Jan': 1, 'Feb': 2, 'Mar': 3, 'Apr': 4, 'May': 5, 'Jun': 6, 'Jul': 7, 'Aug': 8, 'Sep': 9, 'Oct': 10, 'Nov': 11, 'Dec': 12} @@ -779,6 +905,14 @@ def Time2Internaldate(date_time): +if __debug__: + + def _trunc(m, s): + if len(s) <= m: return s + return '%.*s..' % (m, s) + + + if __debug__ and __name__ == '__main__': host = '' @@ -798,8 +932,8 @@ if __debug__ and __name__ == '__main__': ('CREATE', ('/tmp/yyz 2',)), ('append', ('/tmp/yyz 2', None, None, 'From: anon@x.y.z\n\ndata...')), ('select', ('/tmp/yyz 2',)), - ('uid', ('SEARCH', 'ALL')), - ('fetch', ('1', '(INTERNALDATE RFC822)')), + ('search', (None, '(TO zork)')), + ('partial', ('1', 'RFC822', 1, 1024)), ('store', ('1', 'FLAGS', '(\Deleted)')), ('expunge', ()), ('recent', ()), @@ -820,7 +954,7 @@ if __debug__ and __name__ == '__main__': print ' %s %s\n => %s %s' % (cmd, args, typ, dat) return dat - Debug = 4 + Debug = 5 M = IMAP4(host) print 'PROTOCOL_VERSION = %s' % M.PROTOCOL_VERSION @@ -839,6 +973,6 @@ if __debug__ and __name__ == '__main__': if (cmd,args) != ('uid', ('SEARCH', 'ALL')): continue - uid = string.split(dat[0])[-1] + uid = string.split(dat[-1])[-1] run('uid', ('FETCH', '%s' % uid, - '(FLAGS INTERNALDATE RFC822.SIZE RFC822.HEADER RFC822)')) + '(FLAGS INTERNALDATE RFC822.SIZE RFC822.HEADER RFC822.TEXT)')) |