Skip to content

Commit 4e56fc0

Browse files
committed
pregenerate the pycodestyle plugin to avoid call overhead
1 parent 1e5f861 commit 4e56fc0

File tree

5 files changed

+255
-37
lines changed

5 files changed

+255
-37
lines changed

bin/gen-pycodestyle-plugin

+96
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
#!/usr/bin/env python3
2+
import inspect
3+
import os.path
4+
from typing import Any
5+
from typing import Callable
6+
from typing import Generator
7+
from typing import NamedTuple
8+
from typing import Tuple
9+
10+
import pycodestyle
11+
12+
13+
def _too_long(s: str) -> str:
14+
if len(s) >= 80:
15+
return f"{s} # noqa: E501"
16+
else:
17+
return s
18+
19+
20+
class Call(NamedTuple):
21+
name: str
22+
is_generator: bool
23+
params: Tuple[str, ...]
24+
25+
def to_src(self) -> str:
26+
params_s = ", ".join(self.params)
27+
if self.is_generator:
28+
return _too_long(f" yield from _{self.name}({params_s})")
29+
else:
30+
lines = (
31+
_too_long(f" ret = _{self.name}({params_s})"),
32+
" if ret is not None:",
33+
" yield ret",
34+
)
35+
return "\n".join(lines)
36+
37+
@classmethod
38+
def from_func(cls, func: Callable[..., Any]) -> "Call":
39+
spec = inspect.getfullargspec(func)
40+
params = tuple(spec.args)
41+
return cls(func.__name__, inspect.isgeneratorfunction(func), params)
42+
43+
44+
def lines() -> Generator[str, None, None]:
45+
logical = []
46+
physical = []
47+
48+
logical = [
49+
Call.from_func(check) for check in pycodestyle._checks["logical_line"]
50+
]
51+
physical = [
52+
Call.from_func(check) for check in pycodestyle._checks["physical_line"]
53+
]
54+
assert not pycodestyle._checks["tree"]
55+
56+
yield f'"""Generated using ./bin/{os.path.basename(__file__)}."""'
57+
yield "# fmt: off"
58+
yield "from typing import Any"
59+
yield "from typing import Generator"
60+
yield "from typing import Tuple"
61+
yield ""
62+
imports = sorted(call.name for call in logical + physical)
63+
for name in imports:
64+
yield _too_long(f"from pycodestyle import {name} as _{name}")
65+
yield ""
66+
yield ""
67+
68+
yield "def pycodestyle_logical("
69+
logical_params = {param for call in logical for param in call.params}
70+
for param in sorted(logical_params):
71+
yield f" {param}: Any,"
72+
yield ") -> Generator[Tuple[int, str], None, None]:"
73+
yield ' """Run pycodestyle logical checks."""'
74+
for call in sorted(logical):
75+
yield call.to_src()
76+
yield ""
77+
yield ""
78+
79+
yield "def pycodestyle_physical("
80+
physical_params = {param for call in physical for param in call.params}
81+
for param in sorted(physical_params):
82+
yield f" {param}: Any,"
83+
yield ") -> Generator[Tuple[int, str], None, None]:"
84+
yield ' """Run pycodestyle physical checks."""'
85+
for call in sorted(physical):
86+
yield call.to_src()
87+
88+
89+
def main() -> int:
90+
for line in lines():
91+
print(line)
92+
return 0
93+
94+
95+
if __name__ == "__main__":
96+
raise SystemExit(main())

setup.cfg

