~launchpad-pqm/launchpad/devel

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
# Copyright 2009 Canonical Ltd.  This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).

# pylint: disable-msg=E0211,E0213

"""Interfaces to handle translation files imports."""

__metaclass__ = type

__all__ = [
    'ITranslationFormatImporter',
    'ITranslationImporter',
    'OutdatedTranslationError',
    'NotExportedFromLaunchpad',
    'TooManyPluralFormsError',
    'TranslationFormatSyntaxError',
    'TranslationFormatInvalidInputError',
    ]

from zope.interface import Interface
from zope.schema import (
    Bool,
    Int,
    List,
    TextLine,
    )

from lp.translations.interfaces.translationcommonformat import (
    TranslationImportExportBaseException,
    )


class OutdatedTranslationError(TranslationImportExportBaseException):
    """A newer file has already been imported."""


class NotExportedFromLaunchpad(TranslationImportExportBaseException):
    """An imported file lacks the Launchpad export time."""


class TooManyPluralFormsError(TranslationImportExportBaseException):
    """Translation defines more plural forms than we can handle."""


class TranslationFormatBaseError(TranslationImportExportBaseException):
    """Base exception for errors in translation format files."""

    def __init__(self, filename=None, line_number=None, message=None):
        """Initialize the exception information.

        :param filename: The file name that is being parsed.
        :param line_number: The line number where the error was found.
        :param message: The concrete syntax error found. If we get a not None
            value here, filename and line_number are ignored.
        """
        TranslationImportExportBaseException.__init__(self, message)

        self.filename = filename
        self.line_number = line_number
        self.message = message

    def represent(self, default_message):
        """Return human-readable description of error location."""
        if self.filename is not None:
            safe_filename = self.filename.encode("ascii", "backslashreplace")

        if self.line_number is not None and self.line_number > 0:
            if self.filename is not None:
                location = "%s, line %d" % (safe_filename, self.line_number)
            else:
                location = "Line %d" % self.line_number
        elif self.filename is not None:
            location = safe_filename
        else:
            location = None

        if location is not None:
            location_prefix = "%s: " % location
        else:
            location_prefix = ""

        if self.message is not None:
            text = self.message.encode("ascii", "backslashreplace")
        else:
            text = default_message

        return "%s%s" % (location_prefix, text)


class TranslationFormatSyntaxError(TranslationFormatBaseError):
    """A syntax error occurred while parsing a translation file."""

    def __str__(self):
        return self.represent("Unknown syntax error")


class TranslationFormatInvalidInputError(TranslationFormatBaseError):
    """Some fields in the parsed file contain bad content."""

    def __str__(self):
        return self.represent("Invalid input")


class ITranslationImporter(Interface):
    """Importer of translation files."""

    supported_file_extensions = List(
        title=u'List of file extensions we have imports for.',
        required=True, readonly=True)

    # Filename suffixes that identify templates.  These do not have to be
    # CP/M-style "filename extensions" separated from the base file name by a
    # dot; any suffix will do.
    template_suffixes = TextLine(
        title=u'Filename endings that identify templates.', required=True,
        readonly=True)

    def isTemplateName(path):
        """Based on filename, is this a template file?

        :param path: file name, possibly including directory component.
        :return: Boolean: True if `path` identifies this file as a template.
        """

    def isTranslationName(path):
        """Based on filename, is this file a translation?

        :param path: file name, possibly including directory component.
        :return: Boolean: True if `path` identifies this file as a
            translation.  A template, or a file with an unknown filename
            extension, is not considered a translation.
        """

    def isHidden(path):
        """Based on filename, is this a hidden file?

        :param path: file name, possibly including directory component.
        :return: Boolean: True if `path` contains either a hidden file
            or a hidden directory.
        """

    def getTranslationFileFormat(file_extension, file_contents):
        """Return the translation file format for the given file extension.

        :param file_extension: File extension including the dot.
        :param file_contents: File contents.
        :return: A `TranslationFileFormat` for the given file extension
            and file contents or None if it's not supported format.
        """

    def getTranslationFormatImporter(file_format):
        """Return the translation format importer for the given file format.

        :param file_format: A TranslationFileFormat entry.
        :return: An `ITranslationFormatImporter` or None if there is no
            handler for the given file format.
        """

    def importFile(translation_import_queue_entry):
        """Import an `ITranslationImportQueueEntry` file into the system.

        :param translation_import_queue_entry: An
            `ITranslationImportQueueEntry` entry.
        :raise OutdatedTranslationError: If the entry is older than the
            previously imported file.
        :raise NotExportedFromLaunchpad: If the entry imported is not
            from upstream and doesn't have the tag added by Launchpad on
            export time.
        :return: a tuple of two lists: error descriptors, and warnings.

        The errors list contains dictionaries describing messages that
        couldn't be imported.  Each of those has three keys:
            - 'pomsgset': An `IPOMsgSet` associated with this error.
            - 'pomessage': The original message text in its native format.
            - 'error-message': The error message text.

        The warnings list is simply a list of warning strings.
        """


class ITranslationFormatImporter(Interface):
    """Translation file format importer."""

    def getFormat(file_contents):
        """The file format of the import.

        :param file_contents: A unicode string with the contents of the file
            being imported.  A returned format may sometimes be different
            from the base format of the `ITranslationFormatImporter`, and
            that is determined based on the `contents`.
        :return: A `TranslationFileFormat` value.
        """

    priority = Int(
        title=u'Priority among importers for the same file extension.',
        description=u'''
            Priority an `ITranslationFormatImporter` has if there are
            multiple importers for the same file extension.

            Higher value indicates higher priority, i.e. that importer
            is tried first.
            ''',
        required=True,
        default=0
        )

    content_type = TextLine(
        title=u'Content type string for this file format.',
        required=True, readonly=True)

    file_extensions = List(
        title=u'File extensions handable by this importer.',
        required=True, readonly=True)

    uses_source_string_msgids = Bool(
        title=u'A flag indicating whether uses source string as the id',
        description=u'''
            A flag indicating whether this file format importer uses source
            string msgids as the English strings.
            ''',
        required=True, readonly=True)

    def parse(translation_import_queue_entry):
        """Parse an `ITranslationImportQueueEntry` into an
        `ITranslationFileData`.

        :param translation_import_queue: An `ITranslationImportQueueEntry` to
            parse.
        :return: An `ITranslationFileData` representing the parsed file.
        """

    def getHeaderFromString(header_string):
        """Return the `ITranslationHeaderData` for the given header string.

        :param header_string: A text representing a header for this concrete
            file format.
        :return: An `ITranslationHeaderData` based on the header string.
        """