{
  "instance_id": "django__django-12700",
  "repo": "django/django",
  "created_at": "2020-04-11T01:58:27Z",
  "problem_statement": "Settings are cleaned insufficiently.\nDescription\n\t\nPosting publicly after checking with the rest of the security team.\nI just ran into a case where django.views.debug.SafeExceptionReporterFilter.get_safe_settings() would return several un-cleansed values. Looking at cleanse_setting() I realized that we \u200bonly take care of `dict`s but don't take other types of iterables into account but \u200breturn them as-is.\nExample:\nIn my settings.py I have this:\nMY_SETTING = {\n\t\"foo\": \"value\",\n\t\"secret\": \"value\",\n\t\"token\": \"value\",\n\t\"something\": [\n\t\t{\"foo\": \"value\"},\n\t\t{\"secret\": \"value\"},\n\t\t{\"token\": \"value\"},\n\t],\n\t\"else\": [\n\t\t[\n\t\t\t{\"foo\": \"value\"},\n\t\t\t{\"secret\": \"value\"},\n\t\t\t{\"token\": \"value\"},\n\t\t],\n\t\t[\n\t\t\t{\"foo\": \"value\"},\n\t\t\t{\"secret\": \"value\"},\n\t\t\t{\"token\": \"value\"},\n\t\t],\n\t]\n}\nOn Django 3.0 and below:\n>>> import pprint\n>>> from django.views.debug import get_safe_settings\n>>> pprint.pprint(get_safe_settings()[\"MY_SETTING\"])\n{'else': [[{'foo': 'value'}, {'secret': 'value'}, {'token': 'value'}],\n\t\t [{'foo': 'value'}, {'secret': 'value'}, {'token': 'value'}]],\n 'foo': 'value',\n 'secret': '********************',\n 'something': [{'foo': 'value'}, {'secret': 'value'}, {'token': 'value'}],\n 'token': '********************'}\nOn Django 3.1 and up:\n>>> from django.views.debug import SafeExceptionReporterFilter\n>>> import pprint\n>>> pprint.pprint(SafeExceptionReporterFilter().get_safe_settings()[\"MY_SETTING\"])\n{'else': [[{'foo': 'value'}, {'secret': 'value'}, {'token': 'value'}],\n\t\t [{'foo': 'value'}, {'secret': 'value'}, {'token': 'value'}]],\n 'foo': 'value',\n 'secret': '********************',\n 'something': [{'foo': 'value'}, {'secret': 'value'}, {'token': 'value'}],\n 'token': '********************'}\n",
  "patch": "diff --git a/django/views/debug.py b/django/views/debug.py\n--- a/django/views/debug.py\n+++ b/django/views/debug.py\n@@ -90,6 +90,10 @@ def cleanse_setting(self, key, value):\n                 cleansed = self.cleansed_substitute\n             elif isinstance(value, dict):\n                 cleansed = {k: self.cleanse_setting(k, v) for k, v in value.items()}\n+            elif isinstance(value, list):\n+                cleansed = [self.cleanse_setting('', v) for v in value]\n+            elif isinstance(value, tuple):\n+                cleansed = tuple([self.cleanse_setting('', v) for v in value])\n             else:\n                 cleansed = value\n         except TypeError:\n",
  "similar_bug_items": [
    {
      "pr_number": 11306,
      "pr_title": "Fixed #30408 -- Fixed crash when adding check constraints with LIKE operator on Oracle and PostgreSQL.",
      "pr_body": "The LIKE operator wildcard generated for contains, startswith, endswith and\r\ntheir case-insensitive variant lookups was conflicting with parameter\r\ninterpolation on CREATE constraint statement execution.\r\n\r\nIdeally we'd delegate parameters interpolation in DDL statements on backends\r\nthat support it but that would require backward incompatible changes to the\r\nIndex and Constraint SQL generating methods.\r\n\r\nThanks David Sanders for the report.",
      "issue_id": 30408,
      "issue_title": "CheckConstraint with lookup using LIKE & % crash on Oracle and PostgreSQL.",
      "issue_body": "",
      "issue_closed_at": "2019-04-30T01:54:41",
      "base_commit": "673fe2e3ec63614259e86e7a370b9d1e91fcc1e1",
      "changes": [
        {
          "file": "django/db/backends/oracle/schema.py",
          "type": "function",
          "name": "quote_value",
          "class_name": "DatabaseSchemaEditor",
          "code": "def quote_value(self, value):\n        if isinstance(value, (datetime.date, datetime.time, datetime.datetime)):\n            return \"'%s'\" % value\n        elif isinstance(value, str):\n            return \"'%s'\" % value.replace(\"\\'\", \"\\'\\'\")\n        elif isinstance(value, (bytes, bytearray, memoryview)):\n            return \"'%s'\" % value.hex()\n        elif isinstance(value, bool):\n            return \"1\" if value else \"0\"\n        else:\n            return str(value)"
        },
        {
          "file": "django/db/backends/postgresql/schema.py",
          "type": "class",
          "name": "DatabaseSchemaEditor",
          "code": "class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):\n\n    sql_alter_column_type = \"ALTER COLUMN %(column)s TYPE %(type)s USING %(column)s::%(type)s\"\n\n    sql_create_sequence = \"CREATE SEQUENCE %(sequence)s\"\n    sql_delete_sequence = \"DROP SEQUENCE IF EXISTS %(sequence)s CASCADE\"\n    sql_set_sequence_max = \"SELECT setval('%(sequence)s', MAX(%(column)s)) FROM %(table)s\"\n    sql_set_sequence_owner = 'ALTER SEQUENCE %(sequence)s OWNED BY %(table)s.%(column)s'\n\n    sql_create_index = \"CREATE INDEX %(name)s ON %(table)s%(using)s (%(columns)s)%(extra)s%(condition)s\"\n    sql_delete_index = \"DROP INDEX IF EXISTS %(name)s\"\n\n    sql_create_column_inline_fk = 'REFERENCES %(to_table)s(%(to_column)s)%(deferrable)s'\n    # Setting the constraint to IMMEDIATE runs any deferred checks to allow\n    # dropping it in the same transaction.\n    sql_delete_fk = \"SET CONSTRAINTS %(name)s IMMEDIATE; ALTER TABLE %(table)s DROP CONSTRAINT %(name)s\"\n\n    sql_delete_procedure = 'DROP FUNCTION %(procedure)s(%(param_types)s)'\n\n    def quote_value(self, value):\n        # getquoted() returns a quoted bytestring of the adapted value.\n        return psycopg2.extensions.adapt(value).getquoted().decode()\n\n    def _field_indexes_sql(self, model, field):\n        output = super()._field_indexes_sql(model, field)\n        like_index_statement = self._create_like_index_sql(model, field)\n        if like_index_statement is not None:\n            output.append(like_index_statement)\n        return output\n\n    def _create_like_index_sql(self, model, field):\n        \"\"\"\n        Return the statement to create an index with varchar operator pattern\n        when the column type is 'varchar' or 'text', otherwise return None.\n        \"\"\"\n        db_type = field.db_type(connection=self.connection)\n        if db_type is not None and (field.db_index or field.unique):\n            # Fields with database column types of `varchar` and `text` need\n            # a second index that specifies their operator class, which is\n            # needed when performing correct LIKE queries outside the\n            # C locale. See #12234.\n            #\n            # The same doesn't apply to array fields such as varchar[size]\n            # and text[size], so skip them.\n            if '[' in db_type:\n                return None\n            if db_type.startswith('varchar'):\n                return self._create_index_sql(model, [field], suffix='_like', opclasses=['varchar_pattern_ops'])\n            elif db_type.startswith('text'):\n                return self._create_index_sql(model, [field], suffix='_like', opclasses=['text_pattern_ops'])\n        return None\n\n    def _alter_column_type_sql(self, model, old_field, new_field, new_type):\n        \"\"\"Make ALTER TYPE with SERIAL make sense.\"\"\"\n        table = model._meta.db_table\n        if new_type.lower() in (\"serial\", \"bigserial\"):\n            column = new_field.column\n            sequence_name = \"%s_%s_seq\" % (table, column)\n            col_type = \"integer\" if new_type.lower() == \"serial\" else \"bigint\"\n            return (\n                (\n                    self.sql_alter_column_type % {\n                        \"column\": self.quote_name(column),\n                        \"type\": col_type,\n                    },\n                    [],\n                ),\n                [\n                    (\n                        self.sql_delete_sequence % {\n                            \"sequence\": self.quote_name(sequence_name),\n                        },\n                        [],\n                    ),\n                    (\n                        self.sql_create_sequence % {\n                            \"sequence\": self.quote_name(sequence_name),\n                        },\n                        [],\n                    ),\n                    (\n                        self.sql_alter_column % {\n                            \"table\": self.quote_name(table),\n                            \"changes\": self.sql_alter_column_default % {\n                                \"column\": self.quote_name(column),\n                                \"default\": \"nextval('%s')\" % self.quote_name(sequence_name),\n                            }\n                        },\n                        [],\n                    ),\n                    (\n                        self.sql_set_sequence_max % {\n                            \"table\": self.quote_name(table),\n                            \"column\": self.quote_name(column),\n                            \"sequence\": self.quote_name(sequence_name),\n                        },\n                        [],\n                    ),\n                    (\n                        self.sql_set_sequence_owner % {\n                            'table': self.quote_name(table),\n                            'column': self.quote_name(column),\n                            'sequence': self.quote_name(sequence_name),\n                        },\n                        [],\n                    ),\n                ],\n            )\n        else:\n            return super()._alter_column_type_sql(model, old_field, new_field, new_type)\n\n    def _alter_field(self, model, old_field, new_field, old_type, new_type,\n                     old_db_params, new_db_params, strict=False):\n        # Drop indexes on varchar/text/citext columns that are changing to a\n        # different type.\n        if (old_field.db_index or old_field.unique) and (\n            (old_type.startswith('varchar') and not new_type.startswith('varchar')) or\n            (old_type.startswith('text') and not new_type.startswith('text')) or\n            (old_type.startswith('citext') and not new_type.startswith('citext'))\n        ):\n            index_name = self._create_index_name(model._meta.db_table, [old_field.column], suffix='_like')\n            self.execute(self._delete_index_sql(model, index_name))\n\n        super()._alter_field(\n            model, old_field, new_field, old_type, new_type, old_db_params,\n            new_db_params, strict,\n        )\n        # Added an index? Create any PostgreSQL-specific indexes.\n        if ((not (old_field.db_index or old_field.unique) and new_field.db_index) or\n                (not old_field.unique and new_field.unique)):\n            like_index_statement = self._create_like_index_sql(model, new_field)\n            if like_index_statement is not None:\n                self.execute(like_index_statement)\n\n        # Removed an index? Drop any PostgreSQL-specific indexes.\n        if old_field.unique and not (new_field.db_index or new_field.unique):\n            index_to_remove = self._create_index_name(model._meta.db_table, [old_field.column], suffix='_like')\n            self.execute(self._delete_index_sql(model, index_to_remove))\n\n    def _index_columns(self, table, columns, col_suffixes, opclasses):\n        if opclasses:\n            return IndexColumns(table, columns, self.quote_name, col_suffixes=col_suffixes, opclasses=opclasses)\n        return super()._index_columns(table, columns, col_suffixes, opclasses)"
        }
      ]
    },
    {
      "pr_number": 11170,
      "pr_title": "Fixed #30324 -- Forced utf-8 encoding when loading debug templates.",
      "pr_body": "Refs #28007 and #29654.\r\n\r\nRegression in ea542a9c7239b5b665797b7c809f1aceb0b412cf and 50b8493581fea3d7137dd8db33bac7008868d23a.",
      "issue_id": 30324,
      "issue_title": "UnicodeDecodeError when loading debug templates.",
      "issue_body": "",
      "issue_closed_at": "2019-04-05T09:35:37",
      "base_commit": "9012033138fa41b573d3e4e3f0dfa8b94a4719c6",
      "changes": [
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "get_traceback_data",
          "class_name": "ExceptionReporter",
          "code": "def get_traceback_data(self):\n        \"\"\"Return a dictionary containing traceback information.\"\"\"\n        if self.exc_type and issubclass(self.exc_type, TemplateDoesNotExist):\n            self.template_does_not_exist = True\n            self.postmortem = self.exc_value.chain or [self.exc_value]\n\n        frames = self.get_traceback_frames()\n        for i, frame in enumerate(frames):\n            if 'vars' in frame:\n                frame_vars = []\n                for k, v in frame['vars']:\n                    v = pprint(v)\n                    # Trim large blobs of data\n                    if len(v) > 4096:\n                        v = '%s\u2026 <trimmed %d bytes string>' % (v[0:4096], len(v))\n                    frame_vars.append((k, v))\n                frame['vars'] = frame_vars\n            frames[i] = frame\n\n        unicode_hint = ''\n        if self.exc_type and issubclass(self.exc_type, UnicodeError):\n            start = getattr(self.exc_value, 'start', None)\n            end = getattr(self.exc_value, 'end', None)\n            if start is not None and end is not None:\n                unicode_str = self.exc_value.args[1]\n                unicode_hint = force_str(\n                    unicode_str[max(start - 5, 0):min(end + 5, len(unicode_str))],\n                    'ascii', errors='replace'\n                )\n        from django import get_version\n\n        if self.request is None:\n            user_str = None\n        else:\n            try:\n                user_str = str(self.request.user)\n            except Exception:\n                # request.user may raise OperationalError if the database is\n                # unavailable, for example.\n                user_str = '[unable to retrieve the current user]'\n\n        c = {\n            'is_email': self.is_email,\n            'unicode_hint': unicode_hint,\n            'frames': frames,\n            'request': self.request,\n            'user_str': user_str,\n            'filtered_POST_items': list(self.filter.get_post_parameters(self.request).items()),\n            'settings': get_safe_settings(),\n            'sys_executable': sys.executable,\n            'sys_version_info': '%d.%d.%d' % sys.version_info[0:3],\n            'server_time': timezone.now(),\n            'django_version_info': get_version(),\n            'sys_path': sys.path,\n            'template_info': self.template_info,\n            'template_does_not_exist': self.template_does_not_exist,\n            'postmortem': self.postmortem,\n        }\n        if self.request is not None:\n            c['request_GET_items'] = self.request.GET.items()\n            c['request_FILES_items'] = self.request.FILES.items()\n            c['request_COOKIES_items'] = self.request.COOKIES.items()\n        # Check whether exception info is available\n        if self.exc_type:\n            c['exception_type'] = self.exc_type.__name__\n        if self.exc_value:\n            c['exception_value'] = str(self.exc_value)\n        if frames:\n            c['lastframe'] = frames[-1]\n        return c"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "technical_404_response",
          "class_name": null,
          "code": "def technical_404_response(request, exception):\n    \"\"\"Create a technical 404 error response. `exception` is the Http404.\"\"\"\n    try:\n        error_url = exception.args[0]['path']\n    except (IndexError, TypeError, KeyError):\n        error_url = request.path_info[1:]  # Trim leading slash\n\n    try:\n        tried = exception.args[0]['tried']\n    except (IndexError, TypeError, KeyError):\n        tried = []\n    else:\n        if (not tried or (                  # empty URLconf\n            request.path == '/' and\n            len(tried) == 1 and             # default URLconf\n            len(tried[0]) == 1 and\n            getattr(tried[0][0], 'app_name', '') == getattr(tried[0][0], 'namespace', '') == 'admin'\n        )):\n            return default_urlconf(request)\n\n    urlconf = getattr(request, 'urlconf', settings.ROOT_URLCONF)\n    if isinstance(urlconf, types.ModuleType):\n        urlconf = urlconf.__name__\n\n    caller = ''\n    try:\n        resolver_match = resolve(request.path)\n    except Resolver404:\n        pass\n    else:\n        obj = resolver_match.func\n\n        if hasattr(obj, '__name__'):\n            caller = obj.__name__\n        elif hasattr(obj, '__class__') and hasattr(obj.__class__, '__name__'):\n            caller = obj.__class__.__name__\n\n        if hasattr(obj, '__module__'):\n            module = obj.__module__\n            caller = '%s.%s' % (module, caller)\n\n    with Path(CURRENT_DIR, 'templates', 'technical_404.html').open() as fh:\n        t = DEBUG_ENGINE.from_string(fh.read())\n    c = Context({\n        'urlconf': urlconf,\n        'root_urlconf': settings.ROOT_URLCONF,\n        'request_path': error_url,\n        'urlpatterns': tried,\n        'reason': str(exception),\n        'request': request,\n        'settings': get_safe_settings(),\n        'raising_view_name': caller,\n    })\n    return HttpResponseNotFound(t.render(c), content_type='text/html')"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "technical_404_response",
          "class_name": null,
          "code": "def technical_404_response(request, exception):\n    \"\"\"Create a technical 404 error response. `exception` is the Http404.\"\"\"\n    try:\n        error_url = exception.args[0]['path']\n    except (IndexError, TypeError, KeyError):\n        error_url = request.path_info[1:]  # Trim leading slash\n\n    try:\n        tried = exception.args[0]['tried']\n    except (IndexError, TypeError, KeyError):\n        tried = []\n    else:\n        if (not tried or (                  # empty URLconf\n            request.path == '/' and\n            len(tried) == 1 and             # default URLconf\n            len(tried[0]) == 1 and\n            getattr(tried[0][0], 'app_name', '') == getattr(tried[0][0], 'namespace', '') == 'admin'\n        )):\n            return default_urlconf(request)\n\n    urlconf = getattr(request, 'urlconf', settings.ROOT_URLCONF)\n    if isinstance(urlconf, types.ModuleType):\n        urlconf = urlconf.__name__\n\n    caller = ''\n    try:\n        resolver_match = resolve(request.path)\n    except Resolver404:\n        pass\n    else:\n        obj = resolver_match.func\n\n        if hasattr(obj, '__name__'):\n            caller = obj.__name__\n        elif hasattr(obj, '__class__') and hasattr(obj.__class__, '__name__'):\n            caller = obj.__class__.__name__\n\n        if hasattr(obj, '__module__'):\n            module = obj.__module__\n            caller = '%s.%s' % (module, caller)\n\n    with Path(CURRENT_DIR, 'templates', 'technical_404.html').open() as fh:\n        t = DEBUG_ENGINE.from_string(fh.read())\n    c = Context({\n        'urlconf': urlconf,\n        'root_urlconf': settings.ROOT_URLCONF,\n        'request_path': error_url,\n        'urlpatterns': tried,\n        'reason': str(exception),\n        'request': request,\n        'settings': get_safe_settings(),\n        'raising_view_name': caller,\n    })\n    return HttpResponseNotFound(t.render(c), content_type='text/html')"
        }
      ]
    },
    {
      "pr_number": 6754,
      "pr_title": "Fixed #26736 -- Improved unicode handling for SpatialReference.",
      "pr_body": "https://code.djangoproject.com/ticket/26736\n",
      "issue_id": 26736,
      "issue_title": "SpatialReference crashes when initialized with WKT containining unicode characters",
      "issue_body": "",
      "issue_closed_at": "2016-06-11T20:00:40",
      "base_commit": "0451dcc2eb2449a988ade8e603846f0508ce76b4",
      "changes": [
        {
          "file": "django/contrib/gis/gdal/prototypes/srs.py",
          "type": "function",
          "name": "units_func",
          "class_name": null,
          "code": "def units_func(f):\n    \"\"\"\n    Creates a ctypes function prototype for OSR units functions, e.g.,\n    OSRGetAngularUnits, OSRGetLinearUnits.\n    \"\"\"\n    return double_output(f, [c_void_p, POINTER(c_char_p)], strarg=True)"
        },
        {
          "file": "django/contrib/gis/gdal/srs.py",
          "type": "function",
          "name": "__init__",
          "class_name": "CoordTransform",
          "code": "def __init__(self, source, target):\n        \"Initializes on a source and target SpatialReference objects.\"\n        if not isinstance(source, SpatialReference) or not isinstance(target, SpatialReference):\n            raise TypeError('source and target must be of type SpatialReference')\n        self.ptr = capi.new_ct(source._ptr, target._ptr)\n        self._srs1_name = source.name\n        self._srs2_name = target.name"
        },
        {
          "file": "django/contrib/gis/gdal/srs.py",
          "type": "function",
          "name": "import_user_input",
          "class_name": "SpatialReference",
          "code": "def import_user_input(self, user_input):\n        \"Imports the Spatial Reference from the given user input string.\"\n        capi.from_user_input(self.ptr, force_bytes(user_input))"
        },
        {
          "file": "django/contrib/gis/gdal/srs.py",
          "type": "function",
          "name": "proj4",
          "class_name": "SpatialReference",
          "code": "def proj4(self):\n        \"Alias for proj().\"\n        return self.proj"
        }
      ]
    },
    {
      "pr_number": 5437,
      "pr_title": "Fixed #25560 -- Made empty string related name invalid.",
      "pr_body": "Thanks to Ali Lotfi for the initial report and patch.\n",
      "issue_id": 25560,
      "issue_title": "Empty string related name should be invalid",
      "issue_body": "",
      "issue_closed_at": "2015-10-16T13:18:21",
      "base_commit": "4dcc2a195595f8d7ddad45bc4baf98ffdeec7f41",
      "changes": [
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "_check_related_name_is_valid",
          "class_name": "RelatedField",
          "code": "def _check_related_name_is_valid(self):\n        import re\n        import keyword\n        related_name = self.remote_field.related_name\n        if not related_name:\n            return []\n        is_valid_id = True\n        if keyword.iskeyword(related_name):\n            is_valid_id = False\n        if six.PY3:\n            if not related_name.isidentifier():\n                is_valid_id = False\n        else:\n            if not re.match(r'^[a-zA-Z_][a-zA-Z0-9_]*\\Z', related_name):\n                is_valid_id = False\n        if not (is_valid_id or related_name.endswith('+')):\n            return [\n                checks.Error(\n                    \"The name '%s' is invalid related_name for field %s.%s\" %\n                    (self.remote_field.related_name, self.model._meta.object_name,\n                     self.name),\n                    hint=\"Related name must be a valid Python identifier or end with a '+'\",\n                    obj=self,\n                    id='fields.E306',\n                )\n            ]\n        return []"
        },
        {
          "file": "django/db/models/fields/reverse_related.py",
          "type": "function",
          "name": "get_db_prep_lookup",
          "class_name": "ForeignObjectRel",
          "code": "def get_db_prep_lookup(self, lookup_type, value, connection, prepared=False):\n        # Defer to the actual field definition for db prep\n        return self.field.get_db_prep_lookup(lookup_type, value, connection=connection, prepared=prepared)"
        }
      ]
    },
    {
      "pr_number": 12332,
      "pr_title": "Fixed #30439 - Added support for different plurals for the same language.",
      "pr_body": "ticket-30439",
      "issue_id": 30439,
      "issue_title": "Translations issues on Django upgrade due to unexpected changes in plural forms",
      "issue_body": "",
      "issue_closed_at": "2020-03-10T09:56:53",
      "base_commit": "591e2270dc8c685625be25dbed908a9a3897ba1d",
      "changes": [
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "reset_cache",
          "class_name": null,
          "code": "def reset_cache(**kwargs):\n    \"\"\"\n    Reset global state when LANGUAGES setting has been changed, as some\n    languages should no longer be accepted.\n    \"\"\"\n    if kwargs['setting'] in ('LANGUAGES', 'LANGUAGE_CODE'):\n        check_for_language.cache_clear()\n        get_languages.cache_clear()\n        get_supported_language_variant.cache_clear()"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "__init__",
          "class_name": "DjangoTranslation",
          "code": "def __init__(self, language, domain=None, localedirs=None):\n        \"\"\"Create a GNUTranslations() using many locale directories\"\"\"\n        gettext_module.GNUTranslations.__init__(self)\n        if domain is not None:\n            self.domain = domain\n\n        self.__language = language\n        self.__to_language = to_language(language)\n        self.__locale = to_locale(language)\n        self._catalog = None\n        # If a language doesn't have a catalog, use the Germanic default for\n        # pluralization: anything except one is pluralized.\n        self.plural = lambda n: int(n != 1)\n\n        if self.domain == 'django':\n            if localedirs is not None:\n                # A module-level cache is used for caching 'django' translations\n                warnings.warn(\"localedirs is ignored when domain is 'django'.\", RuntimeWarning)\n                localedirs = None\n            self._init_translation_catalog()\n\n        if localedirs:\n            for localedir in localedirs:\n                translation = self._new_gnu_trans(localedir)\n                self.merge(translation)\n        else:\n            self._add_installed_apps_translations()\n\n        self._add_local_translations()\n        if self.__language == settings.LANGUAGE_CODE and self.domain == 'django' and self._catalog is None:\n            # default lang should have at least one translation file available.\n            raise OSError('No translation files found for default language %s.' % settings.LANGUAGE_CODE)\n        self._add_fallback(localedirs)\n        if self._catalog is None:\n            # No catalogs found for this language, set an empty catalog.\n            self._catalog = {}"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "merge",
          "class_name": "DjangoTranslation",
          "code": "def merge(self, other):\n        \"\"\"Merge another translation into this catalog.\"\"\"\n        if not getattr(other, '_catalog', None):\n            return  # NullTranslations() has no _catalog\n        if self._catalog is None:\n            # Take plural and _info from first catalog found (generally Django's).\n            self.plural = other.plural\n            self._info = other._info.copy()\n            self._catalog = other._catalog.copy()\n        else:\n            self._catalog.update(other._catalog)\n        if other._fallback:\n            self.add_fallback(other._fallback)"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "to_language",
          "class_name": "DjangoTranslation",
          "code": "def to_language(self):\n        \"\"\"Return the translation language name.\"\"\"\n        return self.__to_language"
        }
      ]
    }
  ]
}