+2-36
Original file line numberDiff line numberDiff line change
@@ -53,42 +53,8 @@ console_scripts =
5353
flake8 = flake8.main.cli:main
5454
flake8.extension =
5555
F = flake8.plugins.pyflakes:FlakesChecker
56-
pycodestyle.ambiguous_identifier = pycodestyle:ambiguous_identifier
57-
pycodestyle.bare_except = pycodestyle:bare_except
58-
pycodestyle.blank_lines = pycodestyle:blank_lines
59-
pycodestyle.break_after_binary_operator = pycodestyle:break_after_binary_operator
60-
pycodestyle.break_before_binary_operator = pycodestyle:break_before_binary_operator
61-
pycodestyle.comparison_negative = pycodestyle:comparison_negative
62-
pycodestyle.comparison_to_singleton = pycodestyle:comparison_to_singleton
63-
pycodestyle.comparison_type = pycodestyle:comparison_type
64-
pycodestyle.compound_statements = pycodestyle:compound_statements
65-
pycodestyle.continued_indentation = pycodestyle:continued_indentation
66-
pycodestyle.explicit_line_join = pycodestyle:explicit_line_join
67-
pycodestyle.extraneous_whitespace = pycodestyle:extraneous_whitespace
68-
pycodestyle.imports_on_separate_lines = pycodestyle:imports_on_separate_lines
69-
pycodestyle.indentation = pycodestyle:indentation
70-
pycodestyle.maximum_doc_length = pycodestyle:maximum_doc_length
71-
pycodestyle.maximum_line_length = pycodestyle:maximum_line_length
72-
pycodestyle.missing_whitespace = pycodestyle:missing_whitespace
73-
pycodestyle.missing_whitespace_after_import_keyword = pycodestyle:missing_whitespace_after_import_keyword
74-
pycodestyle.missing_whitespace_around_operator = pycodestyle:missing_whitespace_around_operator
75-
pycodestyle.module_imports_on_top_of_file = pycodestyle:module_imports_on_top_of_file
76-
pycodestyle.python_3000_async_await_keywords = pycodestyle:python_3000_async_await_keywords
77-
pycodestyle.python_3000_backticks = pycodestyle:python_3000_backticks
78-
pycodestyle.python_3000_has_key = pycodestyle:python_3000_has_key
79-
pycodestyle.python_3000_invalid_escape_sequence = pycodestyle:python_3000_invalid_escape_sequence
80-
pycodestyle.python_3000_not_equal = pycodestyle:python_3000_not_equal
81-
pycodestyle.python_3000_raise_comma = pycodestyle:python_3000_raise_comma
82-
pycodestyle.tabs_obsolete = pycodestyle:tabs_obsolete
83-
pycodestyle.tabs_or_spaces = pycodestyle:tabs_or_spaces
84-
pycodestyle.trailing_blank_lines = pycodestyle:trailing_blank_lines
85-
pycodestyle.trailing_whitespace = pycodestyle:trailing_whitespace
86-
pycodestyle.whitespace_around_comma = pycodestyle:whitespace_around_comma
87-
pycodestyle.whitespace_around_keywords = pycodestyle:whitespace_around_keywords
88-
pycodestyle.whitespace_around_named_parameter_equals = pycodestyle:whitespace_around_named_parameter_equals
89-
pycodestyle.whitespace_around_operator = pycodestyle:whitespace_around_operator
90-
pycodestyle.whitespace_before_comment = pycodestyle:whitespace_before_comment
91-
pycodestyle.whitespace_before_parameters = pycodestyle:whitespace_before_parameters
56+
E = flake8.plugins.pycodestyle:pycodestyle_logical
57+
W = flake8.plugins.pycodestyle:pycodestyle_physical
9258
flake8.report =
9359
default = flake8.formatting.default:Default
9460
pylint = flake8.formatting.default:Pylint

src/flake8/plugins/pycodestyle.py

