{
  "instance_id": "pylint-dev__pylint-5859",
  "repo": "pylint-dev/pylint",
  "created_at": "2022-03-04T00:01:54Z",
  "problem_statement": "\"--notes\" option ignores note tags that are entirely punctuation\n### Bug description\n\nIf a note tag specified with the `--notes` option is entirely punctuation, pylint won't report a fixme warning (W0511).\r\n\r\n```python\r\n# YES: yes\r\n# ???: no\r\n```\r\n\r\n`pylint test.py --notes=\"YES,???\"` will return a fixme warning (W0511) for the first line, but not the second.\n\n### Configuration\n\n```ini\nDefault\n```\n\n\n### Command used\n\n```shell\npylint test.py --notes=\"YES,???\"\n```\n\n\n### Pylint output\n\n```shell\n************* Module test\r\ntest.py:1:1: W0511: YES: yes (fixme)\n```\n\n\n### Expected behavior\n\n```\r\n************* Module test\r\ntest.py:1:1: W0511: YES: yes (fixme)\r\ntest.py:2:1: W0511: ???: no (fixme)\r\n```\n\n### Pylint version\n\n```shell\npylint 2.12.2\r\nastroid 2.9.0\r\nPython 3.10.2 (main, Feb  2 2022, 05:51:25) [Clang 13.0.0 (clang-1300.0.29.3)]\n```\n\n\n### OS / Environment\n\nmacOS 11.6.1\n\n### Additional dependencies\n\n_No response_\n",
  "patch": "diff --git a/pylint/checkers/misc.py b/pylint/checkers/misc.py\n--- a/pylint/checkers/misc.py\n+++ b/pylint/checkers/misc.py\n@@ -121,9 +121,9 @@ def open(self):\n \n         notes = \"|\".join(re.escape(note) for note in self.config.notes)\n         if self.config.notes_rgx:\n-            regex_string = rf\"#\\s*({notes}|{self.config.notes_rgx})\\b\"\n+            regex_string = rf\"#\\s*({notes}|{self.config.notes_rgx})(?=(:|\\s|\\Z))\"\n         else:\n-            regex_string = rf\"#\\s*({notes})\\b\"\n+            regex_string = rf\"#\\s*({notes})(?=(:|\\s|\\Z))\"\n \n         self._fixme_pattern = re.compile(regex_string, re.I)\n \n",
  "similar_bug_items": [
    {
      "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": 1344,
      "pr_title": "Split missing+differing docstring param checks",
      "pr_body": "Add 2 different error codes for different params defined\r\nSplit param name checking in two functions, one for missing checks\r\nand another for different names checks\r\nCheck for missing params and then check for differing params\r\nMinor fix in type of found_argument_names [list --> set]\r\n\r\n### Fixes / new features\r\n- Fixes #1342\r\n",
      "issue_id": 1342,
      "issue_title": "Missing parameter in documentation error code is the same as differing parameter (W9003)",
      "issue_body": "### Steps to reproduce\r\n1. Activate pylint.extensions.docparams\r\n.pylintrc -> `load-plugins=pylint.extensions.docparams`\r\n2. Create a function like the following\r\n````\r\ndef venv_with_kati(venva, kati):\r\n    \"\"\"\r\n    :param venv: virtualenv activation path\r\n    \"\"\"\r\n    print(kati, venva)\r\n````\r\n3. Run pylint\r\n\r\n### Current behavior\r\nCurrent behavior is to group the missing params in the same error with the invalid name params\r\n[W9003(missing-param-doc), bash_venv_or_null] \"kati, venv, venva\" missing or differing in parameter documentation\r\n\r\n### Expected behavior\r\nI would like to be able to disable warnings about missing params but at the same time still have warnings about obsolete named params. Something like the following:\r\n\r\n[W9003(missing-param-doc), bash_venv_or_null] \"kati, venva\" missing in parameter documentation\r\n***still the same***\r\n\r\n---\r\n\r\n[W900`**`(**invalid**-param-doc), bash_venv_or_null] \"venv\" documentation parameter missing from function signature\r\n***new error code***\r\n\r\n---\r\n\r\n### pylint --version output\r\n````\r\npylint --version\r\npylint 1.6.4,\r\nastroid 1.4.5\r\nPython 2.7.13 (default, Jan 12 2017, 17:59:37)\r\n[GCC 6.3.1 20161221 (Red Hat 6.3.1-1)]\r\n````\r\n\r\n",
      "issue_closed_at": "2017-03-11T07:06:58Z",
      "base_commit": "89d766ef8bea922cfacf55f8a8adbbf783955d4a",
      "changes": [
        {
          "file": "pylint/extensions/docparams.py",
          "type": "class",
          "name": "DocstringParameterChecker",
          "code": "class DocstringParameterChecker(BaseChecker):\n    \"\"\"Checker for Sphinx, Google, or Numpy style docstrings\n\n    * Check that all function, method and constructor parameters are mentioned\n      in the params and types part of the docstring.  Constructor parameters\n      can be documented in either the class docstring or ``__init__`` docstring,\n      but not both.\n    * Check that there are no naming inconsistencies between the signature and\n      the documentation, i.e. also report documented parameters that are missing\n      in the signature. This is important to find cases where parameters are\n      renamed only in the code, not in the documentation.\n    * Check that all explicity raised exceptions in a function are documented\n      in the function docstring. Caught exceptions are ignored.\n\n    Activate this checker by adding the line::\n\n        load-plugins=pylint.extensions.docparams\n\n    to the ``MASTER`` section of your ``.pylintrc``.\n\n    :param linter: linter object\n    :type linter: :class:`pylint.lint.PyLinter`\n    \"\"\"\n    __implements__ = IAstroidChecker\n\n    name = 'parameter_documentation'\n    msgs = {\n        'W9003': ('\"%s\" missing or differing in parameter documentation',\n                  'missing-param-doc',\n                  'Please add parameter declarations for all parameters.'),\n        'W9004': ('\"%s\" missing or differing in parameter type documentation',\n                  'missing-type-doc',\n                  'Please add parameter type declarations for all parameters.'),\n        'W9005': ('\"%s\" has constructor parameters documented in class and __init__',\n                  'multiple-constructor-doc',\n                  'Please remove parameter declarations in the class or constructor.'),\n        'W9006': ('\"%s\" not documented as being raised',\n                  'missing-raises-doc',\n                  'Please document exceptions for all raised exception types.'),\n        'W9008': ('Redundant returns documentation',\n                  'redundant-returns-doc',\n                  'Please remove the return/rtype documentation from this method.'),\n        'W9010': ('Redundant yields documentation',\n                  'redundant-yields-doc',\n                  'Please remove the yields documentation from this method.'),\n        'W9011': ('Missing return documentation',\n                  'missing-return-doc',\n                  'Please add documentation about what this method returns.',\n                  {'old_names': [('W9007', 'missing-returns-doc')]}),\n        'W9012': ('Missing return type documentation',\n                  'missing-return-type-doc',\n                  'Please document the type returned by this method.',\n                  # we can't use the same old_name for two different warnings\n                  # {'old_names': [('W9007', 'missing-returns-doc')]},\n                 ),\n        'W9013': ('Missing yield documentation',\n                  'missing-yield-doc',\n                  'Please add documentation about what this generator yields.',\n                  {'old_names': [('W9009', 'missing-yields-doc')]}),\n        'W9014': ('Missing yield type documentation',\n                  'missing-yield-type-doc',\n                  'Please document the type yielded by this method.',\n                  # we can't use the same old_name for two different warnings\n                  # {'old_names': [('W9009', 'missing-yields-doc')]},\n                 ),\n    }\n\n    options = (('accept-no-param-doc',\n                {'default': True, 'type' : 'yn', 'metavar' : '<y or n>',\n                 'help': 'Whether to accept totally missing parameter '\n                         'documentation in the docstring of a function that has '\n                         'parameters.'\n                }),\n               ('accept-no-raise-doc',\n                {'default': True, 'type' : 'yn', 'metavar' : '<y or n>',\n                 'help': 'Whether to accept totally missing raises '\n                         'documentation in the docstring of a function that '\n                         'raises an exception.'\n                }),\n               ('accept-no-return-doc',\n                {'default': True, 'type' : 'yn', 'metavar' : '<y or n>',\n                 'help': 'Whether to accept totally missing return '\n                         'documentation in the docstring of a function that '\n                         'returns a statement.'\n                }),\n               ('accept-no-yields-doc',\n                {'default': True, 'type' : 'yn', 'metavar': '<y or n>',\n                 'help': 'Whether to accept totally missing yields '\n                         'documentation in the docstring of a generator.'\n                }),\n              )\n\n    priority = -2\n\n    constructor_names = {'__init__', '__new__'}\n    not_needed_param_in_docstring = {'self', 'cls'}\n\n    def visit_functiondef(self, node):\n        \"\"\"Called for function and method definitions (def).\n\n        :param node: Node for a function or method definition in the AST\n        :type node: :class:`astroid.scoped_nodes.Function`\n        \"\"\"\n        node_doc = utils.docstringify(node.doc)\n        self.check_functiondef_params(node, node_doc)\n        self.check_functiondef_returns(node, node_doc)\n        self.check_functiondef_yields(node, node_doc)\n\n    def check_functiondef_params(self, node, node_doc):\n        node_allow_no_param = None\n        if node.name in self.constructor_names:\n            class_node = node_frame_class(node)\n            if class_node is not None:\n                class_doc = utils.docstringify(class_node.doc)\n                self.check_single_constructor_params(class_doc, node_doc, class_node)\n\n                # __init__ or class docstrings can have no parameters documented\n                # as long as the other documents them.\n                node_allow_no_param = (\n                    class_doc.has_params() or\n                    class_doc.params_documented_elsewhere() or\n                    None\n                )\n                class_allow_no_param = (\n                    node_doc.has_params() or\n                    node_doc.params_documented_elsewhere() or\n                    None\n                )\n\n                self.check_arguments_in_docstring(\n                    class_doc, node.args, class_node, class_allow_no_param)\n\n        self.check_arguments_in_docstring(\n            node_doc, node.args, node, node_allow_no_param)\n\n    def check_functiondef_returns(self, node, node_doc):\n        if not node_doc.supports_yields and node.is_generator():\n            return\n\n        return_nodes = node.nodes_of_class(astroid.Return)\n        if ((node_doc.has_returns() or node_doc.has_rtype()) and\n                not any(utils.returns_something(ret_node) for ret_node in return_nodes)):\n            self.add_message(\n                'redundant-returns-doc',\n                node=node)\n\n    def check_functiondef_yields(self, node, node_doc):\n        if not node_doc.supports_yields:\n            return\n\n        if ((node_doc.has_yields() or node_doc.has_yields_type()) and\n                not node.is_generator()):\n            self.add_message(\n                'redundant-yields-doc',\n                node=node)\n\n    def visit_raise(self, node):\n        func_node = node.frame()\n        if not isinstance(func_node, astroid.FunctionDef):\n            return\n\n        expected_excs = utils.possible_exc_types(node)\n        if not expected_excs:\n            return\n\n        doc = utils.docstringify(func_node.doc)\n        if not doc.is_valid():\n            if doc.doc:\n                self._handle_no_raise_doc(expected_excs, func_node)\n            return\n\n        found_excs = doc.exceptions()\n        missing_excs = expected_excs - found_excs\n        self._add_raise_message(missing_excs, func_node)\n\n    def visit_return(self, node):\n        if not utils.returns_something(node):\n            return\n\n        func_node = node.frame()\n        if not isinstance(func_node, astroid.FunctionDef):\n            return\n\n        doc = utils.docstringify(func_node.doc)\n        if not doc.is_valid() and self.config.accept_no_return_doc:\n            return\n\n        if not doc.has_returns():\n            self.add_message(\n                'missing-return-doc',\n                node=func_node\n            )\n\n        if not doc.has_rtype():\n            self.add_message(\n                'missing-return-type-doc',\n                node=func_node\n            )\n\n    def visit_yield(self, node):\n        func_node = node.frame()\n        if not isinstance(func_node, astroid.FunctionDef):\n            return\n\n        doc = utils.docstringify(func_node.doc)\n        if not doc.is_valid() and self.config.accept_no_yields_doc:\n            return\n\n        if doc.supports_yields:\n            doc_has_yields = doc.has_yields()\n            doc_has_yields_type = doc.has_yields_type()\n        else:\n            doc_has_yields = doc.has_returns()\n            doc_has_yields_type = doc.has_rtype()\n\n        if not doc_has_yields:\n            self.add_message(\n                'missing-yield-doc',\n                node=func_node\n            )\n\n        if not doc_has_yields_type:\n            self.add_message(\n                'missing-yield-type-doc',\n                node=func_node\n            )\n\n    def visit_yieldfrom(self, node):\n        self.visit_yield(node)\n\n    def check_arguments_in_docstring(self, doc, arguments_node, warning_node,\n                                     accept_no_param_doc=None):\n        \"\"\"Check that all parameters in a function, method or class constructor\n        on the one hand and the parameters mentioned in the parameter\n        documentation (e.g. the Sphinx tags 'param' and 'type') on the other\n        hand are consistent with each other.\n\n        * Undocumented parameters except 'self' are noticed.\n        * Undocumented parameter types except for 'self' and the ``*<args>``\n          and ``**<kwargs>`` parameters are noticed.\n        * Parameters mentioned in the parameter documentation that don't or no\n          longer exist in the function parameter list are noticed.\n        * If the text \"For the parameters, see\" or \"For the other parameters,\n          see\" (ignoring additional whitespace) is mentioned in the docstring,\n          missing parameter documentation is tolerated.\n        * If there's no Sphinx style, Google style or NumPy style parameter\n          documentation at all, i.e. ``:param`` is never mentioned etc., the\n          checker assumes that the parameters are documented in another format\n          and the absence is tolerated.\n\n        :param doc: Docstring for the function, method or class.\n        :type doc: str\n\n        :param arguments_node: Arguments node for the function, method or\n            class constructor.\n        :type arguments_node: :class:`astroid.scoped_nodes.Arguments`\n\n        :param warning_node: The node to assign the warnings to\n        :type warning_node: :class:`astroid.scoped_nodes.Node`\n\n        :param accept_no_param_doc: Whether or not to allow no parameters\n            to be documented.\n            If None then this value is read from the configuration.\n        :type accept_no_param_doc: bool or None\n        \"\"\"\n        # Tolerate missing param or type declarations if there is a link to\n        # another method carrying the same name.\n        if not doc.doc:\n            return\n\n        if accept_no_param_doc is None:\n            accept_no_param_doc = self.config.accept_no_param_doc\n        tolerate_missing_params = doc.params_documented_elsewhere()\n\n        # Collect the function arguments.\n        expected_argument_names = set(arg.name for arg in arguments_node.args)\n        expected_argument_names.update(arg.name for arg in arguments_node.kwonlyargs)\n        not_needed_type_in_docstring = (\n            self.not_needed_param_in_docstring.copy())\n\n        if arguments_node.vararg is not None:\n            expected_argument_names.add(arguments_node.vararg)\n            not_needed_type_in_docstring.add(arguments_node.vararg)\n        if arguments_node.kwarg is not None:\n            expected_argument_names.add(arguments_node.kwarg)\n            not_needed_type_in_docstring.add(arguments_node.kwarg)\n        params_with_doc, params_with_type = doc.match_param_docs()\n\n        # Tolerate no parameter documentation at all.\n        if (not params_with_doc and not params_with_type\n                and accept_no_param_doc):\n            tolerate_missing_params = True\n\n        def _compare_args(found_argument_names, message_id, not_needed_names):\n            \"\"\"Compare the found argument names with the expected ones and\n            generate a message if there are inconsistencies.\n\n            :param list found_argument_names: argument names found in the\n                docstring\n\n            :param str message_id: pylint message id\n\n            :param not_needed_names: names that may be omitted\n            :type not_needed_names: set of str\n            \"\"\"\n            if not tolerate_missing_params:\n                missing_or_differing_argument_names = (\n                    (expected_argument_names ^ found_argument_names)\n                    - not_needed_names)\n            else:\n                missing_or_differing_argument_names = (\n                    (found_argument_names - expected_argument_names)\n                    - not_needed_names)\n\n            if missing_or_differing_argument_names:\n                self.add_message(\n                    message_id,\n                    args=(', '.join(\n                        sorted(missing_or_differing_argument_names)),),\n                    node=warning_node)\n\n        _compare_args(params_with_doc, 'missing-param-doc',\n                      self.not_needed_param_in_docstring)\n        _compare_args(params_with_type, 'missing-type-doc',\n                      not_needed_type_in_docstring)\n\n    def check_single_constructor_params(self, class_doc, init_doc, class_node):\n        if class_doc.has_params() and init_doc.has_params():\n            self.add_message(\n                'multiple-constructor-doc',\n                args=(class_node.name,),\n                node=class_node)\n\n    def _handle_no_raise_doc(self, excs, node):\n        if self.config.accept_no_raise_doc:\n            return\n\n        self._add_raise_message(excs, node)\n\n    def _add_raise_message(self, missing_excs, node):\n        \"\"\"\n        Adds a message on :param:`node` for the missing exception type.\n\n        :param missing_excs: A list of missing exception types.\n        :type missing_excs: list\n\n        :param node: The node show the message on.\n        :type node: astroid.node_classes.NodeNG\n        \"\"\"\n        if not missing_excs:\n            return\n\n        self.add_message(\n            'missing-raises-doc',\n            args=(', '.join(sorted(missing_excs)),),\n            node=node)"
        },
        {
          "file": "pylint/extensions/docparams.py",
          "type": "class",
          "name": "DocstringParameterChecker",
          "code": "class DocstringParameterChecker(BaseChecker):\n    \"\"\"Checker for Sphinx, Google, or Numpy style docstrings\n\n    * Check that all function, method and constructor parameters are mentioned\n      in the params and types part of the docstring.  Constructor parameters\n      can be documented in either the class docstring or ``__init__`` docstring,\n      but not both.\n    * Check that there are no naming inconsistencies between the signature and\n      the documentation, i.e. also report documented parameters that are missing\n      in the signature. This is important to find cases where parameters are\n      renamed only in the code, not in the documentation.\n    * Check that all explicity raised exceptions in a function are documented\n      in the function docstring. Caught exceptions are ignored.\n\n    Activate this checker by adding the line::\n\n        load-plugins=pylint.extensions.docparams\n\n    to the ``MASTER`` section of your ``.pylintrc``.\n\n    :param linter: linter object\n    :type linter: :class:`pylint.lint.PyLinter`\n    \"\"\"\n    __implements__ = IAstroidChecker\n\n    name = 'parameter_documentation'\n    msgs = {\n        'W9003': ('\"%s\" missing or differing in parameter documentation',\n                  'missing-param-doc',\n                  'Please add parameter declarations for all parameters.'),\n        'W9004': ('\"%s\" missing or differing in parameter type documentation',\n                  'missing-type-doc',\n                  'Please add parameter type declarations for all parameters.'),\n        'W9005': ('\"%s\" has constructor parameters documented in class and __init__',\n                  'multiple-constructor-doc',\n                  'Please remove parameter declarations in the class or constructor.'),\n        'W9006': ('\"%s\" not documented as being raised',\n                  'missing-raises-doc',\n                  'Please document exceptions for all raised exception types.'),\n        'W9008': ('Redundant returns documentation',\n                  'redundant-returns-doc',\n                  'Please remove the return/rtype documentation from this method.'),\n        'W9010': ('Redundant yields documentation',\n                  'redundant-yields-doc',\n                  'Please remove the yields documentation from this method.'),\n        'W9011': ('Missing return documentation',\n                  'missing-return-doc',\n                  'Please add documentation about what this method returns.',\n                  {'old_names': [('W9007', 'missing-returns-doc')]}),\n        'W9012': ('Missing return type documentation',\n                  'missing-return-type-doc',\n                  'Please document the type returned by this method.',\n                  # we can't use the same old_name for two different warnings\n                  # {'old_names': [('W9007', 'missing-returns-doc')]},\n                 ),\n        'W9013': ('Missing yield documentation',\n                  'missing-yield-doc',\n                  'Please add documentation about what this generator yields.',\n                  {'old_names': [('W9009', 'missing-yields-doc')]}),\n        'W9014': ('Missing yield type documentation',\n                  'missing-yield-type-doc',\n                  'Please document the type yielded by this method.',\n                  # we can't use the same old_name for two different warnings\n                  # {'old_names': [('W9009', 'missing-yields-doc')]},\n                 ),\n    }\n\n    options = (('accept-no-param-doc',\n                {'default': True, 'type' : 'yn', 'metavar' : '<y or n>',\n                 'help': 'Whether to accept totally missing parameter '\n                         'documentation in the docstring of a function that has '\n                         'parameters.'\n                }),\n               ('accept-no-raise-doc',\n                {'default': True, 'type' : 'yn', 'metavar' : '<y or n>',\n                 'help': 'Whether to accept totally missing raises '\n                         'documentation in the docstring of a function that '\n                         'raises an exception.'\n                }),\n               ('accept-no-return-doc',\n                {'default': True, 'type' : 'yn', 'metavar' : '<y or n>',\n                 'help': 'Whether to accept totally missing return '\n                         'documentation in the docstring of a function that '\n                         'returns a statement.'\n                }),\n               ('accept-no-yields-doc',\n                {'default': True, 'type' : 'yn', 'metavar': '<y or n>',\n                 'help': 'Whether to accept totally missing yields '\n                         'documentation in the docstring of a generator.'\n                }),\n              )\n\n    priority = -2\n\n    constructor_names = {'__init__', '__new__'}\n    not_needed_param_in_docstring = {'self', 'cls'}\n\n    def visit_functiondef(self, node):\n        \"\"\"Called for function and method definitions (def).\n\n        :param node: Node for a function or method definition in the AST\n        :type node: :class:`astroid.scoped_nodes.Function`\n        \"\"\"\n        node_doc = utils.docstringify(node.doc)\n        self.check_functiondef_params(node, node_doc)\n        self.check_functiondef_returns(node, node_doc)\n        self.check_functiondef_yields(node, node_doc)\n\n    def check_functiondef_params(self, node, node_doc):\n        node_allow_no_param = None\n        if node.name in self.constructor_names:\n            class_node = node_frame_class(node)\n            if class_node is not None:\n                class_doc = utils.docstringify(class_node.doc)\n                self.check_single_constructor_params(class_doc, node_doc, class_node)\n\n                # __init__ or class docstrings can have no parameters documented\n                # as long as the other documents them.\n                node_allow_no_param = (\n                    class_doc.has_params() or\n                    class_doc.params_documented_elsewhere() or\n                    None\n                )\n                class_allow_no_param = (\n                    node_doc.has_params() or\n                    node_doc.params_documented_elsewhere() or\n                    None\n                )\n\n                self.check_arguments_in_docstring(\n                    class_doc, node.args, class_node, class_allow_no_param)\n\n        self.check_arguments_in_docstring(\n            node_doc, node.args, node, node_allow_no_param)\n\n    def check_functiondef_returns(self, node, node_doc):\n        if not node_doc.supports_yields and node.is_generator():\n            return\n\n        return_nodes = node.nodes_of_class(astroid.Return)\n        if ((node_doc.has_returns() or node_doc.has_rtype()) and\n                not any(utils.returns_something(ret_node) for ret_node in return_nodes)):\n            self.add_message(\n                'redundant-returns-doc',\n                node=node)\n\n    def check_functiondef_yields(self, node, node_doc):\n        if not node_doc.supports_yields:\n            return\n\n        if ((node_doc.has_yields() or node_doc.has_yields_type()) and\n                not node.is_generator()):\n            self.add_message(\n                'redundant-yields-doc',\n                node=node)\n\n    def visit_raise(self, node):\n        func_node = node.frame()\n        if not isinstance(func_node, astroid.FunctionDef):\n            return\n\n        expected_excs = utils.possible_exc_types(node)\n        if not expected_excs:\n            return\n\n        doc = utils.docstringify(func_node.doc)\n        if not doc.is_valid():\n            if doc.doc:\n                self._handle_no_raise_doc(expected_excs, func_node)\n            return\n\n        found_excs = doc.exceptions()\n        missing_excs = expected_excs - found_excs\n        self._add_raise_message(missing_excs, func_node)\n\n    def visit_return(self, node):\n        if not utils.returns_something(node):\n            return\n\n        func_node = node.frame()\n        if not isinstance(func_node, astroid.FunctionDef):\n            return\n\n        doc = utils.docstringify(func_node.doc)\n        if not doc.is_valid() and self.config.accept_no_return_doc:\n            return\n\n        if not doc.has_returns():\n            self.add_message(\n                'missing-return-doc',\n                node=func_node\n            )\n\n        if not doc.has_rtype():\n            self.add_message(\n                'missing-return-type-doc',\n                node=func_node\n            )\n\n    def visit_yield(self, node):\n        func_node = node.frame()\n        if not isinstance(func_node, astroid.FunctionDef):\n            return\n\n        doc = utils.docstringify(func_node.doc)\n        if not doc.is_valid() and self.config.accept_no_yields_doc:\n            return\n\n        if doc.supports_yields:\n            doc_has_yields = doc.has_yields()\n            doc_has_yields_type = doc.has_yields_type()\n        else:\n            doc_has_yields = doc.has_returns()\n            doc_has_yields_type = doc.has_rtype()\n\n        if not doc_has_yields:\n            self.add_message(\n                'missing-yield-doc',\n                node=func_node\n            )\n\n        if not doc_has_yields_type:\n            self.add_message(\n                'missing-yield-type-doc',\n                node=func_node\n            )\n\n    def visit_yieldfrom(self, node):\n        self.visit_yield(node)\n\n    def check_arguments_in_docstring(self, doc, arguments_node, warning_node,\n                                     accept_no_param_doc=None):\n        \"\"\"Check that all parameters in a function, method or class constructor\n        on the one hand and the parameters mentioned in the parameter\n        documentation (e.g. the Sphinx tags 'param' and 'type') on the other\n        hand are consistent with each other.\n\n        * Undocumented parameters except 'self' are noticed.\n        * Undocumented parameter types except for 'self' and the ``*<args>``\n          and ``**<kwargs>`` parameters are noticed.\n        * Parameters mentioned in the parameter documentation that don't or no\n          longer exist in the function parameter list are noticed.\n        * If the text \"For the parameters, see\" or \"For the other parameters,\n          see\" (ignoring additional whitespace) is mentioned in the docstring,\n          missing parameter documentation is tolerated.\n        * If there's no Sphinx style, Google style or NumPy style parameter\n          documentation at all, i.e. ``:param`` is never mentioned etc., the\n          checker assumes that the parameters are documented in another format\n          and the absence is tolerated.\n\n        :param doc: Docstring for the function, method or class.\n        :type doc: str\n\n        :param arguments_node: Arguments node for the function, method or\n            class constructor.\n        :type arguments_node: :class:`astroid.scoped_nodes.Arguments`\n\n        :param warning_node: The node to assign the warnings to\n        :type warning_node: :class:`astroid.scoped_nodes.Node`\n\n        :param accept_no_param_doc: Whether or not to allow no parameters\n            to be documented.\n            If None then this value is read from the configuration.\n        :type accept_no_param_doc: bool or None\n        \"\"\"\n        # Tolerate missing param or type declarations if there is a link to\n        # another method carrying the same name.\n        if not doc.doc:\n            return\n\n        if accept_no_param_doc is None:\n            accept_no_param_doc = self.config.accept_no_param_doc\n        tolerate_missing_params = doc.params_documented_elsewhere()\n\n        # Collect the function arguments.\n        expected_argument_names = set(arg.name for arg in arguments_node.args)\n        expected_argument_names.update(arg.name for arg in arguments_node.kwonlyargs)\n        not_needed_type_in_docstring = (\n            self.not_needed_param_in_docstring.copy())\n\n        if arguments_node.vararg is not None:\n            expected_argument_names.add(arguments_node.vararg)\n            not_needed_type_in_docstring.add(arguments_node.vararg)\n        if arguments_node.kwarg is not None:\n            expected_argument_names.add(arguments_node.kwarg)\n            not_needed_type_in_docstring.add(arguments_node.kwarg)\n        params_with_doc, params_with_type = doc.match_param_docs()\n\n        # Tolerate no parameter documentation at all.\n        if (not params_with_doc and not params_with_type\n                and accept_no_param_doc):\n            tolerate_missing_params = True\n\n        def _compare_args(found_argument_names, message_id, not_needed_names):\n            \"\"\"Compare the found argument names with the expected ones and\n            generate a message if there are inconsistencies.\n\n            :param list found_argument_names: argument names found in the\n                docstring\n\n            :param str message_id: pylint message id\n\n            :param not_needed_names: names that may be omitted\n            :type not_needed_names: set of str\n            \"\"\"\n            if not tolerate_missing_params:\n                missing_or_differing_argument_names = (\n                    (expected_argument_names ^ found_argument_names)\n                    - not_needed_names)\n            else:\n                missing_or_differing_argument_names = (\n                    (found_argument_names - expected_argument_names)\n                    - not_needed_names)\n\n            if missing_or_differing_argument_names:\n                self.add_message(\n                    message_id,\n                    args=(', '.join(\n                        sorted(missing_or_differing_argument_names)),),\n                    node=warning_node)\n\n        _compare_args(params_with_doc, 'missing-param-doc',\n                      self.not_needed_param_in_docstring)\n        _compare_args(params_with_type, 'missing-type-doc',\n                      not_needed_type_in_docstring)\n\n    def check_single_constructor_params(self, class_doc, init_doc, class_node):\n        if class_doc.has_params() and init_doc.has_params():\n            self.add_message(\n                'multiple-constructor-doc',\n                args=(class_node.name,),\n                node=class_node)\n\n    def _handle_no_raise_doc(self, excs, node):\n        if self.config.accept_no_raise_doc:\n            return\n\n        self._add_raise_message(excs, node)\n\n    def _add_raise_message(self, missing_excs, node):\n        \"\"\"\n        Adds a message on :param:`node` for the missing exception type.\n\n        :param missing_excs: A list of missing exception types.\n        :type missing_excs: list\n\n        :param node: The node show the message on.\n        :type node: astroid.node_classes.NodeNG\n        \"\"\"\n        if not missing_excs:\n            return\n\n        self.add_message(\n            'missing-raises-doc',\n            args=(', '.join(sorted(missing_excs)),),\n            node=node)"
        },
        {
          "file": "pylint/extensions/docparams.py",
          "type": "function",
          "name": "_compare_args",
          "class_name": "DocstringParameterChecker",
          "code": "def _compare_args(found_argument_names, message_id, not_needed_names):\n            \"\"\"Compare the found argument names with the expected ones and\n            generate a message if there are inconsistencies.\n\n            :param list found_argument_names: argument names found in the\n                docstring\n\n            :param str message_id: pylint message id\n\n            :param not_needed_names: names that may be omitted\n            :type not_needed_names: set of str\n            \"\"\"\n            if not tolerate_missing_params:\n                missing_or_differing_argument_names = (\n                    (expected_argument_names ^ found_argument_names)\n                    - not_needed_names)\n            else:\n                missing_or_differing_argument_names = (\n                    (found_argument_names - expected_argument_names)\n                    - not_needed_names)\n\n            if missing_or_differing_argument_names:\n                self.add_message(\n                    message_id,\n                    args=(', '.join(\n                        sorted(missing_or_differing_argument_names)),),\n                    node=warning_node)"
        }
      ]
    },
    {
      "pr_number": 5262,
      "pr_title": "Fix crash on checking private members on ``__class__``",
      "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 #5261\r\n",
      "issue_id": 5261,
      "issue_title": "Crash ``'Attribute' object has no attribute 'name'`` in _check_unused_private_variables due to access via self.__class__",
      "issue_body": "\r\nWhen parsing the following file:\r\n\r\n<!--\r\n If sharing the code is not an option, please state so,\r\n but providing only the stacktrace would still be helpful.\r\n -->\r\n\r\n```python\r\nclass Foo:\r\n    __ham = 1\r\n\r\n    def foo():\r\n        print(self.__class__.__ham)\r\n\r\n```\r\n\r\npylint crashed with a ``AttributeError`` and with the following stacktrace:\r\n```\r\nTraceback (most recent call last):\r\n  File \"/home/graingert/.local/pipx/.cache/1ea90594a10eda9/lib/python3.8/site-packages/pylint/lint/pylinter.py\", line 1008, in _check_files\r\n    self._check_file(get_ast, check_astroid_module, file)\r\n  File \"/home/graingert/.local/pipx/.cache/1ea90594a10eda9/lib/python3.8/site-packages/pylint/lint/pylinter.py\", line 1043, in _check_file\r\n    check_astroid_module(ast_node)\r\n  File \"/home/graingert/.local/pipx/.cache/1ea90594a10eda9/lib/python3.8/site-packages/pylint/lint/pylinter.py\", line 1180, in check_astroid_module\r\n    retval = self._check_astroid_module(\r\n  File \"/home/graingert/.local/pipx/.cache/1ea90594a10eda9/lib/python3.8/site-packages/pylint/lint/pylinter.py\", line 1227, in _check_astroid_module\r\n    walker.walk(node)\r\n  File \"/home/graingert/.local/pipx/.cache/1ea90594a10eda9/lib/python3.8/site-packages/pylint/utils/ast_walker.py\", line 78, in walk\r\n    self.walk(child)\r\n  File \"/home/graingert/.local/pipx/.cache/1ea90594a10eda9/lib/python3.8/site-packages/pylint/utils/ast_walker.py\", line 80, in walk\r\n    callback(astroid)\r\n  File \"/home/graingert/.local/pipx/.cache/1ea90594a10eda9/lib/python3.8/site-packages/pylint/checkers/classes.py\", line 907, in leave_classdef\r\n    self._check_unused_private_variables(node)\r\n  File \"/home/graingert/.local/pipx/.cache/1ea90594a10eda9/lib/python3.8/site-packages/pylint/checkers/classes.py\", line 973, in _check_unused_private_variables\r\n    and child.expr.name in (\"self\", \"cls\", node.name)\r\nAttributeError: 'Attribute' object has no attribute 'name'\r\n```",
      "issue_closed_at": "2021-11-05T20:26:54Z",
      "base_commit": "96e84595194073ea54a8c7730b86125049c0f4f9",
      "changes": [
        {
          "file": "pylint/checkers/classes.py",
          "type": "function",
          "name": "_check_unused_private_functions",
          "class_name": "ClassChecker",
          "code": "def _check_unused_private_functions(self, node: nodes.ClassDef) -> None:\n        for function_def in node.nodes_of_class(nodes.FunctionDef):\n            if not is_attr_private(function_def.name):\n                continue\n            parent_scope = function_def.parent.scope()\n            if isinstance(parent_scope, nodes.FunctionDef):\n                # Handle nested functions\n                if function_def.name in (\n                    n.name for n in parent_scope.nodes_of_class(nodes.Name)\n                ):\n                    continue\n            for attribute in node.nodes_of_class(nodes.Attribute):\n                if (\n                    attribute.attrname != function_def.name\n                    or attribute.scope() == function_def  # We ignore recursive calls\n                ):\n                    continue\n                if isinstance(attribute.expr, nodes.Name) and attribute.expr.name in (\n                    \"self\",\n                    \"cls\",\n                    node.name,\n                ):\n                    # self.__attrname\n                    # cls.__attrname\n                    # node_name.__attrname\n                    break\n                if isinstance(attribute.expr, nodes.Call):\n                    # type(self).__attrname\n                    inferred = safe_infer(attribute.expr)\n                    if (\n                        isinstance(inferred, nodes.ClassDef)\n                        and inferred.name == node.name\n                    ):\n                        break\n            else:\n                name_stack = []\n                curr = parent_scope\n                # Generate proper names for nested functions\n                while curr != node:\n                    name_stack.append(curr.name)\n                    curr = curr.parent.scope()\n\n                outer_level_names = f\"{'.'.join(reversed(name_stack))}\"\n                function_repr = f\"{outer_level_names}.{function_def.name}({function_def.args.as_string()})\"\n                self.add_message(\n                    \"unused-private-member\",\n                    node=function_def,\n                    args=(node.name, function_repr.lstrip(\".\")),\n                )"
        },
        {
          "file": "pylint/checkers/classes.py",
          "type": "function",
          "name": "_check_unused_private_variables",
          "class_name": "ClassChecker",
          "code": "def _check_unused_private_variables(self, node: nodes.ClassDef) -> None:\n        for assign_name in node.nodes_of_class(nodes.AssignName):\n            if isinstance(assign_name.parent, nodes.Arguments):\n                continue  # Ignore function arguments\n            if not is_attr_private(assign_name.name):\n                continue\n            for child in node.nodes_of_class((nodes.Name, nodes.Attribute)):\n                if isinstance(child, nodes.Name) and child.name == assign_name.name:\n                    break\n                if (\n                    isinstance(child, nodes.Attribute)\n                    and child.attrname == assign_name.name\n                    and child.expr.name in (\"self\", \"cls\", node.name)\n                ):\n                    break\n            else:\n                args = (node.name, assign_name.name)\n                self.add_message(\"unused-private-member\", node=assign_name, args=args)"
        }
      ]
    },
    {
      "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": 3071,
      "pr_title": "useless-suppression detection now ignores cyclic-import",
      "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/master/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- [X] 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 `doc/whatsnew/<current release.rst>`.\r\n- [X] Write a good description on what the PR does.\r\n\r\n## Description\r\nThis simply tweaks iter_spurious_suppression_messages() to ignore any 'R0401' warnings it comes across (cyclic-import). The cyclic-import checker seems to have incomplete context at the point when this runs so is always flagged as a useless suppression.\r\nPlease holler if this is not the best place to add the ignore; happy to revise the PR if needed.\r\n\r\n## Type of Changes\r\n<!-- Leave the corresponding lines for the applicable type of change: -->\r\n|   | Type |\r\n| ------------- | ------------- |\r\n| \u2713  | :bug: Bug fix  |\r\n\r\n## Related Issue\r\nCloses #3064 \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 #xxx\r\n-->",
      "issue_id": 3064,
      "issue_title": "False positive for useless-suppression with cyclic-import",
      "issue_body": "I have a number of import cycles in my codebase, though I try to be safe about them by keeping them in functions and not at the module level.  In this case I suppress the cyclic-import warning within those individual functions.  I just enabled useless-suppression detection, however, and found that it is reporting (seemingly all?) of these cyclic-import suppressions as unnecessary, though when I remove them I once again get cyclic-import warnings.\r\n\r\n### Steps to reproduce\r\n1. create a.py with the following code:\r\n```\r\n\"\"\"Test Module A.\"\"\"\r\nimport b\r\n\r\nprint(b)\r\n```\r\n\r\n2. create b.py with the following code:\r\n```\r\n\"\"\"Test Module B.\"\"\"\r\n\r\ndef bfunc():\r\n    \"\"\"Create a 'safe-ish' import cycle by doing it from a function.\"\"\"\r\n    import a\r\n    print(a)\r\n```\r\n3. pylint --enable=cyclic-import a.py b.py -> should generate a warning\r\n4. add  # pylint: disable=cyclic-import under bfunc()\r\n5. pylint --enable=cyclic-import a.py b.py -> no warning now\r\n6. pylint --enable=cyclic-import,useless-suppression a.py b.py -> incorrectly states that the line we added in step 4 is unnecessary\r\n\r\n### Current behavior\r\nI'm finding all of my cyclic-import suppressions are generating these warnings.\r\n\r\n### Expected behavior\r\nShould useless-suppression detection simply ignore cyclic-import suppressions?  Even if it was behaving as expected in this test case, would it still trigger incorrectly if only a.py or b.py were tested by itself?  (pylint only generates the cycle error when both a.py and b.py are tested together)\r\n\r\n### pylint --version output\r\npylint 2.3.1\r\nastroid 2.2.5\r\nPython 3.7.4 (default, Jul  9 2019, 18:13:23) \r\n[Clang 10.0.1 (clang-1001.0.46.4)]\r\n",
      "issue_closed_at": "2019-08-22T12:37:04Z",
      "base_commit": "ac25a275afe489e7ddf24ff6fd078414854ebaa4",
      "changes": [
        {
          "file": "pylint/utils/file_state.py",
          "type": "function",
          "name": "iter_spurious_suppression_messages",
          "class_name": "FileState",
          "code": "def iter_spurious_suppression_messages(self, msgs_store):\n        for warning, lines in self._raw_module_msgs_state.items():\n            for line, enable in lines.items():\n                if not enable and (warning, line) not in self._ignored_msgs:\n                    yield \"useless-suppression\", line, (\n                        msgs_store.get_msg_display_string(warning),\n                    )\n        # don't use iteritems here, _ignored_msgs may be modified by add_message\n        for (warning, from_), lines in list(self._ignored_msgs.items()):\n            for line in lines:\n                yield \"suppressed-message\", line, (\n                    msgs_store.get_msg_display_string(warning),\n                    from_,\n                )"
        }
      ]
    }
  ]
}