{
  "Selected_candidate": {
    "pr_number": 4720,
    "pr_title": "Pylint fix for invalid TOML config",
    "pr_body": "<!--\r\n\r\nThank you for submitting a PR to pylint!\r\n\r\nTo ease the process of reviewing your PR, do make sure to complete the following boxes.\r\n\r\nYou can also read more about contributing to pylint in this document:\r\nhttps://github.com/PyCQA/pylint/blob/main/doc/development_guide/contribute.rst#repository\r\n-->\r\n\r\n## Steps\r\n\r\n- [x] Add yourself to CONTRIBUTORS if you are a new contributor.\r\n- [ ] Add a ChangeLog entry describing what your PR does.\r\n- [ ] If it's a new feature or an important bug fix, add a What's New entry in\r\n      `doc/whatsnew/<current release.rst>`.\r\n- [x] Write a good description on what the PR does.\r\n\r\n## Description\r\nThis is a fix for handling bad or invalid toml configuration in pyproject.toml\r\n\r\n## Type of Changes\r\n\r\n<!-- Leave the corresponding lines for the applicable type of change: -->\r\n\r\n|     | Type                   |\r\n| --- | ---------------------- |\r\n| ✓   | :bug: Bug fix          |\r\n| ✓   | :sparkles: New feature |\r\n| ✓   | :hammer: Refactoring   |\r\n| ✓   | :scroll: Docs          |\r\n\r\n## Related Issue\r\n\r\n<!--\r\nIf this PR fixes a particular issue, use the following to automatically close that issue\r\nonce this PR gets merged:\r\n\r\nCloses #4580\r\n-->\r\nCloses #4580",
    "issue_id": 4580,
    "issue_title": "Pylint Crash on invalid TOML config",
    "issue_body": "### Steps to reproduce\r\n\r\nIt was not immediately clear to me that the main table should be `[tool.pylint.master]` and not `[tool.pylint]` (I searched in the docs but did not find an example).\r\n\r\nGiven any python file and a config file called `repro.toml` (I initially encountered this on my `pyproject.toml`, but for the sake of a simple example):\r\n\r\n```toml\r\n[tool.pylint]\r\nload-plugins = []\r\n```\r\n\r\n### Current behavior\r\n\r\n```\r\npylint --rcfile=repro.toml repro.py       \r\nTraceback (most recent call last):\r\n  File \"C:\\Program Files\\Python310\\lib\\runpy.py\", line 196, in _run_module_as_main\r\n    return _run_code(code, main_globals, None,\r\n  File \"C:\\Program Files\\Python310\\lib\\runpy.py\", line 86, in _run_code\r\n    exec(code, run_globals)\r\n  File \"D:\\...\\venv\\Scripts\\pylint.exe\\__main__.py\", line 7, in <module>\r\n  File \"d:\\...\\venv\\lib\\site-packages\\pylint\\__init__.py\", line 24, in run_pylint\r\n    PylintRun(sys.argv[1:])\r\n  File \"d:\\...\\venv\\lib\\site-packages\\pylint\\lint\\run.py\", line 316, in __init__\r\n    linter.read_config_file(verbose=self.verbose)\r\n  File \"d:\\...\\venv\\lib\\site-packages\\pylint\\config\\option_manager_mixin.py\", line 281, in read_config_file\r\n    for option, value in values.items():\r\nAttributeError: 'str' object has no attribute 'items'\r\n```\r\n\r\n### Expected behavior\r\n\r\nIt should not crash.\r\n\r\n### pylint --version output\r\n\r\nI just upgraded to prerelease versions as suggested in this template.\r\n\r\n```\r\npylint 3.0.0a3\r\nastroid 2.5.8\r\nPython 3.10.0b1 (tags/v3.10.0b1:ba42175, May  3 2021, 20:22:30) [MSC v.1928 64 bit (AMD64)]\r\n```\r\n\r\n### Additional crashes for invalid data\r\n\r\nAdditional crashes are possible for \"mapping\" configs where I thought a TOML table might be used rather than a list of strings with a built in separator as is used in `.pylintrc` (again, I could not find examples).\r\n\r\nGiven config:\r\n\r\n```toml\r\n[tool.pylint.imports]\r\npreferred-modules = { \"a\"=\"b\" }\r\n```\r\n\r\nOr config:\r\n\r\n```toml\r\n[tool.pylint.basic]\r\nname-group = { \"a\"=\"b\" }\r\n```\r\n\r\nResults in (same traceback for both configs):\r\n\r\n```\r\npylint --rcfile=repro.toml repro.py\r\nTraceback (most recent call last):\r\n  File \"C:\\Program Files\\Python310\\lib\\runpy.py\", line 196, in _run_module_as_main\r\n    return _run_code(code, main_globals, None,\r\n  File \"C:\\Program Files\\Python310\\lib\\runpy.py\", line 86, in _run_code\r\n    exec(code, run_globals)\r\n  File \"D:\\...\\venv\\Scripts\\pylint.exe\\__main__.py\", line 7, in <module>\r\n  File \"d:\\...\\venv\\lib\\site-packages\\pylint\\__init__.py\", line 24, in run_pylint\r\n    PylintRun(sys.argv[1:])\r\n  File \"d:\\...\\venv\\lib\\site-packages\\pylint\\lint\\run.py\", line 333, in __init__\r\n    linter.load_config_file()\r\n  File \"d:\\...\\venv\\lib\\site-packages\\pylint\\config\\option_manager_mixin.py\", line 313, in load_config_file\r\n    for option, value in parser.items(section):\r\n  File \"C:\\Program Files\\Python310\\lib\\configparser.py\", line 860, in items\r\n    return [(option, value_getter(option)) for option in orig_keys]\r\n  File \"C:\\Program Files\\Python310\\lib\\configparser.py\", line 860, in <listcomp>\r\n    return [(option, value_getter(option)) for option in orig_keys]\r\n  File \"C:\\Program Files\\Python310\\lib\\configparser.py\", line 856, in <lambda>\r\n    value_getter = lambda option: self._interpolation.before_get(self,\r\n  File \"C:\\Program Files\\Python310\\lib\\configparser.py\", line 395, in before_get\r\n    self._interpolate_some(parser, option, L, value, section, defaults, 1)\r\n  File \"C:\\Program Files\\Python310\\lib\\configparser.py\", line 412, in _interpolate_some\r\n    p = rest.find(\"%\")\r\nAttributeError: 'DynamicInlineTableDict' object has no attribute 'find'\r\n```\r\n\r\nRelated issues: #4518 #4204 ",
    "issue_closed_at": "2021-11-13T12:06:02Z",
    "base_commit": "7976857b94146f6ad81ade63797905cd662238c2",
    "changes": [
      {
        "file": "pylint/config/option_manager_mixin.py",
        "type": "function",
        "name": "read_config_file",
        "class_name": "OptionsManagerMixIn",
        "code": "def read_config_file(self, config_file=None, verbose=None):\n        \"\"\"Read the configuration file but do not load it (i.e. dispatching\n        values to each options provider)\n        \"\"\"\n        for help_level in range(1, self._maxlevel + 1):\n            opt = \"-\".join([\"long\"] * help_level) + \"-help\"\n            if opt in self._all_options:\n                break  # already processed\n            help_function = functools.partial(self.helpfunc, level=help_level)\n            help_msg = f\"{' '.join(['more'] * help_level)} verbose help.\"\n            optdict = {\n                \"action\": \"callback\",\n                \"callback\": help_function,\n                \"help\": help_msg,\n            }\n            provider = self.options_providers[0]\n            self.add_optik_option(provider, self.cmdline_parser, opt, optdict)\n            provider.options += ((opt, optdict),)\n\n        if config_file is None:\n            config_file = self.config_file\n        if config_file is not None:\n            config_file = os.path.expandvars(os.path.expanduser(config_file))\n            if not os.path.exists(config_file):\n                raise OSError(f\"The config file {config_file} doesn't exist!\")\n\n        use_config_file = config_file and os.path.exists(config_file)\n        if use_config_file:\n            self.set_current_module(config_file)\n            parser = self.cfgfile_parser\n            if config_file.endswith(\".toml\"):\n                self._parse_toml(config_file, parser)\n            else:\n                # Use this encoding in order to strip the BOM marker, if any.\n                with open(config_file, encoding=\"utf_8_sig\") as fp:\n                    parser.read_file(fp)\n                # normalize sections'title\n                for sect, values in list(parser._sections.items()):\n                    if sect.startswith(\"pylint.\"):\n                        sect = sect[len(\"pylint.\") :]\n                    if not sect.isupper() and values:\n                        parser._sections[sect.upper()] = values\n        if not verbose:\n            return\n        if use_config_file:\n            msg = f\"Using config file {os.path.abspath(config_file)}\"\n        else:\n            msg = \"No config file found, using default configuration\"\n        print(msg, file=sys.stderr)"
      },
      {
        "file": "pylint/config/option_manager_mixin.py",
        "type": "function",
        "name": "_parse_toml",
        "class_name": "OptionsManagerMixIn",
        "code": "def _parse_toml(\n        config_file: Union[Path, str], parser: configparser.ConfigParser\n    ) -> None:\n        \"\"\"Parse and handle errors of a toml configuration file.\"\"\"\n        with open(config_file, encoding=\"utf-8\") as fp:\n            content = toml.load(fp)\n        try:\n            sections_values = content[\"tool\"][\"pylint\"]\n        except KeyError:\n            return\n        for section, values in sections_values.items():\n            # TOML has rich types, convert values to\n            # strings as ConfigParser expects.\n            for option, value in values.items():\n                if isinstance(value, bool):\n                    values[option] = \"yes\" if value else \"no\"\n                elif isinstance(value, (int, float)):\n                    values[option] = str(value)\n                elif isinstance(value, list):\n                    values[option] = \",\".join(value)\n            parser._sections[section.upper()] = values"
      },
      {
        "file": "pylint/lint/pylinter.py",
        "type": "function",
        "name": "_load_reporter_by_class",
        "class_name": null,
        "code": "def _load_reporter_by_class(reporter_class: str) -> type:\n    qname = reporter_class\n    module_part = astroid.modutils.get_module_part(qname)\n    module = astroid.modutils.load_module_from_name(module_part)\n    class_name = qname.split(\".\")[-1]\n    return getattr(module, class_name)"
      }
    ]
  },
  "Justification": "Candidate A is the most helpful choice because it involves a crash in Pylint due to an invalid configuration, which parallels the CURRENT bug where an unrecognized command line option leads to an error message. Both deal with misconfigurations – one in TOML and the other in command line options – and both evoke a traceback from Pylint indicating a misunderstanding of user input. Insights into how Pylint parses options/configurations and the handling of invalid input would directly contribute to understanding and rectifying the CURRENT issue.",
  "instance_id": "pylint-dev__pylint-6506",
  "repo": "pylint-dev/pylint",
  "created_at": "2022-05-05T13:01:41Z",
  "problem_statement": "Traceback printed for unrecognized option\n### Bug description\n\nA traceback is printed when an unrecognized option is passed to pylint.\n\n### Configuration\n\n_No response_\n\n### Command used\n\n```shell\npylint -Q\n```\n\n\n### Pylint output\n\n```shell\n************* Module Command line\r\nCommand line:1:0: E0015: Unrecognized option found: Q (unrecognized-option)\r\nTraceback (most recent call last):\r\n  File \"/Users/markbyrne/venv310/bin/pylint\", line 33, in <module>\r\n    sys.exit(load_entry_point('pylint', 'console_scripts', 'pylint')())\r\n  File \"/Users/markbyrne/programming/pylint/pylint/__init__.py\", line 24, in run_pylint\r\n    PylintRun(argv or sys.argv[1:])\r\n  File \"/Users/markbyrne/programming/pylint/pylint/lint/run.py\", line 135, in __init__\r\n    args = _config_initialization(\r\n  File \"/Users/markbyrne/programming/pylint/pylint/config/config_initialization.py\", line 85, in _config_initialization\r\n    raise _UnrecognizedOptionError(options=unrecognized_options)\r\npylint.config.exceptions._UnrecognizedOptionError\n```\n\n\n### Expected behavior\n\nThe top part of the current output is handy:\r\n`Command line:1:0: E0015: Unrecognized option found: Q (unrecognized-option)`\r\n\r\nThe traceback I don't think is expected & not user-friendly.\r\nA usage tip, for example:\r\n```python\r\nmypy -Q\r\nusage: mypy [-h] [-v] [-V] [more options; see below]\r\n            [-m MODULE] [-p PACKAGE] [-c PROGRAM_TEXT] [files ...]\r\nmypy: error: unrecognized arguments: -Q\r\n```\n\n### Pylint version\n\n```shell\npylint 2.14.0-dev0\r\nastroid 2.11.3\r\nPython 3.10.0b2 (v3.10.0b2:317314165a, May 31 2021, 10:02:22) [Clang 12.0.5 (clang-1205.0.22.9)]\n```\n\n\n### OS / Environment\n\n_No response_\n\n### Additional dependencies\n\n_No response_\n",
  "patch": "diff --git a/pylint/config/config_initialization.py b/pylint/config/config_initialization.py\n--- a/pylint/config/config_initialization.py\n+++ b/pylint/config/config_initialization.py\n@@ -81,8 +81,7 @@ def _config_initialization(\n             unrecognized_options.append(opt[1:])\n     if unrecognized_options:\n         msg = \", \".join(unrecognized_options)\n-        linter.add_message(\"unrecognized-option\", line=0, args=msg)\n-        raise _UnrecognizedOptionError(options=unrecognized_options)\n+        linter._arg_parser.error(f\"Unrecognized option found: {msg}\")\n \n     # Set the current module to configuration as we don't know where\n     # the --load-plugins key is coming from\n"
}