imaplib.py 33.2 KB
Newer Older
1 2 3 4
"""IMAP4 client.

Based on RFC 2060.

Tim Peters's avatar
Tim Peters committed
5 6 7 8 9 10
Public class:           IMAP4
Public variable:        Debug
Public functions:       Internaldate2tuple
                        Int2AP
                        ParseFlags
                        Time2Internaldate
11
"""
12

13
# Author: Piers Lauder <piers@cs.su.oz.au> December 1997.
Tim Peters's avatar
Tim Peters committed
14
#
15
# Authentication code contributed by Donn Cave <donn@u.washington.edu> June 1998.
16
# String method conversion by ESR, February 2001.
17

18
__version__ = "2.40"
19

20
import binascii, re, socket, time, random, sys
21

22 23
__all__ = ["IMAP4", "Internaldate2tuple",
           "Int2AP", "ParseFlags", "Time2Internaldate"]
24

Tim Peters's avatar
Tim Peters committed
25
#       Globals
26 27 28 29

CRLF = '\r\n'
Debug = 0
IMAP4_PORT = 143
Tim Peters's avatar
Tim Peters committed
30
AllowedVersions = ('IMAP4REV1', 'IMAP4')        # Most recent first
31

Tim Peters's avatar
Tim Peters committed
32
#       Commands
33 34

Commands = {
Tim Peters's avatar
Tim Peters committed
35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
        # name            valid states
        'APPEND':       ('AUTH', 'SELECTED'),
        'AUTHENTICATE': ('NONAUTH',),
        'CAPABILITY':   ('NONAUTH', 'AUTH', 'SELECTED', 'LOGOUT'),
        'CHECK':        ('SELECTED',),
        'CLOSE':        ('SELECTED',),
        'COPY':         ('SELECTED',),
        'CREATE':       ('AUTH', 'SELECTED'),
        'DELETE':       ('AUTH', 'SELECTED'),
        'EXAMINE':      ('AUTH', 'SELECTED'),
        'EXPUNGE':      ('SELECTED',),
        'FETCH':        ('SELECTED',),
        'LIST':         ('AUTH', 'SELECTED'),
        'LOGIN':        ('NONAUTH',),
        'LOGOUT':       ('NONAUTH', 'AUTH', 'SELECTED', 'LOGOUT'),
        'LSUB':         ('AUTH', 'SELECTED'),
        'NOOP':         ('NONAUTH', 'AUTH', 'SELECTED', 'LOGOUT'),
        'PARTIAL':      ('SELECTED',),
        'RENAME':       ('AUTH', 'SELECTED'),
        'SEARCH':       ('SELECTED',),
        'SELECT':       ('AUTH', 'SELECTED'),
        'STATUS':       ('AUTH', 'SELECTED'),
        'STORE':        ('SELECTED',),
        'SUBSCRIBE':    ('AUTH', 'SELECTED'),
        'UID':          ('SELECTED',),
        'UNSUBSCRIBE':  ('AUTH', 'SELECTED'),
        }

#       Patterns to match server responses
64

65
Continuation = re.compile(r'\+( (?P<data>.*))?')
66 67
Flags = re.compile(r'.*FLAGS \((?P<flags>[^\)]*)\)')
InternalDate = re.compile(r'.*INTERNALDATE "'
Tim Peters's avatar
Tim Peters committed
68 69 70 71
        r'(?P<day>[ 123][0-9])-(?P<mon>[A-Z][a-z][a-z])-(?P<year>[0-9][0-9][0-9][0-9])'
        r' (?P<hour>[0-9][0-9]):(?P<min>[0-9][0-9]):(?P<sec>[0-9][0-9])'
        r' (?P<zonen>[-+])(?P<zoneh>[0-9][0-9])(?P<zonem>[0-9][0-9])'
        r'"')
72
Literal = re.compile(r'.*{(?P<size>\d+)}$')
73
Response_code = re.compile(r'\[(?P<type>[A-Z-]+)( (?P<data>[^\]]*))?\]')
74
Untagged_response = re.compile(r'\* (?P<type>[A-Z-]+)( (?P<data>.*))?')
75 76 77 78 79 80
Untagged_status = re.compile(r'\* (?P<data>\d+) (?P<type>[A-Z-]+)( (?P<data2>.*))?')



class IMAP4:

Tim Peters's avatar
Tim Peters committed
81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
    """IMAP4 client class.

    Instantiate with: IMAP4([host[, port]])

            host - host's name (default: localhost);
            port - port number (default: standard IMAP4 port).

    All IMAP4rev1 commands are supported by methods of the same
    name (in lower-case).

    All arguments to commands are converted to strings, except for
    AUTHENTICATE, and the last argument to APPEND which is passed as
    an IMAP4 literal.  If necessary (the string contains any
    non-printing characters or white-space and isn't enclosed with
    either parentheses or double quotes) each string is quoted.
    However, the 'password' argument to the LOGIN command is always
    quoted.  If you want to avoid having an argument string quoted
    (eg: the 'flags' argument to STORE) then enclose the string in
    parentheses (eg: "(\Deleted)").
100

Tim Peters's avatar
Tim Peters committed
101 102 103
    Each command returns a tuple: (type, [data, ...]) where 'type'
    is usually 'OK' or 'NO', and 'data' is either the text from the
    tagged response, or untagged results from command.
104

Tim Peters's avatar
Tim Peters committed
105 106 107 108 109
    Errors raise the exception class <instance>.error("<reason>").
    IMAP4 server errors raise <instance>.abort("<reason>"),
    which is a sub-class of 'error'. Mailbox status changes
    from READ-WRITE to READ-ONLY raise the exception class
    <instance>.readonly("<reason>"), which is a sub-class of 'abort'.
110

Tim Peters's avatar
Tim Peters committed
111 112 113 114
    "error" exceptions imply a program error.
    "abort" exceptions imply the connection should be reset, and
            the command re-tried.
    "readonly" exceptions imply the command should be re-tried.
Guido van Rossum's avatar
Guido van Rossum committed
115

Tim Peters's avatar
Tim Peters committed
116 117 118 119 120
    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.
    """
