{
  "instance_id": "django__django-13551",
  "repo": "django/django",
  "created_at": "2020-10-17T17:22:01Z",
  "problem_statement": "Changing user's email could invalidate password reset tokens\nDescription\n\t\nSequence:\nHave account with email address foo@…\nPassword reset request for that email (unused)\nfoo@… account changes their email address\nPassword reset email is used\nThe password reset email's token should be rejected at that point, but in fact it is allowed.\nThe fix is to add the user's email address into ​PasswordResetTokenGenerator._make_hash_value()\nNothing forces a user to even have an email as per AbstractBaseUser. Perhaps the token generation method could be factored out onto the model, ala get_session_auth_hash().\n",
  "patch": "diff --git a/django/contrib/auth/tokens.py b/django/contrib/auth/tokens.py\n--- a/django/contrib/auth/tokens.py\n+++ b/django/contrib/auth/tokens.py\n@@ -78,9 +78,9 @@ def _make_token_with_timestamp(self, user, timestamp, legacy=False):\n \n     def _make_hash_value(self, user, timestamp):\n         \"\"\"\n-        Hash the user's primary key and some user state that's sure to change\n-        after a password reset to produce a token that invalidated when it's\n-        used:\n+        Hash the user's primary key, email (if available), and some user state\n+        that's sure to change after a password reset to produce a token that is\n+        invalidated when it's used:\n         1. The password field will change upon a password reset (even if the\n            same password is chosen, due to password salting).\n         2. The last_login field will usually be updated very shortly after\n@@ -94,7 +94,9 @@ def _make_hash_value(self, user, timestamp):\n         # Truncate microseconds so that tokens are consistent even if the\n         # database doesn't support microseconds.\n         login_timestamp = '' if user.last_login is None else user.last_login.replace(microsecond=0, tzinfo=None)\n-        return str(user.pk) + user.password + str(login_timestamp) + str(timestamp)\n+        email_field = user.get_email_field_name()\n+        email = getattr(user, email_field, '') or ''\n+        return f'{user.pk}{user.password}{login_timestamp}{timestamp}{email}'\n \n     def _num_seconds(self, dt):\n         return int((dt - datetime(2001, 1, 1)).total_seconds())\n",
  "similar_bug_items": [
    {
      "pr_number": 5484,
      "pr_title": "Fixed #25596 -- Fixed regression in password change view with custom user model.",
      "pr_body": "https://code.djangoproject.com/ticket/25596\n",
      "issue_id": 25596,
      "issue_title": "Can't change user's password in admin when using custom User model",
      "issue_body": "Django 1.9b1\nI'm using custom User model which is defined as:\nAUTH_USER_MODEL = 'users.User'\n\nINSTALLED_APPS = [\n    'django.contrib.admin',\n    'django.contrib.auth',\n    ...\n    'apps.users',\n]\nWhen I tried to change user's password (using /admin/users/user/ID/password/) I've got an error:\nTraceback:\nFile \"/src/django/django/core/handlers/base.py\" in get_response\n  149.                     response = self.process_exception_by_middleware(e, request)\n\nFile \"/src/django/django/core/handlers/base.py\" in get_response\n  147.                     response = wrapped_callback(request, *callback_args, **callback_kwargs)\n\nFile \"/src/django/django/utils/decorators.py\" in _wrapped_view\n  149.                     response = view_func(request, *args, **kwargs)\n\nFile \"/src/django/django/views/decorators/cache.py\" in _wrapped_view_func\n  57.         response = view_func(request, *args, **kwargs)\n\nFile \"/src/django/django/contrib/admin/sites.py\" in inner\n  244.             return view(request, *args, **kwargs)\n\nFile \"/src/django/django/utils/decorators.py\" in _wrapper\n  67.             return bound_func(*args, **kwargs)\n\nFile \"/src/django/django/views/decorators/debug.py\" in sensitive_post_parameters_wrapper\n  76.             return view(request, *args, **kwargs)\n\nFile \"/src/django/django/utils/decorators.py\" in bound_func\n  63.                 return func.__get__(self, type(self))(*args2, **kwargs2)\n\nFile \"/src/django/django/contrib/auth/admin.py\" in user_change_password\n  155.                         args=(user.pk,),\n\nFile \"/src/django/django/core/urlresolvers.py\" in reverse\n  600.     return force_text(iri_to_uri(resolver._reverse_with_prefix(view, prefix, *args, **kwargs)))\n\nFile \"/src/django/django/core/urlresolvers.py\" in _reverse_with_prefix\n  508.                              (lookup_view_s, args, kwargs, len(patterns), patterns))\n\nException Type: NoReverseMatch at /panel/users/user/8/password/\nException Value: Reverse for 'auth_user_change' with arguments '(8,)' and keyword arguments '{}' not found. 0 pattern(s) tried: []\ndjango/auth/admin.py:151\nreverse(\n                        '%s:auth_%s_change' % (\n                            self.admin_site.name,\n                            user._meta.model_name,\n                        ),\n                        args=(user.pk,),\n                    )\nThere should not be fixed \"auth_\" prefix, but something like user._meta.app_name(?)",
      "issue_closed_at": "2015-10-27T07:38:10",
      "base_commit": "1f07da3e29c7c3d47968e1c4531dd9bf902575b7",
      "changes": [
        {
          "file": "django/contrib/auth/admin.py",
          "type": "function",
          "name": "user_change_password",
          "class_name": "UserAdmin",
          "code": "def user_change_password(self, request, id, form_url=''):\n        if not self.has_change_permission(request):\n            raise PermissionDenied\n        user = self.get_object(request, unquote(id))\n        if user is None:\n            raise Http404(_('%(name)s object with primary key %(key)r does not exist.') % {\n                'name': force_text(self.model._meta.verbose_name),\n                'key': escape(id),\n            })\n        if request.method == 'POST':\n            form = self.change_password_form(user, request.POST)\n            if form.is_valid():\n                form.save()\n                change_message = self.construct_change_message(request, form, None)\n                self.log_change(request, user, change_message)\n                msg = ugettext('Password changed successfully.')\n                messages.success(request, msg)\n                update_session_auth_hash(request, form.user)\n                return HttpResponseRedirect(\n                    reverse(\n                        '%s:auth_%s_change' % (\n                            self.admin_site.name,\n                            user._meta.model_name,\n                        ),\n                        args=(user.pk,),\n                    )\n                )\n        else:\n            form = self.change_password_form(user)\n\n        fieldsets = [(None, {'fields': list(form.base_fields)})]\n        adminForm = admin.helpers.AdminForm(form, fieldsets, {})\n\n        context = {\n            'title': _('Change password: %s') % escape(user.get_username()),\n            'adminForm': adminForm,\n            'form_url': form_url,\n            'form': form,\n            'is_popup': (IS_POPUP_VAR in request.POST or\n                         IS_POPUP_VAR in request.GET),\n            'add': True,\n            'change': False,\n            'has_delete_permission': False,\n            'has_change_permission': True,\n            'has_absolute_url': False,\n            'opts': self.model._meta,\n            'original': user,\n            'save_as': False,\n            'show_save': True,\n        }\n        context.update(admin.site.each_context(request))\n\n        request.current_app = self.admin_site.name\n\n        return TemplateResponse(request,\n            self.change_user_password_template or\n            'admin/auth/user/change_password.html',\n            context)"
        },
        {
          "file": "django/contrib/auth/admin.py",
          "type": "function",
          "name": "user_change_password",
          "class_name": "UserAdmin",
          "code": "def user_change_password(self, request, id, form_url=''):\n        if not self.has_change_permission(request):\n            raise PermissionDenied\n        user = self.get_object(request, unquote(id))\n        if user is None:\n            raise Http404(_('%(name)s object with primary key %(key)r does not exist.') % {\n                'name': force_text(self.model._meta.verbose_name),\n                'key': escape(id),\n            })\n        if request.method == 'POST':\n            form = self.change_password_form(user, request.POST)\n            if form.is_valid():\n                form.save()\n                change_message = self.construct_change_message(request, form, None)\n                self.log_change(request, user, change_message)\n                msg = ugettext('Password changed successfully.')\n                messages.success(request, msg)\n                update_session_auth_hash(request, form.user)\n                return HttpResponseRedirect(\n                    reverse(\n                        '%s:auth_%s_change' % (\n                            self.admin_site.name,\n                            user._meta.model_name,\n                        ),\n                        args=(user.pk,),\n                    )\n                )\n        else:\n            form = self.change_password_form(user)\n\n        fieldsets = [(None, {'fields': list(form.base_fields)})]\n        adminForm = admin.helpers.AdminForm(form, fieldsets, {})\n\n        context = {\n            'title': _('Change password: %s') % escape(user.get_username()),\n            'adminForm': adminForm,\n            'form_url': form_url,\n            'form': form,\n            'is_popup': (IS_POPUP_VAR in request.POST or\n                         IS_POPUP_VAR in request.GET),\n            'add': True,\n            'change': False,\n            'has_delete_permission': False,\n            'has_change_permission': True,\n            'has_absolute_url': False,\n            'opts': self.model._meta,\n            'original': user,\n            'save_as': False,\n            'show_save': True,\n        }\n        context.update(admin.site.each_context(request))\n\n        request.current_app = self.admin_site.name\n\n        return TemplateResponse(request,\n            self.change_user_password_template or\n            'admin/auth/user/change_password.html',\n            context)"
        }
      ]
    },
    {
      "pr_number": 13204,
      "pr_title": "Fixed #17653 -- Allowed using zero as AutoFields value on MySQL if NO_AUTO_VALUE_ON_ZERO SQL mode is enabled.",
      "pr_body": "I think it's time to implement [Ansi's comment](https://code.djangoproject.com/ticket/17653#comment:27).\r\n\r\nticket-17653\r\n\r\nhttps://mariadb.com/kb/en/sql-mode/#no_auto_value_on_zero\r\nhttps://dev.mysql.com/doc/refman/en/sql-mode.html#sqlmode_no_auto_value_on_zero",
      "issue_id": 17653,
      "issue_title": "using id = 0 on get_or_create",
      "issue_body": "When using get_or_create with id=0, django does create an object, but doesn't write the id back in the returned object. The object is therefore not usable. We should get the new object's id, or at least get an error message to prevent using 0 as an id.",
      "issue_closed_at": "2020-07-20T02:48:46",
      "base_commit": "e7fa8aff432a90b6df9914d63aad239164b6b4d4",
      "changes": [
        {
          "file": "django/db/backends/mysql/features.py",
          "type": "class",
          "name": "DatabaseFeatures",
          "code": "class DatabaseFeatures(BaseDatabaseFeatures):\n    empty_fetchmany_value = ()\n    allows_group_by_pk = True\n    related_fields_match_type = True\n    # MySQL doesn't support sliced subqueries with IN/ALL/ANY/SOME.\n    allow_sliced_subqueries_with_in = False\n    has_select_for_update = True\n    supports_forward_references = False\n    supports_regex_backreferencing = False\n    supports_date_lookup_using_string = False\n    supports_index_column_ordering = False\n    supports_timezones = False\n    requires_explicit_null_ordering_when_grouping = True\n    allows_auto_pk_0 = False\n    can_release_savepoints = True\n    atomic_transactions = False\n    can_clone_databases = True\n    supports_temporal_subtraction = True\n    supports_select_intersection = False\n    supports_select_difference = False\n    supports_slicing_ordering_in_compound = True\n    supports_index_on_text_field = False\n    has_case_insensitive_like = False\n    create_test_procedure_without_params_sql = \"\"\"\n        CREATE PROCEDURE test_procedure ()\n        BEGIN\n            DECLARE V_I INTEGER;\n            SET V_I = 1;\n        END;\n    \"\"\"\n    create_test_procedure_with_int_param_sql = \"\"\"\n        CREATE PROCEDURE test_procedure (P_I INTEGER)\n        BEGIN\n            DECLARE V_I INTEGER;\n            SET V_I = P_I;\n        END;\n    \"\"\"\n    # Neither MySQL nor MariaDB support partial indexes.\n    supports_partial_indexes = False\n    supports_order_by_nulls_modifier = False\n    order_by_nulls_first = True\n\n    @cached_property\n    def _mysql_storage_engine(self):\n        \"Internal method used in Django tests. Don't rely on this from your code\"\n        return self.connection.mysql_server_data['default_storage_engine']\n\n    @cached_property\n    def update_can_self_select(self):\n        return self.connection.mysql_is_mariadb and self.connection.mysql_version >= (10, 3, 2)\n\n    @cached_property\n    def can_introspect_foreign_keys(self):\n        \"Confirm support for introspected foreign keys\"\n        return self._mysql_storage_engine != 'MyISAM'\n\n    @cached_property\n    def introspected_field_types(self):\n        return {\n            **super().introspected_field_types,\n            'BinaryField': 'TextField',\n            'BooleanField': 'IntegerField',\n            'DurationField': 'BigIntegerField',\n            'GenericIPAddressField': 'CharField',\n        }\n\n    @cached_property\n    def can_return_columns_from_insert(self):\n        return self.connection.mysql_is_mariadb and self.connection.mysql_version >= (10, 5, 0)\n\n    can_return_rows_from_bulk_insert = property(operator.attrgetter('can_return_columns_from_insert'))\n\n    @cached_property\n    def has_zoneinfo_database(self):\n        return self.connection.mysql_server_data['has_zoneinfo_database']\n\n    @cached_property\n    def is_sql_auto_is_null_enabled(self):\n        return self.connection.mysql_server_data['sql_auto_is_null']\n\n    @cached_property\n    def supports_over_clause(self):\n        if self.connection.mysql_is_mariadb:\n            return True\n        return self.connection.mysql_version >= (8, 0, 2)\n\n    supports_frame_range_fixed_distance = property(operator.attrgetter('supports_over_clause'))\n\n    @cached_property\n    def supports_column_check_constraints(self):\n        if self.connection.mysql_is_mariadb:\n            return self.connection.mysql_version >= (10, 2, 1)\n        return self.connection.mysql_version >= (8, 0, 16)\n\n    supports_table_check_constraints = property(operator.attrgetter('supports_column_check_constraints'))\n\n    @cached_property\n    def can_introspect_check_constraints(self):\n        if self.connection.mysql_is_mariadb:\n            version = self.connection.mysql_version\n            return (version >= (10, 2, 22) and version < (10, 3)) or version >= (10, 3, 10)\n        return self.connection.mysql_version >= (8, 0, 16)\n\n    @cached_property\n    def has_select_for_update_skip_locked(self):\n        return not self.connection.mysql_is_mariadb and self.connection.mysql_version >= (8, 0, 1)\n\n    @cached_property\n    def has_select_for_update_nowait(self):\n        if self.connection.mysql_is_mariadb:\n            return self.connection.mysql_version >= (10, 3, 0)\n        return self.connection.mysql_version >= (8, 0, 1)\n\n    @cached_property\n    def has_select_for_update_of(self):\n        return not self.connection.mysql_is_mariadb and self.connection.mysql_version >= (8, 0, 1)\n\n    @cached_property\n    def supports_explain_analyze(self):\n        return self.connection.mysql_is_mariadb or self.connection.mysql_version >= (8, 0, 18)\n\n    @cached_property\n    def supported_explain_formats(self):\n        # Alias MySQL's TRADITIONAL to TEXT for consistency with other\n        # backends.\n        formats = {'JSON', 'TEXT', 'TRADITIONAL'}\n        if not self.connection.mysql_is_mariadb and self.connection.mysql_version >= (8, 0, 16):\n            formats.add('TREE')\n        return formats\n\n    @cached_property\n    def supports_transactions(self):\n        \"\"\"\n        All storage engines except MyISAM support transactions.\n        \"\"\"\n        return self._mysql_storage_engine != 'MyISAM'\n\n    @cached_property\n    def ignores_table_name_case(self):\n        return self.connection.mysql_server_data['lower_case_table_names']\n\n    @cached_property\n    def supports_default_in_lead_lag(self):\n        # To be added in https://jira.mariadb.org/browse/MDEV-12981.\n        return not self.connection.mysql_is_mariadb\n\n    @cached_property\n    def supports_json_field(self):\n        if self.connection.mysql_is_mariadb:\n            return self.connection.mysql_version >= (10, 2, 7)\n        return self.connection.mysql_version >= (5, 7, 8)\n\n    @cached_property\n    def can_introspect_json_field(self):\n        if self.connection.mysql_is_mariadb:\n            return self.supports_json_field and self.can_introspect_check_constraints\n        return self.supports_json_field"
        },
        {
          "file": "django/db/backends/mysql/features.py",
          "type": "function",
          "name": "_mysql_storage_engine",
          "class_name": "DatabaseFeatures",
          "code": "def _mysql_storage_engine(self):\n        \"Internal method used in Django tests. Don't rely on this from your code\"\n        return self.connection.mysql_server_data['default_storage_engine']"
        },
        {
          "file": "django/db/backends/mysql/operations.py",
          "type": "function",
          "name": "sequence_reset_by_name_sql",
          "class_name": "DatabaseOperations",
          "code": "def sequence_reset_by_name_sql(self, style, sequences):\n        return [\n            '%s %s %s %s = 1;' % (\n                style.SQL_KEYWORD('ALTER'),\n                style.SQL_KEYWORD('TABLE'),\n                style.SQL_FIELD(self.quote_name(sequence_info['table'])),\n                style.SQL_FIELD('AUTO_INCREMENT'),\n            ) for sequence_info in sequences\n        ]"
        },
        {
          "file": "django/db/backends/mysql/operations.py",
          "type": "function",
          "name": "adapt_timefield_value",
          "class_name": "DatabaseOperations",
          "code": "def adapt_timefield_value(self, value):\n        if value is None:\n            return None\n\n        # Expression values are adapted by the database.\n        if hasattr(value, 'resolve_expression'):\n            return value\n\n        # MySQL doesn't support tz-aware times\n        if timezone.is_aware(value):\n            raise ValueError(\"MySQL backend does not support timezone-aware times.\")\n\n        return str(value)"
        }
      ]
    },
    {
      "pr_number": 10773,
      "pr_title": "Fixed #30050 -- Fixed InlineModelAdmin.has_change_permission() being called with non-None obj during add.",
      "pr_body": "https://code.djangoproject.com/ticket/30050",
      "issue_id": 30050,
      "issue_title": "InlineModelAdmin.has_change_permission() incorrectly called with non-None obj during add",
      "issue_body": "Fine with Django 2.1.3, bug with Django 2.1.4.\nIf I have an admin Inline of a\nModel\n, with fk to a\nParentModel\n, when I try to add a new\nParentModel\nin the admin panel (\"/admin/foo/parentmodel/add/\"), the Inline\nhas_change_permission\nis called 3 times instead of 2, after this commit:\n​\nhttps://github.com/django/django/commit/27f5b0aff3442e5c25e84972dff4f5fe1edd4e68\nat line 1962\nhas_change_permission\nis passed an\nobj\nthat is not\nNone\n.\nBefore this commit,\nobj\nwas\nNone\nboth of the times, now it's\nNone\nthe first time, an empty instance of\nParentModel\nthe second time, and\nNone\nthe third time.\nLine 1962 should be something like:\npermission_obj = obj if change else None\nif not inline.has_change_permission(request, permission_obj):",
      "issue_closed_at": "2019-01-01T09:05:00",
      "base_commit": "0123b67f6b8304a5c32a0fe98f97ae506977454b",
      "changes": [
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "user_deleted_form",
          "class_name": "ModelAdmin",
          "code": "def user_deleted_form(request, obj, formset, index):\n                \"\"\"Return whether or not the user deleted the form.\"\"\"\n                return (\n                    inline.has_delete_permission(request, obj) and\n                    '{}-{}-DELETE'.format(formset.prefix, index) in request.POST\n                )"
        }
      ]
    },
    {
      "pr_number": 9966,
      "pr_title": "Fixed #29417 -- Corrected two admin page titles for view-only users.",
      "pr_body": "The Changelist and 'View object' views still contained the word 'change'\r\nin their title and header.",
      "issue_id": 29417,
      "issue_title": "Admin title still says \"Change [model]\" when user has view-only permission",
      "issue_body": "When the admin user has the \"view\" permission for a model but doesn't have the \"change\" permission, the title of the change view still says \"Change [model]\", and the title of the change list view still says \"Select [model] to change.\" Since the admin index page displays \"View\" instead of \"Change\" on the link to the changelist when the user only has the view permission, it would make sense to change the title of the changelist and change views as well.",
      "issue_closed_at": "2018-05-23T10:03:08",
      "base_commit": "40ff93310f03dc89a6281a846b1a1ec4cb672bd0",
      "changes": [
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "_changeform_view",
          "class_name": "ModelAdmin",
          "code": "def _changeform_view(self, request, object_id, form_url, extra_context):\n        to_field = request.POST.get(TO_FIELD_VAR, request.GET.get(TO_FIELD_VAR))\n        if to_field and not self.to_field_allowed(request, to_field):\n            raise DisallowedModelAdminToField(\"The field %s cannot be referenced.\" % to_field)\n\n        model = self.model\n        opts = model._meta\n\n        if request.method == 'POST' and '_saveasnew' in request.POST:\n            object_id = None\n\n        add = object_id is None\n\n        if add:\n            if not self.has_add_permission(request):\n                raise PermissionDenied\n            obj = None\n\n        else:\n            obj = self.get_object(request, unquote(object_id), to_field)\n\n            if not self.has_view_permission(request, obj) and not self.has_change_permission(request, obj):\n                raise PermissionDenied\n\n            if obj is None:\n                return self._get_obj_does_not_exist_redirect(request, opts, object_id)\n\n        ModelForm = self.get_form(request, obj, change=not add)\n        if request.method == 'POST':\n            form = ModelForm(request.POST, request.FILES, instance=obj)\n            form_validated = form.is_valid()\n            if form_validated:\n                new_object = self.save_form(request, form, change=not add)\n            else:\n                new_object = form.instance\n            formsets, inline_instances = self._create_formsets(request, new_object, change=not add)\n            if all_valid(formsets) and form_validated:\n                self.save_model(request, new_object, form, not add)\n                self.save_related(request, form, formsets, not add)\n                change_message = self.construct_change_message(request, form, formsets, add)\n                if add:\n                    self.log_addition(request, new_object, change_message)\n                    return self.response_add(request, new_object)\n                else:\n                    self.log_change(request, new_object, change_message)\n                    return self.response_change(request, new_object)\n            else:\n                form_validated = False\n        else:\n            if add:\n                initial = self.get_changeform_initial_data(request)\n                form = ModelForm(initial=initial)\n                formsets, inline_instances = self._create_formsets(request, form.instance, change=False)\n            else:\n                form = ModelForm(instance=obj)\n                formsets, inline_instances = self._create_formsets(request, obj, change=True)\n\n        if not add and not self.has_change_permission(request):\n            readonly_fields = flatten_fieldsets(self.get_fieldsets(request, obj))\n        else:\n            readonly_fields = self.get_readonly_fields(request, obj)\n        adminForm = helpers.AdminForm(\n            form,\n            list(self.get_fieldsets(request, obj)),\n            self.get_prepopulated_fields(request, obj),\n            readonly_fields,\n            model_admin=self)\n        media = self.media + adminForm.media\n\n        inline_formsets = self.get_inline_formsets(request, formsets, inline_instances, obj)\n        for inline_formset in inline_formsets:\n            media = media + inline_formset.media\n\n        context = {\n            **self.admin_site.each_context(request),\n            'title': (_('Add %s') if add else _('Change %s')) % opts.verbose_name,\n            'adminform': adminForm,\n            'object_id': object_id,\n            'original': obj,\n            'is_popup': IS_POPUP_VAR in request.POST or IS_POPUP_VAR in request.GET,\n            'to_field': to_field,\n            'media': media,\n            'inline_admin_formsets': inline_formsets,\n            'errors': helpers.AdminErrorList(form, formsets),\n            'preserved_filters': self.get_preserved_filters(request),\n        }\n\n        # Hide the \"Save\" and \"Save and continue\" buttons if \"Save as New\" was\n        # previously chosen to prevent the interface from getting confusing.\n        if request.method == 'POST' and not form_validated and \"_saveasnew\" in request.POST:\n            context['show_save'] = False\n            context['show_save_and_continue'] = False\n            # Use the change template instead of the add template.\n            add = False\n\n        context.update(extra_context or {})\n\n        return self.render_change_form(request, context, add=add, change=not add, obj=obj, form_url=form_url)"
        },
        {
          "file": "django/contrib/admin/views/main.py",
          "type": "function",
          "name": "__init__",
          "class_name": "ChangeList",
          "code": "def __init__(self, request, model, list_display, list_display_links,\n                 list_filter, date_hierarchy, search_fields, list_select_related,\n                 list_per_page, list_max_show_all, list_editable, model_admin, sortable_by):\n        self.model = model\n        self.opts = model._meta\n        self.lookup_opts = self.opts\n        self.root_queryset = model_admin.get_queryset(request)\n        self.list_display = list_display\n        self.list_display_links = list_display_links\n        self.list_filter = list_filter\n        self.date_hierarchy = date_hierarchy\n        self.search_fields = search_fields\n        self.list_select_related = list_select_related\n        self.list_per_page = list_per_page\n        self.list_max_show_all = list_max_show_all\n        self.model_admin = model_admin\n        self.preserved_filters = model_admin.get_preserved_filters(request)\n        self.sortable_by = sortable_by\n\n        # Get search parameters from the query string.\n        try:\n            self.page_num = int(request.GET.get(PAGE_VAR, 0))\n        except ValueError:\n            self.page_num = 0\n        self.show_all = ALL_VAR in request.GET\n        self.is_popup = IS_POPUP_VAR in request.GET\n        to_field = request.GET.get(TO_FIELD_VAR)\n        if to_field and not model_admin.to_field_allowed(request, to_field):\n            raise DisallowedModelAdminToField(\"The field %s cannot be referenced.\" % to_field)\n        self.to_field = to_field\n        self.params = dict(request.GET.items())\n        if PAGE_VAR in self.params:\n            del self.params[PAGE_VAR]\n        if ERROR_FLAG in self.params:\n            del self.params[ERROR_FLAG]\n\n        if self.is_popup:\n            self.list_editable = ()\n        else:\n            self.list_editable = list_editable\n        self.query = request.GET.get(SEARCH_VAR, '')\n        self.queryset = self.get_queryset(request)\n        self.get_results(request)\n        if self.is_popup:\n            title = gettext('Select %s')\n        else:\n            title = gettext('Select %s to change')\n        self.title = title % self.opts.verbose_name\n        self.pk_attname = self.lookup_opts.pk.attname"
        }
      ]
    },
    {
      "pr_number": 3259,
      "pr_title": "Fixed #23537 -- Added Oracle GIS SchemaEditor.",
      "pr_body": "",
      "issue_id": 23537,
      "issue_title": "Oracle GIS backend missing SchemaEditor",
      "issue_body": "I think this results in missing indexes and metadata for GIS fields.\nThere is one test failure that will be fixed by this:\n======================================================================\nFAIL: test_add_gis_field (django.contrib.gis.tests.gis_migrations.test_operations.OperationTests)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/tim/code/django/django/contrib/gis/tests/gis_migrations/test_operations.py\", line 77, in test_add_gis_field\n    2\nAssertionError: 0 != 2",
      "issue_closed_at": "2014-09-25T19:33:18",
      "base_commit": "45bd7b3bd9008941580c100b9fc7361e3ff3ff0d",
      "changes": [
        {
          "file": "django/contrib/gis/db/backends/oracle/base.py",
          "type": "line",
          "name": "line 6",
          "code": "from django.contrib.gis.db.backends.oracle.creation import OracleCreation\nfrom django.contrib.gis.db.backends.oracle.introspection import OracleIntrospection\nfrom django.contrib.gis.db.backends.oracle.operations import OracleOperations\n\n\nclass DatabaseFeatures(BaseSpatialFeatures, OracleDatabaseFeatures):"
        },
        {
          "file": "django/contrib/gis/db/backends/oracle/base.py",
          "type": "function",
          "name": "__init__",
          "class_name": "DatabaseWrapper",
          "code": "def __init__(self, *args, **kwargs):\n        super(DatabaseWrapper, self).__init__(*args, **kwargs)\n        self.features = DatabaseFeatures(self)\n        self.ops = OracleOperations(self)\n        self.creation = OracleCreation(self)\n        self.introspection = OracleIntrospection(self)"
        },
        {
          "file": "django/contrib/gis/db/backends/oracle/operations.py",
          "type": "class",
          "name": "OracleOperations",
          "code": "class OracleOperations(DatabaseOperations, BaseSpatialOperations):\n    compiler_module = \"django.contrib.gis.db.backends.oracle.compiler\"\n\n    name = 'oracle'\n    oracle = True\n    valid_aggregates = {'Union', 'Extent'}\n\n    Adapter = OracleSpatialAdapter\n    Adaptor = Adapter  # Backwards-compatibility alias.\n\n    area = 'SDO_GEOM.SDO_AREA'\n    gml = 'SDO_UTIL.TO_GMLGEOMETRY'\n    centroid = 'SDO_GEOM.SDO_CENTROID'\n    difference = 'SDO_GEOM.SDO_DIFFERENCE'\n    distance = 'SDO_GEOM.SDO_DISTANCE'\n    extent = 'SDO_AGGR_MBR'\n    intersection = 'SDO_GEOM.SDO_INTERSECTION'\n    length = 'SDO_GEOM.SDO_LENGTH'\n    num_geom = 'SDO_UTIL.GETNUMELEM'\n    num_points = 'SDO_UTIL.GETNUMVERTICES'\n    perimeter = length\n    point_on_surface = 'SDO_GEOM.SDO_POINTONSURFACE'\n    reverse = 'SDO_UTIL.REVERSE_LINESTRING'\n    sym_difference = 'SDO_GEOM.SDO_XOR'\n    transform = 'SDO_CS.TRANSFORM'\n    union = 'SDO_GEOM.SDO_UNION'\n    unionagg = 'SDO_AGGR_UNION'\n\n    # We want to get SDO Geometries as WKT because it is much easier to\n    # instantiate GEOS proxies from WKT than SDO_GEOMETRY(...) strings.\n    # However, this adversely affects performance (i.e., Java is called\n    # to convert to WKT on every query).  If someone wishes to write a\n    # SDO_GEOMETRY(...) parser in Python, let me know =)\n    select = 'SDO_UTIL.TO_WKTGEOMETRY(%s)'\n\n    distance_functions = {\n        'distance_gt': (SDODistance('>'), dtypes),\n        'distance_gte': (SDODistance('>='), dtypes),\n        'distance_lt': (SDODistance('<'), dtypes),\n        'distance_lte': (SDODistance('<='), dtypes),\n        'dwithin': (SDODWithin(), dtypes),\n    }\n\n    geometry_functions = {\n        'contains': SDOOperation('SDO_CONTAINS'),\n        'coveredby': SDOOperation('SDO_COVEREDBY'),\n        'covers': SDOOperation('SDO_COVERS'),\n        'disjoint': SDOGeomRelate('DISJOINT'),\n        'intersects': SDOOperation('SDO_OVERLAPBDYINTERSECT'),  # TODO: Is this really the same as ST_Intersects()?\n        'equals': SDOOperation('SDO_EQUAL'),\n        'exact': SDOOperation('SDO_EQUAL'),\n        'overlaps': SDOOperation('SDO_OVERLAPS'),\n        'same_as': SDOOperation('SDO_EQUAL'),\n        'relate': (SDORelate, six.string_types),  # Oracle uses a different syntax, e.g., 'mask=inside+touch'\n        'touches': SDOOperation('SDO_TOUCH'),\n        'within': SDOOperation('SDO_INSIDE'),\n    }\n    geometry_functions.update(distance_functions)\n\n    gis_terms = set(['isnull'])\n    gis_terms.update(geometry_functions)\n\n    truncate_params = {'relate': None}\n\n    def get_db_converters(self, internal_type):\n        converters = super(OracleOperations, self).get_db_converters(internal_type)\n        geometry_fields = (\n            'PointField', 'GeometryField', 'LineStringField',\n            'PolygonField', 'MultiPointField', 'MultiLineStringField',\n            'MultiPolygonField', 'GeometryCollectionField', 'GeomField',\n            'GMLField',\n        )\n        if internal_type in geometry_fields:\n            converters.append(self.convert_textfield_value)\n        return converters\n\n    def convert_extent(self, clob):\n        if clob:\n            # Generally, Oracle returns a polygon for the extent -- however,\n            # it can return a single point if there's only one Point in the\n            # table.\n            ext_geom = Geometry(clob.read())\n            gtype = str(ext_geom.geom_type)\n            if gtype == 'Polygon':\n                # Construct the 4-tuple from the coordinates in the polygon.\n                shell = ext_geom.shell\n                ll, ur = shell[0][:2], shell[2][:2]\n            elif gtype == 'Point':\n                ll = ext_geom.coords[:2]\n                ur = ll\n            else:\n                raise Exception('Unexpected geometry type returned for extent: %s' % gtype)\n            xmin, ymin = ll\n            xmax, ymax = ur\n            return (xmin, ymin, xmax, ymax)\n        else:\n            return None\n\n    def convert_geom(self, clob, geo_field):\n        if clob:\n            return Geometry(clob.read(), geo_field.srid)\n        else:\n            return None\n\n    def geo_db_type(self, f):\n        \"\"\"\n        Returns the geometry database type for Oracle.  Unlike other spatial\n        backends, no stored procedure is necessary and it's the same for all\n        geometry types.\n        \"\"\"\n        return 'MDSYS.SDO_GEOMETRY'\n\n    def get_distance(self, f, value, lookup_type):\n        \"\"\"\n        Returns the distance parameters given the value and the lookup type.\n        On Oracle, geometry columns with a geodetic coordinate system behave\n        implicitly like a geography column, and thus meters will be used as\n        the distance parameter on them.\n        \"\"\"\n        if not value:\n            return []\n        value = value[0]\n        if isinstance(value, Distance):\n            if f.geodetic(self.connection):\n                dist_param = value.m\n            else:\n                dist_param = getattr(value, Distance.unit_attname(f.units_name(self.connection)))\n        else:\n            dist_param = value\n\n        # dwithin lookups on Oracle require a special string parameter\n        # that starts with \"distance=\".\n        if lookup_type == 'dwithin':\n            dist_param = 'distance=%s' % dist_param\n\n        return [dist_param]\n\n    def get_geom_placeholder(self, f, value):\n        \"\"\"\n        Provides a proper substitution value for Geometries that are not in the\n        SRID of the field.  Specifically, this routine will substitute in the\n        SDO_CS.TRANSFORM() function call.\n        \"\"\"\n        if value is None:\n            return 'NULL'\n\n        def transform_value(val, srid):\n            return val.srid != srid\n\n        if hasattr(value, 'expression'):\n            if transform_value(value, f.srid):\n                placeholder = '%s(%%s, %s)' % (self.transform, f.srid)\n            else:\n                placeholder = '%s'\n            # No geometry value used for F expression, substitute in\n            # the column name instead.\n            return placeholder % self.get_expression_column(value)\n        else:\n            if transform_value(value, f.srid):\n                return '%s(SDO_GEOMETRY(%%s, %s), %s)' % (self.transform, value.srid, f.srid)\n            else:\n                return 'SDO_GEOMETRY(%%s, %s)' % f.srid\n\n    def spatial_lookup_sql(self, lvalue, lookup_type, value, field, qn):\n        \"Returns the SQL WHERE clause for use in Oracle spatial SQL construction.\"\n        geo_col, db_type = lvalue\n\n        # See if an Oracle Geometry function matches the lookup type next\n        lookup_info = self.geometry_functions.get(lookup_type, False)\n        if lookup_info:\n            # Lookup types that are tuples take tuple arguments, e.g., 'relate' and\n            # 'dwithin' lookup types.\n            if isinstance(lookup_info, tuple):\n                # First element of tuple is lookup type, second element is the type\n                # of the expected argument (e.g., str, float)\n                sdo_op, arg_type = lookup_info\n                geom = value[0]\n\n                # Ensuring that a tuple _value_ was passed in from the user\n                if not isinstance(value, tuple):\n                    raise ValueError('Tuple required for `%s` lookup type.' % lookup_type)\n                if len(value) != 2:\n                    raise ValueError('2-element tuple required for %s lookup type.' % lookup_type)\n\n                # Ensuring the argument type matches what we expect.\n                if not isinstance(value[1], arg_type):\n                    raise ValueError('Argument type should be %s, got %s instead.' % (arg_type, type(value[1])))\n\n                if lookup_type == 'relate':\n                    # The SDORelate class handles construction for these queries,\n                    # and verifies the mask argument.\n                    return sdo_op(value[1]).as_sql(geo_col, self.get_geom_placeholder(field, geom))\n                else:\n                    # Otherwise, just call the `as_sql` method on the SDOOperation instance.\n                    return sdo_op.as_sql(geo_col, self.get_geom_placeholder(field, geom))\n            else:\n                # Lookup info is a SDOOperation instance, whose `as_sql` method returns\n                # the SQL necessary for the geometry function call. For example:\n                #  SDO_CONTAINS(\"geoapp_country\".\"poly\", SDO_GEOMTRY('POINT(5 23)', 4326)) = 'TRUE'\n                return lookup_info.as_sql(geo_col, self.get_geom_placeholder(field, value))\n        elif lookup_type == 'isnull':\n            # Handling 'isnull' lookup type\n            return \"%s IS %sNULL\" % (geo_col, ('' if value else 'NOT ')), []\n\n        raise TypeError(\"Got invalid lookup_type: %s\" % repr(lookup_type))\n\n    def spatial_aggregate_sql(self, agg):\n        \"\"\"\n        Returns the spatial aggregate SQL template and function for the\n        given Aggregate instance.\n        \"\"\"\n        agg_name = agg.__class__.__name__.lower()\n        if agg_name == 'union':\n            agg_name += 'agg'\n        if agg.is_extent:\n            sql_template = '%(function)s(%(field)s)'\n        else:\n            sql_template = '%(function)s(SDOAGGRTYPE(%(field)s,%(tolerance)s))'\n        sql_function = getattr(self, agg_name)\n        return self.select % sql_template, sql_function\n\n    # Routines for getting the OGC-compliant models.\n    def geometry_columns(self):\n        from django.contrib.gis.db.backends.oracle.models import OracleGeometryColumns\n        return OracleGeometryColumns\n\n    def spatial_ref_sys(self):\n        from django.contrib.gis.db.backends.oracle.models import OracleSpatialRefSys\n        return OracleSpatialRefSys\n\n    def modify_insert_params(self, placeholders, params):\n        \"\"\"Drop out insert parameters for NULL placeholder. Needed for Oracle Spatial\n        backend due to #10888\n        \"\"\"\n        # This code doesn't work for bulk insert cases.\n        assert len(placeholders) == 1\n        return [[param for pholder, param\n                 in six.moves.zip(placeholders[0], params[0]) if pholder != 'NULL'], ]"
        }
      ]
    }
  ]
}