Skip to content

Commit 04c0264

Browse files
committed
rename error codes to ASYNCxxx, introducing temp functions to pretend test files are renamed.
1 parent a91e5ce commit 04c0264

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

62 files changed

+586
-560
lines changed

README.md

+33-33
Original file line numberDiff line numberDiff line change
@@ -24,47 +24,47 @@ pip install flake8-async
2424

2525
## List of warnings
2626

27-
- **TRIO100**: A `with trio.fail_after(...):` or `with trio.move_on_after(...):`
27+
- **ASYNC100**: A `with trio.fail_after(...):` or `with trio.move_on_after(...):`
2828
context does not contain any `await` statements. This makes it pointless, as
2929
the timeout can only be triggered by a checkpoint.
30-
- **TRIO101**: `yield` inside a nursery or cancel scope is only safe when implementing a context manager - otherwise, it breaks exception handling.
31-
- **TRIO102**: It's unsafe to await inside `finally:` or `except BaseException/trio.Cancelled` unless you use a shielded
30+
- **ASYNC101**: `yield` inside a nursery or cancel scope is only safe when implementing a context manager - otherwise, it breaks exception handling.
31+
- **ASYNC102**: It's unsafe to await inside `finally:` or `except BaseException/trio.Cancelled` unless you use a shielded
3232
cancel scope with a timeout.
33-
- **TRIO103**: `except BaseException`, `except trio.Cancelled` or a bare `except:` with a code path that doesn't re-raise. If you don't want to re-raise `BaseException`, add a separate handler for `trio.Cancelled` before.
34-
- **TRIO104**: `Cancelled` and `BaseException` must be re-raised - when a user tries to `return` or `raise` a different exception.
35-
- **TRIO105**: Calling a trio async function without immediately `await`ing it.
36-
- **TRIO106**: `trio`/`anyio` must be imported with `import trio`/`import anyio` for the linter to work.
37-
- **TRIO107**: Renamed to TRIO910
38-
- **TRIO108**: Renamed to TRIO911
39-
- **TRIO109**: Async function definition with a `timeout` parameter - use `trio.[fail/move_on]_[after/at]` instead
40-
- **TRIO110**: `while <condition>: await trio.sleep()` should be replaced by a `trio.Event`.
41-
- **TRIO111**: Variable, from context manager opened inside nursery, passed to `start[_soon]` might be invalidly accessed while in use, due to context manager closing before the nursery. This is usually a bug, and nurseries should generally be the inner-most context manager.
42-
- **TRIO112**: Nursery body with only a call to `nursery.start[_soon]` and not passing itself as a parameter can be replaced with a regular function call.
43-
- **TRIO113**: Using `nursery.start_soon` in `__aenter__` doesn't wait for the task to begin. Consider replacing with `nursery.start`.
44-
- **TRIO114**: Startable function (i.e. has a `task_status` keyword parameter) not in `--startable-in-context-manager` parameter list, please add it so TRIO113 can catch errors when using it.
45-
- **TRIO115**: Replace `trio.sleep(0)` with the more suggestive `trio.lowlevel.checkpoint()`.
46-
- **TRIO116**: `trio.sleep()` with >24 hour interval should usually be `trio.sleep_forever()`.
47-
- **TRIO117**: Don't raise or catch `trio.[NonBase]MultiError`, prefer `[exceptiongroup.]BaseExceptionGroup`. Even if Trio still raises `MultiError` for legacy code, it can be caught with `BaseExceptionGroup` so it's fully redundant.
48-
- **TRIO118**: Don't assign the value of `anyio.get_cancelled_exc_class()` to a variable, since that breaks linter checks and multi-backend programs.
33+
- **ASYNC103**: `except BaseException`, `except trio.Cancelled` or a bare `except:` with a code path that doesn't re-raise. If you don't want to re-raise `BaseException`, add a separate handler for `trio.Cancelled` before.
34+
- **ASYNC104**: `Cancelled` and `BaseException` must be re-raised - when a user tries to `return` or `raise` a different exception.
35+
- **ASYNC105**: Calling a trio async function without immediately `await`ing it.
36+
- **ASYNC106**: `trio`/`anyio` must be imported with `import trio`/`import anyio` for the linter to work.
37+
- **ASYNC107**: Renamed to ASYNC910
38+
- **ASYNC108**: Renamed to ASYNC911
39+
- **ASYNC109**: Async function definition with a `timeout` parameter - use `trio.[fail/move_on]_[after/at]` instead
40+
- **ASYNC110**: `while <condition>: await trio.sleep()` should be replaced by a `trio.Event`.
41+
- **ASYNC111**: Variable, from context manager opened inside nursery, passed to `start[_soon]` might be invalidly accessed while in use, due to context manager closing before the nursery. This is usually a bug, and nurseries should generally be the inner-most context manager.
42+
- **ASYNC112**: Nursery body with only a call to `nursery.start[_soon]` and not passing itself as a parameter can be replaced with a regular function call.
43+
- **ASYNC113**: Using `nursery.start_soon` in `__aenter__` doesn't wait for the task to begin. Consider replacing with `nursery.start`.
44+
- **ASYNC114**: Startable function (i.e. has a `task_status` keyword parameter) not in `--startable-in-context-manager` parameter list, please add it so ASYNC113 can catch errors when using it.
45+
- **ASYNC115**: Replace `trio.sleep(0)` with the more suggestive `trio.lowlevel.checkpoint()`.
46+
- **ASYNC116**: `trio.sleep()` with >24 hour interval should usually be `trio.sleep_forever()`.
47+
- **ASYNC117**: Don't raise or catch `trio.[NonBase]MultiError`, prefer `[exceptiongroup.]BaseExceptionGroup`. Even if Trio still raises `MultiError` for legacy code, it can be caught with `BaseExceptionGroup` so it's fully redundant.
48+
- **ASYNC118**: Don't assign the value of `anyio.get_cancelled_exc_class()` to a variable, since that breaks linter checks and multi-backend programs.
4949

