1"""Mailcap file handling.  See RFC 1524."""
2
3import os
4import warnings
5
6__all__ = ["getcaps","findmatch"]
7
8
9def lineno_sort_key(entry):
10    # Sort in ascending order, with unspecified entries at the end
11    if 'lineno' in entry:
12        return 0, entry['lineno']
13    else:
14        return 1, 0
15
16
17# Part 1: top-level interface.
18
19def getcaps():
20    """Return a dictionary containing the mailcap database.
21
22    The dictionary maps a MIME type (in all lowercase, e.g. 'text/plain')
23    to a list of dictionaries corresponding to mailcap entries.  The list
24    collects all the entries for that MIME type from all available mailcap
25    files.  Each dictionary contains key-value pairs for that MIME type,
26    where the viewing command is stored with the key "view".
27
28    """
29    caps = {}
30    lineno = 0
31    for mailcap in listmailcapfiles():
32        try:
33            fp = open(mailcap, 'r')
34        except OSError:
35            continue
36        with fp:
37            morecaps, lineno = _readmailcapfile(fp, lineno)
38        for key, value in morecaps.items():
39            if not key in caps:
40                caps[key] = value
41            else:
42                caps[key] = caps[key] + value
43    return caps
44
45def listmailcapfiles():
46    """Return a list of all mailcap files found on the system."""
47    # This is mostly a Unix thing, but we use the OS path separator anyway
48    if 'MAILCAPS' in os.environ:
49        pathstr = os.environ['MAILCAPS']
50        mailcaps = pathstr.split(os.pathsep)
51    else:
52        if 'HOME' in os.environ:
53            home = os.environ['HOME']
54        else:
55            # Don't bother with getpwuid()
56            home = '.' # Last resort
57        mailcaps = [home + '/.mailcap', '/etc/mailcap',
58                '/usr/etc/mailcap', '/usr/local/etc/mailcap']
59    return mailcaps
60
61
62# Part 2: the parser.
63def readmailcapfile(fp):
64    """Read a mailcap file and return a dictionary keyed by MIME type."""
65    warnings.warn('readmailcapfile is deprecated, use getcaps instead',
66                  DeprecationWarning, 2)
67    caps, _ = _readmailcapfile(fp, None)
68    return caps
69
70
71def _readmailcapfile(fp, lineno):
72    """Read a mailcap file and return a dictionary keyed by MIME type.
73
74    Each MIME type is mapped to an entry consisting of a list of
75    dictionaries; the list will contain more than one such dictionary
76    if a given MIME type appears more than once in the mailcap file.
77    Each dictionary contains key-value pairs for that MIME type, where
78    the viewing command is stored with the key "view".
79    """
80    caps = {}
81    while 1:
82        line = fp.readline()
83        if not line: break
84        # Ignore comments and blank lines
85        if line[0] == '#' or line.strip() == '':
86            continue
87        nextline = line
88        # Join continuation lines
89        while nextline[-2:] == '\\\n':
90            nextline = fp.readline()
91            if not nextline: nextline = '\n'
92            line = line[:-2] + nextline
93        # Parse the line
94        key, fields = parseline(line)
95        if not (key and fields):
96            continue
97        if lineno is not None:
98            fields['lineno'] = lineno
99            lineno += 1
100        # Normalize the key
101        types = key.split('/')
102        for j in range(len(types)):
103            types[j] = types[j].strip()
104        key = '/'.join(types).lower()
105        # Update the database
106        if key in caps:
107            caps[key].append(fields)
108        else:
109            caps[key] = [fields]
110    return caps, lineno
111
112def parseline(line):
113    """Parse one entry in a mailcap file and return a dictionary.
114
115    The viewing command is stored as the value with the key "view",
116    and the rest of the fields produce key-value pairs in the dict.
117    """
118    fields = []
119    i, n = 0, len(line)
120    while i < n:
121        field, i = parsefield(line, i, n)
122        fields.append(field)
123        i = i+1 # Skip semicolon
124    if len(fields) < 2:
125        return None, None
126    key, view, rest = fields[0], fields[1], fields[2:]
127    fields = {'view': view}
128    for field in rest:
129        i = field.find('=')
130        if i < 0:
131            fkey = field
132            fvalue = ""
133        else:
134            fkey = field[:i].strip()
135            fvalue = field[i+1:].strip()
136        if fkey in fields:
137            # Ignore it
138            pass
139        else:
140            fields[fkey] = fvalue
141    return key, fields
142
143def parsefield(line, i, n):
144    """Separate one key-value pair in a mailcap entry."""
145    start = i
146    while i < n:
147        c = line[i]
148        if c == ';':
149            break
150        elif c == '\\':
151            i = i+2
152        else:
153            i = i+1
154    return line[start:i].strip(), i
155
156
157# Part 3: using the database.
158
159def findmatch(caps, MIMEtype, key='view', filename="/dev/null", plist=[]):
160    """Find a match for a mailcap entry.
161
162    Return a tuple containing the command line, and the mailcap entry
163    used; (None, None) if no match is found.  This may invoke the
164    'test' command of several matching entries before deciding which
165    entry to use.
166
167    """
168    entries = lookup(caps, MIMEtype, key)
169    # XXX This code should somehow check for the needsterminal flag.
170    for e in entries:
171        if 'test' in e:
172            test = subst(e['test'], filename, plist)
173            if test and os.system(test) != 0:
174                continue
175        command = subst(e[key], MIMEtype, filename, plist)
176        return command, e
177    return None, None
178
179def lookup(caps, MIMEtype, key=None):
180    entries = []
181    if MIMEtype in caps:
182        entries = entries + caps[MIMEtype]
183    MIMEtypes = MIMEtype.split('/')
184    MIMEtype = MIMEtypes[0] + '/*'
185    if MIMEtype in caps:
186        entries = entries + caps[MIMEtype]
187    if key is not None:
188        entries = [e for e in entries if key in e]
189    entries = sorted(entries, key=lineno_sort_key)
190    return entries
191
192def subst(field, MIMEtype, filename, plist=[]):
193    # XXX Actually, this is Unix-specific
194    res = ''
195    i, n = 0, len(field)
196    while i < n:
197        c = field[i]; i = i+1
198        if c != '%':
199            if c == '\\':
200                c = field[i:i+1]; i = i+1
201            res = res + c
202        else:
203            c = field[i]; i = i+1
204            if c == '%':
205                res = res + c
206            elif c == 's':
207                res = res + filename
208            elif c == 't':
209                res = res + MIMEtype
210            elif c == '{':
211                start = i
212                while i < n and field[i] != '}':
213                    i = i+1
214                name = field[start:i]
215                i = i+1
216                res = res + findparam(name, plist)
217            # XXX To do:
218            # %n == number of parts if type is multipart/*
219            # %F == list of alternating type and filename for parts
220            else:
221                res = res + '%' + c
222    return res
223
224def findparam(name, plist):
225    name = name.lower() + '='
226    n = len(name)
227    for p in plist:
228        if p[:n].lower() == name:
229            return p[n:]
230    return ''
231
232
233# Part 4: test program.
234
235def test():
236    import sys
237    caps = getcaps()
238    if not sys.argv[1:]:
239        show(caps)
240        return
241    for i in range(1, len(sys.argv), 2):
242        args = sys.argv[i:i+2]
243        if len(args) < 2:
244            print("usage: mailcap [MIMEtype file] ...")
245            return
246        MIMEtype = args[0]
247        file = args[1]
248        command, e = findmatch(caps, MIMEtype, 'view', file)
249        if not command:
250            print("No viewer found for", type)
251        else:
252            print("Executing:", command)
253            sts = os.system(command)
254            sts = os.waitstatus_to_exitcode(sts)
255            if sts:
256                print("Exit status:", sts)
257
258def show(caps):
259    print("Mailcap files:")
260    for fn in listmailcapfiles(): print("\t" + fn)
261    print()
262    if not caps: caps = getcaps()
263    print("Mailcap entries:")
264    print()
265    ckeys = sorted(caps)
266    for type in ckeys:
267        print(type)
268        entries = caps[type]
269        for e in entries:
270            keys = sorted(e)
271            for k in keys:
272                print("  %-15s" % k, e[k])
273            print()
274
275if __name__ == '__main__':
276    test()
277