121

Tim Peters's avatar
Tim Peters committed
122 123 124
    class error(Exception): pass    # Logical errors - debug required
    class abort(error): pass        # Service errors - close and retry
    class readonly(abort): pass     # Mailbox status changed to READ-ONLY
125

Tim Peters's avatar
Tim Peters committed
126
    mustquote = re.compile(r"[^\w!#$%&'*+,.:;<=>?^`|~-]")
127

Tim Peters's avatar
Tim Peters committed
128 129 130 131 132 133 134 135 136 137 138
    def __init__(self, host = '', port = IMAP4_PORT):
        self.host = host
        self.port = port
        self.debug = Debug
        self.state = 'LOGOUT'
        self.literal = None             # A literal argument to a command
        self.tagged_commands = {}       # Tagged commands awaiting response
        self.untagged_responses = {}    # {typ: [data, ...], ...}
        self.continuation_response = '' # Last continuation response
        self.is_readonly = None         # READ-ONLY desired state
        self.tagnum = 0
139

Tim Peters's avatar
Tim Peters committed
140
        # Open socket to server.
141

Tim Peters's avatar
Tim Peters committed
142
        self.open(host, port)
143

Tim Peters's avatar
Tim Peters committed
144 145
        # Create unique tag for this session,
        # and compile tagged response matcher.
146

Tim Peters's avatar
Tim Peters committed
147 148 149 150
        self.tagpre = Int2AP(random.randint(0, 31999))
        self.tagre = re.compile(r'(?P<tag>'
                        + self.tagpre
                        + r'\d+) (?P<type>[A-Z]+) (?P<data>.*)')
151

Tim Peters's avatar
Tim Peters committed
152 153
        # Get server welcome message,
        # request and store CAPABILITY response.
154

Tim Peters's avatar
Tim Peters committed
155 156 157
        if __debug__:
            if self.debug >= 1:
                _mesg('new IMAP4 connection, tag=%s' % self.tagpre)
158

Tim Peters's avatar
Tim Peters committed
159 160 161 162 163 164 165
        self.welcome = self._get_response()
        if self.untagged_responses.has_key('PREAUTH'):
            self.state = 'AUTH'
        elif self.untagged_responses.has_key('OK'):
            self.state = 'NONAUTH'
        else:
            raise self.error(self.welcome)
166

Tim Peters's avatar
Tim Peters committed
167 168 169 170
        cap = 'CAPABILITY'
        self._simple_command(cap)
        if not self.untagged_responses.has_key(cap):
            raise self.error('no CAPABILITY response from server')
171
        self.capabilities = tuple(self.untagged_responses[cap][-1].upper().split())
172

Tim Peters's avatar
Tim Peters committed
173 174 175
        if __debug__:
            if self.debug >= 3:
                _mesg('CAPABILITIES: %s' % `self.capabilities`)
176

Tim Peters's avatar
Tim Peters committed
177 178 179 180 181
        for version in AllowedVersions:
            if not version in self.capabilities:
                continue
            self.PROTOCOL_VERSION = version
            return
182

Tim Peters's avatar
Tim Peters committed
183
        raise self.error('server not IMAP4 compliant')
184

185

Tim Peters's avatar
Tim Peters committed
186 187 188
    def __getattr__(self, attr):
        #       Allow UPPERCASE variants of IMAP4 command methods.
        if Commands.has_key(attr):
189
            return eval("self.%s" % attr.lower())
Tim Peters's avatar
Tim Peters committed
190 191
        raise AttributeError("Unknown IMAP4 command: '%s'" % attr)

192 193


Tim Peters's avatar
Tim Peters committed
194
    #       Public methods
195 196


Tim Peters's avatar
Tim Peters committed
197 198 199 200 201
    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')
202 203


Tim Peters's avatar
Tim Peters committed
204 205 206
    def recent(self):
        """Return most recent 'RECENT' responses if any exist,
        else prompt server for an update using the 'NOOP' command.
207

Tim Peters's avatar
Tim Peters committed
208
        (typ, [data]) = <instance>.recent()
209

Tim Peters's avatar
Tim Peters committed
210 211 212 213 214 215 216 217 218
        'data' is None if no new messages,
        else list of RECENT responses, most recent last.
        """
        name = 'RECENT'
        typ, dat = self._untagged_response('OK', [None], name)
        if dat[-1]:
            return typ, dat
        typ, dat = self.noop()  # Prod server for response
        return self._untagged_response(typ, dat, name)
219 220


Tim Peters's avatar
Tim Peters committed
221 222
    def response(self, code):
        """Return data for response 'code' if received, or None.
223

Tim Peters's avatar
Tim Peters committed
224
        Old value for response 'code' is cleared.
225

Tim Peters's avatar
Tim Peters committed
226 227
        (code, [data]) = <instance>.response(code)
        """
228
        return self._untagged_response(code, [None], code.upper())
229 230


Tim Peters's avatar
Tim Peters committed
231 232
    def socket(self):
        """Return socket instance used to connect to IMAP4 server.
233

Tim Peters's avatar
Tim Peters committed
234 235 236
        socket = <instance>.socket()
        """
        return self.sock
237

238 239


Tim Peters's avatar
Tim Peters committed
240
    #       IMAP4 commands
241 242