5050
### Warnings for blocking sync calls in async functions
51-
- **TRIO200**: User-configured error for blocking sync calls in async functions. Does nothing by default, see [`trio200-blocking-calls`](#trio200-blocking-calls) for how to configure it.
52-
- **TRIO210**: Sync HTTP call in async function, use `httpx.AsyncClient`.
53-
- **TRIO211**: Likely sync HTTP call in async function, use `httpx.AsyncClient`. Looks for `urllib3` method calls on pool objects, but only matching on the method signature and not the object.
54-
- **TRIO212**: Blocking sync HTTP call on httpx object, use httpx.AsyncClient.
55-
- **TRIO220**: Sync process call in async function, use `await nursery.start(trio.run_process, ...)`.
56-
- **TRIO221**: Sync process call in async function, use `await trio.run_process(...)`.
57-
- **TRIO222**: Sync `os.*` call in async function, wrap in `await trio.to_thread.run_sync()`.
58-
- **TRIO230**: Sync IO call in async function, use `trio.open_file(...)`.
59-
- **TRIO231**: Sync IO call in async function, use `trio.wrap_file(...)`.
60-
- **TRIO232**: Blocking sync call on file object, wrap the file object in `trio.wrap_file()` to get an async file object.
61-
- **TRIO240**: Avoid using `os.path` in async functions, prefer using `trio.Path` objects.
51+
- **ASYNC200**: User-configured error for blocking sync calls in async functions. Does nothing by default, see [`trio200-blocking-calls`](#trio200-blocking-calls) for how to configure it.
52+
- **ASYNC210**: Sync HTTP call in async function, use `httpx.AsyncClient`.
53+
- **ASYNC211**: Likely sync HTTP call in async function, use `httpx.AsyncClient`. Looks for `urllib3` method calls on pool objects, but only matching on the method signature and not the object.
54+
- **ASYNC212**: Blocking sync HTTP call on httpx object, use httpx.AsyncClient.
55+
- **ASYNC220**: Sync process call in async function, use `await nursery.start(trio.run_process, ...)`.
56+
- **ASYNC221**: Sync process call in async function, use `await trio.run_process(...)`.
57+
- **ASYNC222**: Sync `os.*` call in async function, wrap in `await trio.to_thread.run_sync()`.
58+
- **ASYNC230**: Sync IO call in async function, use `trio.open_file(...)`.
59+
- **ASYNC231**: Sync IO call in async function, use `trio.wrap_file(...)`.
60+
- **ASYNC232**: Blocking sync call on file object, wrap the file object in `trio.wrap_file()` to get an async file object.
61+
- **ASYNC240**: Avoid using `os.path` in async functions, prefer using `trio.Path` objects.
6262

6363

6464
### Warnings disabled by default
65-
- **TRIO900**: Async generator without `@asynccontextmanager` not allowed.
66-
- **TRIO910**: Exit or `return` from async function with no guaranteed checkpoint or exception since function definition.
67-
- **TRIO911**: Exit, `yield` or `return` from async iterable with no guaranteed checkpoint since possible function entry (yield or function definition)
65+
- **ASYNC900**: Async generator without `@asynccontextmanager` not allowed.
66+
- **ASYNC910**: Exit or `return` from async function with no guaranteed checkpoint or exception since function definition.
67+
- **ASYNC911**: Exit, `yield` or `return` from async iterable with no guaranteed checkpoint since possible function entry (yield or function definition)
6868
Checkpoints are `await`, `async for`, and `async with` (on one of enter/exit).
6969

7070
## Examples

flake8_trio/__init__.py

+13-9
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
import functools
1616
import keyword
1717
import os
18+
import re
1819
import subprocess
1920
import sys
2021
import tokenize
@@ -199,7 +200,7 @@ def add_options(option_manager: OptionManager | ArgumentParser):
199200
)
200201
else: # if run as a flake8 plugin
201202
Plugin.standalone = False
202-
# Disable TRIO9xx calls by default
203+
# Disable ASYNC9xx calls by default
203204
option_manager.extend_default_ignore(default_disabled_error_codes)
204205
# add parameter to parse from flake8 config
205206
add_argument = functools.partial( # type: ignore
@@ -209,7 +210,7 @@ def add_options(option_manager: OptionManager | ArgumentParser):
209210
add_argument(
210211
"--enable",
211212
type=comma_separated_list,
212-
default="TRIO",
213+
default="ASYNC",
213214
required=False,
214215
help=(
215216
"Comma-separated list of error codes to enable, similar to flake8"
@@ -221,7 +222,7 @@ def add_options(option_manager: OptionManager | ArgumentParser):
221222
add_argument(
222223
"--disable",
223224
type=comma_separated_list,
224-
default="TRIO9" if Plugin.standalone else "",
225+
default="ASYNC9" if Plugin.standalone else "",
225226
required=False,
226227
help=(
227228
"Comma-separated list of error codes to disable, similar to flake8"
@@ -253,7 +254,7 @@ def add_options(option_manager: OptionManager | ArgumentParser):
253254
required=False,
254255
type=comma_separated_list,
255256
help=(
256-
"Comma-separated list of decorators to disable TRIO910 & TRIO911 "
257+
"Comma-separated list of decorators to disable ASYNC910 & ASYNC911 "
257258
"checkpoint warnings for. "
258259
"Decorators can be dotted or not, as well as support * as a wildcard. "
259260
"For example, ``--no-checkpoint-warning-decorators=app.route,"
@@ -266,7 +267,7 @@ def add_options(option_manager: OptionManager | ArgumentParser):
266267
default="",
267268
required=False,
268269
help=(
269-
"Comma-separated list of method calls to additionally enable TRIO113 "
270+
"Comma-separated list of method calls to additionally enable ASYNC113 "
270271
"warnings for. Will also check for the pattern inside function calls. "
271272
"Methods must be valid identifiers as per `str.isidientifier()` and "
272273
"not reserved keywords. "
@@ -281,7 +282,7 @@ def add_options(option_manager: OptionManager | ArgumentParser):
281282
required=False,
282283
help=(
283284
"Comma-separated list of key->value pairs, where key is a [dotted] "
284-
"function that if found inside an async function will raise TRIO200, "
285+
"function that if found inside an async function will raise ASYNC200, "
285286
"suggesting it be replaced with {value}"
286287
),
287288
)
@@ -313,8 +314,9 @@ def get_matching_codes(
313314
err_code
314315
for err_class in (*ERROR_CLASSES, *ERROR_CLASSES_CST)
315316
for err_code in err_class.error_codes # type: ignore[attr-defined]
316-
if len(err_code) == 7 # exclude e.g. TRIO103_anyio_trio
317+
if re.match("ASYNC...", err_code) # exclude e.g. ASYNC103_anyio_trio
317318
}
319+
assert all_codes
318320

319321
if options.autofix and not Plugin.standalone:
320322
print("Cannot autofix when run as a flake8 plugin.", file=sys.stderr)
@@ -328,8 +330,10 @@ def get_matching_codes(
328330
enabled_codes -= set(get_matching_codes(options.disable, enabled_codes))
329331

330332
# if disable has default value, re-enable explicitly enabled codes
331-
if options.disable == ["TRIO9"]:
332-
enabled_codes.update(code for code in options.enable if len(code) == 7)
333+
if options.disable == ["ASYNC9"]:
334+
enabled_codes.update(
335+
code for code in options.enable if re.match("ASYNC...", code)
336+
)
333337

334338
Plugin._options = Options(
335339
enabled_codes=enabled_codes,

flake8_trio/visitors/flake8triovisitor.py

+7-6
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,9 @@
2020
ast.expr, ast.stmt, ast.arg, ast.excepthandler, ast.alias, Statement
2121
]
2222

23+
# 8 == len('ASYNCxxx'), so alt messages raise the original code
24+
ERROR_CODE_LEN = 8
25+
2326

2427
class Flake8TrioVisitor(ast.NodeVisitor, ABC):
2528
# abstract attribute by not providing a value
@@ -98,13 +101,12 @@ def error(
98101
), "No error code defined, but class has multiple codes"
99102
error_code = next(iter(self.error_codes))
100103
# don't emit an error if this code is disabled in a multi-code visitor
101-
elif error_code[:7] not in self.options.enabled_codes:
104+
elif error_code[:ERROR_CODE_LEN] not in self.options.enabled_codes:
102105
return
103106

104107
self.__state.problems.append(
105108
Error(
106-
# 7 == len('TRIO...'), so alt messages raise the original code
107-
error_code[:7],
109+
error_code[:ERROR_CODE_LEN],
108110
node.lineno,
109111
node.col_offset,
110112
self.error_codes[error_code],
@@ -218,7 +220,7 @@ def error(
218220
error_code = next(iter(self.error_codes))
219221
# don't emit an error if this code is disabled in a multi-code visitor
220222
# TODO: write test for only one of 910/911 enabled/autofixed
221-
elif error_code[:7] not in self.options.enabled_codes:
223+
elif error_code[:ERROR_CODE_LEN] not in self.options.enabled_codes:
222224
return False # pragma: no cover
223225

224226
if self.is_noqa(node, error_code):
@@ -228,8 +230,7 @@ def error(
228230
pos = self.get_metadata(PositionProvider, node).start # type: ignore
229231
self.__state.problems.append(
230232
Error(
231-
# 7 == len('TRIO...'), so alt messages raise the original code
232-
error_code[:7],
233+
error_code[:ERROR_CODE_LEN],
233234
pos.line, # type: ignore
234235
pos.column, # type: ignore
235236
self.error_codes[error_code],

flake8_trio/visitors/helpers.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -364,7 +364,7 @@ def get_comments(node: cst.CSTNode | Iterable[cst.CSTNode]) -> Iterator[cst.Empt
364364
yield from get_comments(n)
365365

366366

367-
# used in TRIO100
367+
# used in ASYNC100
368368
def flatten_preserving_comments(node: cst.BaseCompoundStatement):
369369
# add leading lines (comments and empty lines) for the node to be removed
370370
new_leading_lines = list(node.leading_lines)

flake8_trio/visitors/visitor100.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
"""Contains visitor for TRIO100.
1+
"""Contains visitor for ASYNC100.
22
33
A `with trio.fail_after(...):` or `with trio.move_on_after(...):`
44
context does not contain any `await` statements. This makes it pointless, as
@@ -28,7 +28,7 @@
2828
@error_class_cst
2929
class Visitor100_libcst(Flake8TrioVisitor_cst):
3030
error_codes: Mapping[str, str] = {
31-
"TRIO100": (
31+
"ASYNC100": (
3232
"{0}.{1} context contains no checkpoints, remove the context or add"
3333
" `await {0}.lowlevel.checkpoint()`."
3434
),

flake8_trio/visitors/visitor101.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
"""Contains visitor for TRIO101.
1+
"""Contains visitor for ASYNC101.
22
33
`yield` inside a nursery or cancel scope is only safe when implementing a context manager
44
- otherwise, it breaks exception handling.
@@ -25,7 +25,7 @@
2525
@error_class_cst
2626
class Visitor101(Flake8TrioVisitor_cst):
2727
error_codes: Mapping[str, str] = {
28-
"TRIO101": (
28+
"ASYNC101": (
2929
"`yield` inside a nursery or cancel scope is only safe when implementing "
3030
"a context manager - otherwise, it breaks exception handling."
3131
),

flake8_trio/visitors/visitor102.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@
1919
@error_class
2020
class Visitor102(Flake8TrioVisitor):
2121
error_codes: Mapping[str, str] = {
22-
"TRIO102": (
22+
"ASYNC102": (
2323
"await inside {0.name} on line {0.lineno} must have shielded cancel "
2424
"scope with a timeout."
2525
),

flake8_trio/visitors/visitor103_104.py

+13-13
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@
1717
if TYPE_CHECKING:
1818
from collections.abc import Mapping
1919

20-
_trio103_common_msg = "{} block with a code path that doesn't re-raise the error."
20+
_async103_common_msg = "{} block with a code path that doesn't re-raise the error."
2121
_suggestion = " Consider adding an `except {}: raise` before this exception handler."
2222
_suggestion_dict: dict[tuple[str, ...], str] = {
2323
("anyio",): "anyio.get_cancelled_exc_class()",
@@ -26,12 +26,12 @@
2626
_suggestion_dict[("anyio", "trio")] = "[" + "|".join(_suggestion_dict.values()) + "]"
2727

2828
_error_codes = {
29-
"TRIO103": _trio103_common_msg,
30-
"TRIO104": "Cancelled (and therefore BaseException) must be re-raised.",
29+
"ASYNC103": _async103_common_msg,
30+
"ASYNC104": "Cancelled (and therefore BaseException) must be re-raised.",
3131
}
3232
for poss_library in _suggestion_dict:
33-
_error_codes[f"TRIO103_{'_'.join(poss_library)}"] = (
34-
_trio103_common_msg + _suggestion.format(_suggestion_dict[poss_library])
33+
_error_codes[f"ASYNC103_{'_'.join(poss_library)}"] = (
34+
_async103_common_msg + _suggestion.format(_suggestion_dict[poss_library])
3535
)
3636

3737

@@ -61,21 +61,21 @@ def visit_ExceptHandler(self, node: ast.ExceptHandler):
6161
# If previous excepts have handled trio.Cancelled, don't do anything - namely
6262
# don't set self.unraised (so 104 isn't triggered) nor check for 103.
6363
if marker.name == "trio.Cancelled":
64-
error_code = "TRIO103"
64+
error_code = "ASYNC103"
6565
self.cancelled_caught.add("trio")
6666
elif marker.name in (
6767
"anyio.get_cancelled_exc_class()",
6868
"get_cancelled_exc_class()",
6969
):
70-
error_code = "TRIO103"
70+
error_code = "ASYNC103"
7171
self.cancelled_caught.add("anyio")
7272
else:
7373
if self.cancelled_caught:
7474
return
7575
if len(self.library) < 2:
76-
error_code = f"TRIO103_{self.library_str}"
76+
error_code = f"ASYNC103_{self.library_str}"
7777
else:
78-
error_code = f"TRIO103_{'_'.join(sorted(self.library))}"
78+
error_code = f"ASYNC103_{'_'.join(sorted(self.library))}"
7979
self.cancelled_caught.update("trio", "anyio")
8080

8181
# Don't save the state of cancelled_caught, that's handled in Try and would
@@ -117,15 +117,15 @@ def visit_Raise(self, node: ast.Raise):
117117
and node.exc is not None
118118
and not (isinstance(node.exc, ast.Name) and node.exc.id == self.except_name)
119119
):
120-
self.error(node, error_code="TRIO104")
120+
self.error(node, error_code="ASYNC104")
121121

122122
# treat it as safe regardless, to avoid unnecessary error messages.
123123
self.unraised = False
124124

125125
def visit_Return(self, node: ast.Return | ast.Yield):
126126
if self.unraised:
127127
# Error: must re-raise
128-
self.error(node, error_code="TRIO104")
128+
self.error(node, error_code="ASYNC104")
129129

130130
visit_Yield = visit_Return
131131

@@ -209,10 +209,10 @@ def visit_For(self, node: ast.For | ast.While):
209209

210210
def visit_Break(self, node: ast.Break):
211211
if self.unraised and self.loop_depth == 0:
212-
self.error(node, error_code="TRIO104")
212+
self.error(node, error_code="ASYNC104")
213213
self.unraised_break |= self.unraised
214214

215215
def visit_Continue(self, node: ast.Continue):
216216
if self.unraised and self.loop_depth == 0:
217-
self.error(node, error_code="TRIO104")
217+
self.error(node, error_code="ASYNC104")
218218
self.unraised_continue |= self.unraised

0 commit comments

Comments
 (0)