1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
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
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
|
Language Pack Exports
=====================
Launchpad has a way to export all distribution series translations in a
tarball to provide language packs.
Some initialization tasks:
>>> import datetime
>>> import pytz
>>> import transaction
>>> from lp.testing import login
>>> from lp.services.helpers import string_to_tarfile
>>> from lp.app.interfaces.launchpad import ILaunchpadCelebrities
>>> from lp.translations.scripts.language_pack import (
... export_language_pack)
>>> from canonical.database.sqlbase import flush_database_caches
>>> from lp.services.librarian.interfaces.client import ILibrarianClient
>>> rosetta_experts = getUtility(ILaunchpadCelebrities).rosetta_experts
>>> login("daf@canonical.com")
>>> from lp.services.log.logger import BufferLogger
This is handy for examining the tar files that are generated.
>>> def get_name(entry):
... """For sorting: get tar entry's name."""
... return entry.name
>>> def examine_tarfile(tarfile):
... """Summarize the contents of a tar file."""
... for member in sorted(tarfile.getmembers(), key=get_name):
... if not member.isreg():
... # Not a regular file. No size to print.
... size = '-'
... elif member.name.endswith('.xpi'):
... # XPI file. Binary, so don't try counting lines.
... size = 'bin'
... else:
... size = len(tarfile.extractfile(member).readlines())
... # XXX: 2010-04-26, Salgado, bug=570244: The .rstrip('/') is to
... # make this pass on python2.5 and 2.6.
... print "| %5s | %s" % (size, member.name.rstrip('/'))
Base language pack export using Librarian
-----------------------------------------
When a new language pack export is requested, you can select to use the
Librarian to store the language pack. That is noted with output_file set
to None.
>>> logger = BufferLogger()
>>> language_pack = export_language_pack(
... distribution_name='ubuntu',
... series_name='hoary',
... component='main',
... force_utf8=True,
... output_file=None,
... logger=logger)
>>> transaction.commit()
Check that the log looks ok.
>>> print logger.getLogBuffer()
DEBUG Selecting PO files for export
INFO Number of PO files to export: 12
DEBUG Exporting PO file ... (1/12)
DEBUG Exporting PO file ... (2/12)
...
INFO Adding timestamp file
INFO Adding mapping file
INFO Done.
DEBUG Upload complete, file alias: ...
# Get the generated tarball.
>>> tarfile = string_to_tarfile(language_pack.file.read())
The tarball has the right members.
>>> for member in tarfile.getmembers():
... # XXX: 2010-04-26, Salgado, bug=570244: The .rstrip('/') is to
... # make this pass on python2.5 and 2.6.
... print member.name.rstrip('/')
rosetta-hoary
...
rosetta-hoary/es
rosetta-hoary/es/LC_MESSAGES
rosetta-hoary/es/LC_MESSAGES/pmount.po
...
rosetta-hoary/timestamp.txt
rosetta-hoary/mapping.txt
Directory permissions allow correct use of those directories:
# XXX: 2010-04-26, Salgado, bug=570244: This try/except is needed to make
# the test pass on python2.5 and 2.6.
>>> try:
... directory = tarfile.getmember('rosetta-hoary/')
... except KeyError:
... directory = tarfile.getmember('rosetta-hoary')
>>> oct(directory.mode)
'0755'
And one of the included .po files look like what we expected.
>>> fh = tarfile.extractfile(
... 'rosetta-hoary/es/LC_MESSAGES/evolution-2.2.po')
>>> fh.readline()
'# traducci\xc3\xb3n de es.po al Spanish\n'
Language pack with XPI translations
-----------------------------------
Launchpad supports XPI file imports. However, we don't have an export
process ready, and thus, we do it with an external script that does that
last part based on .po files and the original en-US.xpi file. To achieve
that, we export all translations for XPI files in a special directory:
xpi/translation_domain/
>>> from lp.registry.interfaces.distribution import IDistributionSet
>>> from lp.registry.interfaces.person import IPersonSet
>>> from lp.registry.model.sourcepackagename import SourcePackageName
>>> from lp.translations.model.potemplate import POTemplate
Get hold of a person.
>>> mark = getUtility(IPersonSet).getByName('mark')
>>> print mark.displayname
Mark Shuttleworth
Get the Grumpy distro series.
>>> series = getUtility(IDistributionSet)['ubuntu'].getSeries('grumpy')
Sample data initialization
..........................
We need to import an XPI template and a translation to see those files
exported as part of language packs.
>>> from lp.translations.enums import RosettaImportStatus
>>> from lp.translations.interfaces.translationimportqueue import (
... ITranslationImportQueue)
>>> from lp.translations.utilities.tests.test_xpi_import \
... import get_en_US_xpi_file_to_import
We are going to import translations for mozilla-firefox package in
grumpy distro series.
>>> series = getUtility(IDistributionSet)['ubuntu'].getSeries('grumpy')
>>> spn = SourcePackageName.byName('mozilla-firefox')
>>> pot_header = 'Content-Type: text/plain; charset=UTF-8\n'
>>> firefox_template = POTemplate(
... name='firefox', translation_domain='firefox',
... distroseries=series, sourcepackagename=spn,
... owner=mark, languagepack=True, path='en-US.xpi',
... header=pot_header)
Attach the en-US.xpi (the template) file.
>>> en_US_xpi = get_en_US_xpi_file_to_import('en-US')
>>> translation_import_queue = getUtility(ITranslationImportQueue)
>>> by_maintainer = True
>>> template_entry = translation_import_queue.addOrUpdateEntry(
... firefox_template.path, en_US_xpi.read(), by_maintainer,
... mark, distroseries=series, sourcepackagename=spn,
... potemplate=firefox_template)
Attach the es.xpi file (the translation) file.
>>> es_xpi = get_en_US_xpi_file_to_import('en-US')
>>> firefox_es_translation = firefox_template.newPOFile('es')
>>> translation_entry = translation_import_queue.addOrUpdateEntry(
... 'es.xpi', es_xpi.read(), by_maintainer,
... mark, distroseries=series, sourcepackagename=spn,
... potemplate=firefox_template,
... pofile=firefox_es_translation)
Before we are ready to import the attached files, we need to approve
them first.
>>> template_entry.setStatus(
... RosettaImportStatus.APPROVED, rosetta_experts)
>>> translation_entry.setStatus(
... RosettaImportStatus.APPROVED, rosetta_experts)
Given that the files are attached to Librarian, we need to commit the
transaction to make sure it's stored properly and available.
>>> transaction.commit()
We do now the import from the queue:
>>> (subject, body) = firefox_template.importFromQueue(template_entry)
>>> (subject, body) = firefox_es_translation.importFromQueue(
... translation_entry)
>>> flush_database_caches()
>>> transaction.commit()
Language pack export with XPI files
...................................
We are now ready to get an exported language pack with XPI files.
>>> language_pack = export_language_pack(
... distribution_name='ubuntu',
... series_name='grumpy',
... component=None,
... force_utf8=True,
... output_file=None,
... logger=logger)
>>> transaction.commit()
We get other entries in language pack + en-US.xpi file and the
translations in .po file format.
>>> tarfile = string_to_tarfile(language_pack.file.read())
>>> examine_tarfile(tarfile)
| - | rosetta-grumpy
| 1 | rosetta-grumpy/mapping.txt
| 1 | rosetta-grumpy/timestamp.txt
| - | rosetta-grumpy/xpi
| - | rosetta-grumpy/xpi/firefox
| bin | rosetta-grumpy/xpi/firefox/en-US.xpi
| 94 | rosetta-grumpy/xpi/firefox/en.po
| 102 | rosetta-grumpy/xpi/firefox/es.po
We got a valid en-US.xpi file.
>>> fh = tarfile.extractfile('rosetta-grumpy/xpi/firefox/en-US.xpi')
>>> from zipfile import ZipFile
>>> zip = ZipFile(fh, 'r')
>>> sorted(zip.namelist())
['chrome.manifest', 'chrome/en-US.jar', 'copyover3.png', 'install.rdf']
Base language pack export using Librarian with date limits
----------------------------------------------------------
Launchpad is also able to generate a tarball of all files for a
distribution series that only includes translation files which have been
changed since a certain date.
First we need to set up some data to test with, and for this we need
some DB classes.
>>> from StringIO import StringIO
Get a source package name to go with our distro series.
>>> spn = SourcePackageName.byName('evolution')
Put a dummy file in the Librarian required by the new template we are
creating.
>>> contents = '# Test PO template.'
>>> file_alias = getUtility(ILibrarianClient).addFile(
... name = 'test.po',
... size = len(contents),
... file = StringIO(contents),
... contentType = 'application/x-po')
Get some dates.
>>> UTC = pytz.timezone('UTC')
>>> d2000_01_01 = datetime.datetime(year=2000, month=1, day=1, tzinfo=UTC)
>>> d2000_01_02 = datetime.datetime(year=2000, month=1, day=2, tzinfo=UTC)
>>> d2000_01_03 = datetime.datetime(year=2000, month=1, day=3, tzinfo=UTC)
Create a PO template and put a single message set in it.
>>> pot_header = 'Content-Type: text/plain; charset=UTF-8\n'
>>> template = POTemplate(
... name='test', translation_domain='test',
... distroseries=series, sourcepackagename=spn,
... owner=mark, languagepack=True, path='po/test.pot',
... header=pot_header)
>>> potmsgset = template.createMessageSetFromText(u'blah', None)
>>> item = potmsgset.setSequence(template, 1)
We set the template last update date to the oldest date we are going to
play with, so it doesn't affect translations export.
>>> template.date_last_updated = d2000_01_01
Create a Spanish PO file, with an active translation submission created
on 2000/01/01.
>>> pofile_es = template.newPOFile('es')
>>> translations = { 0: u'blah (es)' }
>>> new_translation_message = factory.makeCurrentTranslationMessage(
... pofile_es, potmsgset, mark, translations=translations)
>>> pofile_es.date_changed = d2000_01_01
Create a Welsh PO file, with an active translation submission created on
2000/01/03.
>>> pofile_cy = template.newPOFile('cy')
>>> translations = { 0: u'blah (cy)' }
>>> new_translation_message = factory.makeCurrentTranslationMessage(
... pofile_cy, potmsgset, mark, translations=translations)
>>> pofile_cy.date_changed = d2000_01_03
>>> transaction.commit()
First, export without any existing base language pack: should get both
PO files.
>>> print series.language_pack_base
None
>>> logger = BufferLogger()
>>> flush_database_caches()
>>> language_pack = export_language_pack(
... distribution_name='ubuntu',
... series_name='grumpy',
... component=None,
... force_utf8=True,
... output_file=None,
... logger=logger)
>>> transaction.commit()
Check that the log looks ok.
>>> print logger.getLogBuffer()
DEBUG Selecting PO files for export
INFO Number of PO files to export: 4
DEBUG Exporting PO file ... (1/4)
DEBUG Exporting PO file ... (2/4)
DEBUG Exporting PO file ... (3/4)
DEBUG Exporting PO file ... (4/4)
INFO Exporting XPI template files.
INFO Adding timestamp file
INFO Adding mapping file
INFO Done.
DEBUG Upload complete, file alias: ...
INFO Registered the language pack.
# Get the generated tarball.
>>> tarfile = string_to_tarfile(language_pack.file.read())
>>> examine_tarfile(tarfile)
| - | rosetta-grumpy
| - | rosetta-grumpy/cy
| - | rosetta-grumpy/cy/LC_MESSAGES
| 21 | rosetta-grumpy/cy/LC_MESSAGES/test.po
| - | rosetta-grumpy/es
| - | rosetta-grumpy/es/LC_MESSAGES
| 21 | rosetta-grumpy/es/LC_MESSAGES/test.po
| 2 | rosetta-grumpy/mapping.txt
| 1 | rosetta-grumpy/timestamp.txt
| - | rosetta-grumpy/xpi
| - | rosetta-grumpy/xpi/firefox
| bin | rosetta-grumpy/xpi/firefox/en-US.xpi
| 94 | rosetta-grumpy/xpi/firefox/en.po
| 102 | rosetta-grumpy/xpi/firefox/es.po
Check the files look OK.
>>> fh = tarfile.extractfile('rosetta-grumpy/es/LC_MESSAGES/test.po')
>>> print fh.read().decode('UTF-8')
# Spanish translation for evolution
# Copyright (c) ... Rosetta Contributors and Canonical Ltd ...
# This file is distributed under the same license as the evolution pack...
# FIRST AUTHOR <EMAIL@ADDRESS>, ...
#
msgid ""
msgstr ""
"Project-Id-Version: evolution\n"
"Report-Msgid-Bugs-To: FULL NAME <EMAIL@ADDRESS>\n"
"POT-Creation-Date: ...\n"
"PO-Revision-Date: ...\n"
"Last-Translator: Mark Shuttleworth <mark@example.com>\n"
"Language-Team: Spanish <es@li.org>\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"X-Launchpad-Export-Date: ...-...-... ...:...+...\n"
"X-Generator: Launchpad (build ...)\n"
<BLANKLINE>
msgid "blah"
msgstr "blah (es)"
>>> fh = tarfile.extractfile('rosetta-grumpy/cy/LC_MESSAGES/test.po')
>>> print fh.read().decode('UTF-8')
# Welsh translation for evolution
# Copyright (c) ... Rosetta Contributors and Canonical Ltd ...
# This file is distributed under the same license as the evolution pack...
# FIRST AUTHOR <EMAIL@ADDRESS>, ...
#
msgid ""
msgstr ""
"Project-Id-Version: evolution\n"
"Report-Msgid-Bugs-To: FULL NAME <EMAIL@ADDRESS>\n"
"POT-Creation-Date: ...\n"
"PO-Revision-Date: ...\n"
"Last-Translator: Mark Shuttleworth <mark@example.com>\n"
"Language-Team: Welsh <cy@li.org>\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"X-Launchpad-Export-Date: ...-...-... ...:...+...\n"
"X-Generator: Launchpad (build ...)\n"
<BLANKLINE>
msgid "blah"
msgstr "blah (cy)"
We set this language pack as the base package for the distroseries. Next
export will be a delta based on that one.
>>> series.language_pack_base = language_pack
This is needed to make the PO export cache work, since it uses the
Librarian.
>>> transaction.commit()
Then, export with a date limit: we should only get the second PO file.
The way to set date limits is setting when the base language pack was
exported, thus, we set it and request an update export, which means we
should get only files that where updated after 2000-01-02.
>>> series.language_pack_base.date_exported = d2000_01_02
>>> transaction.commit()
>>> language_pack = export_language_pack(
... distribution_name='ubuntu',
... series_name='grumpy',
... component=None,
... force_utf8=True,
... output_file=None,
... logger=logger)
>>> transaction.commit()
# Get the generated tarball.
>>> tarfile = string_to_tarfile(language_pack.file.read())
Now, there is only one file exported for the 'test' domain, the one that
had the modification date after the last generated language pack. We
ignore the xpi entries because those are outside the scope of this test.
>>> examine_tarfile(tarfile)
| - | rosetta-grumpy
| - | rosetta-grumpy/cy
| - | rosetta-grumpy/cy/LC_MESSAGES
| 21 | rosetta-grumpy/cy/LC_MESSAGES/test.po
| 2 | rosetta-grumpy/mapping.txt
| 1 | rosetta-grumpy/timestamp.txt
| - | rosetta-grumpy/xpi
| - | rosetta-grumpy/xpi/firefox
| bin | rosetta-grumpy/xpi/firefox/en-US.xpi
| 94 | rosetta-grumpy/xpi/firefox/en.po
| 102 | rosetta-grumpy/xpi/firefox/es.po
There is another situation where a translation file is exported again as
part of a language pack update, even without being changed. It is re-
exported if its template has been changed since the last language pack
was produced.
The latest template change is noted in IPOTemplate.date_last_updated. An
update to the template causes that field to be updated, so that the next
export will include all its translations as well.
>>> template.date_last_updated = datetime.datetime.now(UTC)
# Save changes.
>>> transaction.commit()
We export a language pack with changes relative to a base language pack
that was exported on 2000-01-03:
>>> series.language_pack_base.date_exported = d2000_01_03
>>> flush_database_caches()
>>> language_pack = export_language_pack(
... distribution_name='ubuntu',
... series_name='grumpy',
... component=None,
... force_utf8=True,
... output_file=None,
... logger=logger)
>>> transaction.commit()
The Spanish translation has not changed since 2000-01-03, but the
template has. That's why we get both translations:
>>> tarfile = string_to_tarfile(language_pack.file.read())
>>> examine_tarfile(tarfile)
| - | rosetta-grumpy
| - | rosetta-grumpy/cy
| - | rosetta-grumpy/cy/LC_MESSAGES
| 21 | rosetta-grumpy/cy/LC_MESSAGES/test.po
| - | rosetta-grumpy/es
| - | rosetta-grumpy/es/LC_MESSAGES
| 21 | rosetta-grumpy/es/LC_MESSAGES/test.po
| 2 | rosetta-grumpy/mapping.txt
| 1 | rosetta-grumpy/timestamp.txt
| - | rosetta-grumpy/xpi
| - | rosetta-grumpy/xpi/firefox
| bin | rosetta-grumpy/xpi/firefox/en-US.xpi
| 94 | rosetta-grumpy/xpi/firefox/en.po
| 102 | rosetta-grumpy/xpi/firefox/es.po
Script arguments and concurrency
--------------------------------
The language-pack-exporter script requires two arguments to run: the
distribution name, and series name. The script promptly exits if the
number of command-line arguments is wrong.
>>> import subprocess
>>> def get_subprocess(command):
... return subprocess.Popen(
... command, shell=True,
... stdin=subprocess.PIPE, stdout=subprocess.PIPE,
... stderr=subprocess.PIPE)
>>> proc = get_subprocess('cronscripts/language-pack-exporter.py')
>>> (out, err) = proc.communicate()
>>> print err
Traceback (most recent call last):
...
lp.services.scripts.base.LaunchpadScriptFailure:
Wrong number of arguments: should include distribution and series name.
>>> proc.returncode
1
The script runs when 'ubuntu' and 'hoary' are passed as the distribution
and series names.
Several instances of the language-pack-exporter script can be run
concurrently so long as each instance is exporting a different
combination of distribution and series. LaunchpadScript instanced uses
the lockfilename to prevent the script from running concurrently.
RosettaLangPackExporter incorporates the distribution and series names
into the lockfilename to allow multiple exports to run concurrently for
different distribution and series combinations.
>>> proc = get_subprocess(
... 'cronscripts/language-pack-exporter.py ubuntu hoary')
>>> (out, err) = proc.communicate()
>>> print err
INFO Setting lockfile name to
launchpad-language-pack-exporter__ubuntu__hoary.lock.
INFO Creating lockfile:
/var/lock/launchpad-language-pack-exporter__ubuntu__hoary.lock
INFO Exporting translations for series hoary of distribution ubuntu.
INFO Number of PO files to export: 12
INFO Exporting XPI template files.
INFO Adding timestamp file
INFO Adding mapping file
INFO Done.
INFO Registered the language pack.
>>> print out
<BLANKLINE>
>>> proc.returncode
0
|