Tim Peters's avatar
Tim Peters committed
243 244
    def append(self, mailbox, flags, date_time, message):
        """Append message to named mailbox.
245

Tim Peters's avatar
Tim Peters committed
246
        (typ, [data]) = <instance>.append(mailbox, flags, date_time, message)
Guido van Rossum's avatar
Guido van Rossum committed
247

Tim Peters's avatar
Tim Peters committed
248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263
                All args except `message' can be None.
        """
        name = 'APPEND'
        if not mailbox:
            mailbox = 'INBOX'
        if flags:
            if (flags[0],flags[-1]) != ('(',')'):
                flags = '(%s)' % flags
        else:
            flags = None
        if date_time:
            date_time = Time2Internaldate(date_time)
        else:
            date_time = None
        self.literal = message
        return self._simple_command(name, mailbox, flags, date_time)
264 265


Tim Peters's avatar
Tim Peters committed
266 267
    def authenticate(self, mechanism, authobject):
        """Authenticate command - requires response processing.
268

Tim Peters's avatar
Tim Peters committed
269 270 271
        'mechanism' specifies which authentication mechanism is to
        be used - it must appear in <instance>.capabilities in the
        form AUTH=<mechanism>.
272

Tim Peters's avatar
Tim Peters committed
273
        'authobject' must be a callable object:
274

Tim Peters's avatar
Tim Peters committed
275
                data = authobject(response)
276

Tim Peters's avatar
Tim Peters committed
277 278 279 280 281
        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.
        """
282
        mech = mechanism.upper()
Tim Peters's avatar
Tim Peters committed
283 284 285 286 287 288 289 290 291
        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[-1])
        self.state = 'AUTH'
        return typ, dat
292 293


Tim Peters's avatar
Tim Peters committed
294 295
    def check(self):
        """Checkpoint mailbox on server.
296

Tim Peters's avatar
Tim Peters committed
297 298 299
        (typ, [data]) = <instance>.check()
        """
        return self._simple_command('CHECK')
300 301


Tim Peters's avatar
Tim Peters committed
302 303
    def close(self):
        """Close currently selected mailbox.
304

Tim Peters's avatar
Tim Peters committed
305 306
        Deleted messages are removed from writable mailbox.
        This is the recommended command before 'LOGOUT'.
307

Tim Peters's avatar
Tim Peters committed
308 309 310 311 312 313 314
        (typ, [data]) = <instance>.close()
        """
        try:
            typ, dat = self._simple_command('CLOSE')
        finally:
            self.state = 'AUTH'
        return typ, dat
315 316


Tim Peters's avatar
Tim Peters committed
317 318
    def copy(self, message_set, new_mailbox):
        """Copy 'message_set' messages onto end of 'new_mailbox'.
319

Tim Peters's avatar
Tim Peters committed
320 321 322
        (typ, [data]) = <instance>.copy(message_set, new_mailbox)
        """
        return self._simple_command('COPY', message_set, new_mailbox)
323 324


Tim Peters's avatar
Tim Peters committed
325 326
    def create(self, mailbox):
        """Create new mailbox.
327

Tim Peters's avatar
Tim Peters committed
328 329 330
        (typ, [data]) = <instance>.create(mailbox)
        """
        return self._simple_command('CREATE', mailbox)
331 332


Tim Peters's avatar
Tim Peters committed
333 334
    def delete(self, mailbox):
        """Delete old mailbox.
335

Tim Peters's avatar
Tim Peters committed
336 337 338
        (typ, [data]) = <instance>.delete(mailbox)
        """
        return self._simple_command('DELETE', mailbox)
339 340


Tim Peters's avatar
Tim Peters committed
341 342
    def expunge(self):
        """Permanently remove deleted items from selected mailbox.
343

Tim Peters's avatar
Tim Peters committed
344
        Generates 'EXPUNGE' response for each deleted message.
345

Tim Peters's avatar
Tim Peters committed
346
        (typ, [data]) = <instance>.expunge()
347

Tim Peters's avatar
Tim Peters committed
348 349 350 351 352
        'data' is list of 'EXPUNGE'd message numbers in order received.
        """
        name = 'EXPUNGE'
        typ, dat = self._simple_command(name)
        return self._untagged_response(typ, dat, name)
353 354


Tim Peters's avatar
Tim Peters committed
355 356
    def fetch(self, message_set, message_parts):
        """Fetch (parts of) messages.
357

Tim Peters's avatar
Tim Peters committed
358
        (typ, [data, ...]) = <instance>.fetch(message_set, message_parts)
359

Tim Peters's avatar
Tim Peters committed
360 361
        'message_parts' should be a string of selected parts
        enclosed in parentheses, eg: "(UID BODY[TEXT])".
362

Tim Peters's avatar
Tim Peters committed
363 364 365 366 367
        'data' are tuples of message part envelope and data.
        """
        name = 'FETCH'
        typ, dat = self._simple_command(name, message_set, message_parts)
        return self._untagged_response(typ, dat, name)
368 369


Tim Peters's avatar
Tim Peters committed
370 371
    def list(self, directory='""', pattern='*'):
        """List mailbox names in directory matching pattern.
372

Tim Peters's avatar
Tim Peters committed
373
        (typ, [data]) = <instance>.list(directory='""', pattern='*')
374

Tim Peters's avatar
Tim Peters committed
375 376 377 378 379
        'data' is list of LIST responses.
        """
        name = 'LIST'
        typ, dat = self._simple_command(name, directory, pattern)
        return self._untagged_response(typ, dat, name)
380 381


Tim Peters's avatar
Tim Peters committed
382 383
    def login(self, user, password):
        """Identify client using plaintext password.
384

Tim Peters's avatar
Tim Peters committed
385
        (typ, [data]) = <instance>.login(user, password)
Guido van Rossum's avatar
Guido van Rossum committed
386

Tim Peters's avatar
Tim Peters committed
387 388 389 390 391 392 393 394 395
        NB: 'password' will be quoted.
        """
        #if not 'AUTH=LOGIN' in self.capabilities:
        #       raise self.error("Server doesn't allow LOGIN authentication." % mech)
        typ, dat = self._simple_command('LOGIN', user, self._quote(password))
        if typ != 'OK':
            raise self.error(dat[-1])
        self.state = 'AUTH'
        return typ, dat
