{
  "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 C is particularly useful because it addresses configuration issues in Pylint, which is directly relevant to the CURRENT bug involving message templates. The error message in Candidate C originates from incorrect TOML configurations, similar to how the CURRENT bug report involves a misuse of message formatting in a command. Given both cases involve the handling of configuration files and the impact on Pylint's operation, insights gained from the fix in Candidate C can guide debugging the CURRENT issue. Additionally, both involve the structural handling of configuration inputs, making it easier to adapt solutions from Candidate C to solve the CURRENT bug.",
  "instance_id": "pylint-dev__pylint-7993",
  "repo": "pylint-dev/pylint",
  "created_at": "2022-12-27T18:20:50Z",
  "problem_statement": "Using custom braces in message template does not work\n### Bug description\n\nHave any list of errors:\r\n\r\nOn pylint 1.7 w/ python3.6 - I am able to use this as my message template\r\n```\r\n$ pylint test.py --msg-template='{{ \"Category\": \"{category}\" }}'\r\nNo config file found, using default configuration\r\n************* Module [redacted].test\r\n{ \"Category\": \"convention\" }\r\n{ \"Category\": \"error\" }\r\n{ \"Category\": \"error\" }\r\n{ \"Category\": \"convention\" }\r\n{ \"Category\": \"convention\" }\r\n{ \"Category\": \"convention\" }\r\n{ \"Category\": \"error\" }\r\n```\r\n\r\nHowever, on Python3.9 with Pylint 2.12.2, I get the following:\r\n```\r\n$ pylint test.py --msg-template='{{ \"Category\": \"{category}\" }}'\r\n[redacted]/site-packages/pylint/reporters/text.py:206: UserWarning: Don't recognize the argument '{ \"Category\"' in the --msg-template. Are you sure it is supported on the current version of pylint?\r\n  warnings.warn(\r\n************* Module [redacted].test\r\n\" }\r\n\" }\r\n\" }\r\n\" }\r\n\" }\r\n\" }\r\n```\r\n\r\nIs this intentional or a bug?\n\n### Configuration\n\n_No response_\n\n### Command used\n\n```shell\npylint test.py --msg-template='{{ \"Category\": \"{category}\" }}'\n```\n\n\n### Pylint output\n\n```shell\n[redacted]/site-packages/pylint/reporters/text.py:206: UserWarning: Don't recognize the argument '{ \"Category\"' in the --msg-template. Are you sure it is supported on the current version of pylint?\r\n  warnings.warn(\r\n************* Module [redacted].test\r\n\" }\r\n\" }\r\n\" }\r\n\" }\r\n\" }\r\n\" }\n```\n\n\n### Expected behavior\n\nExpect the dictionary to print out with `\"Category\"` as the key.\n\n### Pylint version\n\n```shell\nAffected Version:\r\npylint 2.12.2\r\nastroid 2.9.2\r\nPython 3.9.9+ (heads/3.9-dirty:a2295a4, Dec 21 2021, 22:32:52) \r\n[GCC 4.8.5 20150623 (Red Hat 4.8.5-44)]\r\n\r\n\r\nPreviously working version:\r\nNo config file found, using default configuration\r\npylint 1.7.4, \r\nastroid 1.6.6\r\nPython 3.6.8 (default, Nov 16 2020, 16:55:22) \r\n[GCC 4.8.5 20150623 (Red Hat 4.8.5-44)]\n```\n\n\n### OS / Environment\n\n_No response_\n\n### Additional dependencies\n\n_No response_\n",
  "patch": "diff --git a/pylint/reporters/text.py b/pylint/reporters/text.py\n--- a/pylint/reporters/text.py\n+++ b/pylint/reporters/text.py\n@@ -175,7 +175,7 @@ def on_set_current_module(self, module: str, filepath: str | None) -> None:\n         self._template = template\n \n         # Check to see if all parameters in the template are attributes of the Message\n-        arguments = re.findall(r\"\\{(.+?)(:.*)?\\}\", template)\n+        arguments = re.findall(r\"\\{(\\w+?)(:.*)?\\}\", template)\n         for argument in arguments:\n             if argument[0] not in MESSAGE_FIELDS:\n                 warnings.warn(\n"
}