+123
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,123 @@
1+
"""Generated using ./bin/gen-pycodestyle-plugin."""
2+
# fmt: off
3+
from typing import Any
4+
from typing import Generator
5+
from typing import Tuple
6+
7+
from pycodestyle import ambiguous_identifier as _ambiguous_identifier
8+
from pycodestyle import bare_except as _bare_except
9+
from pycodestyle import blank_lines as _blank_lines
10+
from pycodestyle import break_after_binary_operator as _break_after_binary_operator # noqa: E501
11+
from pycodestyle import break_before_binary_operator as _break_before_binary_operator # noqa: E501
12+
from pycodestyle import comparison_negative as _comparison_negative
13+
from pycodestyle import comparison_to_singleton as _comparison_to_singleton
14+
from pycodestyle import comparison_type as _comparison_type
15+
from pycodestyle import compound_statements as _compound_statements
16+
from pycodestyle import continued_indentation as _continued_indentation
17+
from pycodestyle import explicit_line_join as _explicit_line_join
18+
from pycodestyle import extraneous_whitespace as _extraneous_whitespace
19+
from pycodestyle import imports_on_separate_lines as _imports_on_separate_lines
20+
from pycodestyle import indentation as _indentation
21+
from pycodestyle import maximum_doc_length as _maximum_doc_length
22+
from pycodestyle import maximum_line_length as _maximum_line_length
23+
from pycodestyle import missing_whitespace as _missing_whitespace
24+
from pycodestyle import missing_whitespace_after_import_keyword as _missing_whitespace_after_import_keyword # noqa: E501
25+
from pycodestyle import missing_whitespace_around_operator as _missing_whitespace_around_operator # noqa: E501
26+
from pycodestyle import module_imports_on_top_of_file as _module_imports_on_top_of_file # noqa: E501
27+
from pycodestyle import python_3000_async_await_keywords as _python_3000_async_await_keywords # noqa: E501
28+
from pycodestyle import python_3000_backticks as _python_3000_backticks
29+
from pycodestyle import python_3000_has_key as _python_3000_has_key
30+
from pycodestyle import python_3000_invalid_escape_sequence as _python_3000_invalid_escape_sequence # noqa: E501
31+
from pycodestyle import python_3000_not_equal as _python_3000_not_equal
32+
from pycodestyle import python_3000_raise_comma as _python_3000_raise_comma
33+
from pycodestyle import tabs_obsolete as _tabs_obsolete
34+
from pycodestyle import tabs_or_spaces as _tabs_or_spaces
35+
from pycodestyle import trailing_blank_lines as _trailing_blank_lines
36+
from pycodestyle import trailing_whitespace as _trailing_whitespace
37+
from pycodestyle import whitespace_around_comma as _whitespace_around_comma
38+
from pycodestyle import whitespace_around_keywords as _whitespace_around_keywords # noqa: E501
39+
from pycodestyle import whitespace_around_named_parameter_equals as _whitespace_around_named_parameter_equals # noqa: E501
40+
from pycodestyle import whitespace_around_operator as _whitespace_around_operator # noqa: E501
41+
from pycodestyle import whitespace_before_comment as _whitespace_before_comment
42+
from pycodestyle import whitespace_before_parameters as _whitespace_before_parameters # noqa: E501
43+
44+
45+
def pycodestyle_logical(
46+
blank_before: Any,
47+
blank_lines: Any,
48+
checker_state: Any,
49+
hang_closing: Any,
50+
indent_char: Any,
51+
indent_level: Any,
52+
indent_size: Any,
53+
line_number: Any,
54+
lines: Any,
55+
logical_line: Any,
56+
max_doc_length: Any,
57+
noqa: Any,
58+
previous_indent_level: Any,
59+
previous_logical: Any,
60+
previous_unindented_logical_line: Any,
61+
tokens: Any,
62+
verbose: Any,
63+
) -> Generator[Tuple[int, str], None, None]:
64+
"""Run pycodestyle logical checks."""
65+
yield from _ambiguous_identifier(logical_line, tokens)
66+
yield from _bare_except(logical_line, noqa)
67+
yield from _blank_lines(logical_line, blank_lines, indent_level, line_number, blank_before, previous_logical, previous_unindented_logical_line, previous_indent_level, lines) # noqa: E501
68+
yield from _break_after_binary_operator(logical_line, tokens)
69+
yield from _break_before_binary_operator(logical_line, tokens)
70+
yield from _comparison_negative(logical_line)
71+
yield from _comparison_to_singleton(logical_line, noqa)
72+
yield from _comparison_type(logical_line, noqa)
73+
yield from _compound_statements(logical_line)
74+
yield from _continued_indentation(logical_line, tokens, indent_level, hang_closing, indent_char, indent_size, noqa, verbose) # noqa: E501
75+
yield from _explicit_line_join(logical_line, tokens)
76+
yield from _extraneous_whitespace(logical_line)
77+
yield from _imports_on_separate_lines(logical_line)
78+
yield from _indentation(logical_line, previous_logical, indent_char, indent_level, previous_indent_level, indent_size) # noqa: E501
79+
yield from _maximum_doc_length(logical_line, max_doc_length, noqa, tokens)
80+
yield from _missing_whitespace(logical_line)
81+
yield from _missing_whitespace_after_import_keyword(logical_line)
82+
yield from _missing_whitespace_around_operator(logical_line, tokens)
83+
yield from _module_imports_on_top_of_file(logical_line, indent_level, checker_state, noqa) # noqa: E501
84+
yield from _python_3000_async_await_keywords(logical_line, tokens)
85+
yield from _python_3000_backticks(logical_line)
86+
yield from _python_3000_has_key(logical_line, noqa)
87+
yield from _python_3000_invalid_escape_sequence(logical_line, tokens, noqa)
88+
yield from _python_3000_not_equal(logical_line)
89+
yield from _python_3000_raise_comma(logical_line)
90+
yield from _whitespace_around_comma(logical_line)
91+
yield from _whitespace_around_keywords(logical_line)
92+
yield from _whitespace_around_named_parameter_equals(logical_line, tokens)
93+
yield from _whitespace_around_operator(logical_line)
94+
yield from _whitespace_before_comment(logical_line, tokens)
95+
yield from _whitespace_before_parameters(logical_line, tokens)
96+
97+
98+
def pycodestyle_physical(
99+
indent_char: Any,
100+
line_number: Any,
101+
lines: Any,
102+
max_line_length: Any,
103+
multiline: Any,
104+
noqa: Any,
105+
physical_line: Any,
106+
total_lines: Any,
107+
) -> Generator[Tuple[int, str], None, None]:
108+
"""Run pycodestyle physical checks."""
109+
ret = _maximum_line_length(physical_line, max_line_length, multiline, line_number, noqa) # noqa: E501
110+
if ret is not None:
111+
yield ret
112+
ret = _tabs_obsolete(physical_line)
113+
if ret is not None:
114+
yield ret
115+
ret = _tabs_or_spaces(physical_line, indent_char)
116+
if ret is not None:
117+
yield ret
118+
ret = _trailing_blank_lines(physical_line, lines, line_number, total_lines)
119+
if ret is not None:
120+
yield ret
121+
ret = _trailing_whitespace(physical_line)
122+
if ret is not None:
123+
yield ret

