web/lib/django/utils/translation/trans_real.py
author ymh <ymh.work@gmail.com>
Thu, 05 Aug 2010 17:28:09 +0200
changeset 50 012451a812f1
parent 38 77b6da96e6f1
permissions -rw-r--r--
Merge with a2711e44ba5de8b1675d7e0ee6aaa4a6c56a9b46
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
38
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
     1
"""Translation helper functions."""
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
     2
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
     3
import locale
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
     4
import os
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
     5
import re
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
     6
import sys
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
     7
import warnings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
     8
import gettext as gettext_module
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
     9
from cStringIO import StringIO
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    10
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    11
from django.utils.importlib import import_module
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    12
from django.utils.safestring import mark_safe, SafeData
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    13
from django.utils.thread_support import currentThread
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    14
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    15
# Translations are cached in a dictionary for every language+app tuple.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    16
# The active translations are stored by threadid to make them thread local.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    17
_translations = {}
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    18
_active = {}
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    19
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    20
# The default translation is based on the settings file.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    21
_default = None
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    22
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    23
# This is a cache for normalized accept-header languages to prevent multiple
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    24
# file lookups when checking the same locale on repeated requests.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    25
_accepted = {}
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    26
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    27
# Format of Accept-Language header values. From RFC 2616, section 14.4 and 3.9.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    28
accept_language_re = re.compile(r'''
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    29
        ([A-Za-z]{1,8}(?:-[A-Za-z]{1,8})*|\*)   # "en", "en-au", "x-y-z", "*"
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    30
        (?:;q=(0(?:\.\d{,3})?|1(?:.0{,3})?))?   # Optional "q=1.00", "q=0.8"
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    31
        (?:\s*,\s*|$)                            # Multiple accepts per header.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    32
        ''', re.VERBOSE)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    33
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    34
def to_locale(language, to_lower=False):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    35
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    36
    Turns a language name (en-us) into a locale name (en_US). If 'to_lower' is
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    37
    True, the last component is lower-cased (en_us).
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    38
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    39
    p = language.find('-')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    40
    if p >= 0:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    41
        if to_lower:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    42
            return language[:p].lower()+'_'+language[p+1:].lower()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    43
        else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    44
            # Get correct locale for sr-latn
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    45
            if len(language[p+1:]) > 2:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    46
                return language[:p].lower()+'_'+language[p+1].upper()+language[p+2:].lower()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    47
            return language[:p].lower()+'_'+language[p+1:].upper()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    48
    else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    49
        return language.lower()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    50
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    51
def to_language(locale):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    52
    """Turns a locale name (en_US) into a language name (en-us)."""
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    53
    p = locale.find('_')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    54
    if p >= 0:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    55
        return locale[:p].lower()+'-'+locale[p+1:].lower()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    56
    else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    57
        return locale.lower()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    58
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    59
class DjangoTranslation(gettext_module.GNUTranslations):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    60
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    61
    This class sets up the GNUTranslations context with regard to output
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    62
    charset. Django uses a defined DEFAULT_CHARSET as the output charset on
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    63
    Python 2.4.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    64
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    65
    def __init__(self, *args, **kw):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    66
        from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    67
        gettext_module.GNUTranslations.__init__(self, *args, **kw)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    68
        # Starting with Python 2.4, there's a function to define
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    69
        # the output charset. Before 2.4, the output charset is
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    70
        # identical with the translation file charset.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    71
        try:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    72
            self.set_output_charset('utf-8')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    73
        except AttributeError:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    74
            pass
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    75
        self.django_output_charset = 'utf-8'
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    76
        self.__language = '??'
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    77
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    78
    def merge(self, other):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    79
        self._catalog.update(other._catalog)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    80
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    81
    def set_language(self, language):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    82
        self.__language = language
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    83
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    84
    def language(self):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    85
        return self.__language
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    86
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    87
    def __repr__(self):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    88
        return "<DjangoTranslation lang:%s>" % self.__language
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    89
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    90
def translation(language):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    91
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    92
    Returns a translation object.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    93
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    94
    This translation object will be constructed out of multiple GNUTranslations
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    95
    objects by merging their catalogs. It will construct a object for the
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    96
    requested language and add a fallback to the default language, if it's
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    97
    different from the requested language.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    98
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
    99
    global _translations
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   100
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   101
    t = _translations.get(language, None)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   102
    if t is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   103
        return t
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   104
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   105
    from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   106
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   107
    globalpath = os.path.join(os.path.dirname(sys.modules[settings.__module__].__file__), 'locale')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   108
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   109
    if settings.SETTINGS_MODULE is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   110
        parts = settings.SETTINGS_MODULE.split('.')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   111
        project = import_module(parts[0])
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   112
        projectpath = os.path.join(os.path.dirname(project.__file__), 'locale')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   113
    else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   114
        projectpath = None
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   115
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   116
    def _fetch(lang, fallback=None):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   117
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   118
        global _translations
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   119
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   120
        loc = to_locale(lang)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   121
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   122
        res = _translations.get(lang, None)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   123
        if res is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   124
            return res
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   125
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   126
        def _translation(path):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   127
            try:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   128
                t = gettext_module.translation('django', path, [loc], DjangoTranslation)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   129
                t.set_language(lang)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   130
                return t
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   131
            except IOError, e:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   132
                return None
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   133
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   134
        res = _translation(globalpath)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   135
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   136
        # We want to ensure that, for example,  "en-gb" and "en-us" don't share
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   137
        # the same translation object (thus, merging en-us with a local update
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   138
        # doesn't affect en-gb), even though they will both use the core "en"
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   139
        # translation. So we have to subvert Python's internal gettext caching.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   140
        base_lang = lambda x: x.split('-', 1)[0]
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   141
        if base_lang(lang) in [base_lang(trans) for trans in _translations]:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   142
            res._info = res._info.copy()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   143
            res._catalog = res._catalog.copy()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   144
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   145
        def _merge(path):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   146
            t = _translation(path)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   147
            if t is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   148
                if res is None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   149
                    return t
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   150
                else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   151
                    res.merge(t)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   152
            return res
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   153
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   154
        for localepath in settings.LOCALE_PATHS:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   155
            if os.path.isdir(localepath):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   156
                res = _merge(localepath)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   157
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   158
        for appname in settings.INSTALLED_APPS:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   159
            app = import_module(appname)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   160
            apppath = os.path.join(os.path.dirname(app.__file__), 'locale')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   161
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   162
            if os.path.isdir(apppath):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   163
                res = _merge(apppath)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   164
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   165
        if projectpath and os.path.isdir(projectpath):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   166
            res = _merge(projectpath)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   167
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   168
        if res is None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   169
            if fallback is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   170
                res = fallback
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   171
            else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   172
                return gettext_module.NullTranslations()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   173
        _translations[lang] = res
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   174
        return res
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   175
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   176
    default_translation = _fetch(settings.LANGUAGE_CODE)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   177
    current_translation = _fetch(language, fallback=default_translation)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   178
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   179
    return current_translation
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   180
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   181
def activate(language):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   182
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   183
    Fetches the translation object for a given tuple of application name and
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   184
    language and installs it as the current translation object for the current
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   185
    thread.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   186
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   187
    if isinstance(language, basestring) and language == 'no':
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   188
        warnings.warn(
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   189
            "The use of the language code 'no' is deprecated. "
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   190
            "Please use the 'nb' translation instead.",
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   191
            PendingDeprecationWarning
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   192
        )
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   193
    _active[currentThread()] = translation(language)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   194
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   195
def deactivate():
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   196
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   197
    Deinstalls the currently active translation object so that further _ calls
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   198
    will resolve against the default translation object, again.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   199
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   200
    global _active
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   201
    if currentThread() in _active:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   202
        del _active[currentThread()]
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   203
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   204
def deactivate_all():
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   205
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   206
    Makes the active translation object a NullTranslations() instance. This is
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   207
    useful when we want delayed translations to appear as the original string
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   208
    for some reason.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   209
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   210
    _active[currentThread()] = gettext_module.NullTranslations()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   211
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   212
def get_language():
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   213
    """Returns the currently selected language."""
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   214
    t = _active.get(currentThread(), None)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   215
    if t is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   216
        try:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   217
            return to_language(t.language())
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   218
        except AttributeError:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   219
            pass
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   220
    # If we don't have a real translation object, assume it's the default language.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   221
    from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   222
    return settings.LANGUAGE_CODE
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   223
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   224
def get_language_bidi():
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   225
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   226
    Returns selected language's BiDi layout.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   227
    
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   228
    * False = left-to-right layout
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   229
    * True = right-to-left layout
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   230
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   231
    from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   232
    
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   233
    base_lang = get_language().split('-')[0]
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   234
    return base_lang in settings.LANGUAGES_BIDI
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   235
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   236
def catalog():
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   237
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   238
    Returns the current active catalog for further processing.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   239
    This can be used if you need to modify the catalog or want to access the
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   240
    whole message catalog instead of just translating one string.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   241
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   242
    global _default, _active
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   243
    t = _active.get(currentThread(), None)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   244
    if t is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   245
        return t
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   246
    if _default is None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   247
        from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   248
        _default = translation(settings.LANGUAGE_CODE)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   249
    return _default
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   250
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   251
def do_translate(message, translation_function):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   252
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   253
    Translates 'message' using the given 'translation_function' name -- which
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   254
    will be either gettext or ugettext. It uses the current thread to find the
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   255
    translation object to use. If no current translation is activated, the
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   256
    message will be run through the default translation object.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   257
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   258
    eol_message = message.replace('\r\n', '\n').replace('\r', '\n')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   259
    global _default, _active
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   260
    t = _active.get(currentThread(), None)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   261
    if t is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   262
        result = getattr(t, translation_function)(eol_message)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   263
    else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   264
        if _default is None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   265
            from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   266
            _default = translation(settings.LANGUAGE_CODE)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   267
        result = getattr(_default, translation_function)(eol_message)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   268
    if isinstance(message, SafeData):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   269
        return mark_safe(result)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   270
    return result
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   271
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   272
def gettext(message):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   273
    return do_translate(message, 'gettext')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   274
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   275
def ugettext(message):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   276
    return do_translate(message, 'ugettext')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   277
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   278
def gettext_noop(message):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   279
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   280
    Marks strings for translation but doesn't translate them now. This can be
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   281
    used to store strings in global variables that should stay in the base
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   282
    language (because they might be used externally) and will be translated
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   283
    later.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   284
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   285
    return message
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   286
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   287
def do_ntranslate(singular, plural, number, translation_function):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   288
    global _default, _active
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   289
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   290
    t = _active.get(currentThread(), None)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   291
    if t is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   292
        return getattr(t, translation_function)(singular, plural, number)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   293
    if _default is None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   294
        from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   295
        _default = translation(settings.LANGUAGE_CODE)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   296
    return getattr(_default, translation_function)(singular, plural, number)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   297
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   298
def ngettext(singular, plural, number):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   299
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   300
    Returns a UTF-8 bytestring of the translation of either the singular or
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   301
    plural, based on the number.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   302
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   303
    return do_ntranslate(singular, plural, number, 'ngettext')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   304
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   305
def ungettext(singular, plural, number):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   306
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   307
    Returns a unicode strings of the translation of either the singular or
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   308
    plural, based on the number.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   309
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   310
    return do_ntranslate(singular, plural, number, 'ungettext')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   311
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   312
def check_for_language(lang_code):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   313
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   314
    Checks whether there is a global language file for the given language
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   315
    code. This is used to decide whether a user-provided language is
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   316
    available. This is only used for language codes from either the cookies or
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   317
    session.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   318
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   319
    from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   320
    globalpath = os.path.join(os.path.dirname(sys.modules[settings.__module__].__file__), 'locale')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   321
    if gettext_module.find('django', globalpath, [to_locale(lang_code)]) is not None:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   322
        return True
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   323
    else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   324
        return False
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   325
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   326
def get_language_from_request(request):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   327
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   328
    Analyzes the request to find what language the user wants the system to
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   329
    show. Only languages listed in settings.LANGUAGES are taken into account.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   330
    If the user requests a sublanguage where we have a main language, we send
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   331
    out the main language.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   332
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   333
    global _accepted
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   334
    from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   335
    globalpath = os.path.join(os.path.dirname(sys.modules[settings.__module__].__file__), 'locale')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   336
    supported = dict(settings.LANGUAGES)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   337
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   338
    if hasattr(request, 'session'):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   339
        lang_code = request.session.get('django_language', None)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   340
        if lang_code in supported and lang_code is not None and check_for_language(lang_code):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   341
            return lang_code
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   342
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   343
    lang_code = request.COOKIES.get(settings.LANGUAGE_COOKIE_NAME)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   344
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   345
    if lang_code and lang_code not in supported:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   346
        lang_code = lang_code.split('-')[0] # e.g. if fr-ca is not supported fallback to fr
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   347
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   348
    if lang_code and lang_code in supported and check_for_language(lang_code):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   349
        return lang_code
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   350
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   351
    accept = request.META.get('HTTP_ACCEPT_LANGUAGE', '')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   352
    for accept_lang, unused in parse_accept_lang_header(accept):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   353
        if accept_lang == '*':
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   354
            break
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   355
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   356
        # We have a very restricted form for our language files (no encoding
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   357
        # specifier, since they all must be UTF-8 and only one possible
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   358
        # language each time. So we avoid the overhead of gettext.find() and
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   359
        # work out the MO file manually.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   360
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   361
        # 'normalized' is the root name of the locale in POSIX format (which is
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   362
        # the format used for the directories holding the MO files).
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   363
        normalized = locale.locale_alias.get(to_locale(accept_lang, True))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   364
        if not normalized:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   365
            continue
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   366
        # Remove the default encoding from locale_alias.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   367
        normalized = normalized.split('.')[0]
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   368
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   369
        if normalized in _accepted:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   370
            # We've seen this locale before and have an MO file for it, so no
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   371
            # need to check again.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   372
            return _accepted[normalized]
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   373
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   374
        for lang, dirname in ((accept_lang, normalized),
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   375
                (accept_lang.split('-')[0], normalized.split('_')[0])):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   376
            if lang.lower() not in supported:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   377
                continue
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   378
            langfile = os.path.join(globalpath, dirname, 'LC_MESSAGES',
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   379
                    'django.mo')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   380
            if os.path.exists(langfile):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   381
                _accepted[normalized] = lang
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   382
                return lang
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   383
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   384
    return settings.LANGUAGE_CODE
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   385
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   386
dot_re = re.compile(r'\S')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   387
def blankout(src, char):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   388
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   389
    Changes every non-whitespace character to the given char.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   390
    Used in the templatize function.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   391
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   392
    return dot_re.sub(char, src)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   393
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   394
inline_re = re.compile(r"""^\s*trans\s+((?:".*?")|(?:'.*?'))\s*""")
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   395
block_re = re.compile(r"""^\s*blocktrans(?:\s+|$)""")
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   396
endblock_re = re.compile(r"""^\s*endblocktrans$""")
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   397
plural_re = re.compile(r"""^\s*plural$""")
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   398
constant_re = re.compile(r"""_\(((?:".*?")|(?:'.*?'))\)""")
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   399
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   400
def templatize(src):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   401
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   402
    Turns a Django template into something that is understood by xgettext. It
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   403
    does so by translating the Django translation tags into standard gettext
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   404
    function invocations.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   405
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   406
    from django.template import Lexer, TOKEN_TEXT, TOKEN_VAR, TOKEN_BLOCK
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   407
    out = StringIO()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   408
    intrans = False
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   409
    inplural = False
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   410
    singular = []
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   411
    plural = []
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   412
    for t in Lexer(src, None).tokenize():
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   413
        if intrans:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   414
            if t.token_type == TOKEN_BLOCK:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   415
                endbmatch = endblock_re.match(t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   416
                pluralmatch = plural_re.match(t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   417
                if endbmatch:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   418
                    if inplural:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   419
                        out.write(' ngettext(%r,%r,count) ' % (''.join(singular), ''.join(plural)))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   420
                        for part in singular:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   421
                            out.write(blankout(part, 'S'))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   422
                        for part in plural:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   423
                            out.write(blankout(part, 'P'))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   424
                    else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   425
                        out.write(' gettext(%r) ' % ''.join(singular))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   426
                        for part in singular:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   427
                            out.write(blankout(part, 'S'))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   428
                    intrans = False
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   429
                    inplural = False
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   430
                    singular = []
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   431
                    plural = []
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   432
                elif pluralmatch:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   433
                    inplural = True
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   434
                else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   435
                    raise SyntaxError("Translation blocks must not include other block tags: %s" % t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   436
            elif t.token_type == TOKEN_VAR:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   437
                if inplural:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   438
                    plural.append('%%(%s)s' % t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   439
                else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   440
                    singular.append('%%(%s)s' % t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   441
            elif t.token_type == TOKEN_TEXT:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   442
                if inplural:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   443
                    plural.append(t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   444
                else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   445
                    singular.append(t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   446
        else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   447
            if t.token_type == TOKEN_BLOCK:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   448
                imatch = inline_re.match(t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   449
                bmatch = block_re.match(t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   450
                cmatches = constant_re.findall(t.contents)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   451
                if imatch:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   452
                    g = imatch.group(1)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   453
                    if g[0] == '"': g = g.strip('"')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   454
                    elif g[0] == "'": g = g.strip("'")
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   455
                    out.write(' gettext(%r) ' % g)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   456
                elif bmatch:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   457
                    for fmatch in constant_re.findall(t.contents):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   458
                        out.write(' _(%s) ' % fmatch)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   459
                    intrans = True
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   460
                    inplural = False
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   461
                    singular = []
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   462
                    plural = []
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   463
                elif cmatches:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   464
                    for cmatch in cmatches:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   465
                        out.write(' _(%s) ' % cmatch)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   466
                else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   467
                    out.write(blankout(t.contents, 'B'))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   468
            elif t.token_type == TOKEN_VAR:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   469
                parts = t.contents.split('|')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   470
                cmatch = constant_re.match(parts[0])
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   471
                if cmatch:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   472
                    out.write(' _(%s) ' % cmatch.group(1))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   473
                for p in parts[1:]:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   474
                    if p.find(':_(') >= 0:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   475
                        out.write(' %s ' % p.split(':',1)[1])
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   476
                    else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   477
                        out.write(blankout(p, 'F'))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   478
            else:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   479
                out.write(blankout(t.contents, 'X'))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   480
    return out.getvalue()
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   481
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   482
def parse_accept_lang_header(lang_string):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   483
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   484
    Parses the lang_string, which is the body of an HTTP Accept-Language
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   485
    header, and returns a list of (lang, q-value), ordered by 'q' values.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   486
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   487
    Any format errors in lang_string results in an empty list being returned.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   488
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   489
    result = []
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   490
    pieces = accept_language_re.split(lang_string)
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   491
    if pieces[-1]:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   492
        return []
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   493
    for i in range(0, len(pieces) - 1, 3):
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   494
        first, lang, priority = pieces[i : i + 3]
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   495
        if first:
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   496
            return []
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   497
        priority = priority and float(priority) or 1.0
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   498
        result.append((lang, priority))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   499
    result.sort(lambda x, y: -cmp(x[1], y[1]))
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   500
    return result
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   501
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   502
# get_date_formats and get_partial_date_formats aren't used anymore by Django
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   503
# and are kept for backward compatibility.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   504
# Note, it's also important to keep format names marked for translation.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   505
# For compatibility we still want to have formats on translation catalogs.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   506
# That makes template code like {{ my_date|date:_('DATE_FORMAT') }} still work
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   507
def get_date_formats():
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   508
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   509
    Checks whether translation files provide a translation for some technical
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   510
    message ID to store date and time formats. If it doesn't contain one, the
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   511
    formats provided in the settings will be used.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   512
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   513
    warnings.warn(
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   514
        "'django.utils.translation.get_date_formats' is deprecated. "
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   515
        "Please update your code to use the new i18n aware formatting.",
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   516
        PendingDeprecationWarning
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   517
    )
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   518
    from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   519
    date_format = ugettext('DATE_FORMAT')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   520
    datetime_format = ugettext('DATETIME_FORMAT')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   521
    time_format = ugettext('TIME_FORMAT')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   522
    if date_format == 'DATE_FORMAT':
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   523
        date_format = settings.DATE_FORMAT
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   524
    if datetime_format == 'DATETIME_FORMAT':
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   525
        datetime_format = settings.DATETIME_FORMAT
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   526
    if time_format == 'TIME_FORMAT':
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   527
        time_format = settings.TIME_FORMAT
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   528
    return date_format, datetime_format, time_format
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   529
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   530
def get_partial_date_formats():
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   531
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   532
    Checks whether translation files provide a translation for some technical
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   533
    message ID to store partial date formats. If it doesn't contain one, the
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   534
    formats provided in the settings will be used.
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   535
    """
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   536
    warnings.warn(
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   537
        "'django.utils.translation.get_partial_date_formats' is deprecated. "
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   538
        "Please update your code to use the new i18n aware formatting.",
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   539
        PendingDeprecationWarning
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   540
    )
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   541
    from django.conf import settings
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   542
    year_month_format = ugettext('YEAR_MONTH_FORMAT')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   543
    month_day_format = ugettext('MONTH_DAY_FORMAT')
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   544
    if year_month_format == 'YEAR_MONTH_FORMAT':
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   545
        year_month_format = settings.YEAR_MONTH_FORMAT
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   546
    if month_day_format == 'MONTH_DAY_FORMAT':
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   547
        month_day_format = settings.MONTH_DAY_FORMAT
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   548
    return year_month_format, month_day_format
77b6da96e6f1 update django
ymh <ymh.work@gmail.com>
parents:
diff changeset
   549