396 397


Tim Peters's avatar
Tim Peters committed
398 399
    def logout(self):
        """Shutdown connection to server.
400

Tim Peters's avatar
Tim Peters committed
401
        (typ, [data]) = <instance>.logout()
402

Tim Peters's avatar
Tim Peters committed
403 404 405 406 407 408 409 410 411 412
        Returns server 'BYE' response.
        """
        self.state = 'LOGOUT'
        try: typ, dat = self._simple_command('LOGOUT')
        except: typ, dat = 'NO', ['%s: %s' % sys.exc_info()[:2]]
        self.file.close()
        self.sock.close()
        if self.untagged_responses.has_key('BYE'):
            return 'BYE', self.untagged_responses['BYE']
        return typ, dat
413 414


Tim Peters's avatar
Tim Peters committed
415 416
    def lsub(self, directory='""', pattern='*'):
        """List 'subscribed' mailbox names in directory matching pattern.
417

Tim Peters's avatar
Tim Peters committed
418
        (typ, [data, ...]) = <instance>.lsub(directory='""', pattern='*')
419

Tim Peters's avatar
Tim Peters committed
420 421 422 423 424
        'data' are tuples of message part envelope and data.
        """
        name = 'LSUB'
        typ, dat = self._simple_command(name, directory, pattern)
        return self._untagged_response(typ, dat, name)
425 426


Tim Peters's avatar
Tim Peters committed
427 428
    def noop(self):
        """Send NOOP command.
429

Tim Peters's avatar
Tim Peters committed
430 431 432 433 434 435
        (typ, data) = <instance>.noop()
        """
        if __debug__:
            if self.debug >= 3:
                _dump_ur(self.untagged_responses)
        return self._simple_command('NOOP')
436 437


Tim Peters's avatar
Tim Peters committed
438 439
    def partial(self, message_num, message_part, start, length):
        """Fetch truncated part of a message.
440

Tim Peters's avatar
Tim Peters committed
441
        (typ, [data, ...]) = <instance>.partial(message_num, message_part, start, length)
442

Tim Peters's avatar
Tim Peters committed
443 444 445 446 447
        '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, dat, 'FETCH')
448 449


Tim Peters's avatar
Tim Peters committed
450 451
    def rename(self, oldmailbox, newmailbox):
        """Rename old mailbox name to new.
452

Tim Peters's avatar
Tim Peters committed
453 454 455
        (typ, data) = <instance>.rename(oldmailbox, newmailbox)
        """
        return self._simple_command('RENAME', oldmailbox, newmailbox)
456 457


Tim Peters's avatar
Tim Peters committed
458 459
    def search(self, charset, *criteria):
        """Search mailbox for matching messages.
460

Tim Peters's avatar
Tim Peters committed
461
        (typ, [data]) = <instance>.search(charset, criterium, ...)
462

Tim Peters's avatar
Tim Peters committed
463 464 465 466 467 468 469
        'data' is space separated list of matching message numbers.
        """
        name = 'SEARCH'
        if charset:
            charset = 'CHARSET ' + charset
        typ, dat = apply(self._simple_command, (name, charset) + criteria)
        return self._untagged_response(typ, dat, name)
470 471


Tim Peters's avatar
Tim Peters committed
472 473
    def select(self, mailbox='INBOX', readonly=None):
        """Select a mailbox.
474

Tim Peters's avatar
Tim Peters committed
475
        Flush all untagged responses.
476

Tim Peters's avatar
Tim Peters committed
477
        (typ, [data]) = <instance>.select(mailbox='INBOX', readonly=None)
478

Tim Peters's avatar
Tim Peters committed
479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499
        'data' is count of messages in mailbox ('EXISTS' response).
        """
        # Mandated responses are ('FLAGS', 'EXISTS', 'RECENT', 'UIDVALIDITY')
        self.untagged_responses = {}    # Flush old responses.
        self.is_readonly = readonly
        if readonly:
            name = 'EXAMINE'
        else:
            name = 'SELECT'
        typ, dat = self._simple_command(name, mailbox)
        if typ != 'OK':
            self.state = 'AUTH'     # Might have been 'SELECTED'
            return typ, dat
        self.state = 'SELECTED'
        if self.untagged_responses.has_key('READ-ONLY') \
                and not readonly:
            if __debug__:
                if self.debug >= 1:
                    _dump_ur(self.untagged_responses)
            raise self.readonly('%s is not writable' % mailbox)
        return typ, self.untagged_responses.get('EXISTS', [None])
500 501


Tim Peters's avatar
Tim Peters committed
502 503
    def status(self, mailbox, names):
        """Request named status conditions for mailbox.
504

Tim Peters's avatar
Tim Peters committed
505 506 507 508 509 510 511
        (typ, [data]) = <instance>.status(mailbox, names)
        """
        name = 'STATUS'
        if self.PROTOCOL_VERSION == 'IMAP4':
            raise self.error('%s unimplemented in IMAP4 (obtain IMAP4rev1 server, or re-code)' % name)
        typ, dat = self._simple_command(name, mailbox, names)
        return self._untagged_response(typ, dat, name)
512 513


Tim Peters's avatar
Tim Peters committed
514 515
    def store(self, message_set, command, flags):
        """Alters flag dispositions for messages in mailbox.
516

Tim Peters's avatar
Tim Peters committed
517 518 519 520 521 522
        (typ, [data]) = <instance>.store(message_set, command, flags)
        """
        if (flags[0],flags[-1]) != ('(',')'):
            flags = '(%s)' % flags  # Avoid quoting the flags
        typ, dat = self._simple_command('STORE', message_set, command, flags)
        return self._untagged_response(typ, dat, 'FETCH')
523 524


Tim Peters's avatar
Tim Peters committed
525 526
    def subscribe(self, mailbox):
        """Subscribe to new mailbox.
