{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 6351,
      "pr_title": "Add exception handling for broken config files",
      "pr_body": "Extends the error handling of bad config files, to catch even more errors.\r\n\r\n## Type of Changes\r\n\r\n|     | Type                   |\r\n| --- | ---------------------- |\r\n| \u2713   | :bug: Bug fix          |\r\n\r\n## Description\r\n\r\nCloses #5667\r\n",
      "issue_id": 5667,
      "issue_title": "Add a flag to disable config parsing",
      "issue_body": "### Current problem\r\n\r\nHere's a sample broken `.pylintrc`, placed in the root folder:\r\n\r\n```ini\r\n[pylint]\r\ntest = '%A'\r\n```\r\n\r\nThis is what happens when I try to use pylint now:\r\n```console\r\n$ pylint myfile.py\r\nTraceback (most recent call last):\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv3/bin/pylint\", line 8, in <module>\r\n    sys.exit(run_pylint())\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv3/lib/python3.10/site-packages/pylint/__init__.py\", line 24, in run_pylint\r\n    PylintRun(sys.argv[1:])\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv3/lib/python3.10/site-packages/pylint/lint/run.py\", line 324, in __init__\r\n    linter.load_config_file()\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv3/lib/python3.10/site-packages/pylint/config/option_manager_mixin.py\", line 319, in load_config_file\r\n    for option, value in parser.items(section):\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 860, in items\r\n    return [(option, value_getter(option)) for option in orig_keys]\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 860, in <listcomp>\r\n    return [(option, value_getter(option)) for option in orig_keys]\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 856, in <lambda>\r\n    value_getter = lambda option: self._interpolation.before_get(self,\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 395, in before_get\r\n    self._interpolate_some(parser, option, L, value, section, defaults, 1)\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 442, in _interpolate_some\r\n    raise InterpolationSyntaxError(\r\nconfigparser.InterpolationSyntaxError: '%' must be followed by '%' or '(', found: \"%A'\"\r\n\r\n$ pylint --help   \r\nTraceback (most recent call last):\r\n[...]\r\nconfigparser.InterpolationSyntaxError: '%' must be followed by '%' or '(', found: \"%A'\"\r\n\r\n$ pylint       \r\nTraceback (most recent call last):\r\n[...]\r\nconfigparser.InterpolationSyntaxError: '%' must be followed by '%' or '(', found: \"%A'\"\r\n```\r\n\r\nIs it possible to add a flag that disables config parsing entirely, so that the pylint isn't completely broken when this happens? `--ignore-config-file` for example?\r\n\r\nP.S. this is a feature I want, and I'd be willing to contribute it.",
      "issue_closed_at": "2022-04-15T20:15:52Z",
      "base_commit": "98bb5bf8ebf15fbce960533e5983a71351a1fe3f",
      "changes": [
        {
          "file": "pylint/config/config_file_parser.py",
          "type": "function",
          "name": "parse_config_file",
          "class_name": "_ConfigurationFileParser",
          "code": "def parse_config_file(\n        self, file_path: Path | None\n    ) -> tuple[dict[str, str], list[str]]:\n        \"\"\"Parse a config file and return str-str pairs.\"\"\"\n        if file_path is None:\n            if self.verbose_mode:\n                print(\n                    \"No config file found, using default configuration\", file=sys.stderr\n                )\n            return {}, []\n\n        file_path = Path(os.path.expandvars(file_path)).expanduser()\n        if not file_path.exists():\n            raise OSError(f\"The config file {file_path} doesn't exist!\")\n\n        if self.verbose_mode:\n            print(f\"Using config file {file_path}\", file=sys.stderr)\n\n        if file_path.suffix == \".toml\":\n            return self._parse_toml_file(file_path)\n        return self._parse_ini_file(file_path)"
        }
      ]
    },
    {
      "pr_number": 6664,
      "pr_title": "Fix ``unsupported-binary-operation`` on classes that overload or",
      "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\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- [ ] Write a good description on what the PR does.\r\n- [ ] 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\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| \u2713   | :bug: Bug fix          |\r\n\r\n## Description\r\n\r\nWe have a specific checker looking for people using `|` with two types, apparently on the assumption that anyone doing that is mistakenly trying to use union syntax where it's not supported. This seems to overlook the fact that it is possible for class definitions to use a metaclass that implements `__or__` or `__ror__` as appropriate, which is the case in the permission classes for Django REST Framework, for example.\r\n\r\nCloses #4951\r\n",
      "issue_id": 4951,
      "issue_title": "DjangoRestFramework: E1131: unsupported operand type(s) for | (unsupported-binary-operation)",
      "issue_body": "### Bug description\r\n\r\n```python\r\n# pylint: disable=missing-docstring\r\n\r\nfrom rest_framework.permissions import BasePermission, IsAuthenticated, SAFE_METHODS\r\n\r\nclass ReadOnly(BasePermission):\r\n    def has_permission(self, request, view):\r\n        return request.method in SAFE_METHODS\r\n\r\npermission_classes = [IsAuthenticated|ReadOnly]\r\n```\r\n\r\n\r\n### Configuration\r\n\r\n_No response_\r\n\r\n### Command used\r\n\r\n```shell\r\npylint binary.py\r\n```\r\n\r\n\r\n### Pylint output\r\n\r\n```shell\r\n************* Module binary\r\nbinary.py:9:22: E1131: unsupported operand type(s) for | (unsupported-binary-operation)\r\n\r\n-----------------------------------\r\nYour code has been rated at 0.00/10\r\n```\r\n\r\n\r\n### Expected behavior\r\n\r\nFor pylint to not throw this error. \r\n\r\nDjango Rest Framework supports binary operands for permissions [1]\r\nThis example code is taken from the documentation [2]\r\n[1] https://github.com/encode/django-rest-framework/blob/master/rest_framework/permissions.py\r\n[2] https://www.django-rest-framework.org/api-guide/permissions/\r\n\r\n### Pylint version\r\n\r\n```shell\r\npylint 2.10.2\r\nastroid 2.7.3\r\nPython 3.9.4 (default, Apr 10 2021,\r\n```\r\n\r\n\r\n### OS / Environment\r\n\r\nDebian GNU/Linux 10 (buster)\r\n\r\n### Additional dependencies\r\n\r\ndjangorestframework==3.12.4\r\n\r\n\r\n*edit*: Added link [2] to django rest frameworks documentation on permissions",
      "issue_closed_at": "2022-06-23T05:46:57Z",
      "base_commit": "a4b9ef560166f35f9ecca95d0ec4c82b70fc031f",
      "changes": [
        {
          "file": "pylint/checkers/typecheck.py",
          "type": "line",
          "name": "line 20",
          "code": "from re import Pattern\nfrom typing import TYPE_CHECKING, Any, Union\n\nimport astroid.exceptions\nfrom astroid import bases, nodes\n\nfrom pylint.checkers import BaseChecker, utils"
        },
        {
          "file": "pylint/checkers/typecheck.py",
          "type": "function",
          "name": "_detect_unsupported_alternative_union_syntax",
          "class_name": "TypeChecker",
          "code": "def _detect_unsupported_alternative_union_syntax(self, node: nodes.BinOp) -> None:\n        \"\"\"Detect if unsupported alternative Union syntax (PEP 604) was used.\"\"\"\n        if self._py310_plus:  # 310+ supports the new syntax\n            return\n\n        if isinstance(\n            node.parent, TYPE_ANNOTATION_NODES_TYPES\n        ) and not is_postponed_evaluation_enabled(node):\n            # Use in type annotations only allowed if\n            # postponed evaluation is enabled.\n            self._check_unsupported_alternative_union_syntax(node)\n\n        if isinstance(\n            node.parent,\n            (\n                nodes.Assign,\n                nodes.Call,\n                nodes.Keyword,\n                nodes.Dict,\n                nodes.Tuple,\n                nodes.Set,\n                nodes.List,\n                nodes.BinOp,\n            ),\n        ):\n            # Check other contexts the syntax might appear, but are invalid.\n            # Make sure to filter context if postponed evaluation is enabled\n            # and parent is allowed node type.\n            allowed_nested_syntax = False\n            if is_postponed_evaluation_enabled(node):\n                parent_node = node.parent\n                while True:\n                    if isinstance(parent_node, TYPE_ANNOTATION_NODES_TYPES):\n                        allowed_nested_syntax = True\n                        break\n                    parent_node = parent_node.parent\n                    if isinstance(parent_node, nodes.Module):\n                        break\n            if not allowed_nested_syntax:\n                self._check_unsupported_alternative_union_syntax(node)"
        }
      ]
    },
    {
      "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| \u2713   | :bug: Bug fix          |\r\n| \u2713   | :sparkles: New feature |\r\n| \u2713   | :hammer: Refactoring   |\r\n| \u2713   | :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)"
        }
      ]
    },
    {
      "pr_number": 4614,
      "pr_title": "Fix false positive useless type annotation for pep484 typing",
      "pr_body": "## Description\r\n\r\nFix false positive ``useless-type-doc`` on ignored argument using ``pylint.extensions.docparams``\r\n  when a function was typed using pep484 but not inside the docstring.\r\n\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| \u2713   | :bug: Bug fix          |\r\n\r\n## Related Issue\r\n\r\n  Closes #4117\r\n  Closes #4593",
      "issue_id": 4593,
      "issue_title": "useless-type-doc issued for undocumented parameters that have PEP484 type annotations",
      "issue_body": "<!--\r\n  Hi there! Thank you for discovering and submitting an issue.\r\n\r\n  Before you submit this, make sure that the issue doesn't already exist\r\n  or if it is not closed.\r\n\r\n  Is your issue fixed on the preview release?:\r\n    pip install pylint astroid --pre -U\r\n-->\r\n\r\nThis likely relates to (and solves) issue #4117.\r\n\r\n### Steps to reproduce\r\n\r\nCall pylint with the docparams extension enabled on the following code:\r\n\r\n```python\r\n'''demonstrate FP with useless-type-doc'''\r\n\r\ndef function(public_param: int, _some_private_param: bool = False) -> None:\r\n    '''does things\r\n\r\n    Args:\r\n        public_param: an ordinary parameter\r\n    '''\r\n    for _ in range(public_param):\r\n        ...\r\n    if _some_private_param:\r\n        ...\r\n    else:\r\n        ...\r\n```\r\n\r\n### Current behavior\r\n\r\n```\r\n$ python3 -m pylint --load-plugins=pylint.extensions.docparams --rcfile=/dev/null bug.py\r\n************* Module bug\r\nbug.py:4:0: W9020: \"_some_private_param\" useless ignored parameter type documentation (useless-type-doc)\r\n\r\n------------------------------------------------------------------\r\nYour code has been rated at 8.33/10 (previous run: 5.00/10, +3.33)\r\n```\r\n\r\n### Expected behavior\r\n\r\nPylint should issue no errors.\r\n\r\nIt would be reasonable for pylint to issue this error if the type information appeared in the docstring, however, pylint is actually responding to the type annotation (as in PEP484). The docparams module should only issue errors in response to docstrings (or lack thereof), and not PEP484 annotations.\r\n\r\n### pylint --version output\r\n\r\nResult of `pylint --version` output:\r\n\r\n```\r\npylint 2.8.1\r\nastroid 2.5.6\r\nPython 3.8.3 (default, Jul  6 2020, 09:12:34)\r\n[Clang 10.0.1 (clang-1001.0.46.4)]\r\n```\r\n",
      "issue_closed_at": "2021-06-25T12:34:49Z",
      "base_commit": "bebdf9b3aee66c5d8937ed2a8993db398995c6b3",
      "changes": [
        {
          "file": "pylint/extensions/_check_docs_utils.py",
          "type": "function",
          "name": "__init__",
          "class_name": "Docstring",
          "code": "def __init__(self, doc):\n        doc = doc or \"\"\n        self.doc = doc.expandtabs()"
        },
        {
          "file": "pylint/extensions/docparams.py",
          "type": "line",
          "name": "line 23",
          "code": "\"\"\"Pylint plugin for checking in Sphinx, Google, or Numpy style docstrings\n\"\"\"\nimport re\n\nimport astroid\n\nfrom pylint.checkers import BaseChecker\nfrom pylint.checkers import utils as checker_utils\nfrom pylint.extensions import _check_docs_utils as utils\nfrom pylint.interfaces import IAstroidChecker\nfrom pylint.utils import get_global_option\n"
        },
        {
          "file": "pylint/extensions/docparams.py",
          "type": "function",
          "name": "_compare_ignored_args",
          "class_name": "DocstringParameterChecker",
          "code": "def _compare_ignored_args(\n        self,\n        found_argument_names,\n        message_id,\n        ignored_argument_names,\n        warning_node,\n    ):\n        \"\"\"Compare the found argument names with the ignored ones and\n        generate a message if there are ignored arguments found.\n\n        :param found_argument_names: argument names found in the docstring\n        :type found_argument_names: set\n\n        :param message_id: pylint message id\n        :type message_id: str\n\n        :param ignored_argument_names: Expected argument names\n        :type ignored_argument_names: set\n\n        :param warning_node: The node to be analyzed\n        :type warning_node: :class:`astroid.scoped_nodes.Node`\n        \"\"\"\n        existing_ignored_argument_names = ignored_argument_names & found_argument_names\n\n        if existing_ignored_argument_names:\n            self.add_message(\n                message_id,\n                args=(\", \".join(sorted(existing_ignored_argument_names)),),\n                node=warning_node,\n            )"
        }
      ]
    },
    {
      "pr_number": 6181,
      "pr_title": "Only emit ``lru-cache-decorating-method`` when ``maxsize`` is ``None``",
      "pr_body": "- [x] Add yourself to CONTRIBUTORS if you are a new contributor.\r\n- [x] Add a ChangeLog entry describing what your PR does.\r\n- [x] 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## Type of Changes\r\n\r\n|     | Type                   |\r\n| --- | ---------------------- |\r\n| \u2713   | :bug: Bug fix          |\r\n\r\n## Description\r\n\r\nCloses #6180.",
      "issue_id": 6180,
      "issue_title": "pylint reports false positive error for lru_cache with maxsize omitted",
      "issue_body": "### Bug description\n\nStarting in `v2.13.0` pylint reports warning if you decorate a method with `lru_cache` but do not provide a value for `maxsize` (#5670) . It does this even if you omit a value for `maxsize` and just use `@lru_cache` or `@lru_cache()`.\r\n\r\nThis is incorrect, the `lru_cache` decorator [defaults maxsize to 128 if omitted](https://github.com/python/cpython/blob/main/Lib/functools.py#L479) so it should not warn about this. It should only warn if `maxsize` is explicitly set to `None` by doing `@lru_cache(maxsize=None)` as that is what can create memory leaks.\n\n### Configuration\n\n_No response_\n\n### Command used\n\n```shell\npylint supervisor/host/manager.py\n```\n\n\n### Pylint output\n\n```shell\n************* Module supervisor.host.manager\r\nsupervisor/host/manager.py:71:5: W1516: 'lru_cache' without 'maxsize' will keep all method args alive indefinitely, including 'self' (lru-cache-decorating-method)\n```\n\n\n### Expected behavior\n\nNo warning since we are using the default maxsize of 128.\n\n### Pylint version\n\n```shell\npylint 2.13.4\n```\n\n\n### OS / Environment\n\nmacOS 12.2.1\r\nPython 3.10.2\r\npip 22.0.4\n\n### Additional dependencies\n\n_No response_",
      "issue_closed_at": "2022-04-05T14:25:43Z",
      "base_commit": "7a9234f681b3c046b97e4371c84fadcc3ac7c721",
      "changes": [
        {
          "file": "pylint/checkers/stdlib.py",
          "type": "class",
          "name": "StdlibChecker",
          "code": "class StdlibChecker(DeprecatedMixin, BaseChecker):\n    __implements__ = (IAstroidChecker,)\n    name = \"stdlib\"\n\n    msgs = {\n        \"W1501\": (\n            '\"%s\" is not a valid mode for open.',\n            \"bad-open-mode\",\n            \"Python supports: r, w, a[, x] modes with b, +, \"\n            \"and U (only with r) options. \"\n            \"See https://docs.python.org/3/library/functions.html#open\",\n        ),\n        \"W1502\": (\n            \"Using datetime.time in a boolean context.\",\n            \"boolean-datetime\",\n            \"Using datetime.time in a boolean context can hide \"\n            \"subtle bugs when the time they represent matches \"\n            \"midnight UTC. This behaviour was fixed in Python 3.5. \"\n            \"See https://bugs.python.org/issue13936 for reference.\",\n            {\"maxversion\": (3, 5)},\n        ),\n        \"W1503\": (\n            \"Redundant use of %s with constant value %r\",\n            \"redundant-unittest-assert\",\n            \"The first argument of assertTrue and assertFalse is \"\n            \"a condition. If a constant is passed as parameter, that \"\n            \"condition will be always true. In this case a warning \"\n            \"should be emitted.\",\n        ),\n        \"W1505\": (\n            \"Using deprecated method %s()\",\n            \"deprecated-method\",\n            \"The method is marked as deprecated and will be removed in \"\n            \"a future version of Python. Consider looking for an \"\n            \"alternative in the documentation.\",\n        ),\n        \"W1506\": (\n            \"threading.Thread needs the target function\",\n            \"bad-thread-instantiation\",\n            \"The warning is emitted when a threading.Thread class \"\n            \"is instantiated without the target function being passed. \"\n            \"By default, the first parameter is the group param, not the target param.\",\n        ),\n        \"W1507\": (\n            \"Using copy.copy(os.environ). Use os.environ.copy() instead. \",\n            \"shallow-copy-environ\",\n            \"os.environ is not a dict object but proxy object, so \"\n            \"shallow copy has still effects on original object. \"\n            \"See https://bugs.python.org/issue15373 for reference.\",\n        ),\n        \"E1507\": (\n            \"%s does not support %s type argument\",\n            \"invalid-envvar-value\",\n            \"Env manipulation functions support only string type arguments. \"\n            \"See https://docs.python.org/3/library/os.html#os.getenv.\",\n        ),\n        \"W1508\": (\n            \"%s default type is %s. Expected str or None.\",\n            \"invalid-envvar-default\",\n            \"Env manipulation functions return None or str values. \"\n            \"Supplying anything different as a default may cause bugs. \"\n            \"See https://docs.python.org/3/library/os.html#os.getenv.\",\n        ),\n        \"W1509\": (\n            \"Using preexec_fn keyword which may be unsafe in the presence \"\n            \"of threads\",\n            \"subprocess-popen-preexec-fn\",\n            \"The preexec_fn parameter is not safe to use in the presence \"\n            \"of threads in your application. The child process could \"\n            \"deadlock before exec is called. If you must use it, keep it \"\n            \"trivial! Minimize the number of libraries you call into.\"\n            \"https://docs.python.org/3/library/subprocess.html#popen-constructor\",\n        ),\n        \"W1510\": (\n            \"Using subprocess.run without explicitly set `check` is not recommended.\",\n            \"subprocess-run-check\",\n            \"The check parameter should always be used with explicitly set \"\n            \"`check` keyword to make clear what the error-handling behavior is.\"\n            \"https://docs.python.org/3/library/subprocess.html#subprocess.run\",\n        ),\n        \"W1511\": (\n            \"Using deprecated argument %s of method %s()\",\n            \"deprecated-argument\",\n            \"The argument is marked as deprecated and will be removed in the future.\",\n        ),\n        \"W1512\": (\n            \"Using deprecated class %s of module %s\",\n            \"deprecated-class\",\n            \"The class is marked as deprecated and will be removed in the future.\",\n        ),\n        \"W1513\": (\n            \"Using deprecated decorator %s()\",\n            \"deprecated-decorator\",\n            \"The decorator is marked as deprecated and will be removed in the future.\",\n        ),\n        \"W1514\": (\n            \"Using open without explicitly specifying an encoding\",\n            \"unspecified-encoding\",\n            \"It is better to specify an encoding when opening documents. \"\n            \"Using the system default implicitly can create problems on other operating systems. \"\n            \"See https://www.python.org/dev/peps/pep-0597/\",\n        ),\n        \"W1515\": (\n            \"Leaving functions creating breakpoints in production code is not recommended\",\n            \"forgotten-debug-statement\",\n            \"Calls to breakpoint(), sys.breakpointhook() and pdb.set_trace() should be removed \"\n            \"from code that is not actively being debugged.\",\n        ),\n        \"W1516\": (\n            \"'lru_cache' without 'maxsize' will keep all method args alive indefinitely, including 'self'\",\n            \"lru-cache-decorating-method\",\n            \"By decorating a method with lru_cache the 'self' argument will be linked to \"\n            \"the lru_cache function and therefore never garbage collected. Unless your instance \"\n            \"will never need to be garbage collected (singleton) it is recommended to refactor \"\n            \"code to avoid this pattern or add a maxsize to the cache.\",\n        ),\n    }\n\n    def __init__(self, linter: Optional[\"PyLinter\"] = None) -> None:\n        BaseChecker.__init__(self, linter)\n        self._deprecated_methods: Set[str] = set()\n        self._deprecated_arguments: Dict[\n            str, Tuple[Tuple[Optional[int], str], ...]\n        ] = {}\n        self._deprecated_classes: Dict[str, Set[str]] = {}\n        self._deprecated_modules: Set[str] = set()\n        self._deprecated_decorators: Set[str] = set()\n\n        for since_vers, func_list in DEPRECATED_METHODS[sys.version_info[0]].items():\n            if since_vers <= sys.version_info:\n                self._deprecated_methods.update(func_list)\n        for since_vers, func_list in DEPRECATED_ARGUMENTS.items():\n            if since_vers <= sys.version_info:\n                self._deprecated_arguments.update(func_list)\n        for since_vers, class_list in DEPRECATED_CLASSES.items():\n            if since_vers <= sys.version_info:\n                self._deprecated_classes.update(class_list)\n        for since_vers, mod_list in DEPRECATED_MODULES.items():\n            if since_vers <= sys.version_info:\n                self._deprecated_modules.update(mod_list)\n        for since_vers, decorator_list in DEPRECATED_DECORATORS.items():\n            if since_vers <= sys.version_info:\n                self._deprecated_decorators.update(decorator_list)\n\n    def _check_bad_thread_instantiation(self, node):\n        if not node.kwargs and not node.keywords and len(node.args) <= 1:\n            self.add_message(\"bad-thread-instantiation\", node=node)\n\n    def _check_for_preexec_fn_in_popen(self, node):\n        if node.keywords:\n            for keyword in node.keywords:\n                if keyword.arg == \"preexec_fn\":\n                    self.add_message(\"subprocess-popen-preexec-fn\", node=node)\n\n    def _check_for_check_kw_in_run(self, node):\n        kwargs = {keyword.arg for keyword in (node.keywords or ())}\n        if \"check\" not in kwargs:\n            self.add_message(\"subprocess-run-check\", node=node)\n\n    def _check_shallow_copy_environ(self, node: nodes.Call) -> None:\n        arg = utils.get_argument_from_call(node, position=0)\n        try:\n            inferred_args = arg.inferred()\n        except astroid.InferenceError:\n            return\n        for inferred in inferred_args:\n            if inferred.qname() == OS_ENVIRON:\n                self.add_message(\"shallow-copy-environ\", node=node)\n                break\n\n    @utils.check_messages(\n        \"bad-open-mode\",\n        \"redundant-unittest-assert\",\n        \"deprecated-method\",\n        \"deprecated-argument\",\n        \"bad-thread-instantiation\",\n        \"shallow-copy-environ\",\n        \"invalid-envvar-value\",\n        \"invalid-envvar-default\",\n        \"subprocess-popen-preexec-fn\",\n        \"subprocess-run-check\",\n        \"deprecated-class\",\n        \"unspecified-encoding\",\n        \"forgotten-debug-statement\",\n    )\n    def visit_call(self, node: nodes.Call) -> None:\n        \"\"\"Visit a Call node.\"\"\"\n        self.check_deprecated_class_in_call(node)\n        for inferred in utils.infer_all(node.func):\n            if inferred is astroid.Uninferable:\n                continue\n            if inferred.root().name in OPEN_MODULE:\n                if (\n                    isinstance(node.func, nodes.Name)\n                    and node.func.name in OPEN_FILES_MODE\n                ):\n                    self._check_open_mode(node)\n                if (\n                    isinstance(node.func, nodes.Name)\n                    and node.func.name in OPEN_FILES_ENCODING\n                    or isinstance(node.func, nodes.Attribute)\n                    and node.func.attrname in OPEN_FILES_ENCODING\n                ):\n                    self._check_open_encoded(node, inferred.root().name)\n            elif inferred.root().name == UNITTEST_CASE:\n                self._check_redundant_assert(node, inferred)\n            elif isinstance(inferred, nodes.ClassDef):\n                if inferred.qname() == THREADING_THREAD:\n                    self._check_bad_thread_instantiation(node)\n                elif inferred.qname() == SUBPROCESS_POPEN:\n                    self._check_for_preexec_fn_in_popen(node)\n            elif isinstance(inferred, nodes.FunctionDef):\n                name = inferred.qname()\n                if name == COPY_COPY:\n                    self._check_shallow_copy_environ(node)\n                elif name in ENV_GETTERS:\n                    self._check_env_function(node, inferred)\n                elif name == SUBPROCESS_RUN:\n                    self._check_for_check_kw_in_run(node)\n                elif name in DEBUG_BREAKPOINTS:\n                    self.add_message(\"forgotten-debug-statement\", node=node)\n            self.check_deprecated_method(node, inferred)\n\n    @utils.check_messages(\"boolean-datetime\")\n    def visit_unaryop(self, node: nodes.UnaryOp) -> None:\n        if node.op == \"not\":\n            self._check_datetime(node.operand)\n\n    @utils.check_messages(\"boolean-datetime\")\n    def visit_if(self, node: nodes.If) -> None:\n        self._check_datetime(node.test)\n\n    @utils.check_messages(\"boolean-datetime\")\n    def visit_ifexp(self, node: nodes.IfExp) -> None:\n        self._check_datetime(node.test)\n\n    @utils.check_messages(\"boolean-datetime\")\n    def visit_boolop(self, node: nodes.BoolOp) -> None:\n        for value in node.values:\n            self._check_datetime(value)\n\n    @utils.check_messages(\"lru-cache-decorating-method\")\n    def visit_functiondef(self, node: nodes.FunctionDef) -> None:\n        if node.decorators and isinstance(node.parent, nodes.ClassDef):\n            self._check_lru_cache_decorators(node.decorators)\n\n    def _check_lru_cache_decorators(self, decorators: nodes.Decorators) -> None:\n        \"\"\"Check if instance methods are decorated with functools.lru_cache.\"\"\"\n        lru_cache_nodes: List[nodes.NodeNG] = []\n        for d_node in decorators.nodes:\n            try:\n                for infered_node in d_node.infer():\n                    q_name = infered_node.qname()\n                    if q_name in NON_INSTANCE_METHODS:\n                        return\n                    if q_name not in LRU_CACHE:\n                        return\n\n                    # Check if there is a maxsize argument to the call\n                    if isinstance(d_node, nodes.Call):\n                        try:\n                            utils.get_argument_from_call(\n                                d_node, position=0, keyword=\"maxsize\"\n                            )\n                            return\n                        except utils.NoSuchArgumentError:\n                            pass\n\n                    lru_cache_nodes.append(d_node)\n                    break\n            except astroid.InferenceError:\n                pass\n        for lru_cache_node in lru_cache_nodes:\n            self.add_message(\n                \"lru-cache-decorating-method\",\n                node=lru_cache_node,\n                confidence=interfaces.INFERENCE,\n            )\n\n    def _check_redundant_assert(self, node, infer):\n        if (\n            isinstance(infer, astroid.BoundMethod)\n            and node.args\n            and isinstance(node.args[0], nodes.Const)\n            and infer.name in {\"assertTrue\", \"assertFalse\"}\n        ):\n            self.add_message(\n                \"redundant-unittest-assert\",\n                args=(infer.name, node.args[0].value),\n                node=node,\n            )\n\n    def _check_datetime(self, node):\n        \"\"\"Check that a datetime was inferred, if so, emit boolean-datetime warning.\"\"\"\n        try:\n            inferred = next(node.infer())\n        except astroid.InferenceError:\n            return\n        if (\n            isinstance(inferred, astroid.Instance)\n            and inferred.qname() == \"datetime.time\"\n        ):\n            self.add_message(\"boolean-datetime\", node=node)\n\n    def _check_open_mode(self, node):\n        \"\"\"Check that the mode argument of an open or file call is valid.\"\"\"\n        try:\n            mode_arg = utils.get_argument_from_call(node, position=1, keyword=\"mode\")\n        except utils.NoSuchArgumentError:\n            return\n        if mode_arg:\n            mode_arg = utils.safe_infer(mode_arg)\n            if isinstance(mode_arg, nodes.Const) and not _check_mode_str(\n                mode_arg.value\n            ):\n                self.add_message(\n                    \"bad-open-mode\",\n                    node=node,\n                    args=mode_arg.value or str(mode_arg.value),\n                )\n\n    def _check_open_encoded(self, node: nodes.Call, open_module: str) -> None:\n        \"\"\"Check that the encoded argument of an open call is valid.\"\"\"\n        mode_arg = None\n        try:\n            if open_module == \"_io\":\n                mode_arg = utils.get_argument_from_call(\n                    node, position=1, keyword=\"mode\"\n                )\n            elif open_module == \"pathlib\":\n                mode_arg = utils.get_argument_from_call(\n                    node, position=0, keyword=\"mode\"\n                )\n        except utils.NoSuchArgumentError:\n            pass\n\n        if mode_arg:\n            mode_arg = utils.safe_infer(mode_arg)\n\n        if (\n            not mode_arg\n            or isinstance(mode_arg, nodes.Const)\n            and (not mode_arg.value or \"b\" not in mode_arg.value)\n        ):\n            encoding_arg = None\n            try:\n                if open_module == \"pathlib\":\n                    if node.func.attrname == \"read_text\":\n                        encoding_arg = utils.get_argument_from_call(\n                            node, position=0, keyword=\"encoding\"\n                        )\n                    elif node.func.attrname == \"write_text\":\n                        encoding_arg = utils.get_argument_from_call(\n                            node, position=1, keyword=\"encoding\"\n                        )\n                    else:\n                        encoding_arg = utils.get_argument_from_call(\n                            node, position=2, keyword=\"encoding\"\n                        )\n                else:\n                    encoding_arg = utils.get_argument_from_call(\n                        node, position=3, keyword=\"encoding\"\n                    )\n            except utils.NoSuchArgumentError:\n                self.add_message(\"unspecified-encoding\", node=node)\n\n            if encoding_arg:\n                encoding_arg = utils.safe_infer(encoding_arg)\n\n                if isinstance(encoding_arg, nodes.Const) and encoding_arg.value is None:\n                    self.add_message(\"unspecified-encoding\", node=node)\n\n    def _check_env_function(self, node, infer):\n        env_name_kwarg = \"key\"\n        env_value_kwarg = \"default\"\n        if node.keywords:\n            kwargs = {keyword.arg: keyword.value for keyword in node.keywords}\n        else:\n            kwargs = None\n        if node.args:\n            env_name_arg = node.args[0]\n        elif kwargs and env_name_kwarg in kwargs:\n            env_name_arg = kwargs[env_name_kwarg]\n        else:\n            env_name_arg = None\n\n        if env_name_arg:\n            self._check_invalid_envvar_value(\n                node=node,\n                message=\"invalid-envvar-value\",\n                call_arg=utils.safe_infer(env_name_arg),\n                infer=infer,\n                allow_none=False,\n            )\n\n        if len(node.args) == 2:\n            env_value_arg = node.args[1]\n        elif kwargs and env_value_kwarg in kwargs:\n            env_value_arg = kwargs[env_value_kwarg]\n        else:\n            env_value_arg = None\n\n        if env_value_arg:\n            self._check_invalid_envvar_value(\n                node=node,\n                infer=infer,\n                message=\"invalid-envvar-default\",\n                call_arg=utils.safe_infer(env_value_arg),\n                allow_none=True,\n            )\n\n    def _check_invalid_envvar_value(self, node, infer, message, call_arg, allow_none):\n        if call_arg in (astroid.Uninferable, None):\n            return\n\n        name = infer.qname()\n        if isinstance(call_arg, nodes.Const):\n            emit = False\n            if call_arg.value is None:\n                emit = not allow_none\n            elif not isinstance(call_arg.value, str):\n                emit = True\n            if emit:\n                self.add_message(message, node=node, args=(name, call_arg.pytype()))\n        else:\n            self.add_message(message, node=node, args=(name, call_arg.pytype()))\n\n    def deprecated_modules(self):\n        \"\"\"Callback returning the deprecated modules.\"\"\"\n        return self._deprecated_modules\n\n    def deprecated_methods(self):\n        return self._deprecated_methods\n\n    def deprecated_arguments(self, method: str):\n        return self._deprecated_arguments.get(method, ())\n\n    def deprecated_classes(self, module: str):\n        return self._deprecated_classes.get(module, ())\n\n    def deprecated_decorators(self) -> Iterable:\n        return self._deprecated_decorators"
        },
        {
          "file": "pylint/checkers/stdlib.py",
          "type": "function",
          "name": "visit_boolop",
          "class_name": "StdlibChecker",
          "code": "def visit_boolop(self, node: nodes.BoolOp) -> None:\n        for value in node.values:\n            self._check_datetime(value)"
        },
        {
          "file": "pylint/checkers/stdlib.py",
          "type": "function",
          "name": "_check_lru_cache_decorators",
          "class_name": "StdlibChecker",
          "code": "def _check_lru_cache_decorators(self, decorators: nodes.Decorators) -> None:\n        \"\"\"Check if instance methods are decorated with functools.lru_cache.\"\"\"\n        lru_cache_nodes: List[nodes.NodeNG] = []\n        for d_node in decorators.nodes:\n            try:\n                for infered_node in d_node.infer():\n                    q_name = infered_node.qname()\n                    if q_name in NON_INSTANCE_METHODS:\n                        return\n                    if q_name not in LRU_CACHE:\n                        return\n\n                    # Check if there is a maxsize argument to the call\n                    if isinstance(d_node, nodes.Call):\n                        try:\n                            utils.get_argument_from_call(\n                                d_node, position=0, keyword=\"maxsize\"\n                            )\n                            return\n                        except utils.NoSuchArgumentError:\n                            pass\n\n                    lru_cache_nodes.append(d_node)\n                    break\n            except astroid.InferenceError:\n                pass\n        for lru_cache_node in lru_cache_nodes:\n            self.add_message(\n                \"lru-cache-decorating-method\",\n                node=lru_cache_node,\n                confidence=interfaces.INFERENCE,\n            )"
        },
        {
          "file": "pylint/checkers/stdlib.py",
          "type": "function",
          "name": "_check_lru_cache_decorators",
          "class_name": "StdlibChecker",
          "code": "def _check_lru_cache_decorators(self, decorators: nodes.Decorators) -> None:\n        \"\"\"Check if instance methods are decorated with functools.lru_cache.\"\"\"\n        lru_cache_nodes: List[nodes.NodeNG] = []\n        for d_node in decorators.nodes:\n            try:\n                for infered_node in d_node.infer():\n                    q_name = infered_node.qname()\n                    if q_name in NON_INSTANCE_METHODS:\n                        return\n                    if q_name not in LRU_CACHE:\n                        return\n\n                    # Check if there is a maxsize argument to the call\n                    if isinstance(d_node, nodes.Call):\n                        try:\n                            utils.get_argument_from_call(\n                                d_node, position=0, keyword=\"maxsize\"\n                            )\n                            return\n                        except utils.NoSuchArgumentError:\n                            pass\n\n                    lru_cache_nodes.append(d_node)\n                    break\n            except astroid.InferenceError:\n                pass\n        for lru_cache_node in lru_cache_nodes:\n            self.add_message(\n                \"lru-cache-decorating-method\",\n                node=lru_cache_node,\n                confidence=interfaces.INFERENCE,\n            )"
        },
        {
          "file": "pylint/message/message_definition_store.py",
          "type": "function",
          "name": "register_message",
          "class_name": "MessageDefinitionStore",
          "code": "def register_message(self, message: MessageDefinition) -> None:\n        \"\"\"Register a MessageDefinition with consistency in mind.\"\"\"\n        self.message_id_store.register_message_definition(\n            message.msgid, message.symbol, message.old_names\n        )\n        self._messages_definitions[message.msgid] = message\n        self._msgs_by_category[message.msgid[0]].append(message.msgid)"
        }
      ]
    }
  ]
}