tests/integration/test_plugins.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ def test_local_plugin_can_add_option(local_config):
104104

105105
args = aggregator.aggregate_options(option_manager, cfg, cfg_dir, argv)
106106

107-
assert args.extended_default_select == {"XE", "F", "E", "C90"}
107+
assert args.extended_default_select == {"XE", "F", "E", "W", "C90"}
108108
assert args.anopt == "foo"
109109

110110

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
import importlib.machinery
2+
import importlib.util
3+
import os.path
4+
5+
import flake8.plugins.pycodestyle
6+
7+
HERE = os.path.dirname(os.path.abspath(__file__))
8+
9+
10+
def test_up_to_date():
11+
"""Validate that the generated pycodestyle plugin is up to date.
12+
13+
We generate two "meta" plugins for pycodestyle to avoid calling overhead.
14+
15+
To regenerate run:
16+
17+
./bin/gen-pycodestyle-plugin > src/flake8/plugins/pycodestyle.py
18+
"""
19+
20+
path = os.path.join(HERE, "../../../bin/gen-pycodestyle-plugin")
21+
name = os.path.basename(path)
22+
loader = importlib.machinery.SourceFileLoader(name, path)
23+
spec = importlib.util.spec_from_loader(loader.name, loader)
24+
assert spec is not None
25+
mod = importlib.util.module_from_spec(spec)
26+
loader.exec_module(mod)
27+
28+
expected = "".join(f"{line}\n" for line in mod.lines())
29+
30+
with open(flake8.plugins.pycodestyle.__file__) as f:
31+
contents = f.read()
32+
33+
assert contents == expected

0 commit comments

Comments
 (0)