{
  "instance_id": "pylint-dev__pylint-7228",
  "repo": "pylint-dev/pylint",
  "created_at": "2022-07-25T17:19:11Z",
  "problem_statement": "rxg include '\\p{Han}' will throw error\n### Bug description\r\n\r\nconfig rxg in pylintrc with \\p{Han} will throw err\r\n\r\n### Configuration\r\n.pylintrc:\r\n\r\n```ini\r\nfunction-rgx=[\\p{Han}a-z_][\\p{Han}a-z0-9_]{2,30}$\r\n```\r\n\r\n### Command used\r\n\r\n```shell\r\npylint\r\n```\r\n\r\n\r\n### Pylint output\r\n\r\n```shell\r\n(venvtest) tsung-hande-MacBook-Pro:robot_is_comming tsung-han$ pylint\r\nTraceback (most recent call last):\r\n  File \"/Users/tsung-han/PycharmProjects/robot_is_comming/venvtest/bin/pylint\", line 8, in <module>\r\n    sys.exit(run_pylint())\r\n  File \"/Users/tsung-han/PycharmProjects/robot_is_comming/venvtest/lib/python3.9/site-packages/pylint/__init__.py\", line 25, in run_pylint\r\n    PylintRun(argv or sys.argv[1:])\r\n  File \"/Users/tsung-han/PycharmProjects/robot_is_comming/venvtest/lib/python3.9/site-packages/pylint/lint/run.py\", line 161, in __init__\r\n    args = _config_initialization(\r\n  File \"/Users/tsung-han/PycharmProjects/robot_is_comming/venvtest/lib/python3.9/site-packages/pylint/config/config_initialization.py\", line 57, in _config_initialization\r\n    linter._parse_configuration_file(config_args)\r\n  File \"/Users/tsung-han/PycharmProjects/robot_is_comming/venvtest/lib/python3.9/site-packages/pylint/config/arguments_manager.py\", line 244, in _parse_configuration_file\r\n    self.config, parsed_args = self._arg_parser.parse_known_args(\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/argparse.py\", line 1858, in parse_known_args\r\n    namespace, args = self._parse_known_args(args, namespace)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/argparse.py\", line 2067, in _parse_known_args\r\n    start_index = consume_optional(start_index)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/argparse.py\", line 2007, in consume_optional\r\n    take_action(action, args, option_string)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/argparse.py\", line 1919, in take_action\r\n    argument_values = self._get_values(action, argument_strings)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/argparse.py\", line 2450, in _get_values\r\n    value = self._get_value(action, arg_string)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/argparse.py\", line 2483, in _get_value\r\n    result = type_func(arg_string)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/re.py\", line 252, in compile\r\n    return _compile(pattern, flags)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/re.py\", line 304, in _compile\r\n    p = sre_compile.compile(pattern, flags)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/sre_compile.py\", line 788, in compile\r\n    p = sre_parse.parse(p, flags)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/sre_parse.py\", line 955, in parse\r\n    p = _parse_sub(source, state, flags & SRE_FLAG_VERBOSE, 0)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/sre_parse.py\", line 444, in _parse_sub\r\n    itemsappend(_parse(source, state, verbose, nested + 1,\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/sre_parse.py\", line 555, in _parse\r\n    code1 = _class_escape(source, this)\r\n  File \"/usr/local/Cellar/python@3.9/3.9.13_1/Frameworks/Python.framework/Versions/3.9/lib/python3.9/sre_parse.py\", line 350, in _class_escape\r\n    raise source.error('bad escape %s' % escape, len(escape))\r\nre.error: bad escape \\p at position 1\r\n```\r\n\r\n### Expected behavior\r\n\r\nnot throw error\r\n\r\n### Pylint version\r\n\r\n```shell\r\npylint 2.14.4\r\nastroid 2.11.7\r\nPython 3.9.13 (main, May 24 2022, 21:28:44) \r\n[Clang 13.0.0 (clang-1300.0.29.30)]\r\n```\r\n\r\n\r\n### OS / Environment\r\n\r\nmacOS 11.6.7\r\n\n",
  "patch": "diff --git a/pylint/config/argument.py b/pylint/config/argument.py\n--- a/pylint/config/argument.py\n+++ b/pylint/config/argument.py\n@@ -99,11 +99,20 @@ def _py_version_transformer(value: str) -> tuple[int, ...]:\n     return version\n \n \n+def _regex_transformer(value: str) -> Pattern[str]:\n+    \"\"\"Return `re.compile(value)`.\"\"\"\n+    try:\n+        return re.compile(value)\n+    except re.error as e:\n+        msg = f\"Error in provided regular expression: {value} beginning at index {e.pos}: {e.msg}\"\n+        raise argparse.ArgumentTypeError(msg)\n+\n+\n def _regexp_csv_transfomer(value: str) -> Sequence[Pattern[str]]:\n     \"\"\"Transforms a comma separated list of regular expressions.\"\"\"\n     patterns: list[Pattern[str]] = []\n     for pattern in _csv_transformer(value):\n-        patterns.append(re.compile(pattern))\n+        patterns.append(_regex_transformer(pattern))\n     return patterns\n \n \n@@ -130,7 +139,7 @@ def _regexp_paths_csv_transfomer(value: str) -> Sequence[Pattern[str]]:\n     \"non_empty_string\": _non_empty_string_transformer,\n     \"path\": _path_transformer,\n     \"py_version\": _py_version_transformer,\n-    \"regexp\": re.compile,\n+    \"regexp\": _regex_transformer,\n     \"regexp_csv\": _regexp_csv_transfomer,\n     \"regexp_paths_csv\": _regexp_paths_csv_transfomer,\n     \"string\": pylint_utils._unquote,\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": 4446,
      "pr_title": "Fix issues with Python 3.6.0",
      "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- [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## Description\r\n\r\nThis PR fixes compatibility with Python 3.6.0, mainly due to `typing`:\r\n\r\n- `typing.Counter` is added in Python 3.6.1, so make the import conditional on TYPE_CHECKING.\r\n- `typing.NoReturn` is added in Python 3.6.2, split the inconsistent_returns tests that need this to it's own file.\r\n- `black`, `pre-commit` and `pyupgrade` require Python 3.6.1 (or later), add `python_full_version` markers so `tox -e py36` works\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\nCloses #4412 ",
      "issue_id": 4412,
      "issue_title": "Problem importing module strings.py: cannot import name 'Counter'",
      "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\nOn python 3.6.0, usage of pylint 2.8.2 results in an import warning to stdout.  \r\n\r\nN.B. `typing.Counter` was added in python 3.6.1, so this import probably needs to be wrapped in a `TYPE_CHECKING` guard.\r\n\r\n(Yes, I know 3.6.0 is an old patch release. Still have it on some CI servers for unfortunate reasons.)\r\n\r\n### Steps to reproduce\r\n\r\nGiven that this warning seems to be coming from `register_plugins()` in utils.py, it is probably reproducible with any input file.\r\n\r\n### Current behavior\r\n\r\nPrinted message: `Problem importing module strings.py: cannot import name 'Counter'`\r\n\r\n### Expected behavior\r\n\r\nNo warning, and `strings` checker works as expected.\r\n\r\n### pylint --version output\r\n\r\nResult of `pylint --version` output:\r\n\r\n```\r\npylint 2.8.2\r\nastroid 2.5.6\r\nPython 3.6.0 (default, Jan 23 2017, 20:01:14) [MSC v.1900 64 bit (AMD64)]\r\n```\r\n",
      "issue_closed_at": "2021-05-12T11:40:15Z",
      "base_commit": "9b268ecedb1c3dbfd24f5edb0c891266f988b87a",
      "changes": [
        {
          "file": "pylint/checkers/strings.py",
          "type": "line",
          "name": "line 38",
          "code": "import numbers\nimport re\nimport tokenize\nfrom typing import Counter, Iterable\n\nimport astroid\n"
        },
        {
          "file": "pylint/checkers/strings.py",
          "type": "line",
          "name": "line 46",
          "code": "from pylint.checkers.utils import check_messages\nfrom pylint.interfaces import IAstroidChecker, IRawChecker, ITokenChecker\n\n_AST_NODE_STR_TYPES = (\"__builtin__.unicode\", \"__builtin__.str\", \"builtins.str\")\n# Prefixes for both strings and bytes literals per\n# https://docs.python.org/3/reference/lexical_analysis.html#string-and-bytes-literals"
        },
        {
          "file": "pylint/checkers/strings.py",
          "type": "function",
          "name": "check_for_consistent_string_delimiters",
          "class_name": "StringConstantChecker",
          "code": "def check_for_consistent_string_delimiters(\n        self, tokens: Iterable[tokenize.TokenInfo]\n    ) -> None:\n        \"\"\"Adds a message for each string using inconsistent quote delimiters.\n\n        Quote delimiters are used inconsistently if \" and ' are mixed in a module's\n        shortstrings without having done so to avoid escaping an internal quote\n        character.\n\n        Args:\n          tokens: The tokens to be checked against for consistent usage.\n        \"\"\"\n        string_delimiters: Counter[str] = collections.Counter()\n\n        # First, figure out which quote character predominates in the module\n        for tok_type, token, _, _, _ in tokens:\n            if tok_type == tokenize.STRING and _is_quote_delimiter_chosen_freely(token):\n                string_delimiters[_get_quote_delimiter(token)] += 1\n\n        if len(string_delimiters) > 1:\n            # Ties are broken arbitrarily\n            most_common_delimiter = string_delimiters.most_common(1)[0][0]\n            for tok_type, token, start, _, _ in tokens:\n                if tok_type != tokenize.STRING:\n                    continue\n                quote_delimiter = _get_quote_delimiter(token)\n                if (\n                    _is_quote_delimiter_chosen_freely(token)\n                    and quote_delimiter != most_common_delimiter\n                ):\n                    self.add_message(\n                        \"inconsistent-quotes\", line=start[0], args=(quote_delimiter,)\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": 2282,
      "pr_title": "Fix incorrect hanging indent detection for with statements",
      "pr_body": "Add with statement to recognized token keywords for format checker.\r\nAlso recognize with statement as the start of a block\r\nto allow deeper indentation for hanging continuation\r\n\r\nClose #461\r\n",
      "issue_id": 461,
      "issue_title": "incorrect hanging indent detection with 'with' statement",
      "issue_body": "Originally reported by: **Dylan Baker (BitBucket: [dylan_baker](http://bitbucket.org/dylan_baker))**\n\n---\n\nPylint incorrectly marks hanging intdent errors when using multiline context managers, that use a double indent. Below is a trivial example.\n\n``` python\n\n#!/usr/bin/python2.7\n\nfrom __future__ import print_function\nfrom contextlib import contextmanager\n\n\n@contextmanager\ndef a_manager(thing1, thing2, thing3):\n    yield lambda x, y: print(thing1, thing2, thing3, x, y)\n\n\n# Pylint will Mark the arguments to a_manager as a bad hanging indent\nwith a_manager(\n        \"my really freakin' long string, it's long enough for a new line\",\n        \"it's another realy long string.\",\n        \"this one is short.\") as f:\n    f(1, 2)\n\n# This is okay\nwith a_manager(\"A long enough string I need a newline\",\n               \"Cuz this is also a long string or staement.\",\n               \"but not this.\") as f:\n    f(1, 2)\n\n\n# so is this\nif (\"this string is super mega awesome long string\" and\n        \"Also this really long string or comprasison or whatever\"):\n    print(\"Yay\")\n\n\n```\n\n---\n- Bitbucket: https://bitbucket.org/logilab/pylint/issue/461\n",
      "issue_closed_at": "2018-07-15T07:26:23Z",
      "base_commit": "5f9b46c0d45aa1fef67283a96062ff5de25a1263",
      "changes": [
        {
          "file": "pylint/checkers/format.py",
          "type": "line",
          "name": "line 47",
          "code": "from pylint.utils import WarningScope, OPTION_RGX\n\n_ASYNC_TOKEN = 'async'\n_CONTINUATION_BLOCK_OPENERS = ['elif', 'except', 'for', 'if', 'while', 'def', 'class']\n_KEYWORD_TOKENS = ['assert', 'del', 'elif', 'except', 'for', 'if', 'in', 'not',\n                   'raise', 'return', 'while', 'yield']\nif sys.version_info < (3, 0):\n    _KEYWORD_TOKENS.append('print')\n"
        },
        {
          "file": "pylint/checkers/format.py",
          "type": "function",
          "name": "get_valid_indentations",
          "class_name": "ContinuedLineState",
          "code": "def get_valid_indentations(self, idx):\n        \"\"\"Returns the valid offsets for the token at the given position.\"\"\"\n        # The closing brace on a dict or the 'for' in a dict comprehension may\n        # reset two indent levels because the dict value is ended implicitly\n        stack_top = -1\n        if self._tokens.token(idx) in ('}', 'for') and self._cont_stack[-1].token == ':':\n            stack_top = -2\n        indent = self._cont_stack[stack_top]\n        if self._tokens.token(idx) in _CLOSING_BRACKETS:\n            valid_indentations = indent.valid_outdent_strings\n        else:\n            valid_indentations = indent.valid_continuation_strings\n        return indent, valid_indentations.copy()"
        },
        {
          "file": "pylint/checkers/format.py",
          "type": "function",
          "name": "_hanging_indent_after_bracket",
          "class_name": "ContinuedLineState",
          "code": "def _hanging_indent_after_bracket(self, bracket, position):\n        \"\"\"Extracts indentation information for a hanging indent.\"\"\"\n        indentation = self._tokens.line_indent(position)\n        if self._is_block_opener and self._continuation_string == self._block_indent_string:\n            return _ContinuedIndent(\n                HANGING_BLOCK,\n                bracket,\n                position,\n                _Indentations(indentation + self._continuation_string, indentation),\n                _BeforeBlockIndentations(indentation + self._continuation_string,\n                                         indentation + self._continuation_string * 2))\n        if bracket == ':':\n            # If the dict key was on the same line as the open brace, the new\n            # correct indent should be relative to the key instead of the\n            # current indent level\n            paren_align = self._cont_stack[-1].valid_outdent_strings\n            next_align = self._cont_stack[-1].valid_continuation_strings.copy()\n            next_align_keys = list(next_align.keys())\n            next_align[next_align_keys[0] + self._continuation_string] = True\n            # Note that the continuation of\n            # d = {\n            #       'a': 'b'\n            #            'c'\n            # }\n            # is handled by the special-casing for hanging continued string indents.\n            return _ContinuedIndent(HANGING_DICT_VALUE, bracket, position, paren_align, next_align)\n        return _ContinuedIndent(\n            HANGING,\n            bracket,\n            position,\n            _Indentations(indentation, indentation + self._continuation_string),\n            _Indentations(indentation + self._continuation_string))"
        },
        {
          "file": "pylint/checkers/format.py",
          "type": "function",
          "name": "_continuation_inside_bracket",
          "class_name": "ContinuedLineState",
          "code": "def _continuation_inside_bracket(self, bracket, pos):\n        \"\"\"Extracts indentation information for a continued indent.\"\"\"\n        indentation = self._tokens.line_indent(pos)\n        token_indent = self._tokens.token_indent(pos)\n        next_token_indent = self._tokens.token_indent(pos + 1)\n        if self._is_block_opener and next_token_indent == indentation + self._block_indent_string:\n            return _ContinuedIndent(\n                CONTINUED_BLOCK,\n                bracket,\n                pos,\n                _Indentations(token_indent),\n                _BeforeBlockIndentations(\n                    next_token_indent,\n                    next_token_indent + self._continuation_string))\n        return _ContinuedIndent(\n            CONTINUED,\n            bracket,\n            pos,\n            _Indentations(token_indent, next_token_indent),\n            _Indentations(next_token_indent))"
        }
      ]
    }
  ]
}