527

Tim Peters's avatar
Tim Peters committed
528 529 530
        (typ, [data]) = <instance>.subscribe(mailbox)
        """
        return self._simple_command('SUBSCRIBE', mailbox)
531 532


Tim Peters's avatar
Tim Peters committed
533 534 535
    def uid(self, command, *args):
        """Execute "command arg ..." with messages identified by UID,
                rather than message number.
536

Tim Peters's avatar
Tim Peters committed
537
        (typ, [data]) = <instance>.uid(command, arg1, arg2, ...)
538

Tim Peters's avatar
Tim Peters committed
539 540
        Returns response appropriate to 'command'.
        """
541
        command = command.upper()
Tim Peters's avatar
Tim Peters committed
542 543 544 545 546 547 548 549 550 551 552 553
        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, (name, command) + args)
        if command == 'SEARCH':
            name = 'SEARCH'
        else:
            name = 'FETCH'
        return self._untagged_response(typ, dat, name)
554 555


Tim Peters's avatar
Tim Peters committed
556 557
    def unsubscribe(self, mailbox):
        """Unsubscribe from old mailbox.
558

Tim Peters's avatar
Tim Peters committed
559 560 561 562 563 564 565 566
        (typ, [data]) = <instance>.unsubscribe(mailbox)
        """
        return self._simple_command('UNSUBSCRIBE', mailbox)


    def xatom(self, name, *args):
        """Allow simple extension commands
                notified by server in CAPABILITY response.
567

