{
  "Selected_candidate": {
    "pr_number": 2963,
    "pr_title": "Fix #2961：ignore 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"
      }
    ]
  },
  "Justification": "Candidate C is the most helpful as it addresses a problem directly related to how the Flask application handles input for the app variable. The CURRENT bug report involves the handling of file modes in Flask configuration, which is closely aligned with the functionality of `flask.Config.from_file()`. Both reports involve the file handling processes within the Flask CLI, and the fix proposed in Candidate C for the `NoAppException` can provide insights into modifying file access patterns in the CURRENT bug. Additionally, the error message and exceptions discussed in Candidate C may share common ground in terms of debugging strategies, making it significantly useful for the developer resolving the CURRENT issue.",
  "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"
}