{
    "Selected_candidate": {
        "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- [✓] 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 `doc/whatsnew/<current release.rst>`.\r\n- [✓] 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| ✓ | :bug: Bug fix  |\r\n| ✓ | :sparkles: New feature |\r\n|    | :hammer: Refactoring  |\r\n| ✓ | :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)"
            }
        ]
    },
    "Justification": "Candidate E is the most relevant as it deals with configuration handling in pylint, which is at the core of the CURRENT bug report regarding the module name conflict affecting the parsing and validation process. Both bugs raise issues around how pylint interprets configurations and handles file structures, specifically in the context of modules and imports, which could provide useful insights into fixing the CURRENT bug. In addition, the candidate addresses configuration overrides, which aligns with the CURRENT bug's impact on linting behavior. Thus, the understanding gained from Candidate E regarding configuration loading and potential pitfalls can facilitate a more precise debugging approach for the CURRENT bug."
}