Tim Peters's avatar
Tim Peters committed
568 569 570 571 572
        (typ, [data]) = <instance>.xatom(name, arg, ...)
        """
        if name[0] != 'X' or not name in self.capabilities:
            raise self.error('unknown extension command: %s' % name)
        return apply(self._simple_command, (name,) + args)
573 574


Tim Peters's avatar
Tim Peters committed
575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590

    #       Private methods


    def _append_untagged(self, typ, dat):

        if dat is None: dat = ''
        ur = self.untagged_responses
        if __debug__:
            if self.debug >= 5:
                _mesg('untagged_responses[%s] %s += ["%s"]' %
                        (typ, len(ur.get(typ,'')), dat))
        if ur.has_key(typ):
            ur[typ].append(dat)
        else:
            ur[typ] = [dat]
591 592


Tim Peters's avatar
Tim Peters committed
593 594 595 596
    def _check_bye(self):
        bye = self.untagged_responses.get('BYE')
        if bye:
            raise self.abort(bye[-1])
Guido van Rossum's avatar
Guido van Rossum committed
597 598


Tim Peters's avatar
Tim Peters committed
599
    def _command(self, name, *args):
600

Tim Peters's avatar
Tim Peters committed
601 602 603 604
        if self.state not in Commands[name]:
            self.literal = None
            raise self.error(
            'command %s illegal in state %s' % (name, self.state))
605

Tim Peters's avatar
Tim Peters committed
606 607 608
        for typ in ('OK', 'NO', 'BAD'):
            if self.untagged_responses.has_key(typ):
                del self.untagged_responses[typ]
609

Tim Peters's avatar
Tim Peters committed
610 611 612
        if self.untagged_responses.has_key('READ-ONLY') \
        and not self.is_readonly:
            raise self.readonly('mailbox status changed to READ-ONLY')
613

Tim Peters's avatar
Tim Peters committed
614 615 616 617 618
        tag = self._new_tag()
        data = '%s %s' % (tag, name)
        for arg in args:
            if arg is None: continue
            data = '%s %s' % (data, self._checkquote(arg))
619

Tim Peters's avatar
Tim Peters committed
620 621 622 623 624 625 626 627
        literal = self.literal
        if literal is not None:
            self.literal = None
            if type(literal) is type(self._command):
                literator = literal
            else:
                literator = None
                data = '%s {%s}' % (data, len(literal))
628

Tim Peters's avatar
Tim Peters committed
629 630 631 632 633
        if __debug__:
            if self.debug >= 4:
                _mesg('> %s' % data)
            else:
                _log('> %s' % data)
Guido van Rossum's avatar
Guido van Rossum committed
634

Tim Peters's avatar
Tim Peters committed
635 636 637 638
        try:
            self.sock.send('%s%s' % (data, CRLF))
        except socket.error, val:
            raise self.abort('socket error: %s' % val)
639

Tim Peters's avatar
Tim Peters committed
640 641
        if literal is None:
            return tag
642

Tim Peters's avatar
Tim Peters committed
643 644
        while 1:
            # Wait for continuation response
645

Tim Peters's avatar
Tim Peters committed
646 647 648
            while self._get_response():
                if self.tagged_commands[tag]:   # BAD/NO?
                    return tag
649

Tim Peters's avatar
Tim Peters committed
650
            # Send literal
651

Tim Peters's avatar
Tim Peters committed
652 653
            if literator:
                literal = literator(self.continuation_response)
654

Tim Peters's avatar
Tim Peters committed
655 656 657
            if __debug__:
                if self.debug >= 4:
                    _mesg('write literal size %s' % len(literal))
658

Tim Peters's avatar
Tim Peters committed
659 660 661 662 663
            try:
                self.sock.send(literal)
                self.sock.send(CRLF)
            except socket.error, val:
                raise self.abort('socket error: %s' % val)
664

Tim Peters's avatar
Tim Peters committed
665 666
            if not literator:
                break
667

Tim Peters's avatar
Tim Peters committed
668
        return tag
669 670


Tim Peters's avatar
Tim Peters committed
671 672 673 674 675 676 677 678 679 680 681 682
    def _command_complete(self, name, tag):
        self._check_bye()
        try:
            typ, data = self._get_tagged_response(tag)
        except self.abort, val:
            raise self.abort('command: %s => %s' % (name, val))
        except self.error, val:
            raise self.error('command: %s => %s' % (name, val))
        self._check_bye()
        if typ == 'BAD':
            raise self.error('%s command error: %s %s' % (name, typ, data))
        return typ, data
683 684


Tim Peters's avatar
Tim Peters committed
685
    def _get_response(self):
686

Tim Peters's avatar
Tim Peters committed
687 688 689 690
        # Read response and store.
        #
        # Returns None for continuation responses,
        # otherwise first response line received.
691

Tim Peters's avatar
Tim Peters committed
692
        resp = self._get_line()
693

Tim Peters's avatar
Tim Peters committed
694
        # Command completion response?
695

Tim Peters's avatar
Tim Peters committed
696 697 698 699
        if self._match(self.tagre, resp):
            tag = self.mo.group('tag')
            if not self.tagged_commands.has_key(tag):
                raise self.abort('unexpected tagged response: %s' % resp)
700

Tim Peters's avatar
Tim Peters committed
701 702 703 704 705
            typ = self.mo.group('type')
            dat = self.mo.group('data')
            self.tagged_commands[tag] = (typ, [dat])
        else:
            dat2 = None
706

Tim Peters's avatar
Tim Peters committed
707
            # '*' (untagged) responses?
708

Tim Peters's avatar
Tim Peters committed
709 710 711
            if not self._match(Untagged_response, resp):
                if self._match(Untagged_status, resp):
                    dat2 = self.mo.group('data2')
712

Tim Peters's avatar
Tim Peters committed
713 714
            if self.mo is None:
                # Only other possibility is '+' (continuation) response...
715

Tim Peters's avatar
Tim Peters committed
716 717 718
                if self._match(Continuation, resp):
                    self.continuation_response = self.mo.group('data')
                    return None     # NB: indicates continuation
719

Tim Peters's avatar
Tim Peters committed
720
                raise self.abort("unexpected response: '%s'" % resp)
721

Tim Peters's avatar
Tim Peters committed
722 723 724 725
            typ = self.mo.group('type')
            dat = self.mo.group('data')
            if dat is None: dat = ''        # Null untagged response
            if dat2: dat = dat + ' ' + dat2
726

Tim Peters's avatar
Tim Peters committed
727
            # Is there a literal to come?
728

Tim Peters's avatar
Tim Peters committed
729
            while self._match(Literal, dat):
730

Tim Peters's avatar
Tim Peters committed
731
                # Read literal direct from connection.
732

733
                size = int(self.mo.group('size'))
Tim Peters's avatar
Tim Peters committed
734 735 736 737
                if __debug__:
                    if self.debug >= 4:
                        _mesg('read literal size %s' % size)
                data = self.file.read(size)
738

Tim Peters's avatar
Tim Peters committed
739
                # Store response with literal as tuple
740

Tim Peters's avatar
Tim Peters committed
741
                self._append_untagged(typ, (dat, data))
742

Tim Peters's avatar
Tim Peters committed
743
                # Read trailer - possibly containing another literal
744

Tim Peters's avatar
Tim Peters committed
745
                dat = self._get_line()
746

Tim Peters's avatar
Tim Peters committed
747
            self._append_untagged(typ, dat)
748

Tim Peters's avatar
Tim Peters committed
749
        # Bracketed response information?
750

Tim Peters's avatar
Tim Peters committed
751 752
        if typ in ('OK', 'NO', 'BAD') and self._match(Response_code, dat):
            self._append_untagged(self.mo.group('type'), self.mo.group('data'))
753

Tim Peters's avatar
Tim Peters committed
754 755 756
        if __debug__:
            if self.debug >= 1 and typ in ('NO', 'BAD', 'BYE'):
                _mesg('%s response: %s' % (typ, dat))
757

Tim Peters's avatar
Tim Peters committed
758
        return resp
759 760


Tim Peters's avatar
Tim Peters committed
761
    def _get_tagged_response(self, tag):
762

Tim Peters's avatar
Tim Peters committed
763 764 765 766 767
        while 1:
            result = self.tagged_commands[tag]
            if result is not None:
                del self.tagged_commands[tag]
                return result
768

Tim Peters's avatar
Tim Peters committed
769 770 771 772
            # Some have reported "unexpected response" exceptions.
            # Note that ignoring them here causes loops.
            # Instead, send me details of the unexpected response and
            # I'll update the code in `_get_response()'.
773

Tim Peters's avatar
Tim Peters committed
774 775 776 777 778 779 780
            try:
                self._get_response()
            except self.abort, val:
                if __debug__:
                    if self.debug >= 1:
                        print_log()
                raise
781 782


Tim Peters's avatar
Tim Peters committed
783
    def _get_line(self):
784

Tim Peters's avatar
Tim Peters committed
785 786 787
        line = self.file.readline()
        if not line:
            raise self.abort('socket error: EOF')
788

Tim Peters's avatar
Tim Peters committed
789
        # Protocol mandates all lines terminated by CRLF
790

Tim Peters's avatar
Tim Peters committed
791 792 793 794 795 796 797
        line = line[:-2]
        if __debug__:
            if self.debug >= 4:
                _mesg('< %s' % line)
            else:
                _log('< %s' % line)
        return line
798 799


Tim Peters's avatar
Tim Peters committed
800
    def _match(self, cre, s):
801

Tim Peters's avatar
Tim Peters committed
802 803
        # Run compiled regular expression match method on 's'.
        # Save result, return success.
804

Tim Peters's avatar
Tim Peters committed
805 806 807 808 809
        self.mo = cre.match(s)
        if __debug__:
            if self.mo is not None and self.debug >= 5:
                _mesg("\tmatched r'%s' => %s" % (cre.pattern, `self.mo.groups()`))
        return self.mo is not None
810 811


Tim Peters's avatar
Tim Peters committed
812
    def _new_tag(self):
813

Tim Peters's avatar
Tim Peters committed
814 815 816 817
        tag = '%s%s' % (self.tagpre, self.tagnum)
        self.tagnum = self.tagnum + 1
        self.tagged_commands[tag] = None
        return tag
818 819


Tim Peters's avatar
Tim Peters committed
820
    def _checkquote(self, arg):
Guido van Rossum's avatar
Guido van Rossum committed
821

Tim Peters's avatar
Tim Peters committed
822 823
        # Must quote command args if non-alphanumeric chars present,
        # and not already quoted.
Guido van Rossum's avatar
Guido van Rossum committed
824

Tim Peters's avatar
Tim Peters committed
825 826 827 828 829 830 831
        if type(arg) is not type(''):
            return arg
        if (arg[0],arg[-1]) in (('(',')'),('"','"')):
            return arg
        if self.mustquote.search(arg) is None:
            return arg
        return self._quote(arg)
Guido van Rossum's avatar
Guido van Rossum committed
832 833


Tim Peters's avatar
Tim Peters committed
834
    def _quote(self, arg):
Guido van Rossum's avatar
Guido van Rossum committed
835

836 837
        arg = arg.replace('\\', '\\\\')
        arg = arg.replace('"', '\\"')
Guido van Rossum's avatar
Guido van Rossum committed
838

Tim Peters's avatar
Tim Peters committed
839
        return '"%s"' % arg
Guido van Rossum's avatar
Guido van Rossum committed
840 841


Tim Peters's avatar
Tim Peters committed
842
    def _simple_command(self, name, *args):
843

Tim Peters's avatar
Tim Peters committed
844
        return self._command_complete(name, apply(self._command, (name,) + args))
845 846


Tim Peters's avatar
Tim Peters committed
847
    def _untagged_response(self, typ, dat, name):
848

Tim Peters's avatar
Tim Peters committed
849 850 851 852 853 854 855 856 857 858
        if typ == 'NO':
            return typ, dat
        if not self.untagged_responses.has_key(name):
            return typ, [None]
        data = self.untagged_responses[name]
        if __debug__:
            if self.debug >= 5:
                _mesg('untagged_responses[%s] => %s' % (name, data))
        del self.untagged_responses[name]
        return typ, data
859 860 861



862 863
class _Authenticator:

Tim Peters's avatar
Tim Peters committed
864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903
    """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)

