{
  "instance_id": "pallets__flask-4992",
  "repo": "pallets/flask",
  "created_at": "2023-02-22T14:00:17Z",
  "problem_statement": "Add a file mode parameter to flask.Config.from_file()\nPython 3.11 introduced native TOML support with the `tomllib` package. This could work nicely with the `flask.Config.from_file()` method as an easy way to load TOML config files:\r\n\r\n```python\r\napp.config.from_file(\"config.toml\", tomllib.load)\r\n```\r\n\r\nHowever, `tomllib.load()` takes an object readable in binary mode, while `flask.Config.from_file()` opens a file in text mode, resulting in this error:\r\n\r\n```\r\nTypeError: File must be opened in binary mode, e.g. use `open('foo.toml', 'rb')`\r\n```\r\n\r\nWe can get around this with a more verbose expression, like loading from a file opened with the built-in `open()` function and passing the `dict` to `app.Config.from_mapping()`:\r\n\r\n```python\r\n# We have to repeat the path joining that from_file() does\r\nwith open(os.path.join(app.config.root_path, \"config.toml\"), \"rb\") as file:\r\n    app.config.from_mapping(tomllib.load(file))\r\n```\r\n\r\nBut adding a file mode parameter to `flask.Config.from_file()` would enable the use of a simpler expression. E.g.:\r\n\r\n```python\r\napp.config.from_file(\"config.toml\", tomllib.load, mode=\"b\")\r\n```\r\n\n",
  "patch": "diff --git a/src/flask/config.py b/src/flask/config.py\n--- a/src/flask/config.py\n+++ b/src/flask/config.py\n@@ -234,6 +234,7 @@ def from_file(\n         filename: str,\n         load: t.Callable[[t.IO[t.Any]], t.Mapping],\n         silent: bool = False,\n+        text: bool = True,\n     ) -> bool:\n         \"\"\"Update the values in the config from a file that is loaded\n         using the ``load`` parameter. The loaded data is passed to the\n@@ -244,8 +245,8 @@ def from_file(\n             import json\n             app.config.from_file(\"config.json\", load=json.load)\n \n-            import toml\n-            app.config.from_file(\"config.toml\", load=toml.load)\n+            import tomllib\n+            app.config.from_file(\"config.toml\", load=tomllib.load, text=False)\n \n         :param filename: The path to the data file. This can be an\n             absolute path or relative to the config root path.\n@@ -254,14 +255,18 @@ def from_file(\n         :type load: ``Callable[[Reader], Mapping]`` where ``Reader``\n             implements a ``read`` method.\n         :param silent: Ignore the file if it doesn't exist.\n+        :param text: Open the file in text or binary mode.\n         :return: ``True`` if the file was loaded successfully.\n \n+        .. versionchanged:: 2.3\n+            The ``text`` parameter was added.\n+\n         .. versionadded:: 2.0\n         \"\"\"\n         filename = os.path.join(self.root_path, filename)\n \n         try:\n-            with open(filename) as f:\n+            with open(filename, \"r\" if text else \"rb\") as f:\n                 obj = load(f)\n         except OSError as e:\n             if silent and e.errno in (errno.ENOENT, errno.EISDIR):\n",
  "similar_bug_items": [
    {
      "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"
        }
      ]
    },
    {
      "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": 2963,
      "pr_title": "Fix #2961\uff1aignore colon followed by slash when split app_import_path",
      "pr_body": "Flask currently supports importing app through a combination of module path and app variable name, such as '/usr/app.py:my_app'. When the module path contains a colon, it will conflict with this import way and a `flask.cli.NoAppException` will be raised.\r\n\r\nA file path on a Windows system may contain a colon followed by a slash. So we solved this problem on Windows by ignoring the colon followed by a slash when we split `app_import_path`. \r\n\r\nFix issue #2961.",
      "issue_id": 2961,
      "issue_title": "Flask doens't run when APP is specified with full path",
      "issue_body": "### Expected Behavior\r\nPreviously in our development environment we ran flask app from vscode but after we updated to the latest version of flask we started experiencing problems. After some investigation I found out that in our run configuration we specify the whole path to the app.py-file. Changing to just referencing it directly (i.e. app.py instead of C:\\Path\\To\\My\\app.py) I got it to work.\r\n\r\nI recreated the fault in another simpler environment:\r\n\r\nThe following command runs \r\n``` env \"FLASK_APP=`pwd`/app.py\" \"FLASK_ENV=Development\" \"FLASK_DEBUG=1\" env/Scripts/python.exe -m flask run```\r\n\r\nbut when you try to access localhost:5000/ you get the following error\r\n\r\n```Traceback (most recent call last):\r\n  File \"C:\\Users\\AlexanderS\\code\\flask-bugg\\env\\lib\\site-packages\\flask\\cli.py\", line 330, in __call__\r\n    rv = self._load_unlocked()\r\n  File \"C:\\Users\\AlexanderS\\code\\flask-bugg\\env\\lib\\site-packages\\flask\\cli.py\", line 317, in _load_unlocked\r\n    self._app = rv = self.loader()\r\n  File \"C:\\Users\\AlexanderS\\code\\flask-bugg\\env\\lib\\site-packages\\flask\\cli.py\", line 372, in load_app\r\n    app = locate_app(self, import_name, name)\r\n  File \"C:\\Users\\AlexanderS\\code\\flask-bugg\\env\\lib\\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 \"C\".\r\n```\r\n\r\n\r\n### Actual Behavior\r\nThe application should run without errors.\r\n\r\n### Environment\r\n\r\n* Python version: 3.6.5\r\n* Flask version: 1.0.2\r\n* Werkzeug version: 0.14.1\r\n\r\nRecreated the bug with the follow app.py\r\n\r\n```python\r\nfrom flask import Flask\r\n\r\napp = Flask(__name__)\r\n\r\n@app.route('/')\r\ndef hello_world():\r\n    return 'Hello world2'\r\n```\r\n",
      "issue_closed_at": "2018-10-24T16:07:06Z",
      "base_commit": "363205bdc3abcfeed08c7e85e397fb7a7dc9ef07",
      "changes": [
        {
          "file": "flask/cli.py",
          "type": "function",
          "name": "load_app",
          "class_name": "ScriptInfo",
          "code": "def load_app(self):\n        \"\"\"Loads the Flask app (if not yet loaded) and returns it.  Calling\n        this multiple times will just result in the already loaded app to\n        be returned.\n        \"\"\"\n        __traceback_hide__ = True\n\n        if self._loaded_app is not None:\n            return self._loaded_app\n\n        app = None\n\n        if self.create_app is not None:\n            app = call_factory(self, self.create_app)\n        else:\n            if self.app_import_path:\n                path, name = (self.app_import_path.split(':', 1) + [None])[:2]\n                import_name = prepare_import(path)\n                app = locate_app(self, import_name, name)\n            else:\n                for path in ('wsgi.py', 'app.py'):\n                    import_name = prepare_import(path)\n                    app = locate_app(self, import_name, None,\n                                     raise_if_not_found=False)\n\n                    if app:\n                        break\n\n        if not app:\n            raise NoAppException(\n                'Could not locate a Flask application. You did not provide '\n                'the \"FLASK_APP\" environment variable, and a \"wsgi.py\" or '\n                '\"app.py\" module was not found in the current directory.'\n            )\n\n        if self.set_debug_flag:\n            # Update the app's debug flag through the descriptor so that\n            # other values repopulate as well.\n            app.debug = get_debug_flag()\n\n        self._loaded_app = app\n        return app"
        }
      ]
    },
    {
      "pr_number": 4049,
      "pr_title": "Use compat fspath instead of os.fspath in static_folder",
      "pr_body": "When #3678 was merged it introduced the usage of os.fspath which is not supported on Python <3.6\r\n\r\nThis PR updates `static_folder` to use the `fspath` from `_compat` and un-skips the test case.\r\n\r\n- fixes #4050 \r\n\r\nChecklist:\r\n\r\n- [x] Add tests that demonstrate the correct behavior of the change. Tests should fail without the change.\r\n- ~[ ] Add or update relevant docs, in the docs folder and in code.~\r\n- [x] Add an entry in `CHANGES.rst` summarizing the change and linking to the issue.\r\n- ~[ ] Add `.. versionchanged::` entries in any relevant code docs.~\r\n- [x] Run `pre-commit` hooks and fix any issues.\r\n- [x] Run `pytest` and `tox`, no tests failed.\r\n",
      "issue_id": 4050,
      "issue_title": "AttributeError: 'module' object has no attribute 'fspath'",
      "issue_body": "`AttributeError: 'module' object has no attribute 'fspath'` exception is thrown when running `app = flask.Flask(__name__)` for flask v1.13.\r\n\r\nVersion is Python 2.7 and 3.5. I realize `fspath` is only available for Python version >= 3.6 but there shouldn't be any breaking changes from Flask 1.1.2 => 1.1.3, so the classifiers [here](https://pypi.org/project/Flask/1.1.3/) are actually incorrect.",
      "issue_closed_at": "2021-05-14T01:20:00Z",
      "base_commit": "c04b0de558fe8e1ccb8edb4525d40e725ae9a24d",
      "changes": [
        {
          "file": "src/flask/__init__.py",
          "type": "line",
          "name": "line 57",
          "code": "from .templating import render_template\nfrom .templating import render_template_string\n\n__version__ = \"1.1.3\""
        },
        {
          "file": "src/flask/helpers.py",
          "type": "function",
          "name": "static_folder",
          "class_name": "_PackageBoundObject",
          "code": "def static_folder(self, value):\n        if value is not None:\n            value = os.fspath(value).rstrip(r\"\\/\")\n        self._static_folder = value"
        }
      ]
    },
    {
      "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)"
        }
      ]
    }
  ]
}