{
  "instance_id": "pylint-dev__pylint-6506",
  "repo": "pylint-dev/pylint",
  "created_at": "2022-05-05T13:01:41Z",
  "problem_statement": "Traceback printed for unrecognized option\n### Bug description\n\nA traceback is printed when an unrecognized option is passed to pylint.\n\n### Configuration\n\n_No response_\n\n### Command used\n\n```shell\npylint -Q\n```\n\n\n### Pylint output\n\n```shell\n************* Module Command line\r\nCommand line:1:0: E0015: Unrecognized option found: Q (unrecognized-option)\r\nTraceback (most recent call last):\r\n  File \"/Users/markbyrne/venv310/bin/pylint\", line 33, in <module>\r\n    sys.exit(load_entry_point('pylint', 'console_scripts', 'pylint')())\r\n  File \"/Users/markbyrne/programming/pylint/pylint/__init__.py\", line 24, in run_pylint\r\n    PylintRun(argv or sys.argv[1:])\r\n  File \"/Users/markbyrne/programming/pylint/pylint/lint/run.py\", line 135, in __init__\r\n    args = _config_initialization(\r\n  File \"/Users/markbyrne/programming/pylint/pylint/config/config_initialization.py\", line 85, in _config_initialization\r\n    raise _UnrecognizedOptionError(options=unrecognized_options)\r\npylint.config.exceptions._UnrecognizedOptionError\n```\n\n\n### Expected behavior\n\nThe top part of the current output is handy:\r\n`Command line:1:0: E0015: Unrecognized option found: Q (unrecognized-option)`\r\n\r\nThe traceback I don't think is expected & not user-friendly.\r\nA usage tip, for example:\r\n```python\r\nmypy -Q\r\nusage: mypy [-h] [-v] [-V] [more options; see below]\r\n            [-m MODULE] [-p PACKAGE] [-c PROGRAM_TEXT] [files ...]\r\nmypy: error: unrecognized arguments: -Q\r\n```\n\n### Pylint version\n\n```shell\npylint 2.14.0-dev0\r\nastroid 2.11.3\r\nPython 3.10.0b2 (v3.10.0b2:317314165a, May 31 2021, 10:02:22) [Clang 12.0.5 (clang-1205.0.22.9)]\n```\n\n\n### OS / Environment\n\n_No response_\n\n### Additional dependencies\n\n_No response_\n",
  "patch": "diff --git a/pylint/config/config_initialization.py b/pylint/config/config_initialization.py\n--- a/pylint/config/config_initialization.py\n+++ b/pylint/config/config_initialization.py\n@@ -81,8 +81,7 @@ def _config_initialization(\n             unrecognized_options.append(opt[1:])\n     if unrecognized_options:\n         msg = \", \".join(unrecognized_options)\n-        linter.add_message(\"unrecognized-option\", line=0, args=msg)\n-        raise _UnrecognizedOptionError(options=unrecognized_options)\n+        linter._arg_parser.error(f\"Unrecognized option found: {msg}\")\n \n     # Set the current module to configuration as we don't know where\n     # the --load-plugins key is coming from\n",
  "similar_bug_items": [
    {
      "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": 2611,
      "pr_title": "implicit-str-concat-in-sequence: Handling lines with multi-bytes characters - fix #2610",
      "pr_body": "",
      "issue_id": 2610,
      "issue_title": "New implicit-str-concat-in-sequence crashes",
      "issue_body": "### Current behavior\r\n```\r\n00:02:04.979 Traceback (most recent call last):\r\n00:02:04.979   File \"/home/jenkins/venv/4/bin/pylint\", line 11, in <module>\r\n00:02:04.979     sys.exit(run_pylint())\r\n00:02:04.979   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/__init__.py\", line 20, in run_pylint\r\n00:02:04.979     Run(sys.argv[1:])\r\n00:02:04.979   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/lint.py\", line 1608, in __init__\r\n00:02:04.979     linter.check(args)\r\n00:02:04.979   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/lint.py\", line 938, in check\r\n00:02:04.979     self._do_check(files_or_modules)\r\n00:02:04.980   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/lint.py\", line 1071, in _do_check\r\n00:02:04.980     self.check_astroid_module(ast_node, walker, rawcheckers, tokencheckers)\r\n00:02:04.980   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/lint.py\", line 1154, in check_astroid_module\r\n00:02:04.980     walker.walk(ast_node)\r\n00:02:04.980   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/utils.py\", line 1269, in walk\r\n00:02:04.980     self.walk(child)\r\n00:02:04.980   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/utils.py\", line 1269, in walk\r\n00:02:04.980     self.walk(child)\r\n00:02:04.980   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/utils.py\", line 1269, in walk\r\n00:02:04.980     self.walk(child)\r\n00:02:04.980   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/utils.py\", line 1266, in walk\r\n00:02:04.980     cb(astroid)\r\n00:02:04.980   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/checkers/strings.py\", line 613, in visit_tuple\r\n00:02:04.980     self.check_for_concatenated_strings(node, \"tuple\")\r\n00:02:04.980   File \"/home/jenkins/venv/4/lib/python3.6/site-packages/pylint/checkers/strings.py\", line 622, in check_for_concatenated_strings\r\n00:02:04.980     (elt.lineno, elt.col_offset)\r\n00:02:04.980 KeyError: (5, 39)\r\n```\r\n\r\nI'll try to find the crash-causing code.\r\n### Expected behavior\r\nDo not crash ;)\r\n\r\n### pylint --version output\r\npylint 2.2.0\r\nastroid 2.1.0\r\nPython 3.6.7 (default, Nov 22 2018, 09:26:21) \r\n[GCC 8.2.1 20180831]\r\n",
      "issue_closed_at": "2018-11-26T13:22:07Z",
      "base_commit": "1ac83855b38a8d053ec5d0403a5c0e6e8b191fbb",
      "changes": [
        {
          "file": "pylint/checkers/strings.py",
          "type": "function",
          "name": "process_module",
          "class_name": "StringConstantChecker",
          "code": "def process_module(self, module):\n        self._unicode_literals = \"unicode_literals\" in module.future_imports"
        },
        {
          "file": "pylint/checkers/strings.py",
          "type": "function",
          "name": "check_for_concatenated_strings",
          "class_name": "StringConstantChecker",
          "code": "def check_for_concatenated_strings(self, iterable_node, iterable_type):\n        for elt in iterable_node.elts:\n            if isinstance(elt, Const) and elt.pytype() in _AST_NODE_STR_TYPES:\n                if elt.col_offset < 0:\n                    # This can happen in case of escaped newlines\n                    continue\n                matching_token, next_token = self.string_tokens[\n                    (elt.lineno, elt.col_offset)\n                ]\n                if matching_token != elt.value and next_token is not None:\n                    next_token_type, next_token_pos = next_token[0], next_token[2]\n                    # We do not warn if string concatenation happens over a newline\n                    if (\n                        next_token_type == tokenize.STRING\n                        and next_token_pos[0] == elt.lineno\n                    ):\n                        self.add_message(\n                            \"implicit-str-concat-in-sequence\",\n                            line=elt.lineno,\n                            args=(iterable_type,),\n                        )"
        }
      ]
    },
    {
      "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                )"
        }
      ]
    },
    {
      "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": 4202,
      "pr_title": "Fix tox -e pylint and pre commit",
      "pr_body": "## Description\r\n\r\nThe ``pre-commit run pylint`` command was not exactly the same than ``tox -e pylint``\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\n\r\nCloses #4201 \r\n",
      "issue_id": 4201,
      "issue_title": "tox -e pylint failing",
      "issue_body": "Commit 6721cd1c causes a failure in the `pylint` testenv under `tox`.\r\n\r\n### Steps to reproduce\r\n\r\n1. Check out commit 6721cd1c\r\n2. `tox -e pylint` (or `tox`)\r\n\r\n### Current behavior\r\n```\r\n************* Module pylint.checkers.deprecated\r\npylint/checkers/deprecated.py:55:4: W9012: Missing return type documentation (missing-return-type-doc)\r\npylint/checkers/deprecated.py:64:4: W9012: Missing return type documentation (missing-return-type-doc)\r\n\r\n--------------------------------------------------------------------\r\nYour code has been rated at 10.00/10 (previous run: 10.00/10, -0.00)\r\n\r\nERROR: InvocationError for command /home/kolbus/projects/pylint/.tox/pylint/bin/pylint -rn --rcfile=/home/kolbus/projects/pylint/pylintrc --load-plugins=pylint.extensions.docparams, pylint.extensions.mccabe pylint tests/message tests/checkers tests/extensions tests/utils tests/acceptance tests/conftest.py tests/test_config.py tests/test_func.py tests/test_functional.py tests/test_import_graph.py tests/test_pragma_parser.py tests/test_pylint_runners.py tests/test_regr.py tests/test_self.py tests/unittest_config.py tests/lint tests/unittest_pyreverse_diadefs.py tests/unittest_pyreverse_inspector.py tests/unittest_pyreverse_writer.py tests/unittest_reporters_json.py tests/unittest_reporting.py (exited with code 4)\r\n__________________________________________________________________________________________________________ summary __________________________________________________________________________________________________________\r\nERROR:   pylint: commands failed\r\n```\r\n\r\n### Expected behavior\r\n\r\nNo errors\r\n",
      "issue_closed_at": "2021-03-07T09:39:07Z",
      "base_commit": "6721cd1cf2da0294124b75d382a042e23ec27d47",
      "changes": [
        {
          "file": "examples/deprecation_checker.py",
          "type": "class",
          "name": "DeprecationChecker",
          "code": "class DeprecationChecker(DeprecatedMixin, BaseChecker):\n    \"\"\"Class implementing deprecation checker.\"\"\"\n\n    # DeprecationMixin class is Mixin class implementing logic for searching deprecated methods and functions.\n    # The list of deprecated methods/functions is defined by implementing class via deprecated_methods callback.\n    # DeprecatedMixin class is overriding attributes of BaseChecker hence must be specified *before* BaseChecker\n    # in list of base classes.\n\n    __implements__ = (IAstroidChecker,)\n    # The name defines a custom section of the config for this checker.\n    name = \"deprecated\"\n\n    def deprecated_methods(self):\n        \"\"\"Callback method called by DeprecatedMixin for every method/function found in the code.\n\n        Returns:\n            collections.abc.Container of deprecated function/method names.\n        \"\"\"\n        return {\"mymodule.deprecated_function\", \"mymodule.MyClass.deprecated_method\"}\n\n    def deprecated_arguments(self, method: str):\n        \"\"\"Callback returning the deprecated arguments of method/function.\n\n        Returns:\n            collections.abc.Iterable in form:\n                ((POSITION1, PARAM1), (POSITION2: PARAM2) ...)\n            where\n                * POSITIONX - position of deprecated argument PARAMX in function definition.\n                  If argument is keyword-only, POSITIONX should be None.\n                * PARAMX - name of the deprecated argument.\n        \"\"\"\n        if method == \"mymodule.myfunction\":\n            # myfunction() has two deprecated arguments:\n            # * deprecated_arg1 defined at 2nd position and\n            # * deprecated_arg2 defined at 5th position.\n            return ((2, \"deprecated_arg1\"), (5, \"deprecated_arg2\"))\n        if method == \"mymodule.MyClass.mymethod\":\n            # mymethod() has two deprecated arguments:\n            # * deprecated1 defined at 2nd position and\n            # * deprecated2 defined at 4th position.\n            return ((2, \"deprecated1\"), (4, \"deprecated2\"))\n        return ()"
        },
        {
          "file": "examples/deprecation_checker.py",
          "type": "function",
          "name": "deprecated_methods",
          "class_name": "DeprecationChecker",
          "code": "def deprecated_methods(self):\n        \"\"\"Callback method called by DeprecatedMixin for every method/function found in the code.\n\n        Returns:\n            collections.abc.Container of deprecated function/method names.\n        \"\"\"\n        return {\"mymodule.deprecated_function\", \"mymodule.MyClass.deprecated_method\"}"
        },
        {
          "file": "pylint/checkers/deprecated.py",
          "type": "line",
          "name": "line 2",
          "code": "# For details: https://github.com/PyCQA/pylint/blob/master/COPYING\n\n\"\"\"Checker mixin for deprecated functionality.\"\"\"\n\nfrom itertools import chain\nfrom typing import Any\n"
        },
        {
          "file": "pylint/checkers/deprecated.py",
          "type": "function",
          "name": "deprecated_methods",
          "class_name": "DeprecatedMixin",
          "code": "def deprecated_methods(self):\n        \"\"\"Callback returning the deprecated methods/functions.\n\n        Returns:\n            collections.abc.Container of deprecated function/method names.\n        \"\"\"\n        # pylint: disable=no-self-use\n        return ()"
        }
      ]
    }
  ]
}