annotate babel/messages/mofile.py @ 530:85e1beadacb0

Update the copyright line.
author jruigrok
date Sat, 05 Mar 2011 15:22:28 +0000
parents 53c68c594b5f
children f642c81ce8cc
rev   line source
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
1 # -*- coding: utf-8 -*-
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
2 #
530
85e1beadacb0 Update the copyright line.
jruigrok
parents: 420
diff changeset
3 # Copyright (C) 2007-2011 Edgewall Software
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
4 # All rights reserved.
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
5 #
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
6 # This software is licensed as described in the file COPYING, which
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
7 # you should have received as part of this distribution. The terms
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
8 # are also available at http://babel.edgewall.org/wiki/License.
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
9 #
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
10 # This software consists of voluntary contributions made by many
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
11 # individuals. For the exact contribution history, see the revision
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
12 # history and logs, available at http://babel.edgewall.org/log/.
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
13
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
14 """Writing of files in the ``gettext`` MO (machine object) format.
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
15
234
ada322f472ca Add more `since` tags to stuff added in trunk.
cmlenz
parents: 174
diff changeset
16 :since: version 0.9
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
17 :see: `The Format of MO Files
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
18 <http://www.gnu.org/software/gettext/manual/gettext.html#MO-Files>`_
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
19 """
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
20
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
21 import array
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
22 import struct
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
23
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
24 from babel.messages.catalog import Catalog, Message
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
25
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
26 __all__ = ['read_mo', 'write_mo']
161
04c56e82c98b Slightly simplified CLI-frontend class.
cmlenz
parents: 160
diff changeset
27 __docformat__ = 'restructuredtext en'
04c56e82c98b Slightly simplified CLI-frontend class.
cmlenz
parents: 160
diff changeset
28
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
29
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
30 LE_MAGIC = 0x950412deL
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
31 BE_MAGIC = 0xde120495L
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
32
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
33 def read_mo(fileobj):
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
34 """Read a binary MO file from the given file-like object and return a
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
35 corresponding `Catalog` object.
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
36
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
37 :param fileobj: the file-like object to read the MO file from
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
38 :return: a catalog object representing the parsed MO file
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
39 :rtype: `Catalog`
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
40
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
41 :note: The implementation of this function is heavily based on the
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
42 ``GNUTranslations._parse`` method of the ``gettext`` module in the
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
43 standard library.
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
44 """
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
45 catalog = Catalog()
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
46 headers = {}
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
47
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
48 filename = getattr(fileobj, 'name', '')
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
49 charset = None
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
50
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
51 buf = fileobj.read()
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
52 buflen = len(buf)
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
53 unpack = struct.unpack
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
54
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
55 # Parse the .mo file header, which consists of 5 little endian 32
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
56 # bit words.
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
57 magic = unpack('<I', buf[:4])[0] # Are we big endian or little endian?
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
58 if magic == LE_MAGIC:
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
59 version, msgcount, origidx, transidx = unpack('<4I', buf[4:20])
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
60 ii = '<II'
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
61 elif magic == BE_MAGIC:
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
62 version, msgcount, origidx, transidx = unpack('>4I', buf[4:20])
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
63 ii = '>II'
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
64 else:
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
65 raise IOError(0, 'Bad magic number', filename)
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
66
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
67 # Now put all messages from the .mo file buffer into the catalog
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
68 # dictionary
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
69 for i in xrange(0, msgcount):
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
70 mlen, moff = unpack(ii, buf[origidx:origidx + 8])
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
71 mend = moff + mlen
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
72 tlen, toff = unpack(ii, buf[transidx:transidx + 8])
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
73 tend = toff + tlen
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
74 if mend < buflen and tend < buflen:
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
75 msg = buf[moff:mend]
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
76 tmsg = buf[toff:tend]
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
77 else:
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
78 raise IOError(0, 'File is corrupt', filename)
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
79
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
80 # See if we're looking at GNU .mo conventions for metadata
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
81 if mlen == 0:
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
82 # Catalog description
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
83 lastkey = key = None
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
84 for item in tmsg.splitlines():
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
85 item = item.strip()
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
86 if not item:
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
87 continue
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
88 if ':' in item:
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
89 key, value = item.split(':', 1)
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
90 lastkey = key = key.strip().lower()
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
91 headers[key] = value.strip()
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
92 elif lastkey:
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
93 headers[lastkey] += '\n' + item
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
94
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
95 if '\x04' in msg: # context
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
96 ctxt, msg = msg.split('\x04')
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
97 else:
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
98 ctxt = None
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
99
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
100 if '\x00' in msg: # plural forms
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
101 msg = msg.split('\x00')
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
102 tmsg = tmsg.split('\x00')
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
103 if catalog.charset:
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
104 msg = [x.decode(catalog.charset) for x in msg]
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
105 tmsg = [x.decode(catalog.charset) for x in tmsg]
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
106 else:
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
107 if catalog.charset:
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
108 msg = msg.decode(catalog.charset)
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
109 tmsg = tmsg.decode(catalog.charset)
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
110 catalog[msg] = Message(msg, tmsg, context=ctxt)
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
111
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
112 # advance to next entry in the seek tables
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
113 origidx += 8
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
114 transidx += 8
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
115
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
116 catalog.mime_headers = headers.items()
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
117 return catalog
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
118
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
119 def write_mo(fileobj, catalog, use_fuzzy=False):
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
120 """Write a catalog to the specified file-like object using the GNU MO file
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
121 format.
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
122
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
123 >>> from babel.messages import Catalog
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
124 >>> from gettext import GNUTranslations
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
125 >>> from StringIO import StringIO
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
126
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
127 >>> catalog = Catalog(locale='en_US')
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
128 >>> catalog.add('foo', 'Voh')
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
129 >>> catalog.add((u'bar', u'baz'), (u'Bahr', u'Batz'))
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
130 >>> catalog.add('fuz', 'Futz', flags=['fuzzy'])
172
9e35bbb458aa Fix for #28 with updated doctest.
palgarvio
parents: 161
diff changeset
131 >>> catalog.add('Fizz', '')
174
456ee0941987 Extended the doctest to include tests for the fix on [176].
palgarvio
parents: 173
diff changeset
132 >>> catalog.add(('Fuzz', 'Fuzzes'), ('', ''))
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
133 >>> buf = StringIO()
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
134
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
135 >>> write_mo(buf, catalog)
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
136 >>> buf.seek(0)
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
137 >>> translations = GNUTranslations(fp=buf)
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
138 >>> translations.ugettext('foo')
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
139 u'Voh'
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
140 >>> translations.ungettext('bar', 'baz', 1)
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
141 u'Bahr'
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
142 >>> translations.ungettext('bar', 'baz', 2)
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
143 u'Batz'
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
144 >>> translations.ugettext('fuz')
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
145 u'fuz'
172
9e35bbb458aa Fix for #28 with updated doctest.
palgarvio
parents: 161
diff changeset
146 >>> translations.ugettext('Fizz')
9e35bbb458aa Fix for #28 with updated doctest.
palgarvio
parents: 161
diff changeset
147 u'Fizz'
174
456ee0941987 Extended the doctest to include tests for the fix on [176].
palgarvio
parents: 173
diff changeset
148 >>> translations.ugettext('Fuzz')
456ee0941987 Extended the doctest to include tests for the fix on [176].
palgarvio
parents: 173
diff changeset
149 u'Fuzz'
456ee0941987 Extended the doctest to include tests for the fix on [176].
palgarvio
parents: 173
diff changeset
150 >>> translations.ugettext('Fuzzes')
456ee0941987 Extended the doctest to include tests for the fix on [176].
palgarvio
parents: 173
diff changeset
151 u'Fuzzes'
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
152
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
153 :param fileobj: the file-like object to write to
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
154 :param catalog: the `Catalog` instance
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
155 :param use_fuzzy: whether translations marked as "fuzzy" should be included
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
156 in the output
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
157 """
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
158 messages = list(catalog)
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
159 if not use_fuzzy:
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
160 messages[1:] = [m for m in messages[1:] if not m.fuzzy]
248
bedaaeadc1db add a __cmp__ to Message that correctly sorts by id, taking into account plurals
pjenvey
parents: 234
diff changeset
161 messages.sort()
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
162
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
163 ids = strs = ''
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
164 offsets = []
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
165
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
166 for message in messages:
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
167 # For each string, we need size and file offset. Each string is NUL
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
168 # terminated; the NUL does not count into the size.
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
169 if message.pluralizable:
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
170 msgid = '\x00'.join([
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
171 msgid.encode(catalog.charset) for msgid in message.id
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
172 ])
173
743abd4ca3c3 Forgot to fix the pluralizable messages, regarding #28.
palgarvio
parents: 172
diff changeset
173 msgstrs = []
743abd4ca3c3 Forgot to fix the pluralizable messages, regarding #28.
palgarvio
parents: 172
diff changeset
174 for idx, string in enumerate(message.string):
743abd4ca3c3 Forgot to fix the pluralizable messages, regarding #28.
palgarvio
parents: 172
diff changeset
175 if not string:
330
5f67aa47609c Fix for #97, compilation of message catalogs for locales with more than two plural forms where the translations were empty was failing.
cmlenz
parents: 248
diff changeset
176 msgstrs.append(message.id[min(int(idx), 1)])
173
743abd4ca3c3 Forgot to fix the pluralizable messages, regarding #28.
palgarvio
parents: 172
diff changeset
177 else:
743abd4ca3c3 Forgot to fix the pluralizable messages, regarding #28.
palgarvio
parents: 172
diff changeset
178 msgstrs.append(string)
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
179 msgstr = '\x00'.join([
173
743abd4ca3c3 Forgot to fix the pluralizable messages, regarding #28.
palgarvio
parents: 172
diff changeset
180 msgstr.encode(catalog.charset) for msgstr in msgstrs
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
181 ])
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
182 else:
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
183 msgid = message.id.encode(catalog.charset)
172
9e35bbb458aa Fix for #28 with updated doctest.
palgarvio
parents: 161
diff changeset
184 if not message.string:
9e35bbb458aa Fix for #28 with updated doctest.
palgarvio
parents: 161
diff changeset
185 msgstr = message.id.encode(catalog.charset)
9e35bbb458aa Fix for #28 with updated doctest.
palgarvio
parents: 161
diff changeset
186 else:
9e35bbb458aa Fix for #28 with updated doctest.
palgarvio
parents: 161
diff changeset
187 msgstr = message.string.encode(catalog.charset)
335
9c41fe73e2e6 More preparation for msgctxt support (#54).
cmlenz
parents: 334
diff changeset
188 if message.context:
420
53c68c594b5f Add support for `msgctxt`. See #54.
palgarvio
parents: 335
diff changeset
189 msgid = '\x04'.join([message.context.encode(catalog.charset),
53c68c594b5f Add support for `msgctxt`. See #54.
palgarvio
parents: 335
diff changeset
190 msgid])
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
191 offsets.append((len(ids), len(msgid), len(strs), len(msgstr)))
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
192 ids += msgid + '\x00'
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
193 strs += msgstr + '\x00'
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
194
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
195 # The header is 7 32-bit unsigned integers. We don't use hash tables, so
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
196 # the keys start right after the index tables.
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
197 keystart = 7 * 4 + 16 * len(messages)
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
198 valuestart = keystart + len(ids)
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
199
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
200 # The string table first has the list of keys, then the list of values.
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
201 # Each entry has first the size of the string, then the file offset.
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
202 koffsets = []
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
203 voffsets = []
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
204 for o1, l1, o2, l2 in offsets:
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
205 koffsets += [l1, o1 + keystart]
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
206 voffsets += [l2, o2 + valuestart]
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
207 offsets = koffsets + voffsets
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
208
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
209 fileobj.write(struct.pack('Iiiiiii',
334
aa8457401353 Add basic MO file reading in preparation for #54.
cmlenz
parents: 330
diff changeset
210 LE_MAGIC, # magic
160
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
211 0, # version
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
212 len(messages), # number of entries
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
213 7 * 4, # start of key index
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
214 7 * 4 + len(messages) * 8, # start of value index
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
215 0, 0 # size and offset of hash table
80e51aabc440 Add MO file generation. Closes #21.
cmlenz
parents:
diff changeset
216 ) + array.array("i", offsets).tostring() + ids + strs)
Copyright (C) 2012-2017 Edgewall Software