{
    "Selected_candidate": {
        "pr_number": 6944,
        "pr_title": "Mix incorrect parsing of multi-line options in ``ini`` files",
        "pr_body": "- [x] Write a good description on what the PR does.\r\n- [x] Add an entry to the change log describing the change in\r\n  `doc/whatsnew/2/2.15/index.rst` (or ``doc/whatsnew/2/2.14/full.rst``\r\n   if the change needs backporting in 2.14). If necessary you can write\r\n   details or offer examples on how the new change is supposed to work.\r\n- [x] If you used multiple emails or multiple names when contributing, add your mails\r\n      and preferred name in ``script/.contributors_aliases.json``\r\n\r\n## Type of Changes\r\n\r\n|     | Type                   |\r\n| --- | ---------------------- |\r\n| ✓   | :bug: Bug fix          |\r\n\r\n## Description\r\n\r\nCloses #6888.\r\n\r\nWe could also use an `if` statement to check whether the option is `init-hook` and and then not strip the newlines. But that introduce additional overhead, while this makes pylint faster. The fact that the rest of the test suite passes makes me think that this will be fine. We strip most options of whitespaces in other places anyway.\r\n\r\nIf this regresses after `2.14.2` we can simply add the `if` statement as desired.",
        "issue_id": 6888,
        "issue_title": "init-hook no longer supports multi-line hook",
        "issue_body": "### Bug description\n\nUp to version 2.13.9 multi-line init-hook configurations like the following were supported:\r\n\r\n```\r\ninit-hook=\r\n    try: import pylint_venv\r\n    except ImportError: pass\r\n    else: pylint_venv.inithook()\r\n```\r\n\r\nFrom version 2.14.0 on this no longer works. The error looks like the lines are joined into a single line:\r\n\r\n```\r\n❯ pylint test.py\r\nTraceback (most recent call last):\r\n  File \"/Users/jgosmann/Library/Python/3.8/bin/pylint\", line 8, in <module>\r\n    sys.exit(run_pylint())\r\n  File \"/Users/jgosmann/Library/Python/3.8/lib/python/site-packages/pylint/__init__.py\", line 25, in run_pylint\r\n    PylintRun(argv or sys.argv[1:])\r\n  File \"/Users/jgosmann/Library/Python/3.8/lib/python/site-packages/pylint/lint/run.py\", line 151, in __init__\r\n    args = _config_initialization(\r\n  File \"/Users/jgosmann/Library/Python/3.8/lib/python/site-packages/pylint/config/config_initialization.py\", line 48, in _config_initialization\r\n    exec(utils._unquote(config_data[\"init-hook\"]))  # pylint: disable=exec-used\r\n  File \"<string>\", line 1\r\n    try: import pylint_venvexcept ImportError: passelse: pylint_venv.inithook()\r\n                                  ^\r\nSyntaxError: invalid syntax\r\n```\n\n### Configuration\n\n```ini\ninit-hook=\r\n    try: import pylint_venv\r\n    except ImportError: pass\r\n    else: pylint_venv.inithook()\r\n```\n```\n\n\n### Command used\n\n```shell\npylint test.py\n```\n\n\n### Pylint output\n\n```shell\nTraceback (most recent call last):\r\n  File \"/Users/jgosmann/Library/Python/3.8/bin/pylint\", line 8, in <module>\r\n    sys.exit(run_pylint())\r\n  File \"/Users/jgosmann/Library/Python/3.8/lib/python/site-packages/pylint/__init__.py\", line 25, in run_pylint\r\n    PylintRun(argv or sys.argv[1:])\r\n  File \"/Users/jgosmann/Library/Python/3.8/lib/python/site-packages/pylint/lint/run.py\", line 151, in __init__\r\n    args = _config_initialization(\r\n  File \"/Users/jgosmann/Library/Python/3.8/lib/python/site-packages/pylint/config/config_initialization.py\", line 48, in _config_initialization\r\n    exec(utils._unquote(config_data[\"init-hook\"]))  # pylint: disable=exec-used\r\n  File \"<string>\", line 1\r\n    try: import pylint_venvexcept ImportError: passelse: pylint_venv.inithook()\r\n                                  ^\r\nSyntaxError: invalid syntax\n```\n\n\n### Expected behavior\n\nThe init-hook runs without a syntax error.\n\n### Pylint version\n\n```shell\npylint 2.14.0 (and 2.14.1)\r\nastroid 2.11.5\r\nPython 3.8.13 (default, May 10 2022, 11:26:18)\r\n[Clang 13.0.0 (clang-1300.0.29.3)]\n```\n\n\n### OS / Environment\n\n_No response_\n\n### Additional dependencies\n\npylint-venv==2.2.0",
        "issue_closed_at": "2022-06-13T16:24:32Z",
        "base_commit": "c21261eb4c0e2a0ec4c239b6f9af3f3a746b01cc",
        "changes": [
            {
                "file": "pylint/config/config_file_parser.py",
                "type": "function",
                "name": "_parse_ini_file",
                "class_name": "_ConfigurationFileParser",
                "code": "def _parse_ini_file(self, file_path: Path) -> tuple[dict[str, str], list[str]]:\n        \"\"\"Parse and handle errors of a ini configuration file.\"\"\"\n        parser = configparser.ConfigParser(inline_comment_prefixes=(\"#\", \";\"))\n\n        # Use this encoding in order to strip the BOM marker, if any.\n        with open(file_path, encoding=\"utf_8_sig\") as fp:\n            parser.read_file(fp)\n\n        config_content: dict[str, str] = {}\n        options: list[str] = []\n        for section in parser.sections():\n            if self._ini_file_with_sections(str(file_path)) and not section.startswith(\n                \"pylint\"\n            ):\n                if section.lower() == \"master\":\n                    # TODO: 3.0: Remove deprecated handling of master, only allow 'pylint.' sections\n                    warnings.warn(\n                        \"The use of 'MASTER' or 'master' as configuration section for pylint \"\n                        \"has been deprecated, as it's bad practice to not start sections titles with the \"\n                        \"tool name. Please use 'pylint.main' instead.\",\n                        UserWarning,\n                    )\n                else:\n                    continue\n            for opt, value in parser[section].items():\n                value = value.replace(\"\\n\", \"\")\n                config_content[opt] = value\n                options += [f\"--{opt}\", value]\n        return config_content, options"
            },
            {
                "file": "pylint/config/config_initialization.py",
                "type": "function",
                "name": "_config_initialization",
                "class_name": null,
                "code": "def _config_initialization(\n    linter: PyLinter,\n    args_list: list[str],\n    reporter: reporters.BaseReporter | reporters.MultiReporter | None = None,\n    config_file: None | str | Path = None,\n    verbose_mode: bool = False,\n) -> list[str]:\n    \"\"\"Parse all available options, read config files and command line arguments and\n    set options accordingly.\n    \"\"\"\n    config_file = Path(config_file) if config_file else None\n\n    # Set the current module to the configuration file\n    # to allow raising messages on the configuration file.\n    linter.set_current_module(str(config_file) if config_file else \"\")\n\n    # Read the configuration file\n    config_file_parser = _ConfigurationFileParser(verbose_mode, linter)\n    try:\n        config_data, config_args = config_file_parser.parse_config_file(\n            file_path=config_file\n        )\n    except OSError as ex:\n        print(ex, file=sys.stderr)\n        sys.exit(32)\n\n    # Run init hook, if present, before loading plugins\n    if \"init-hook\" in config_data:\n        exec(utils._unquote(config_data[\"init-hook\"]))  # pylint: disable=exec-used\n\n    # Load plugins if specified in the config file\n    if \"load-plugins\" in config_data:\n        linter.load_plugin_modules(utils._splitstrip(config_data[\"load-plugins\"]))\n\n    unrecognized_options_message = None\n    # First we parse any options from a configuration file\n    try:\n        linter._parse_configuration_file(config_args)\n    except _UnrecognizedOptionError as exc:\n        unrecognized_options_message = \", \".join(exc.options)\n\n    # Then, if a custom reporter is provided as argument, it may be overridden\n    # by file parameters, so we re-set it here. We do this before command line\n    # parsing, so it's still overridable by command line options\n    if reporter:\n        linter.set_reporter(reporter)\n\n    # Set the current module to the command line\n    # to allow raising messages on it\n    linter.set_current_module(\"Command line\")\n\n    # Now we parse any options from the command line, so they can override\n    # the configuration file\n    parsed_args_list = linter._parse_command_line_configuration(args_list)\n\n    # Check if there are any options that we do not recognize\n    unrecognized_options: list[str] = []\n    for opt in parsed_args_list:\n        if opt.startswith(\"--\"):\n            unrecognized_options.append(opt[2:])\n        elif opt.startswith(\"-\"):\n            unrecognized_options.append(opt[1:])\n    if unrecognized_options:\n        msg = \", \".join(unrecognized_options)\n        linter._arg_parser.error(f\"Unrecognized option found: {msg}\")\n\n    # Now that config file and command line options have been loaded\n    # with all disables, it is safe to emit messages\n    if unrecognized_options_message is not None:\n        linter.set_current_module(str(config_file) if config_file else \"\")\n        linter.add_message(\n            \"unrecognized-option\", args=unrecognized_options_message, line=0\n        )\n\n    linter._emit_stashed_messages()\n\n    # Set the current module to configuration as we don't know where\n    # the --load-plugins key is coming from\n    linter.set_current_module(\"Command line or configuration file\")\n\n    # We have loaded configuration from config file and command line. Now, we can\n    # load plugin specific configuration.\n    linter.load_plugin_configuration()\n\n    # Now that plugins are loaded, get list of all fail_on messages, and enable them\n    linter.enable_fail_on_messages()\n\n    linter._parse_error_mode()\n\n    # Link the base Namespace object on the current directory\n    linter._directory_namespaces[Path(\".\").resolve()] = (linter.config, {})\n\n    # parsed_args_list should now only be a list of files/directories to lint.\n    # All other options have been removed from the list.\n    return parsed_args_list"
            }
        ]
    },
    "Justification": "Candidate C is the most relevant because it addresses a bug related to the parsing of configuration options in `.ini` files, similar to the `ignore-paths` configuration issue in the CURRENT bug report. Both bugs involve the handling of configuration parsing in pylint, which suggests that insights gained from how multi-line options were mismanaged could help in understanding how the `ignore-paths` setting might be overlooked when using the `--recursive=y` command. Furthermore, since Candidate C's fix revolves around correcting parsing logic, it could lead to improvements or workarounds relevant to the CURRENT bug's context."
}