mboxconvert.py 3.13 KB
Newer Older
1
#! /usr/bin/env python3
Guido van Rossum's avatar
Guido van Rossum committed
2 3 4 5 6 7 8 9 10 11 12

# Convert  MH directories (1 message per file) or MMDF mailboxes (4x^A
# delimited) to unix mailbox (From ... delimited) on stdout.
# If -f is given, files contain one message per file (e.g. MH messages)

import rfc822
import sys
import time
import os
import stat
import getopt
13
import re
Guido van Rossum's avatar
Guido van Rossum committed
14 15

def main():
16 17 18
    dofile = mmdf
    try:
        opts, args = getopt.getopt(sys.argv[1:], 'f')
19
    except getopt.error as msg:
20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
        sys.stderr.write('%s\n' % msg)
        sys.exit(2)
    for o, a in opts:
        if o == '-f':
            dofile = message
    if not args:
        args = ['-']
    sts = 0
    for arg in args:
        if arg == '-' or arg == '':
            sts = dofile(sys.stdin) or sts
        elif os.path.isdir(arg):
            sts = mh(arg) or sts
        elif os.path.isfile(arg):
            try:
                f = open(arg)
36
            except IOError as msg:
37 38 39 40 41 42 43 44 45 46
                sys.stderr.write('%s: %s\n' % (arg, msg))
                sts = 1
                continue
            sts = dofile(f) or sts
            f.close()
        else:
            sys.stderr.write('%s: not found\n' % arg)
            sts = 1
    if sts:
        sys.exit(sts)
Guido van Rossum's avatar
Guido van Rossum committed
47

48
numeric = re.compile('[1-9][0-9]*')
Guido van Rossum's avatar
Guido van Rossum committed
49 50

def mh(dir):
51 52 53 54 55 56 57 58
    sts = 0
    msgs = os.listdir(dir)
    for msg in msgs:
        if numeric.match(msg) != len(msg):
            continue
        fn = os.path.join(dir, msg)
        try:
            f = open(fn)
59
        except IOError as msg:
60 61 62 63 64
            sys.stderr.write('%s: %s\n' % (fn, msg))
            sts = 1
            continue
        sts = message(f) or sts
    return sts
Guido van Rossum's avatar
Guido van Rossum committed
65 66

def mmdf(f):
67 68 69 70 71 72 73 74 75 76 77
    sts = 0
    while 1:
        line = f.readline()
        if not line:
            break
        if line == '\1\1\1\1\n':
            sts = message(f, line) or sts
        else:
            sys.stderr.write(
                    'Bad line in MMFD mailbox: %r\n' % (line,))
    return sts
Guido van Rossum's avatar
Guido van Rossum committed
78

79 80
counter = 0 # for generating unique Message-ID headers

Guido van Rossum's avatar
Guido van Rossum committed
81
def message(f, delimiter = ''):
82 83 84 85 86 87 88 89 90 91
    sts = 0
    # Parse RFC822 header
    m = rfc822.Message(f)
    # Write unix header line
    fullname, email = m.getaddr('From')
    tt = m.getdate('Date')
    if tt:
        t = time.mktime(tt)
    else:
        sys.stderr.write(
92
                'Unparseable date: %r\n' % (m.get('Date'),))
93
        t = os.fstat(f.fileno())[stat.ST_MTIME]
94
    print('From', email, time.ctime(t))
95 96
    # Copy RFC822 header
    for line in m.headers:
97
        print(line, end=' ')
98
    # Invent Message-ID header if none is present
99
    if 'message-id' not in m:
100 101 102 103 104
        global counter
        counter = counter + 1
        msgid = "<%s.%d>" % (hex(t), counter)
        sys.stderr.write("Adding Message-ID %s (From %s)\n" %
                         (msgid, email))
105 106
        print("Message-ID:", msgid)
    print()
107 108 109 110 111 112 113 114 115 116 117
    # Copy body
    while 1:
        line = f.readline()
        if line == delimiter:
            break
        if not line:
            sys.stderr.write('Unexpected EOF in message\n')
            sts = 1
            break
        if line[:5] == 'From ':
            line = '>' + line
118
        print(line, end=' ')
119
    # Print trailing newline
120
    print()
121
    return sts
Guido van Rossum's avatar
Guido van Rossum committed
122

123 124
if __name__ == "__main__":
    main()