{
  "instance_id": "pallets__flask-5063",
  "repo": "pallets/flask",
  "created_at": "2023-04-14T16:36:54Z",
  "problem_statement": "Flask routes to return domain/sub-domains information\nCurrently when checking **flask routes** it provides all routes but **it is no way to see which routes are assigned to which subdomain**.\r\n\r\n**Default server name:**\r\nSERVER_NAME: 'test.local'\r\n\r\n**Domains (sub-domains):**\r\ntest.test.local\r\nadmin.test.local\r\ntest.local\r\n\r\n**Adding blueprints:**\r\napp.register_blueprint(admin_blueprint,url_prefix='',subdomain='admin')\r\napp.register_blueprint(test_subdomain_blueprint,url_prefix='',subdomain='test')\r\n\r\n\r\n```\r\n$ flask routes\r\n * Tip: There are .env or .flaskenv files present. Do \"pip install python-dotenv\" to use them.\r\nEndpoint                                                 Methods    Rule\r\n-------------------------------------------------------  ---------  ------------------------------------------------\r\nadmin_blueprint.home                                      GET        /home\r\ntest_subdomain_blueprint.home                             GET        /home\r\nstatic                                                    GET        /static/<path:filename>\r\n...\r\n```\r\n\r\n\r\n**Feature request**\r\nIt will be good to see something like below (that will make more clear which route for which subdomain, because now need to go and check configuration).\r\n**If it is not possible to fix routes**, can you add or tell which method(s) should be used to get below information from flask? \r\n\r\n```\r\n$ flask routes\r\n * Tip: There are .env or .flaskenv files present. Do \"pip install python-dotenv\" to use them.\r\nDomain                Endpoint                                             Methods    Rule\r\n-----------------   ----------------------------------------------------  ----------  ------------------------------------------------\r\nadmin.test.local     admin_blueprint.home                                  GET        /home\r\ntest.test.local      test_subdomain_blueprint.home                         GET        /home\r\ntest.local           static                                                GET        /static/<path:filename>\r\n...\r\n```\r\n\n",
  "patch": "diff --git a/src/flask/cli.py b/src/flask/cli.py\n--- a/src/flask/cli.py\n+++ b/src/flask/cli.py\n@@ -9,7 +9,7 @@\n import traceback\n import typing as t\n from functools import update_wrapper\n-from operator import attrgetter\n+from operator import itemgetter\n \n import click\n from click.core import ParameterSource\n@@ -989,49 +989,62 @@ def shell_command() -> None:\n @click.option(\n     \"--sort\",\n     \"-s\",\n-    type=click.Choice((\"endpoint\", \"methods\", \"rule\", \"match\")),\n+    type=click.Choice((\"endpoint\", \"methods\", \"domain\", \"rule\", \"match\")),\n     default=\"endpoint\",\n     help=(\n-        'Method to sort routes by. \"match\" is the order that Flask will match '\n-        \"routes when dispatching a request.\"\n+        \"Method to sort routes by. 'match' is the order that Flask will match routes\"\n+        \" when dispatching a request.\"\n     ),\n )\n @click.option(\"--all-methods\", is_flag=True, help=\"Show HEAD and OPTIONS methods.\")\n @with_appcontext\n def routes_command(sort: str, all_methods: bool) -> None:\n     \"\"\"Show all registered routes with endpoints and methods.\"\"\"\n-\n     rules = list(current_app.url_map.iter_rules())\n+\n     if not rules:\n         click.echo(\"No routes were registered.\")\n         return\n \n-    ignored_methods = set(() if all_methods else (\"HEAD\", \"OPTIONS\"))\n+    ignored_methods = set() if all_methods else {\"HEAD\", \"OPTIONS\"}\n+    host_matching = current_app.url_map.host_matching\n+    has_domain = any(rule.host if host_matching else rule.subdomain for rule in rules)\n+    rows = []\n \n-    if sort in (\"endpoint\", \"rule\"):\n-        rules = sorted(rules, key=attrgetter(sort))\n-    elif sort == \"methods\":\n-        rules = sorted(rules, key=lambda rule: sorted(rule.methods))  # type: ignore\n+    for rule in rules:\n+        row = [\n+            rule.endpoint,\n+            \", \".join(sorted((rule.methods or set()) - ignored_methods)),\n+        ]\n \n-    rule_methods = [\n-        \", \".join(sorted(rule.methods - ignored_methods))  # type: ignore\n-        for rule in rules\n-    ]\n+        if has_domain:\n+            row.append((rule.host if host_matching else rule.subdomain) or \"\")\n \n-    headers = (\"Endpoint\", \"Methods\", \"Rule\")\n-    widths = (\n-        max(len(rule.endpoint) for rule in rules),\n-        max(len(methods) for methods in rule_methods),\n-        max(len(rule.rule) for rule in rules),\n-    )\n-    widths = [max(len(h), w) for h, w in zip(headers, widths)]\n-    row = \"{{0:<{0}}}  {{1:<{1}}}  {{2:<{2}}}\".format(*widths)\n+        row.append(rule.rule)\n+        rows.append(row)\n+\n+    headers = [\"Endpoint\", \"Methods\"]\n+    sorts = [\"endpoint\", \"methods\"]\n+\n+    if has_domain:\n+        headers.append(\"Host\" if host_matching else \"Subdomain\")\n+        sorts.append(\"domain\")\n+\n+    headers.append(\"Rule\")\n+    sorts.append(\"rule\")\n+\n+    try:\n+        rows.sort(key=itemgetter(sorts.index(sort)))\n+    except ValueError:\n+        pass\n \n-    click.echo(row.format(*headers).strip())\n-    click.echo(row.format(*(\"-\" * width for width in widths)))\n+    rows.insert(0, headers)\n+    widths = [max(len(row[i]) for row in rows) for i in range(len(headers))]\n+    rows.insert(1, [\"-\" * w for w in widths])\n+    template = \"  \".join(f\"{{{i}:<{w}}}\" for i, w in enumerate(widths))\n \n-    for rule, methods in zip(rules, rule_methods):\n-        click.echo(row.format(rule.endpoint, methods, rule.rule).rstrip())\n+    for row in rows:\n+        click.echo(template.format(*row))\n \n \n cli = FlaskGroup(\n",
  "similar_bug_items": [
    {
      "pr_number": 2746,
      "pr_title": "allow empty prefix and no lead slash in bp route",
      "pr_body": "closes #2742\r\n\r\nFix `url_prefx=''` or `url_prefix='/'` when Blueprint routes don't begin with a leading slash. Technically, routes should start with a leading slash, so don't rely on this behavior.",
      "issue_id": 2742,
      "issue_title": "Flask 1.0 backwards incompatible root url_prefix",
      "issue_body": "Some of the recent PRs in blueprint.py seem to make Flask 1.0 not backwards compatible with 0.x versions.  When `url_prefix=''`, `url_prefix='/'`, or `url_prefix` is not specified, a `ValueError('urls must start with a leading slash')` is now raised. \r\n\r\n### Expected Behavior\r\n\r\nI would have assumed, given the discussion in #2629, that `''` and maybe `/` would be acceptable url_prefixes to indicate the blueprint is meant for the root.\r\n\r\nAn example of a broken library is `flask-sitemap` (https://github.com/inveniosoftware/flask-sitemap/blob/master/flask_sitemap/__init__.py#L120) which uses\r\n\r\n```\r\n            app.register_blueprint(\r\n                self.blueprint,\r\n                url_prefix=app.config.get('SITEMAP_BLUEPRINT_URL_PREFIX') # default is '/'\r\n            )\r\n```\r\n\r\n\r\n### Actual Behavior\r\n\r\n In my testing, I continually get \r\n\r\n```  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/flask_sitemap/__init__.py\", line 79, in __init__\r\n    self.init_app(app)\r\n  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/flask_sitemap/__init__.py\", line 120, in init_app\r\n    url_prefix='' # app.config.get('SITEMAP_BLUEPRINT_URL_PREFIX')\r\n  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/flask/app.py\", line 64, in wrapper_func\r\n    return f(self, *args, **kwargs)\r\n  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/flask/app.py\", line 1113, in register_blueprint\r\n    blueprint.register(self, options, first_registration)\r\n  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/flask/blueprints.py\", line 186, in register\r\n    deferred(state)\r\n  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/flask/blueprints.py\", line 207, in <lambda>\r\n    s.add_url_rule(rule, endpoint, view_func, **options))\r\n  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/flask/blueprints.py\", line 79, in add_url_rule\r\n    view_func, defaults=defaults, **options)\r\n  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/flask/app.py\", line 64, in wrapper_func\r\n    return f(self, *args, **kwargs)\r\n  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/flask/app.py\", line 1211, in add_url_rule\r\n    rule = self.url_rule_class(rule, methods=methods, **options)\r\n  File \"/home/albertyw/.virtualenvs/baseflask/local/lib/python2.7/site-packages/werkzeug/routing.py\", line 603, in __init__\r\n    raise ValueError('urls must start with a leading slash')\r\nValueError: urls must start with a leading slash\r\n```\r\n\r\nunless I set `url_prefix='//'`, in which case the url_prefix is the root.\r\n\r\n### Environment\r\n\r\n* Python version: python 2.7 and 3.6\r\n* Flask version: 1.0\r\n* Werkzeug version: 0.14.1\r\n",
      "issue_closed_at": "2018-04-29T22:47:15Z",
      "base_commit": "6b2127b1e0ef474aed91a25492e18a361ad7f364",
      "changes": [
        {
          "file": "flask/blueprints.py",
          "type": "function",
          "name": "add_url_rule",
          "class_name": "Blueprint",
          "code": "def add_url_rule(self, rule, endpoint=None, view_func=None, **options):\n        \"\"\"Like :meth:`Flask.add_url_rule` but for a blueprint.  The endpoint for\n        the :func:`url_for` function is prefixed with the name of the blueprint.\n        \"\"\"\n        if endpoint:\n            assert '.' not in endpoint, \"Blueprint endpoints should not contain dots\"\n        if view_func and hasattr(view_func, '__name__'):\n            assert '.' not in view_func.__name__, \"Blueprint view function name should not contain dots\"\n        self.record(lambda s:\n            s.add_url_rule(rule, endpoint, view_func, **options))"
        }
      ]
    },
    {
      "pr_number": 2738,
      "pr_title": "merge slashes between blueprint prefix and rule",
      "pr_body": "closes #2731 \r\n\r\nWhen registering a blueprint, strip `/` from the right side of the prefix and the left side of each rule, then join to ensure there's only one slash. #2629 only considered the prefix, and only stripped one slash.",
      "issue_id": 2731,
      "issue_title": "Flask 1.0 backwards-incompat with double-slash/no-slash re. #2629",
      "issue_body": "This is a major backwards-compat breaking change, but I suspect not the intended design and hopefully easy to fix.\r\n\r\nThe issue is related to PR #2629, and this example follows from that:\r\n\r\nGiven blueprint `bp` and app `app`:\r\n\r\n```python\r\n@bp.route('b/')\r\ndef tmp():\r\n    return \"URI should be '/a/b/\"\r\n\r\napp.register_blueprint(bp, url_prefix='/a/')\r\n```\r\n\r\nIn Flask 0.12 the URL is correctly `/a/b`, but in Flask 1.0 it's `/ab`.\r\n\r\nSince issue #2629 relates to resolve double-slashes, I imagine this is a bug (and not a design decision) - and the correct solution would be to remove a slash only when there are two.\r\n",
      "issue_closed_at": "2018-04-27T19:47:01Z",
      "base_commit": "5239458f28c5e3057f9987ceb2e6d33b9d9532d3",
      "changes": [
        {
          "file": "flask/blueprints.py",
          "type": "function",
          "name": "__init__",
          "class_name": "Blueprint",
          "code": "def __init__(self, name, import_name, static_folder=None,\n                 static_url_path=None, template_folder=None,\n                 url_prefix=None, subdomain=None, url_defaults=None,\n                 root_path=None):\n        _PackageBoundObject.__init__(self, import_name, template_folder,\n                                     root_path=root_path)\n        self.name = name\n        self.url_prefix = url_prefix\n        self.subdomain = subdomain\n        self.static_folder = static_folder\n        self.static_url_path = static_url_path\n        self.deferred_functions = []\n        if url_defaults is None:\n            url_defaults = {}\n        self.url_values_defaults = url_defaults"
        },
        {
          "file": "flask/blueprints.py",
          "type": "function",
          "name": "add_url_rule",
          "class_name": "Blueprint",
          "code": "def add_url_rule(self, rule, endpoint=None, view_func=None, **options):\n        \"\"\"Like :meth:`Flask.add_url_rule` but for a blueprint.  The endpoint for\n        the :func:`url_for` function is prefixed with the name of the blueprint.\n        \"\"\"\n        if endpoint:\n            assert '.' not in endpoint, \"Blueprint endpoints should not contain dots\"\n        if view_func and hasattr(view_func, '__name__'):\n            assert '.' not in view_func.__name__, \"Blueprint view function name should not contain dots\"\n        self.record(lambda s:\n            s.add_url_rule(rule, endpoint, view_func, **options))"
        }
      ]
    },
    {
      "pr_number": 4560,
      "pr_title": "Fix old behavior in `register_error_handler()`",
      "pr_body": "- Fixes #4559 ",
      "issue_id": 4559,
      "issue_title": "Fix old behavior in `register_error_handler()` ",
      "issue_body": "## Background\r\n[`Line 700 src/scaffold.py`](https://github.com/pallets/flask/blob/f976d5bb88216e921a96998f767df31d7039e4ef/src/flask/scaffold.py#L700) has a comment `# old broken behavior`.  \r\n```python\r\nif isinstance(code_or_exception, HTTPException):  # old broken behavior\r\n    raise ValueError(\r\n        \"Tried to register a handler for an exception instance\"\r\n        f\" {code_or_exception!r}. Handlers can only be\"\r\n        \" registered for exception classes or HTTP error codes.\"\r\n    )\r\n```\r\nThe purpose of this code is check if the handler is an exception instance.   \r\nIf so, raise an error to warn the user: the handler should be HTTPException classes or HTTP error codes.   \r\n### Trigger test case: \r\nIn [`test/test_user_error_handler.py`](https://github.com/pallets/flask/blob/f976d5bb88216e921a96998f767df31d7039e4ef/tests/test_user_error_handler.py), add:\r\n```python\r\nwith pytest.raises(ValueError):\r\n    app.register_error_handler(HTTPException(), None)\r\n```\r\n### Why is this old and broken?  \r\nFrom version 0.7, `self.register_error_handler` can accept all `Exception` not just `HTTPException` so it is not accurate now. Also, the main check part for handlers is in `self._get_exc_class_and_code`.\r\n\r\n## Purposed Solution\r\nRemove this part of code.  \r\nEnhance the check for handlers in `self.register_error_handler`.  \r\n\r\n## Detail Discussion\r\n- Since the main check is in `self._get_exc_class_and_code`, new check part is written in `self._get_exc_class_and_code` to keep consistency.\r\n- `isinstance(exc_class, Exception)` is kept for warning error-prone cases.\r\n- Idea from [`inspect.isclass`](https://docs.python.org/3/library/inspect.html#inspect.isclass) to solve `Line 735 scr/scaffold.py` `TypeError: issubclass() arg 1 must be a class` when arg1 is a instance.  \r\nThis idea uses `New-style Classes` feature in Python3. Since Flask needs Python>=3.7, it should not cause any compatibility issues.\r\n- Move `except KeyError` after confirm `int` type to improve type safety.\r\n- Print `exc_class` when AssertionError to improve traceback.\r\n\r\nA PR will be linked.",
      "issue_closed_at": "2022-05-03T17:56:11Z",
      "base_commit": "f976d5bb88216e921a96998f767df31d7039e4ef",
      "changes": [
        {
          "file": "src/flask/scaffold.py",
          "type": "function",
          "name": "register_error_handler",
          "class_name": "Scaffold",
          "code": "def register_error_handler(\n        self,\n        code_or_exception: t.Union[t.Type[Exception], int],\n        f: \"ErrorHandlerCallable\",\n    ) -> None:\n        \"\"\"Alternative error attach function to the :meth:`errorhandler`\n        decorator that is more straightforward to use for non decorator\n        usage.\n\n        .. versionadded:: 0.7\n        \"\"\"\n        if isinstance(code_or_exception, HTTPException):  # old broken behavior\n            raise ValueError(\n                \"Tried to register a handler for an exception instance\"\n                f\" {code_or_exception!r}. Handlers can only be\"\n                \" registered for exception classes or HTTP error codes.\"\n            )\n\n        try:\n            exc_class, code = self._get_exc_class_and_code(code_or_exception)\n        except KeyError:\n            raise KeyError(\n                f\"'{code_or_exception}' is not a recognized HTTP error\"\n                \" code. Use a subclass of HTTPException with that code\"\n                \" instead.\"\n            ) from None\n\n        self.error_handler_spec[None][code][exc_class] = f"
        },
        {
          "file": "src/flask/scaffold.py",
          "type": "function",
          "name": "_get_exc_class_and_code",
          "class_name": "Scaffold",
          "code": "def _get_exc_class_and_code(\n        exc_class_or_code: t.Union[t.Type[Exception], int]\n    ) -> t.Tuple[t.Type[Exception], t.Optional[int]]:\n        \"\"\"Get the exception class being handled. For HTTP status codes\n        or ``HTTPException`` subclasses, return both the exception and\n        status code.\n\n        :param exc_class_or_code: Any exception class, or an HTTP status\n            code as an integer.\n        \"\"\"\n        exc_class: t.Type[Exception]\n        if isinstance(exc_class_or_code, int):\n            exc_class = default_exceptions[exc_class_or_code]\n        else:\n            exc_class = exc_class_or_code\n\n        assert issubclass(\n            exc_class, Exception\n        ), \"Custom exceptions must be subclasses of Exception.\"\n\n        if issubclass(exc_class, HTTPException):\n            return exc_class, exc_class.code\n        else:\n            return exc_class, None"
        }
      ]
    },
    {
      "pr_number": 2348,
      "pr_title": "make debugging bad key errors easier",
      "pr_body": "* `TRAP_BAD_REQUEST_ERRORS` is enabled by default in debug mode\r\n* `BadRequestKeyError` has the key in the description in debug mode\r\n\r\nIf `app.debug` is true but we're not running in the interactive debugger, this now raises a 500 error (the traceback in the console has the key error though). Can get the old behavior by setting `TRAP_BAD_REQUEST_ERRORS = False`.\r\n\r\nThis error is one of the most recurring questions about Flask on Stack Overflow. Making the error visible will hopefully answer a lot of questions before they're asked.\r\n\r\ncloses #382",
      "issue_id": 382,
      "issue_title": "Set TRAP_BAD_REQUEST_ERRORS to True as the default when debugging is enabled",
      "issue_body": "When debugging an application, it's surprising to get a simple \"400 Bad Request\" when accessing a form parameter that doesn't exist instead of a full traceback. I always forget that I also need to set TRAP_BAD_REQUEST_ERRORS.\n\nIs there a reason why this isn't set by default when debugging is enabled?\n",
      "issue_closed_at": "2017-05-30T02:53:19Z",
      "base_commit": "fb90c310b9c9099118c06a95183e38157727fa92",
      "changes": [
        {
          "file": "flask/app.py",
          "type": "line",
          "name": "line 18",
          "code": "\nfrom werkzeug.datastructures import ImmutableDict, Headers\nfrom werkzeug.exceptions import BadRequest, HTTPException, \\\n    InternalServerError, MethodNotAllowed, default_exceptions\nfrom werkzeug.routing import BuildError, Map, RequestRedirect, Rule\n\nfrom . import cli, json"
        },
        {
          "file": "flask/app.py",
          "type": "function",
          "name": "_set_request_globals_class",
          "class_name": "Flask",
          "code": "def _set_request_globals_class(self, value):\n        from warnings import warn\n        warn(DeprecationWarning('request_globals_class attribute is now '\n                                'called app_ctx_globals_class'))\n        self.app_ctx_globals_class = value"
        },
        {
          "file": "flask/app.py",
          "type": "function",
          "name": "trap_http_exception",
          "class_name": "Flask",
          "code": "def trap_http_exception(self, e):\n        \"\"\"Checks if an HTTP exception should be trapped or not.  By default\n        this will return ``False`` for all exceptions except for a bad request\n        key error if ``TRAP_BAD_REQUEST_ERRORS`` is set to ``True``.  It\n        also returns ``True`` if ``TRAP_HTTP_EXCEPTIONS`` is set to ``True``.\n\n        This is called for all HTTP exceptions raised by a view function.\n        If it returns ``True`` for any exception the error handler for this\n        exception is not called and it shows up as regular exception in the\n        traceback.  This is helpful for debugging implicitly raised HTTP\n        exceptions.\n\n        .. versionadded:: 0.8\n        \"\"\"\n        if self.config['TRAP_HTTP_EXCEPTIONS']:\n            return True\n        if self.config['TRAP_BAD_REQUEST_ERRORS']:\n            return isinstance(e, BadRequest)\n        return False"
        },
        {
          "file": "flask/app.py",
          "type": "function",
          "name": "handle_user_exception",
          "class_name": "Flask",
          "code": "def handle_user_exception(self, e):\n        \"\"\"This method is called whenever an exception occurs that should be\n        handled.  A special case are\n        :class:`~werkzeug.exception.HTTPException`\\s which are forwarded by\n        this function to the :meth:`handle_http_exception` method.  This\n        function will either return a response value or reraise the\n        exception with the same traceback.\n\n        .. versionadded:: 0.7\n        \"\"\"\n        exc_type, exc_value, tb = sys.exc_info()\n        assert exc_value is e\n\n        # ensure not to trash sys.exc_info() at that point in case someone\n        # wants the traceback preserved in handle_http_exception.  Of course\n        # we cannot prevent users from trashing it themselves in a custom\n        # trap_http_exception method so that's their fault then.\n\n        if isinstance(e, HTTPException) and not self.trap_http_exception(e):\n            return self.handle_http_exception(e)\n\n        handler = self._find_error_handler(e)\n\n        if handler is None:\n            reraise(exc_type, exc_value, tb)\n        return handler(e)"
        }
      ]
    },
    {
      "pr_number": 3711,
      "pr_title": "cleaner message when CLI can't load app",
      "pr_body": "When loading the app fails for the `--help` command, only the error message is shown, then the help text. The full traceback is still shown for other exceptions. Also show the message when loading fails while getting a command, instead of only \"command not found\". The error message goes to `stderr` to match other error behavior, and is in red with an extra newline to make it more obvious next to the help text.\r\n\r\nAlso fixes an issue with the `test_apps` fixture that caused an imported app to still be importable after the test was over and the path was reset. Now the module cache is reset as well. This was causing an issue with the new tests, because previous tests had imported `test_apps/helloworld/wsgi`, one of the default imports that is tried if `FLASK_APP` isn't set, which caused the test to import an app successfully even though it should have failed and shown an error. This also revealed an issue with an existing test relying on the cached behavior.\r\n\r\nfixes #2741 ",
      "issue_id": 2741,
      "issue_title": "Better exception handling when env vars are missing for flask CLI",
      "issue_body": "### Expected Behavior\r\n\r\nAs there is the support for `lazy loading` the app, when running `flask` CLI without providing the proper environment variables we must see a better warning instead of raw exception.\r\n\r\nTell us what should happen.\r\n\r\n**we should see a better warning or message pointing to the problem**\r\n\r\n```pytb\r\n# there is no FLASK_APP env var\r\n$ flask --help\r\nWARNING: You need to define the app e.g: `export FLASK_APP=app.py`\r\n```\r\n### Actual Behavior\r\n\r\nTell us what happens instead.\r\n\r\n**we see traceback before the help message**\r\n\r\n```pytb\r\n# there is no FLASK_APP env var\r\n$ flask --help                                                               Sat 28 Apr 2018 01:25:50 PM -03\r\nTraceback (most recent call last):\r\n  File \"~Projects/personal/flasgger/venv/lib/python3.6/site-packages/flask/cli.py\", line 235, in locate_app\r\n    __import__(module_name)\r\nModuleNotFoundError: No module named 'app'\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nTraceback (most recent call last):\r\n  File \"~/Projects/personal/flasgger/venv/lib/python3.6/site-packages/flask/cli.py\", line 529, in list_commands\r\n    rv.update(info.load_app().cli.list_commands(ctx))\r\n  File \"~/Projects/personal/flasgger/venv/lib/python3.6/site-packages/flask/cli.py\", line 372, in load_app\r\n    app = locate_app(self, import_name, name)\r\n  File \"~/Projects/personal/flasgger/venv/lib/python3.6/site-packages/flask/cli.py\", line 246, in locate_app\r\n    'Could not import \"{name}\".'.format(name=module_name)\r\nflask.cli.NoAppException: Could not import \"app\".\r\nUsage: flask [OPTIONS] COMMAND [ARGS]...\r\n\r\n  A general utility script for Flask applications.\r\n\r\n  Provides commands from Flask, extensions, and the application. Loads the\r\n  application defined in the FLASK_APP environment variable, or from a\r\n  wsgi.py file. Setting the FLASK_ENV environment variable to 'development'\r\n  will enable debug mode.\r\n\r\n    $ export FLASK_APP=hello.py\r\n    $ export FLASK_ENV=development\r\n    $ flask run\r\n\r\nOptions:\r\n  --version  Show the flask version\r\n  --help     Show this message and exit.\r\n\r\nCommands:\r\n  routes  Show the routes for the app.\r\n  run     Runs a development server.\r\n  shell   Runs a shell in the app context.\r\n\r\n```\r\n\r\nThe same happens to `run`\r\n\r\n```pytb\r\n$ flask run                                                          429ms \ue0b3 Sat 28 Apr 2018 01:32:48 PM -03\r\n * Serving Flask app \"app.py\"\r\n * Environment: production\r\n   WARNING: Do not use the development server in a production environment.\r\n   Use a production WSGI server instead.\r\n * Debug mode: off\r\nUsage: flask run [OPTIONS]\r\n\r\nError: Could not import \"app\".\r\n```\r\n\r\nThe `Error: Could not import \"app\".` could include `WARNING: You need to define the app e.g: export FLASK_APP=app.py`\r\n\r\n\r\n### Suggestion\r\n\r\nWe could check the existence of `FLASK_APP` envvar before running any of the commands in the Group Cli, if FLASK_APP does not exist the dispatch of commands never happens.\r\n\r\n\r\n### Environment\r\n\r\n* Python version: 3.6.0\r\n* Flask version: Flask==1.0\r\n* Werkzeug version: Werkzeug==0.14.1\r\n* Click: click==6.7 \r\n",
      "issue_closed_at": "2020-07-31T01:48:46Z",
      "base_commit": "fd0a6084498e9d1990ddcf96366e8ac27b63428a",
      "changes": [
        {
          "file": "src/flask/cli.py",
          "type": "function",
          "name": "_load_plugin_commands",
          "class_name": "FlaskGroup",
          "code": "def _load_plugin_commands(self):\n        if self._loaded_plugin_commands:\n            return\n        try:\n            import pkg_resources\n        except ImportError:\n            self._loaded_plugin_commands = True\n            return\n\n        for ep in pkg_resources.iter_entry_points(\"flask.commands\"):\n            self.add_command(ep.load(), ep.name)\n        self._loaded_plugin_commands = True"
        }
      ]
    }
  ]
}