{
  "instance_id": "django__django-11620",
  "repo": "django/django",
  "created_at": "2019-08-02T13:56:08Z",
  "problem_statement": "When DEBUG is True, raising Http404 in a path converter's to_python method does not result in a technical response\nDescription\n\t\nThis is the response I get (plain text): \nA server error occurred. Please contact the administrator.\nI understand a ValueError should be raised which tells the URL resolver \"this path does not match, try next one\" but Http404 is what came to my mind intuitively and the error message was not very helpful.\nOne could also make a point that raising a Http404 should be valid way to tell the resolver \"this is indeed the right path but the current parameter value does not match anything so stop what you are doing and let the handler return the 404 page (including a helpful error message when DEBUG is True instead of the default 'Django tried these URL patterns')\".\nThis would prove useful for example to implement a path converter that uses get_object_or_404.\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@@ -5,10 +5,10 @@\n from pathlib import Path\n \n from django.conf import settings\n-from django.http import HttpResponse, HttpResponseNotFound\n+from django.http import Http404, HttpResponse, HttpResponseNotFound\n from django.template import Context, Engine, TemplateDoesNotExist\n from django.template.defaultfilters import pprint\n-from django.urls import Resolver404, resolve\n+from django.urls import resolve\n from django.utils import timezone\n from django.utils.datastructures import MultiValueDict\n from django.utils.encoding import force_str\n@@ -483,7 +483,7 @@ def technical_404_response(request, exception):\n     caller = ''\n     try:\n         resolver_match = resolve(request.path)\n-    except Resolver404:\n+    except Http404:\n         pass\n     else:\n         obj = resolver_match.func\n",
  "similar_bug_items": [
    {
      "pr_number": 10027,
      "pr_title": "Fixed #24384 -- Allowed compilemessages to continue running after nonfatal errors.",
      "pr_body": "",
      "issue_id": 24384,
      "issue_title": "compilemessages shouldn't return with exit code 0 when it fails",
      "issue_body": "When compilemessages is run on a read-only location, it displays a messages to standard error and terminates with exit code 0. It should have exit code 1.\nPatch:\ndiff --git a/django/core/management/commands/compilemessages.py b/django/core/management/commands/compilemessages.py\nindex dbadac0..75f3c57 100644\n--- a/django/core/management/commands/compilemessages.py\n+++ b/django/core/management/commands/compilemessages.py\n@@ -108,9 +108,9 @@ class Command(BaseCommand):\n \n             # Check writability on first location\n             if i == 0 and not is_writable(npath(base_path + '.mo')):\n-                self.stderr.write(\"The po files under %s are in a seemingly not writable location. \"\n-                                  \"mo files will not be updated/created.\" % dirpath)\n-                return\n+                raise CommandError(\"The po files under %s are in a seemingly \"\n+                                   \"not writable location. mo files will not \"\n+                                   \"be updated/created.\" % dirpath)\n \n             args = [self.program] + self.program_options + ['-o',\n                     npath(base_path + '.mo'), npath(base_path + '.po')]\nWith this change, the output is exactly the same, but the exit code reflects the failure and the --traceback option works.\nI don't quite have the motivation to write tests for this change :-/",
      "issue_closed_at": "2018-06-13T14:27:08",
      "base_commit": "b30f9b131c9489b9d9f21c311ecb46d0aea91381",
      "changes": [
        {
          "file": "django/core/management/commands/compilemessages.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, **options):\n        locale = options['locale']\n        exclude = options['exclude']\n        self.verbosity = options['verbosity']\n        if options['fuzzy']:\n            self.program_options = self.program_options + ['-f']\n\n        if find_command(self.program) is None:\n            raise CommandError(\"Can't find %s. Make sure you have GNU gettext \"\n                               \"tools 0.15 or newer installed.\" % self.program)\n\n        basedirs = [os.path.join('conf', 'locale'), 'locale']\n        if os.environ.get('DJANGO_SETTINGS_MODULE'):\n            from django.conf import settings\n            basedirs.extend(settings.LOCALE_PATHS)\n\n        # Walk entire tree, looking for locale directories\n        for dirpath, dirnames, filenames in os.walk('.', topdown=True):\n            for dirname in dirnames:\n                if dirname == 'locale':\n                    basedirs.append(os.path.join(dirpath, dirname))\n\n        # Gather existing directories.\n        basedirs = set(map(os.path.abspath, filter(os.path.isdir, basedirs)))\n\n        if not basedirs:\n            raise CommandError(\"This script should be run from the Django Git \"\n                               \"checkout or your project or app tree, or with \"\n                               \"the settings module specified.\")\n\n        # Build locale list\n        all_locales = []\n        for basedir in basedirs:\n            locale_dirs = filter(os.path.isdir, glob.glob('%s/*' % basedir))\n            all_locales.extend(map(os.path.basename, locale_dirs))\n\n        # Account for excluded locales\n        locales = locale or all_locales\n        locales = set(locales).difference(exclude)\n\n        for basedir in basedirs:\n            if locales:\n                dirs = [os.path.join(basedir, l, 'LC_MESSAGES') for l in locales]\n            else:\n                dirs = [basedir]\n            locations = []\n            for ldir in dirs:\n                for dirpath, dirnames, filenames in os.walk(ldir):\n                    locations.extend((dirpath, f) for f in filenames if f.endswith('.po'))\n            if locations:\n                self.compile_messages(locations)"
        },
        {
          "file": "django/core/management/commands/compilemessages.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, **options):\n        locale = options['locale']\n        exclude = options['exclude']\n        self.verbosity = options['verbosity']\n        if options['fuzzy']:\n            self.program_options = self.program_options + ['-f']\n\n        if find_command(self.program) is None:\n            raise CommandError(\"Can't find %s. Make sure you have GNU gettext \"\n                               \"tools 0.15 or newer installed.\" % self.program)\n\n        basedirs = [os.path.join('conf', 'locale'), 'locale']\n        if os.environ.get('DJANGO_SETTINGS_MODULE'):\n            from django.conf import settings\n            basedirs.extend(settings.LOCALE_PATHS)\n\n        # Walk entire tree, looking for locale directories\n        for dirpath, dirnames, filenames in os.walk('.', topdown=True):\n            for dirname in dirnames:\n                if dirname == 'locale':\n                    basedirs.append(os.path.join(dirpath, dirname))\n\n        # Gather existing directories.\n        basedirs = set(map(os.path.abspath, filter(os.path.isdir, basedirs)))\n\n        if not basedirs:\n            raise CommandError(\"This script should be run from the Django Git \"\n                               \"checkout or your project or app tree, or with \"\n                               \"the settings module specified.\")\n\n        # Build locale list\n        all_locales = []\n        for basedir in basedirs:\n            locale_dirs = filter(os.path.isdir, glob.glob('%s/*' % basedir))\n            all_locales.extend(map(os.path.basename, locale_dirs))\n\n        # Account for excluded locales\n        locales = locale or all_locales\n        locales = set(locales).difference(exclude)\n\n        for basedir in basedirs:\n            if locales:\n                dirs = [os.path.join(basedir, l, 'LC_MESSAGES') for l in locales]\n            else:\n                dirs = [basedir]\n            locations = []\n            for ldir in dirs:\n                for dirpath, dirnames, filenames in os.walk(ldir):\n                    locations.extend((dirpath, f) for f in filenames if f.endswith('.po'))\n            if locations:\n                self.compile_messages(locations)"
        },
        {
          "file": "django/core/management/commands/compilemessages.py",
          "type": "function",
          "name": "compile_messages",
          "class_name": "Command",
          "code": "def compile_messages(self, locations):\n        \"\"\"\n        Locations is a list of tuples: [(directory, file), ...]\n        \"\"\"\n        for i, (dirpath, f) in enumerate(locations):\n            if self.verbosity > 0:\n                self.stdout.write('processing file %s in %s\\n' % (f, dirpath))\n            po_path = os.path.join(dirpath, f)\n            if has_bom(po_path):\n                raise CommandError(\"The %s file has a BOM (Byte Order Mark). \"\n                                   \"Django only supports .po files encoded in \"\n                                   \"UTF-8 and without any BOM.\" % po_path)\n            base_path = os.path.splitext(po_path)[0]\n\n            # Check writability on first location\n            if i == 0 and not is_writable(base_path + '.mo'):\n                self.stderr.write(\"The po files under %s are in a seemingly not writable location. \"\n                                  \"mo files will not be updated/created.\" % dirpath)\n                return\n\n            args = [self.program] + self.program_options + [\n                '-o', base_path + '.mo', base_path + '.po'\n            ]\n            output, errors, status = popen_wrapper(args)\n            if status:\n                if errors:\n                    msg = \"Execution of %s failed: %s\" % (self.program, errors)\n                else:\n                    msg = \"Execution of %s failed\" % self.program\n                raise CommandError(msg)"
        },
        {
          "file": "django/core/management/commands/compilemessages.py",
          "type": "function",
          "name": "compile_messages",
          "class_name": "Command",
          "code": "def compile_messages(self, locations):\n        \"\"\"\n        Locations is a list of tuples: [(directory, file), ...]\n        \"\"\"\n        for i, (dirpath, f) in enumerate(locations):\n            if self.verbosity > 0:\n                self.stdout.write('processing file %s in %s\\n' % (f, dirpath))\n            po_path = os.path.join(dirpath, f)\n            if has_bom(po_path):\n                raise CommandError(\"The %s file has a BOM (Byte Order Mark). \"\n                                   \"Django only supports .po files encoded in \"\n                                   \"UTF-8 and without any BOM.\" % po_path)\n            base_path = os.path.splitext(po_path)[0]\n\n            # Check writability on first location\n            if i == 0 and not is_writable(base_path + '.mo'):\n                self.stderr.write(\"The po files under %s are in a seemingly not writable location. \"\n                                  \"mo files will not be updated/created.\" % dirpath)\n                return\n\n            args = [self.program] + self.program_options + [\n                '-o', base_path + '.mo', base_path + '.po'\n            ]\n            output, errors, status = popen_wrapper(args)\n            if status:\n                if errors:\n                    msg = \"Execution of %s failed: %s\" % (self.program, errors)\n                else:\n                    msg = \"Execution of %s failed\" % self.program\n                raise CommandError(msg)"
        }
      ]
    },
    {
      "pr_number": 3989,
      "pr_title": "Fixed #24220 - Allowed lazy objects for success_url",
      "pr_body": "See https://code.djangoproject.com/ticket/24220 for more info\n",
      "issue_id": 24220,
      "issue_title": "get_success_url raises an exception when success_url is lazy object",
      "issue_body": "In\n#24133\nwas added deprecation warning when\nself.success_url\ncontains old %-style Python formatting strings.\nHowever,\nsuccess_url\ncould be a lazy object like\nreverse_lazy\nand trying to call\nre.search\non lazy object results in\nTypeError\n:\n../../../../.envs/lingui-next/lib/python3.4/site-packages/django/views/generic/edit.py:166: in get_success_url\n    if re.search(r'%\\([^\\)]+\\)', self.success_url):\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\npattern = '%\\\\([^\\\\)]+\\\\)'\nstring = <django.utils.functional.lazy.<locals>.__proxy__ object at 0x10678ca90>\nflags = 0\n\n    def search(pattern, string, flags=0):\n        \"\"\"Scan through string looking for a match to the pattern, returning\n        a match object, or None if no match was found.\"\"\"\n>       return _compile(pattern, flags).search(string)\nE       TypeError: expected string or buffer\n\n../../../../.envs/lingui-next/lib/python3.4/re.py:166: TypeError",
      "issue_closed_at": "2015-01-26T11:40:17",
      "base_commit": "ea0ea7859a224225950a4df7c23eb3a7d823ddcd",
      "changes": [
        {
          "file": "django/views/generic/edit.py",
          "type": "function",
          "name": "get_success_url",
          "class_name": "DeletionMixin",
          "code": "def get_success_url(self):\n        if self.success_url:\n            if PERCENT_PLACEHOLDER_REGEX.search(self.success_url):\n                warnings.warn(\n                    \"%()s placeholder style in success_url is deprecated. \"\n                    \"Please replace them by the {} Python format syntax.\",\n                    RemovedInDjango20Warning, stacklevel=2\n                )\n                return self.success_url % self.object.__dict__\n            else:\n                return self.success_url.format(**self.object.__dict__)\n        else:\n            raise ImproperlyConfigured(\n                \"No URL to redirect to. Provide a success_url.\")"
        },
        {
          "file": "django/views/generic/edit.py",
          "type": "function",
          "name": "post",
          "class_name": "DeletionMixin",
          "code": "def post(self, request, *args, **kwargs):\n        return self.delete(request, *args, **kwargs)"
        }
      ]
    },
    {
      "pr_number": 11169,
      "pr_title": "Fixed #30318 --  Added check for importability of arguments of custom error handler views.",
      "pr_body": "…mported\r\n\r\nThanks to Jon on Stack Overflow for reporting the issue.",
      "issue_id": 30318,
      "issue_title": "Add new system check message when custom error handler 'path.to.view' cannot be imported",
      "issue_body": "#29642\nadded checks for the signatures of custom error handlers.\nWhen the 'path.to.view' cannot be imported, it raises ModuleNotFoundError or ViewDoesNotExist, as seen in this Stack Overflow question:\n​\nhttps://stackoverflow.com/q/55481810/113962\nI suggest we catch the exception, and add another check code, e.g.\n* **urls.E008**: The custom ``handlerXXX`` view ``'path.to.view'`` cannot be imported.",
      "issue_closed_at": "2019-04-25T04:38:58",
      "base_commit": "fc9566d42daf28cdaa25a5db1b5ade253ceb064f",
      "changes": [
        {
          "file": "django/urls/resolvers.py",
          "type": "line",
          "name": "line 15",
          "code": "from django.conf import settings\nfrom django.core.checks import Error, Warning\nfrom django.core.checks.urls import check_resolver\nfrom django.core.exceptions import ImproperlyConfigured\nfrom django.utils.datastructures import MultiValueDict\nfrom django.utils.functional import cached_property\nfrom django.utils.http import RFC3986_SUBDELIMS, escape_leading_slashes"
        },
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "_check_custom_error_handlers",
          "class_name": "URLResolver",
          "code": "def _check_custom_error_handlers(self):\n        messages = []\n        # All handlers take (request, exception) arguments except handler500\n        # which takes (request).\n        for status_code, num_parameters in [(400, 2), (403, 2), (404, 2), (500, 1)]:\n            handler, param_dict = self.resolve_error_handler(status_code)\n            signature = inspect.signature(handler)\n            args = [None] * num_parameters\n            try:\n                signature.bind(*args)\n            except TypeError:\n                msg = (\n                    \"The custom handler{status_code} view '{path}' does not \"\n                    \"take the correct number of arguments ({args}).\"\n                ).format(\n                    status_code=status_code,\n                    path=handler.__module__ + '.' + handler.__qualname__,\n                    args='request, exception' if num_parameters == 2 else 'request',\n                )\n                messages.append(Error(msg, id='urls.E007'))\n        return messages"
        }
      ]
    },
    {
      "pr_number": 8396,
      "pr_title": "Fixed #28116 -- Used error code filtering in PostgreSQL test database creation.",
      "pr_body": "Ticket [28116](https://code.djangoproject.com/ticket/28116).",
      "issue_id": 28116,
      "issue_title": "Filtering PostgreSQL exception based on message is too brittle",
      "issue_body": "In commit\n64264c9a19b7dae6cd1d5e112177cdbcafafc93c\n, the behavior of\n_execute_create_test_db\ndepends on the databae error message\n''database %s already exists'\n.\nOn my system for example, the error message is localized so I'm finally getting a message\nGot an error creating the test database: la base de données « test_testgis1 » existe déjà\neven when I provide\n--keepdb\n.",
      "issue_closed_at": "2017-04-24T23:01:44",
      "base_commit": "8ef35468b660e1c25af67a8299202b8bc108679f",
      "changes": [
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "line",
          "name": "line 1",
          "code": "import sys\n\nfrom django.db.backends.base.creation import BaseDatabaseCreation\n\n"
        },
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "function",
          "name": "_execute_create_test_db",
          "class_name": "DatabaseCreation",
          "code": "def _execute_create_test_db(self, cursor, parameters, keepdb=False):\n        try:\n            super()._execute_create_test_db(cursor, parameters, keepdb)\n        except Exception as e:\n            exc_msg = 'database %s already exists' % parameters['dbname']\n            if exc_msg not in str(e):\n                # All errors except \"database already exists\" cancel tests\n                sys.stderr.write('Got an error creating the test database: %s\\n' % e)\n                sys.exit(2)\n            elif not keepdb:\n                # If the database should be kept, ignore \"database already\n                # exists\".\n                raise e"
        }
      ]
    },
    {
      "pr_number": 7471,
      "pr_title": "Fixed #26812 -- Check only path with APPEND_SLASH enabled",
      "pr_body": "",
      "issue_id": 26812,
      "issue_title": "APPEND_SLASH doesn't work with URLs that have query strings",
      "issue_body": "Hi,\nIn\nmiddleware/common.py\nthere is a code, which checks whether the slash should be appended to URL.\ndef should_redirect_with_slash(self, request):\n        \"\"\"\n        Return True if settings.APPEND_SLASH is True and appending a slash to\n        the request path turns an invalid path into a valid one.\n        \"\"\"\n\n        if settings.APPEND_SLASH and not request.get_full_path().endswith('/'):\n            urlconf = getattr(request, 'urlconf', None)\n\n            return (\n                not urlresolvers.is_valid_path(request.path_info, urlconf)\n                and urlresolvers.is_valid_path('%s/' % request.path_info, urlconf)\n            )\n        return False\nIn most cases it works fine, but it fails to append slash, when used with LOGIN_URL without slash and with \"next\" parameter. For example:\nIt works fine:\n/accounts/login/?next=/polls/3/\n-> LOGIN_URL =\n/accounts/login/\nBut this one returns 404:\n/accounts/login?next=/polls/3/\n-> LOGIN_URL =\n/accounts/login (no trailing slash).\nIt is because:\nrequest.get_full_path().endswith('/')\nchecks also query string and therefore returns true and the slash is never appended.",
      "issue_closed_at": "2016-11-06T03:38:42",
      "base_commit": "b741fe397aa567dd43d3e9dbd9fb5ecca6eab412",
      "changes": [
        {
          "file": "django/middleware/common.py",
          "type": "function",
          "name": "should_redirect_with_slash",
          "class_name": "CommonMiddleware",
          "code": "def should_redirect_with_slash(self, request):\n        \"\"\"\n        Return True if settings.APPEND_SLASH is True and appending a slash to\n        the request path turns an invalid path into a valid one.\n        \"\"\"\n        if settings.APPEND_SLASH and not request.get_full_path().endswith('/'):\n            urlconf = getattr(request, 'urlconf', None)\n            return (\n                not is_valid_path(request.path_info, urlconf) and\n                is_valid_path('%s/' % request.path_info, urlconf)\n            )\n        return False"
        }
      ]
    }
  ]
}