{
  "instance_id": "pylint-dev__pylint-7114",
  "repo": "pylint-dev/pylint",
  "created_at": "2022-07-03T04:36:40Z",
  "problem_statement": "Linting fails if module contains module of the same name\n### Steps to reproduce\r\n\r\nGiven multiple files:\r\n```\r\n.\r\n`-- a/\r\n    |-- a.py\r\n    `-- b.py\r\n```\r\nWhich are all empty, running `pylint a` fails:\r\n\r\n```\r\n$ pylint a\r\n************* Module a\r\na/__init__.py:1:0: F0010: error while code parsing: Unable to load file a/__init__.py:\r\n[Errno 2] No such file or directory: 'a/__init__.py' (parse-error)\r\n$\r\n```\r\n\r\nHowever, if I rename `a.py`, `pylint a` succeeds:\r\n\r\n```\r\n$ mv a/a.py a/c.py\r\n$ pylint a\r\n$\r\n```\r\nAlternatively, I can also `touch a/__init__.py`, but that shouldn't be necessary anymore.\r\n\r\n### Current behavior\r\n\r\nRunning `pylint a` if `a/a.py` is present fails while searching for an `__init__.py` file.\r\n\r\n### Expected behavior\r\n\r\nRunning `pylint a` if `a/a.py` is present should succeed.\r\n\r\n### pylint --version output\r\n\r\nResult of `pylint --version` output:\r\n\r\n```\r\npylint 3.0.0a3\r\nastroid 2.5.6\r\nPython 3.8.5 (default, Jan 27 2021, 15:41:15) \r\n[GCC 9.3.0]\r\n```\r\n\r\n### Additional info\r\n\r\nThis also has some side-effects in module resolution. For example, if I create another file `r.py`:\r\n\r\n```\r\n.\r\n|-- a\r\n|   |-- a.py\r\n|   `-- b.py\r\n`-- r.py\r\n```\r\n\r\nWith the content:\r\n\r\n```\r\nfrom a import b\r\n```\r\n\r\nRunning `pylint -E r` will run fine, but `pylint -E r a` will fail. Not just for module a, but for module r as well.\r\n\r\n```\r\n************* Module r\r\nr.py:1:0: E0611: No name 'b' in module 'a' (no-name-in-module)\r\n************* Module a\r\na/__init__.py:1:0: F0010: error while code parsing: Unable to load file a/__init__.py:\r\n[Errno 2] No such file or directory: 'a/__init__.py' (parse-error)\r\n```\r\n\r\nAgain, if I rename `a.py` to `c.py`, `pylint -E r a` will work perfectly.\n",
  "patch": "diff --git a/pylint/lint/expand_modules.py b/pylint/lint/expand_modules.py\n--- a/pylint/lint/expand_modules.py\n+++ b/pylint/lint/expand_modules.py\n@@ -82,8 +82,10 @@ def expand_modules(\n             continue\n         module_path = get_python_path(something)\n         additional_search_path = [\".\", module_path] + path\n-        if os.path.exists(something):\n-            # this is a file or a directory\n+        if os.path.isfile(something) or os.path.exists(\n+            os.path.join(something, \"__init__.py\")\n+        ):\n+            # this is a file or a directory with an explicit __init__.py\n             try:\n                 modname = \".\".join(\n                     modutils.modpath_from_file(something, path=additional_search_path)\n@@ -103,9 +105,7 @@ def expand_modules(\n                 )\n                 if filepath is None:\n                     continue\n-            except (ImportError, SyntaxError) as ex:\n-                # The SyntaxError is a Python bug and should be\n-                # removed once we move away from imp.find_module: https://bugs.python.org/issue10588\n+            except ImportError as ex:\n                 errors.append({\"key\": \"fatal\", \"mod\": modname, \"ex\": ex})\n                 continue\n         filepath = os.path.normpath(filepath)\n",
  "similar_bug_items": [
    {
      "pr_number": 2060,
      "pr_title": "fix for #2052.",
      "pr_body": "Add checker to warn when imported package is renamed as original one.\r\ne.g below imports\r\nfrom collections import OrderedDict as OrderedDict\r\nimport os.path as path\r\n\r\n### Fixes / new features\r\nFixes #2052 \r\n- \r\n",
      "issue_id": 2052,
      "issue_title": "no complaints about \"from collections import OrderedDict as OrderedDict\"",
      "issue_body": "pylint does not complain if you use `import as` but don't change the name.\r\n\r\n### Steps to reproduce\r\nRun pylint on this file:\r\n```py\r\n\"\"\"frobnicate\"\"\"\r\n\r\nfrom collections import OrderedDict as OrderedDict\r\n\r\n\r\nSOME_DICT = OrderedDict()\r\n```\r\n\r\n### Current behavior\r\n\r\n10/10 rating\r\n\r\n### Expected behavior\r\n\r\nLess than 10/10 rating\r\n\r\n### pylint --version output\r\n\r\n```\r\nNo config file found, using default configuration\r\npylint 1.8.4, \r\nastroid 1.6.3\r\nPython 3.6.3 (default, Oct  3 2017, 21:45:48) \r\n[GCC 7.2.0]\r\n```",
      "issue_closed_at": "2018-05-07T08:40:13Z",
      "base_commit": "e090ea0acfa54c70b78d6345be1d32a8ddd36d74",
      "changes": [
        {
          "file": "pylint/checkers/imports.py",
          "type": "function",
          "name": "_make_graph",
          "class_name": null,
          "code": "def _make_graph(filename, dep_info, sect, gtype):\n    \"\"\"generate a dependencies graph and add some information about it in the\n    report's section\n    \"\"\"\n    _dependencies_graph(filename, dep_info)\n    sect.append(Paragraph('%simports graph has been written to %s'\n                          % (gtype, filename)))"
        },
        {
          "file": "pylint/checkers/imports.py",
          "type": "function",
          "name": "close",
          "class_name": "ImportsChecker",
          "code": "def close(self):\n        \"\"\"called before visiting project (i.e set of modules)\"\"\"\n        if self.linter.is_message_enabled('cyclic-import'):\n            graph = self._import_graph_without_ignored_edges()\n            vertices = list(graph)\n            for cycle in get_cycles(graph, vertices=vertices):\n                self.add_message('cyclic-import', args=' -> '.join(cycle))"
        },
        {
          "file": "pylint/checkers/imports.py",
          "type": "function",
          "name": "visit_importfrom",
          "class_name": "ImportsChecker",
          "code": "def visit_importfrom(self, node):\n        \"\"\"triggered when a from statement is seen\"\"\"\n        basename = node.modname\n        imported_module = self._get_imported_module(node, basename)\n\n        self._check_misplaced_future(node)\n        self._check_deprecated_module(node, basename)\n        self._check_wildcard_imports(node, imported_module)\n        self._check_same_line_imports(node)\n        self._check_reimport(node, basename=basename, level=node.level)\n\n        if isinstance(node.parent, astroid.Module):\n            # Allow imports nested\n            self._check_position(node)\n        if isinstance(node.scope(), astroid.Module):\n            self._record_import(node, imported_module)\n        if imported_module is None:\n            return\n        modnode = node.root()\n        self._check_relative_import(modnode, node, imported_module, basename)\n\n        for name, _ in node.names:\n            if name != '*':\n                self._add_imported_module(node, '%s.%s' % (imported_module.name, name))"
        },
        {
          "file": "pylint/checkers/imports.py",
          "type": "function",
          "name": "_check_deprecated_module",
          "class_name": "ImportsChecker",
          "code": "def _check_deprecated_module(self, node, mod_path):\n        \"\"\"check if the module is deprecated\"\"\"\n        for mod_name in self.config.deprecated_modules:\n            if mod_path == mod_name or mod_path.startswith(mod_name + '.'):\n                self.add_message('deprecated-module', node=node, args=mod_path)"
        }
      ]
    },
    {
      "pr_number": 6351,
      "pr_title": "Add exception handling for broken config files",
      "pr_body": "Extends the error handling of bad config files, to catch even more errors.\r\n\r\n## Type of Changes\r\n\r\n|     | Type                   |\r\n| --- | ---------------------- |\r\n| \u2713   | :bug: Bug fix          |\r\n\r\n## Description\r\n\r\nCloses #5667\r\n",
      "issue_id": 5667,
      "issue_title": "Add a flag to disable config parsing",
      "issue_body": "### Current problem\r\n\r\nHere's a sample broken `.pylintrc`, placed in the root folder:\r\n\r\n```ini\r\n[pylint]\r\ntest = '%A'\r\n```\r\n\r\nThis is what happens when I try to use pylint now:\r\n```console\r\n$ pylint myfile.py\r\nTraceback (most recent call last):\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv3/bin/pylint\", line 8, in <module>\r\n    sys.exit(run_pylint())\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv3/lib/python3.10/site-packages/pylint/__init__.py\", line 24, in run_pylint\r\n    PylintRun(sys.argv[1:])\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv3/lib/python3.10/site-packages/pylint/lint/run.py\", line 324, in __init__\r\n    linter.load_config_file()\r\n  File \"/Users/tusharsadhwani/code/marvin-python/venv3/lib/python3.10/site-packages/pylint/config/option_manager_mixin.py\", line 319, in load_config_file\r\n    for option, value in parser.items(section):\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 860, in items\r\n    return [(option, value_getter(option)) for option in orig_keys]\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 860, in <listcomp>\r\n    return [(option, value_getter(option)) for option in orig_keys]\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 856, in <lambda>\r\n    value_getter = lambda option: self._interpolation.before_get(self,\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 395, in before_get\r\n    self._interpolate_some(parser, option, L, value, section, defaults, 1)\r\n  File \"/usr/local/Cellar/python@3.10/3.10.1/Frameworks/Python.framework/Versions/3.10/lib/python3.10/configparser.py\", line 442, in _interpolate_some\r\n    raise InterpolationSyntaxError(\r\nconfigparser.InterpolationSyntaxError: '%' must be followed by '%' or '(', found: \"%A'\"\r\n\r\n$ pylint --help   \r\nTraceback (most recent call last):\r\n[...]\r\nconfigparser.InterpolationSyntaxError: '%' must be followed by '%' or '(', found: \"%A'\"\r\n\r\n$ pylint       \r\nTraceback (most recent call last):\r\n[...]\r\nconfigparser.InterpolationSyntaxError: '%' must be followed by '%' or '(', found: \"%A'\"\r\n```\r\n\r\nIs it possible to add a flag that disables config parsing entirely, so that the pylint isn't completely broken when this happens? `--ignore-config-file` for example?\r\n\r\nP.S. this is a feature I want, and I'd be willing to contribute it.",
      "issue_closed_at": "2022-04-15T20:15:52Z",
      "base_commit": "98bb5bf8ebf15fbce960533e5983a71351a1fe3f",
      "changes": [
        {
          "file": "pylint/config/config_file_parser.py",
          "type": "function",
          "name": "parse_config_file",
          "class_name": "_ConfigurationFileParser",
          "code": "def parse_config_file(\n        self, file_path: Path | None\n    ) -> tuple[dict[str, str], list[str]]:\n        \"\"\"Parse a config file and return str-str pairs.\"\"\"\n        if file_path is None:\n            if self.verbose_mode:\n                print(\n                    \"No config file found, using default configuration\", file=sys.stderr\n                )\n            return {}, []\n\n        file_path = Path(os.path.expandvars(file_path)).expanduser()\n        if not file_path.exists():\n            raise OSError(f\"The config file {file_path} doesn't exist!\")\n\n        if self.verbose_mode:\n            print(f\"Using config file {file_path}\", file=sys.stderr)\n\n        if file_path.suffix == \".toml\":\n            return self._parse_toml_file(file_path)\n        return self._parse_ini_file(file_path)"
        }
      ]
    },
    {
      "pr_number": 4703,
      "pr_title": "Fix cyclic import with TYPE_CHECKING",
      "pr_body": "## Description\r\nImports guarded by `typing.TYPE_CHECKING` should be added to the `excluded_edges` map so that they don't result in `cyclic-import` errors.\r\n\r\n## Type of Changes\r\n\r\n|     | Type                   |\r\n| --- | ---------------------- |\r\n| \u2713   | :bug: Bug fix          |\r\n\r\n## Related Issue\r\nCloses #3525",
      "issue_id": 3525,
      "issue_title": "Imports within TYPE_CHECKING can induce \"cyclic-import\"",
      "issue_body": "Suppose you have two modules: a.py and b.py.  `a` imports `b`, but `b` needs `a` in order to do type annotations, then this is usually done by importing `a` within a `TYPE_CHECKING` block.  Unfortunately, this causes pylint to report `cyclic-import`.\r\n\r\nPossibly related to https://github.com/PyCQA/pylint/issues/3285",
      "issue_closed_at": "2021-07-19T19:57:13Z",
      "base_commit": "3a6f08e4a1155e5098c3bec2d779cb3e654a1b11",
      "changes": [
        {
          "file": "pylint/checkers/imports.py",
          "type": "function",
          "name": "_get_imported_module",
          "class_name": "ImportsChecker",
          "code": "def _get_imported_module(self, importnode, modname):\n        try:\n            return importnode.do_import_module(modname)\n        except astroid.TooManyLevelsError:\n            if _ignore_import_failure(importnode, modname, self._ignored_modules):\n                return None\n            self.add_message(\"relative-beyond-top-level\", node=importnode)\n        except astroid.AstroidSyntaxError as exc:\n            message = \"Cannot import {!r} due to syntax error {!r}\".format(\n                modname, str(exc.error)  # pylint: disable=no-member; false positive\n            )\n            self.add_message(\"syntax-error\", line=importnode.lineno, args=message)\n\n        except astroid.AstroidBuildingException:\n            if not self.linter.is_message_enabled(\"import-error\"):\n                return None\n            if _ignore_import_failure(importnode, modname, self._ignored_modules):\n                return None\n            if not self.config.analyse_fallback_blocks and is_from_fallback_block(\n                importnode\n            ):\n                return None\n\n            dotted_modname = get_import_name(importnode, modname)\n            self.add_message(\"import-error\", args=repr(dotted_modname), node=importnode)\n        return None"
        },
        {
          "file": "pylint/checkers/imports.py",
          "type": "function",
          "name": "_add_imported_module",
          "class_name": "ImportsChecker",
          "code": "def _add_imported_module(self, node, importedmodname):\n        \"\"\"notify an imported module, used to analyze dependencies\"\"\"\n        module_file = node.root().file\n        context_name = node.root().name\n        base = os.path.splitext(os.path.basename(module_file))[0]\n\n        try:\n            importedmodname = astroid.modutils.get_module_part(\n                importedmodname, module_file\n            )\n        except ImportError:\n            pass\n\n        if context_name == importedmodname:\n            self.add_message(\"import-self\", node=node)\n\n        elif not astroid.modutils.is_standard_module(importedmodname):\n            # if this is not a package __init__ module\n            if base != \"__init__\" and context_name not in self._module_pkg:\n                # record the module's parent, or the module itself if this is\n                # a top level module, as the package it belongs to\n                self._module_pkg[context_name] = context_name.rsplit(\".\", 1)[0]\n\n            # handle dependencies\n            importedmodnames = self.stats[\"dependencies\"].setdefault(\n                importedmodname, set()\n            )\n            if context_name not in importedmodnames:\n                importedmodnames.add(context_name)\n\n            # update import graph\n            self.import_graph[context_name].add(importedmodname)\n            if not self.linter.is_message_enabled(\"cyclic-import\", line=node.lineno):\n                self._excluded_edges[context_name].add(importedmodname)"
        },
        {
          "file": "pylint/checkers/imports.py",
          "type": "function",
          "name": "_add_imported_module",
          "class_name": "ImportsChecker",
          "code": "def _add_imported_module(self, node, importedmodname):\n        \"\"\"notify an imported module, used to analyze dependencies\"\"\"\n        module_file = node.root().file\n        context_name = node.root().name\n        base = os.path.splitext(os.path.basename(module_file))[0]\n\n        try:\n            importedmodname = astroid.modutils.get_module_part(\n                importedmodname, module_file\n            )\n        except ImportError:\n            pass\n\n        if context_name == importedmodname:\n            self.add_message(\"import-self\", node=node)\n\n        elif not astroid.modutils.is_standard_module(importedmodname):\n            # if this is not a package __init__ module\n            if base != \"__init__\" and context_name not in self._module_pkg:\n                # record the module's parent, or the module itself if this is\n                # a top level module, as the package it belongs to\n                self._module_pkg[context_name] = context_name.rsplit(\".\", 1)[0]\n\n            # handle dependencies\n            importedmodnames = self.stats[\"dependencies\"].setdefault(\n                importedmodname, set()\n            )\n            if context_name not in importedmodnames:\n                importedmodnames.add(context_name)\n\n            # update import graph\n            self.import_graph[context_name].add(importedmodname)\n            if not self.linter.is_message_enabled(\"cyclic-import\", line=node.lineno):\n                self._excluded_edges[context_name].add(importedmodname)"
        }
      ]
    },
    {
      "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": 2644,
      "pr_title": "Load plugin configuration hook",
      "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- [\u2713] Add yourself to CONTRIBUTORS if you are a new contributor.\r\n- [\u2713] Add a ChangeLog entry describing what your PR does.\r\n- [\u2713] 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- [\u2713] Write a good description on what the PR does.\r\n\r\n## Description\r\n\r\nAdded ``load_configuration()`` hook for plugins.\r\n    \r\nNew optional hook for plugins is added: ``load_configuration()``. This hook is called by pylint after configuration is loaded to prevent against overwriting plugin specific configuration via user-based configuration. Plugins are supposed to adjust configuration e.g. by extending ``good_names`` or extending ``black_list``, etc...\r\n\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| \u2713 | :sparkles: New feature |\r\n|    | :hammer: Refactoring  |\r\n| \u2713 | :scroll: Docs |\r\n\r\n## Related Issue\r\nCloses #2635  \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 ####\r\n-->\r\n",
      "issue_id": 2635,
      "issue_title": "pylint custom configuration overrides plugin defaults settings.",
      "issue_body": "pylint overrides default configuration set by plugins using parameters set by rc file or command line arguments:\r\nhttps://github.com/PyCQA/pylint/blob/fcc01516ae176ad3fdedc4497328105f3314e376/pylint/lint.py#L1568-L1571\r\n\r\npylint-django extends good-names in `register()` method [1]:\r\n```python\r\n    name_checker = get_checker(linter, NameChecker)\r\n    name_checker.config.good_names += ('qs', 'urlpatterns', 'register', 'app_name', 'handler500')\r\n    # we don't care about South migrations\r\n    linter.config.black_list += ('migrations', 'south_migrations')\r\n```\r\n\r\nCurrently plugin is not able to avoid overriding of configuration. pylint should provide a hook for plugins which is executed *after* configuration is set. Currently there are multiple open issues in pylint-django project affected by this defect: \r\n* https://github.com/PyCQA/pylint-django/issues/181 \r\n* https://github.com/PyCQA/pylint-django/issues/167\r\n\r\n### Steps to reproduce\r\n1. Install django and pylint-django using pip\r\n2. Create empty project: `django-admin startproject myproject`\r\n3. Execute pylint with pylint-django plugin and with default configuration:\r\n```\r\npylint --load-plugins=pylint_django myproject/myproject/urls.py\r\n\r\n------------------------------------\r\nYour code has been rated at 10.00/10\r\n```\r\n4. Execute pylint with django-pylint and custom configuration:\r\n```\r\npylint --load-plugins=pylint_django --good-names=foo,bar  myproject/myproject/urls.py\r\n\r\n************* Module myproject.urls\r\nmyproject/myproject/urls.py:19:0: C0103: Constant name \"urlpatterns\" doesn't conform to UPPER_CASE naming style (invalid-name)\r\n\r\n-------------------------------------------------------------------\r\nYour code has been rated at 6.67/10 (previous run: 10.00/10, -3.33)\r\n```\r\n### Current behavior\r\nCurrently, configuration provided by pylint-django is overriden by custom configuration (see [1] pylint-django configuration)\r\n\r\n### Expected behavior\r\nCustom configuration should not override pylint-django customization\r\n\r\n### pylint --version output\r\npylint 2.1.1\r\nastroid 2.0.4\r\nPython 3.6.4 (default, Jan  7 2018, 15:53:53)\r\n[GCC 6.4.0]\r\n\r\n[1] https://github.com/PyCQA/pylint-django/blob/c5b5bfdef66453575074b36018ee716411bcb0a4/pylint_django/plugin.py#L18-L22\r\n",
      "issue_closed_at": "2018-12-20T08:11:51Z",
      "base_commit": "75cecdb1b88cc759223e83fd325aeafd09fec37e",
      "changes": [
        {
          "file": "pylint/lint.py",
          "type": "function",
          "name": "_run_linter",
          "class_name": "ChildLinter",
          "code": "def _run_linter(self, file_or_module):\n            linter = PyLinter()\n\n            # Register standard checkers.\n            linter.load_default_plugins()\n            # Load command line plugins.\n            if self._plugins:\n                linter.load_plugin_modules(self._plugins)\n\n            linter.load_configuration_from_config(self._config)\n            linter.set_reporter(reporters.CollectingReporter())\n\n            # Enable the Python 3 checker mode. This option is\n            # passed down from the parent linter up to here, since\n            # the Python 3 porting flag belongs to the Run class,\n            # instead of the Linter class.\n            if self._python3_porting_mode:\n                linter.python3_porting_mode()\n\n            # Run the checks.\n            linter.check(file_or_module)\n\n            msgs = [_get_new_args(m) for m in linter.reporter.messages]\n            return (\n                file_or_module,\n                linter.file_state.base_name,\n                linter.current_name,\n                msgs,\n                linter.stats,\n                linter.msg_status,\n            )"
        },
        {
          "file": "pylint/lint.py",
          "type": "function",
          "name": "load_plugin_modules",
          "class_name": "PyLinter",
          "code": "def load_plugin_modules(self, modnames):\n        \"\"\"take a list of module names which are pylint plugins and load\n        and register them\n        \"\"\"\n        for modname in modnames:\n            if modname in self._dynamic_plugins:\n                continue\n            self._dynamic_plugins.add(modname)\n            module = modutils.load_module_from_name(modname)\n            module.register(self)"
        },
        {
          "file": "pylint/lint.py",
          "type": "function",
          "name": "__init__",
          "class_name": "Run",
          "code": "def __init__(self, args, reporter=None, do_exit=True):\n        self._rcfile = None\n        self._plugins = []\n        self.verbose = None\n        try:\n            preprocess_options(\n                args,\n                {\n                    # option: (callback, takearg)\n                    \"init-hook\": (cb_init_hook, True),\n                    \"rcfile\": (self.cb_set_rcfile, True),\n                    \"load-plugins\": (self.cb_add_plugins, True),\n                    \"verbose\": (self.cb_verbose_mode, False),\n                },\n            )\n        except ArgumentPreprocessingError as ex:\n            print(ex, file=sys.stderr)\n            sys.exit(32)\n\n        self.linter = linter = self.LinterClass(\n            (\n                (\n                    \"rcfile\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": lambda *args: 1,\n                        \"type\": \"string\",\n                        \"metavar\": \"<file>\",\n                        \"help\": \"Specify a configuration file.\",\n                    },\n                ),\n                (\n                    \"init-hook\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": lambda *args: 1,\n                        \"type\": \"string\",\n                        \"metavar\": \"<code>\",\n                        \"level\": 1,\n                        \"help\": \"Python code to execute, usually for sys.path \"\n                        \"manipulation such as pygtk.require().\",\n                    },\n                ),\n                (\n                    \"help-msg\",\n                    {\n                        \"action\": \"callback\",\n                        \"type\": \"string\",\n                        \"metavar\": \"<msg-id>\",\n                        \"callback\": self.cb_help_message,\n                        \"group\": \"Commands\",\n                        \"help\": \"Display a help message for the given message id and \"\n                        \"exit. The value may be a comma separated list of message ids.\",\n                    },\n                ),\n                (\n                    \"list-msgs\",\n                    {\n                        \"action\": \"callback\",\n                        \"metavar\": \"<msg-id>\",\n                        \"callback\": self.cb_list_messages,\n                        \"group\": \"Commands\",\n                        \"level\": 1,\n                        \"help\": \"Generate pylint's messages.\",\n                    },\n                ),\n                (\n                    \"list-conf-levels\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": cb_list_confidence_levels,\n                        \"group\": \"Commands\",\n                        \"level\": 1,\n                        \"help\": \"Generate pylint's confidence levels.\",\n                    },\n                ),\n                (\n                    \"full-documentation\",\n                    {\n                        \"action\": \"callback\",\n                        \"metavar\": \"<msg-id>\",\n                        \"callback\": self.cb_full_documentation,\n                        \"group\": \"Commands\",\n                        \"level\": 1,\n                        \"help\": \"Generate pylint's full documentation.\",\n                    },\n                ),\n                (\n                    \"generate-rcfile\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_generate_config,\n                        \"group\": \"Commands\",\n                        \"help\": \"Generate a sample configuration file according to \"\n                        \"the current configuration. You can put other options \"\n                        \"before this one to get them in the generated \"\n                        \"configuration.\",\n                    },\n                ),\n                (\n                    \"generate-man\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_generate_manpage,\n                        \"group\": \"Commands\",\n                        \"help\": \"Generate pylint's man page.\",\n                        \"hide\": True,\n                    },\n                ),\n                (\n                    \"errors-only\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_error_mode,\n                        \"short\": \"E\",\n                        \"help\": \"In error mode, checkers without error messages are \"\n                        \"disabled and for others, only the ERROR messages are \"\n                        \"displayed, and no reports are done by default.\",\n                    },\n                ),\n                (\n                    \"py3k\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_python3_porting_mode,\n                        \"help\": \"In Python 3 porting mode, all checkers will be \"\n                        \"disabled and only messages emitted by the porting \"\n                        \"checker will be displayed.\",\n                    },\n                ),\n                (\n                    \"verbose\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_verbose_mode,\n                        \"short\": \"v\",\n                        \"help\": \"In verbose mode, extra non-checker-related info \"\n                        \"will be displayed.\",\n                    },\n                ),\n            ),\n            option_groups=self.option_groups,\n            pylintrc=self._rcfile,\n        )\n        # register standard checkers\n        linter.load_default_plugins()\n        # load command line plugins\n        linter.load_plugin_modules(self._plugins)\n        # add some help section\n        linter.add_help_section(\"Environment variables\", config.ENV_HELP, level=1)\n        # pylint: disable=bad-continuation\n        linter.add_help_section(\n            \"Output\",\n            \"Using the default text output, the message format is :                          \\n\"\n            \"                                                                                \\n\"\n            \"        MESSAGE_TYPE: LINE_NUM:[OBJECT:] MESSAGE                                \\n\"\n            \"                                                                                \\n\"\n            \"There are 5 kind of message types :                                             \\n\"\n            \"    * (C) convention, for programming standard violation                        \\n\"\n            \"    * (R) refactor, for bad code smell                                          \\n\"\n            \"    * (W) warning, for python specific problems                                 \\n\"\n            \"    * (E) error, for probable bugs in the code                                  \\n\"\n            \"    * (F) fatal, if an error occurred which prevented pylint from doing further\\n\"\n            \"processing.\\n\",\n            level=1,\n        )\n        linter.add_help_section(\n            \"Output status code\",\n            \"Pylint should leave with following status code:                                 \\n\"\n            \"    * 0 if everything went fine                                                 \\n\"\n            \"    * 1 if a fatal message was issued                                           \\n\"\n            \"    * 2 if an error message was issued                                          \\n\"\n            \"    * 4 if a warning message was issued                                         \\n\"\n            \"    * 8 if a refactor message was issued                                        \\n\"\n            \"    * 16 if a convention message was issued                                     \\n\"\n            \"    * 32 on usage error                                                         \\n\"\n            \"                                                                                \\n\"\n            \"status 1 to 16 will be bit-ORed so you can know which different categories has\\n\"\n            \"been issued by analysing pylint output status code\\n\",\n            level=1,\n        )\n        # read configuration\n        linter.disable(\"I\")\n        linter.enable(\"c-extension-no-member\")\n        linter.read_config_file(verbose=self.verbose)\n        config_parser = linter.cfgfile_parser\n        # run init hook, if present, before loading plugins\n        if config_parser.has_option(\"MASTER\", \"init-hook\"):\n            cb_init_hook(\n                \"init-hook\", utils._unquote(config_parser.get(\"MASTER\", \"init-hook\"))\n            )\n        # is there some additional plugins in the file configuration, in\n        if config_parser.has_option(\"MASTER\", \"load-plugins\"):\n            plugins = utils._splitstrip(config_parser.get(\"MASTER\", \"load-plugins\"))\n            linter.load_plugin_modules(plugins)\n        # now we can load file config and command line, plugins (which can\n        # provide options) have been registered\n        linter.load_config_file()\n        if reporter:\n            # if a custom reporter is provided as argument, it may be overridden\n            # by file parameters, so re-set it here, but before command line\n            # parsing so it's still overrideable by command line option\n            linter.set_reporter(reporter)\n        try:\n            args = linter.load_command_line_configuration(args)\n        except SystemExit as exc:\n            if exc.code == 2:  # bad options\n                exc.code = 32\n            raise\n        if not args:\n            print(linter.help())\n            sys.exit(32)\n\n        if linter.config.jobs < 0:\n            print(\n                \"Jobs number (%d) should be greater than or equal to 0\"\n                % linter.config.jobs,\n                file=sys.stderr,\n            )\n            sys.exit(32)\n        if linter.config.jobs > 1 or linter.config.jobs == 0:\n            if multiprocessing is None:\n                print(\n                    \"Multiprocessing library is missing, \" \"fallback to single process\",\n                    file=sys.stderr,\n                )\n                linter.set_option(\"jobs\", 1)\n            else:\n                if linter.config.jobs == 0:\n                    linter.config.jobs = _cpu_count()\n\n        # insert current working directory to the python path to have a correct\n        # behaviour\n        with fix_import_path(args):\n            linter.check(args)\n            linter.generate_reports()\n        if do_exit:\n            if linter.config.exit_zero:\n                sys.exit(0)\n            else:\n                sys.exit(self.linter.msg_status)"
        },
        {
          "file": "pylint/lint.py",
          "type": "function",
          "name": "__init__",
          "class_name": "Run",
          "code": "def __init__(self, args, reporter=None, do_exit=True):\n        self._rcfile = None\n        self._plugins = []\n        self.verbose = None\n        try:\n            preprocess_options(\n                args,\n                {\n                    # option: (callback, takearg)\n                    \"init-hook\": (cb_init_hook, True),\n                    \"rcfile\": (self.cb_set_rcfile, True),\n                    \"load-plugins\": (self.cb_add_plugins, True),\n                    \"verbose\": (self.cb_verbose_mode, False),\n                },\n            )\n        except ArgumentPreprocessingError as ex:\n            print(ex, file=sys.stderr)\n            sys.exit(32)\n\n        self.linter = linter = self.LinterClass(\n            (\n                (\n                    \"rcfile\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": lambda *args: 1,\n                        \"type\": \"string\",\n                        \"metavar\": \"<file>\",\n                        \"help\": \"Specify a configuration file.\",\n                    },\n                ),\n                (\n                    \"init-hook\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": lambda *args: 1,\n                        \"type\": \"string\",\n                        \"metavar\": \"<code>\",\n                        \"level\": 1,\n                        \"help\": \"Python code to execute, usually for sys.path \"\n                        \"manipulation such as pygtk.require().\",\n                    },\n                ),\n                (\n                    \"help-msg\",\n                    {\n                        \"action\": \"callback\",\n                        \"type\": \"string\",\n                        \"metavar\": \"<msg-id>\",\n                        \"callback\": self.cb_help_message,\n                        \"group\": \"Commands\",\n                        \"help\": \"Display a help message for the given message id and \"\n                        \"exit. The value may be a comma separated list of message ids.\",\n                    },\n                ),\n                (\n                    \"list-msgs\",\n                    {\n                        \"action\": \"callback\",\n                        \"metavar\": \"<msg-id>\",\n                        \"callback\": self.cb_list_messages,\n                        \"group\": \"Commands\",\n                        \"level\": 1,\n                        \"help\": \"Generate pylint's messages.\",\n                    },\n                ),\n                (\n                    \"list-conf-levels\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": cb_list_confidence_levels,\n                        \"group\": \"Commands\",\n                        \"level\": 1,\n                        \"help\": \"Generate pylint's confidence levels.\",\n                    },\n                ),\n                (\n                    \"full-documentation\",\n                    {\n                        \"action\": \"callback\",\n                        \"metavar\": \"<msg-id>\",\n                        \"callback\": self.cb_full_documentation,\n                        \"group\": \"Commands\",\n                        \"level\": 1,\n                        \"help\": \"Generate pylint's full documentation.\",\n                    },\n                ),\n                (\n                    \"generate-rcfile\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_generate_config,\n                        \"group\": \"Commands\",\n                        \"help\": \"Generate a sample configuration file according to \"\n                        \"the current configuration. You can put other options \"\n                        \"before this one to get them in the generated \"\n                        \"configuration.\",\n                    },\n                ),\n                (\n                    \"generate-man\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_generate_manpage,\n                        \"group\": \"Commands\",\n                        \"help\": \"Generate pylint's man page.\",\n                        \"hide\": True,\n                    },\n                ),\n                (\n                    \"errors-only\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_error_mode,\n                        \"short\": \"E\",\n                        \"help\": \"In error mode, checkers without error messages are \"\n                        \"disabled and for others, only the ERROR messages are \"\n                        \"displayed, and no reports are done by default.\",\n                    },\n                ),\n                (\n                    \"py3k\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_python3_porting_mode,\n                        \"help\": \"In Python 3 porting mode, all checkers will be \"\n                        \"disabled and only messages emitted by the porting \"\n                        \"checker will be displayed.\",\n                    },\n                ),\n                (\n                    \"verbose\",\n                    {\n                        \"action\": \"callback\",\n                        \"callback\": self.cb_verbose_mode,\n                        \"short\": \"v\",\n                        \"help\": \"In verbose mode, extra non-checker-related info \"\n                        \"will be displayed.\",\n                    },\n                ),\n            ),\n            option_groups=self.option_groups,\n            pylintrc=self._rcfile,\n        )\n        # register standard checkers\n        linter.load_default_plugins()\n        # load command line plugins\n        linter.load_plugin_modules(self._plugins)\n        # add some help section\n        linter.add_help_section(\"Environment variables\", config.ENV_HELP, level=1)\n        # pylint: disable=bad-continuation\n        linter.add_help_section(\n            \"Output\",\n            \"Using the default text output, the message format is :                          \\n\"\n            \"                                                                                \\n\"\n            \"        MESSAGE_TYPE: LINE_NUM:[OBJECT:] MESSAGE                                \\n\"\n            \"                                                                                \\n\"\n            \"There are 5 kind of message types :                                             \\n\"\n            \"    * (C) convention, for programming standard violation                        \\n\"\n            \"    * (R) refactor, for bad code smell                                          \\n\"\n            \"    * (W) warning, for python specific problems                                 \\n\"\n            \"    * (E) error, for probable bugs in the code                                  \\n\"\n            \"    * (F) fatal, if an error occurred which prevented pylint from doing further\\n\"\n            \"processing.\\n\",\n            level=1,\n        )\n        linter.add_help_section(\n            \"Output status code\",\n            \"Pylint should leave with following status code:                                 \\n\"\n            \"    * 0 if everything went fine                                                 \\n\"\n            \"    * 1 if a fatal message was issued                                           \\n\"\n            \"    * 2 if an error message was issued                                          \\n\"\n            \"    * 4 if a warning message was issued                                         \\n\"\n            \"    * 8 if a refactor message was issued                                        \\n\"\n            \"    * 16 if a convention message was issued                                     \\n\"\n            \"    * 32 on usage error                                                         \\n\"\n            \"                                                                                \\n\"\n            \"status 1 to 16 will be bit-ORed so you can know which different categories has\\n\"\n            \"been issued by analysing pylint output status code\\n\",\n            level=1,\n        )\n        # read configuration\n        linter.disable(\"I\")\n        linter.enable(\"c-extension-no-member\")\n        linter.read_config_file(verbose=self.verbose)\n        config_parser = linter.cfgfile_parser\n        # run init hook, if present, before loading plugins\n        if config_parser.has_option(\"MASTER\", \"init-hook\"):\n            cb_init_hook(\n                \"init-hook\", utils._unquote(config_parser.get(\"MASTER\", \"init-hook\"))\n            )\n        # is there some additional plugins in the file configuration, in\n        if config_parser.has_option(\"MASTER\", \"load-plugins\"):\n            plugins = utils._splitstrip(config_parser.get(\"MASTER\", \"load-plugins\"))\n            linter.load_plugin_modules(plugins)\n        # now we can load file config and command line, plugins (which can\n        # provide options) have been registered\n        linter.load_config_file()\n        if reporter:\n            # if a custom reporter is provided as argument, it may be overridden\n            # by file parameters, so re-set it here, but before command line\n            # parsing so it's still overrideable by command line option\n            linter.set_reporter(reporter)\n        try:\n            args = linter.load_command_line_configuration(args)\n        except SystemExit as exc:\n            if exc.code == 2:  # bad options\n                exc.code = 32\n            raise\n        if not args:\n            print(linter.help())\n            sys.exit(32)\n\n        if linter.config.jobs < 0:\n            print(\n                \"Jobs number (%d) should be greater than or equal to 0\"\n                % linter.config.jobs,\n                file=sys.stderr,\n            )\n            sys.exit(32)\n        if linter.config.jobs > 1 or linter.config.jobs == 0:\n            if multiprocessing is None:\n                print(\n                    \"Multiprocessing library is missing, \" \"fallback to single process\",\n                    file=sys.stderr,\n                )\n                linter.set_option(\"jobs\", 1)\n            else:\n                if linter.config.jobs == 0:\n                    linter.config.jobs = _cpu_count()\n\n        # insert current working directory to the python path to have a correct\n        # behaviour\n        with fix_import_path(args):\n            linter.check(args)\n            linter.generate_reports()\n        if do_exit:\n            if linter.config.exit_zero:\n                sys.exit(0)\n            else:\n                sys.exit(self.linter.msg_status)"
        }
      ]
    }
  ]
}