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
|
#!/usr/bin/python2.4
# Copyright 2007 Canonical Ltd. All rights reserved.
"""Substitute some chars with the right value in POTranslation.translation."""
__metaclass__ = type
__all__ = []
from optparse import OptionParser
import sys
from canonical.lp import initZopeless
from canonical.database.sqlbase import cursor, sqlvalues
from canonical.launchpad.scripts import logger_options, logger
mapping = {
# u'\u2022': u' ', # space represented as a dot.
u'\u21b5': u'\n', # new line represented with a graphical char.
}
def change_char(ztm, log, character, newchar):
"""Substitute character with newchar in POTranslation.translation."""
log.info("Changing all ocurrences of %r with %r" % (character, newchar))
cur = cursor()
cur.execute("""
SELECT POTranslation.id, POTranslation.translation
FROM POTranslation
WHERE translation like %s
""" % sqlvalues('%%%s%%' % character.encode('UTF-8')))
log.info("Found %d translations that may need a fix" % cur.rowcount)
for potranslation_id, translation in list(cur.fetchall()):
cur = cursor()
# Get the list of POMsgSet that will be affected by this change.
cur.execute("""
SELECT DISTINCT POSubmission.pomsgset
FROM POSubmission
JOIN POMsgSet ON POSubmission.pomsgset = POMsgSet.id
JOIN POTMsgSet ON POMsgSet.potmsgset = POTMsgSet.id
JOIN POMsgID ON POTMsgSet.primemsgid = POMsgID.id
WHERE
POSubmission.potranslation = %s AND
POSubmission.active IS TRUE AND
POMsgID.msgid NOT like %s
""" % sqlvalues(
potranslation_id, '%%%s%%' % character.encode('UTF-8')))
log.info("Changing %d submissions for IPOTranslation %d" % (
cur.rowcount, potranslation_id))
pomsgset_ids = [str(id) for [id] in cur.fetchall()]
if len(pomsgset_ids) > 0:
# There is someone using this translation, update its review date
# so new file exports will get this change.
cur.execute("""
UPDATE POMsgSet
SET date_reviewed = (CURRENT_TIMESTAMP AT TIME ZONE 'UTC')
WHERE id IN (%s)
""" % ','.join(pomsgset_ids))
log.info("Updated %d submissions for IPOTranslation %d" % (
cur.rowcount, potranslation_id))
fixed_translation = translation.replace(
character, newchar).encode('UTF-8')
cur.execute("""
SELECT id
FROM POTranslation
WHERE sha1(translation) = sha1(%s)
""" % sqlvalues(fixed_translation))
if cur.rowcount == 0:
# The fixed translation is not yet in our database, let's add
# it.
cur.execute(
"INSERT INTO POTranslation(translation) VALUES(%s)" % (
sqlvalues(fixed_translation)))
cur.execute("""
SELECT id
FROM POTranslation
WHERE translation = %s
""" % sqlvalues(fixed_translation))
# Get the id of the fixed POTranslation.
[existing_translation_id] = cur.fetchone()
# Get the submissions that should point to the fixed translation.
cur.execute("""
SELECT id
FROM POSubmission
WHERE
potranslation = %d AND
pomsgset IN (%s) AND
active IS TRUE
""" % (potranslation_id, ','.join(pomsgset_ids)))
submission_ids = [str(id) for [id] in cur.fetchall()]
# Link to the new fixed translation.
cur.execute("""
UPDATE POSubmission
SET potranslation = %d
WHERE id IN (%s)
""" % (existing_translation_id, ','.join(submission_ids)))
# Commit is done per IPOTranslation.
ztm.commit()
def read_options():
"""Read the command-line options and return an options object."""
parser = OptionParser()
logger_options(parser)
(options, args) = parser.parse_args()
return options
def main():
options = read_options()
log = logger(options)
ztm = None
try:
log.debug("Connecting to database")
ztm = initZopeless()
for oldchar, newchar in mapping.iteritems():
change_char(ztm, log, oldchar, newchar)
return 0
except:
log.exception("Unhandled exception")
log.debug("Rolling back")
if ztm is not None:
ztm.abort()
return 1
if __name__ == '__main__':
sys.exit(main())
|