904 905


906
Mon2num = {'Jan': 1, 'Feb': 2, 'Mar': 3, 'Apr': 4, 'May': 5, 'Jun': 6,
Tim Peters's avatar
Tim Peters committed
907
        'Jul': 7, 'Aug': 8, 'Sep': 9, 'Oct': 10, 'Nov': 11, 'Dec': 12}
908 909

def Internaldate2tuple(resp):
Tim Peters's avatar
Tim Peters committed
910
    """Convert IMAP4 INTERNALDATE to UT.
911

Tim Peters's avatar
Tim Peters committed
912 913
    Returns Python time module tuple.
    """
914

Tim Peters's avatar
Tim Peters committed
915 916 917
    mo = InternalDate.match(resp)
    if not mo:
        return None
918

Tim Peters's avatar
Tim Peters committed
919 920
    mon = Mon2num[mo.group('mon')]
    zonen = mo.group('zonen')
921

922 923 924 925 926 927 928
    day = int(mo.group('day'))
    year = int(mo.group('year'))
    hour = int(mo.group('hour'))
    min = int(mo.group('min'))
    sec = int(mo.group('sec'))
    zoneh = int(mo.group('zoneh'))
    zonem = int(mo.group('zonem'))
929

Tim Peters's avatar
Tim Peters committed
930
    # INTERNALDATE timezone must be subtracted to get UT
931

Tim Peters's avatar
Tim Peters committed
932 933 934
    zone = (zoneh*60 + zonem)*60
    if zonen == '-':
        zone = -zone
935

Tim Peters's avatar
Tim Peters committed
936
    tt = (year, mon, day, hour, min, sec, -1, -1, -1)
937

Tim Peters's avatar
Tim Peters committed
938
    utc = time.mktime(tt)
939

Tim Peters's avatar
Tim Peters committed
940 941
    # Following is necessary because the time module has no 'mkgmtime'.
    # 'mktime' assumes arg in local timezone, so adds timezone/altzone.
942

Tim Peters's avatar
Tim Peters committed
943 944 945 946 947
    lt = time.localtime(utc)
    if time.daylight and lt[-1]:
        zone = zone + time.altzone
    else:
        zone = zone + time.timezone
948

Tim Peters's avatar
Tim Peters committed
949
    return time.localtime(utc - zone)
950 951 952 953 954



def Int2AP(num):

Tim Peters's avatar
Tim Peters committed
955
    """Convert integer to A-P string representation."""
956

Tim Peters's avatar
Tim Peters committed
957 958 959 960 961 962
    val = ''; AP = 'ABCDEFGHIJKLMNOP'
    num = int(abs(num))
    while num:
        num, mod = divmod(num, 16)
        val = AP[mod] + val
    return val
963 964 965 966 967



def ParseFlags(resp):

Tim Peters's avatar
Tim Peters committed
968
    """Convert IMAP4 flags response to python tuple."""
969

Tim Peters's avatar
Tim Peters committed
970 971 972
    mo = Flags.match(resp)
    if not mo:
        return ()
973

974
    return tuple(mo.group('flags').split())
975 976 977 978


def Time2Internaldate(date_time):

Tim Peters's avatar
Tim Peters committed
979
    """Convert 'date_time' to IMAP4 INTERNALDATE representation.
980

Tim Peters's avatar
Tim Peters committed
981 982
    Return string in form: '"DD-Mmm-YYYY HH:MM:SS +HHMM"'
    """
983

