{
  "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 ​only take care of `dict`s but don't take other types of iterables into account but ​return 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": "Given the following model:\nclass Foo(models.Model):\n    bar = models.CharField(max_length=255)\n\n    class Meta:\n        constraints = (\n            models.CheckConstraint(\n                check=models.Q(bar__startswith='BAR'),\n                name='check_bar_starts_with_BAR',\n            ),\n        )\nRunning migrate with PostgreSQL will result in an exception:\ndavids ~/projects/test_startswith_constraint $ ./manage.py migrate\nOperations to perform:\n  Apply all migrations: admin, auth, contenttypes, sample, sessions\nRunning migrations:\n  Applying contenttypes.0001_initial... OK\n  Applying auth.0001_initial... OK\n  Applying admin.0001_initial... OK\n  Applying admin.0002_logentry_remove_auto_add... OK\n  Applying admin.0003_logentry_add_action_flag_choices... OK\n  Applying contenttypes.0002_remove_content_type_name... OK\n  Applying auth.0002_alter_permission_name_max_length... OK\n  Applying auth.0003_alter_user_email_max_length... OK\n  Applying auth.0004_alter_user_username_opts... OK\n  Applying auth.0005_alter_user_last_login_null... OK\n  Applying auth.0006_require_contenttypes_0002... OK\n  Applying auth.0007_alter_validators_add_error_messages... OK\n  Applying auth.0008_alter_user_username_max_length... OK\n  Applying auth.0009_alter_user_last_name_max_length... OK\n  Applying auth.0010_alter_group_name_max_length... OK\n  Applying auth.0011_update_proxy_permissions... OK\n  Applying sample.0001_initial...Traceback (most recent call last):\n  File \"./manage.py\", line 21, in <module>\n    main()\n  File \"./manage.py\", line 17, in main\n    execute_from_command_line(sys.argv)\n  File \"/Users/davids/src/django/django/core/management/__init__.py\", line 381, in execute_from_command_line\n    utility.execute()\n  File \"/Users/davids/src/django/django/core/management/__init__.py\", line 375, in execute\n    self.fetch_command(subcommand).run_from_argv(self.argv)\n  File \"/Users/davids/src/django/django/core/management/base.py\", line 323, in run_from_argv\n    self.execute(*args, **cmd_options)\n  File \"/Users/davids/src/django/django/core/management/base.py\", line 364, in execute\n    output = self.handle(*args, **options)\n  File \"/Users/davids/src/django/django/core/management/base.py\", line 83, in wrapped\n    res = handle_func(*args, **kwargs)\n  File \"/Users/davids/src/django/django/core/management/commands/migrate.py\", line 233, in handle\n    fake_initial=fake_initial,\n  File \"/Users/davids/src/django/django/db/migrations/executor.py\", line 117, in migrate\n    state = self._migrate_all_forwards(state, plan, full_plan, fake=fake, fake_initial=fake_initial)\n  File \"/Users/davids/src/django/django/db/migrations/executor.py\", line 147, in _migrate_all_forwards\n    state = self.apply_migration(state, migration, fake=fake, fake_initial=fake_initial)\n  File \"/Users/davids/src/django/django/db/migrations/executor.py\", line 245, in apply_migration\n    state = migration.apply(state, schema_editor)\n  File \"/Users/davids/src/django/django/db/migrations/migration.py\", line 124, in apply\n    operation.database_forwards(self.app_label, schema_editor, old_state, project_state)\n  File \"/Users/davids/src/django/django/db/migrations/operations/models.py\", line 827, in database_forwards\n    schema_editor.add_constraint(model, self.constraint)\n  File \"/Users/davids/src/django/django/db/backends/base/schema.py\", line 346, in add_constraint\n    self.execute(sql)\n  File \"/Users/davids/src/django/django/db/backends/base/schema.py\", line 138, in execute\n    cursor.execute(sql, params)\n  File \"/Users/davids/src/django/django/db/backends/utils.py\", line 99, in execute\n    return super().execute(sql, params)\n  File \"/Users/davids/src/django/django/db/backends/utils.py\", line 67, in execute\n    return self._execute_with_wrappers(sql, params, many=False, executor=self._execute)\n  File \"/Users/davids/src/django/django/db/backends/utils.py\", line 76, in _execute_with_wrappers\n    return executor(sql, params, many, context)\n  File \"/Users/davids/src/django/django/db/backends/utils.py\", line 84, in _execute\n    return self.cursor.execute(sql, params)\nIndexError: tuple index out of range\nThis is due to the SQL being passed to\nexecute()\nhas an unescaped\n%\n:\n> /Users/davids/src/django/django/db/backends/utils.py(85)_execute()\n     84                 import ipdb; ipdb.set_trace()\n---> 85                 return self.cursor.execute(sql, params)\n     86\n\nipdb> sql\n'ALTER TABLE \"sample_foo\" ADD CONSTRAINT \"check_bar_starts_with_BAR\" CHECK (\"bar\"::text LIKE \\'BAR%\\')'\nNote that this runs fine with SQLite but is problematic for PostgreSQL.",
      "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": "While studying Django, I had an error where the technical_500.html page was not properly printed.\nIn the log window, UnicodeDecodeError was continuously printed, and in the template, the sentence 'A server error occured. Please contact the administrator' was printed\nSo when I checked the technical_500.html file of Django 2.2version, I found that the dotdotdot wrapped by the <span>tag on the 239th line was changed to ellipsis.\nApparently, the version of Django 2.1.8 was a dotdotdot.\nSo I took steps to change the 239th line's ellipsis to dotdotdot.\nOr, when reading the technical_500.html file from inside the debug.py file, the encoding format was set to utf-8.\nThis enabled me to resolve the error.\nDid you intentionally modify the technical_html file?",
      "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… <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": "from\ndjango.contrib.gis.db.backends.spatialite.models\nimport\nSpatialiteSpatialRefSys\nas\nSpatialRefSys\nfrom\ndjango.contrib.gis.gdal.srs\nimport\nSpatialReference\nSpatialReference\n(\nSpatialRefSys\n.\nobjects\n.\nget\n(\nsrid\n=\n187939\n)\n.\nsrtext\n)\n/home/sergey/dev/django/django/contrib/gis/gdal/srs.py in __init__(self, srs_input, srs_type)\n     58             # Encoding to ASCII if unicode passed in.\n     59             if isinstance(srs_input, six.text_type):\n---> 60                 srs_input = srs_input.encode('ascii')\n     61             try:\n     62                 # If SRID is a string, e.g., '4326', then make acceptable\n\nUnicodeEncodeError: 'ascii' codec can't encode character u'\\xdf' in position 34: ordinal not in range(128)",
      "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": "The actual model field check is\n​\nskipped\nif the\nrelated_name\nis an empty string when a warning should be raised since it's an invalid Python identifier.",
      "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": "When locales have different plural forms, the ngettext can be easily broken because it uses plural equation from the first loaded gettext catalog for all of them.\nFor example in Czech locale, there are different plural equations being used even inside Django with either 3 or 4 plural forms. When the one with 4 forms is loaded first, Django is looking for non existing plural in catalogs.\nReproducer in Django 2.2.1:\n>>> from django.utils import translation\n>>> translation.activate('cs')\n>>> translation.ngettext('This password is too short. It must contain at least %(min_length)d character.', 'This password is too short. It must contain at least %(min_length)d characters.', 1)\n'Heslo je příliš krátké. Musí mít délku aspoň %(min_length)d znak.'\n>>> translation.ngettext('This password is too short. It must contain at least %(min_length)d character.', 'This password is too short. It must contain at least %(min_length)d characters.', 10)\n'This password is too short. It must contain at least %(min_length)d characters.'\nThe second invocation is trying to find 4th plural in 3 plurals catalog.",
      "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"
        }
      ]
    }
  ]
}