Tim Peters's avatar
Tim Peters committed
984 985 986 987 988 989 990 991
    dttype = type(date_time)
    if dttype is type(1) or dttype is type(1.1):
        tt = time.localtime(date_time)
    elif dttype is type(()):
        tt = date_time
    elif dttype is type(""):
        return date_time        # Assume in correct format
    else: raise ValueError
992

Tim Peters's avatar
Tim Peters committed
993 994 995 996 997 998 999 1000
    dt = time.strftime("%d-%b-%Y %H:%M:%S", tt)
    if dt[0] == '0':
        dt = ' ' + dt[1:]
    if time.daylight and tt[-1]:
        zone = -time.altzone
    else:
        zone = -time.timezone
    return '"' + dt + " %+02d%02d" % divmod(zone/60, 60) + '"'
1001 1002 1003



1004 1005
if __debug__:

Tim Peters's avatar
Tim Peters committed
1006 1007 1008 1009 1010 1011
    def _mesg(s, secs=None):
        if secs is None:
            secs = time.time()
        tm = time.strftime('%M:%S', time.localtime(secs))
        sys.stderr.write('  %s.%02d %s\n' % (tm, (secs*100)%100, s))
        sys.stderr.flush()
1012

Tim Peters's avatar
Tim Peters committed
1013 1014 1015 1016 1017
    def _dump_ur(dict):
        # Dump untagged responses (in `dict').
        l = dict.items()
        if not l: return
        t = '\n\t\t'
1018
        l = map(lambda x:'%s: "%s"' % (x[0], x[1][0] and '" "'.join(x[1]) or ''), l)
Tim Peters's avatar
Tim Peters committed
1019
        _mesg('untagged responses dump:%s%s' % (t, j(l, t)))
1020

Tim Peters's avatar
Tim Peters committed
1021 1022
    _cmd_log = []           # Last `_cmd_log_len' interactions
    _cmd_log_len = 10
Guido van Rossum's avatar
Guido van Rossum committed
1023

Tim Peters's avatar
Tim Peters committed
1024 1025 1026 1027 1028
    def _log(line):
        # Keep log of last `_cmd_log_len' interactions for debugging.
        if len(_cmd_log) == _cmd_log_len:
            del _cmd_log[0]
        _cmd_log.append((time.time(), line))
Guido van Rossum's avatar
Guido van Rossum committed
1029

Tim Peters's avatar
Tim Peters committed
1030 1031 1032 1033
    def print_log():
        _mesg('last %d IMAP4 interactions:' % len(_cmd_log))
        for secs,line in _cmd_log:
            _mesg(line, secs)
Guido van Rossum's avatar
Guido van Rossum committed
1034

1035 1036


Guido van Rossum's avatar
Guido van Rossum committed
1037
if __name__ == '__main__':
1038

Tim Peters's avatar
Tim Peters committed
1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054
    import getopt, getpass, sys

    try:
        optlist, args = getopt.getopt(sys.argv[1:], 'd:')
    except getopt.error, val:
        pass

    for opt,val in optlist:
        if opt == '-d':
            Debug = int(val)

    if not args: args = ('',)

    host = args[0]

    USER = getpass.getuser()
1055
    PASSWD = getpass.getpass("IMAP password for %s on %s:" % (USER, host or "localhost"))
Tim Peters's avatar
Tim Peters committed
1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099

    test_mesg = 'From: %s@localhost\nSubject: IMAP4 test\n\ndata...\n' % USER
    test_seq1 = (
    ('login', (USER, PASSWD)),
    ('create', ('/tmp/xxx 1',)),
    ('rename', ('/tmp/xxx 1', '/tmp/yyy')),
    ('CREATE', ('/tmp/yyz 2',)),
    ('append', ('/tmp/yyz 2', None, None, test_mesg)),
    ('list', ('/tmp', 'yy*')),
    ('select', ('/tmp/yyz 2',)),
    ('search', (None, 'SUBJECT', 'test')),
    ('partial', ('1', 'RFC822', 1, 1024)),
    ('store', ('1', 'FLAGS', '(\Deleted)')),
    ('expunge', ()),
    ('recent', ()),
    ('close', ()),
    )

    test_seq2 = (
    ('select', ()),
    ('response',('UIDVALIDITY',)),
    ('uid', ('SEARCH', 'ALL')),
    ('response', ('EXISTS',)),
    ('append', (None, None, None, test_mesg)),
    ('recent', ()),
    ('logout', ()),
    )

    def run(cmd, args):
        _mesg('%s %s' % (cmd, args))
        typ, dat = apply(eval('M.%s' % cmd), args)
        _mesg('%s => %s %s' % (cmd, typ, dat))
        return dat

    try:
        M = IMAP4(host)
        _mesg('PROTOCOL_VERSION = %s' % M.PROTOCOL_VERSION)

        for cmd,args in test_seq1:
            run(cmd, args)

        for ml in run('list', ('/tmp/', 'yy%')):
            mo = re.match(r'.*"([^"]+)"$', ml)
            if mo: path = mo.group(1)
1100
            else: path = ml.split()[-1]
Tim Peters's avatar
Tim Peters committed
1101 1102 1103 1104 1105 1106 1107 1108
            run('delete', (path,))

        for cmd,args in test_seq2:
            dat = run(cmd, args)

            if (cmd,args) != ('uid', ('SEARCH', 'ALL')):
                continue

1109
            uid = dat[-1].split()
Tim Peters's avatar
Tim Peters committed
1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120
            if not uid: continue
            run('uid', ('FETCH', '%s' % uid[-1],
                    '(FLAGS INTERNALDATE RFC822.SIZE RFC822.HEADER RFC822.TEXT)'))

        print '\nAll tests OK.'

    except:
        print '\nTests failed.'

        if not Debug:
            print '''
Guido van Rossum's avatar
Guido van Rossum committed
1121 1122 1123
If you would like to see debugging output,
try: %s -d5
''' % sys.argv[0]
1124

Tim Peters's avatar
Tim Peters committed
1125
        raise