A highly opinionated flake8 plugin for Trio-related problems.
Project description
flake8-trio
A highly opinionated flake8 plugin for Trio-related problems.
This can include anything from outright bugs, to pointless/dead code, to likely performance issues, to minor points of idiom that might signal a misunderstanding.
It may well be too noisy for anyone with different opinions, that's OK.
Pairs well with flake8-bugbear.
Installation
pip install flake8-trio
List of warnings
- TRIO100: a
with trio.fail_after(...):
orwith trio.move_on_after(...):
context does not contain anyawait
statements. This makes it pointless, as the timeout can only be triggered by a checkpoint. - TRIO101:
yield
inside a nursery or cancel scope is only safe when implementing a context manager - otherwise, it breaks exception handling. - TRIO102: it's unsafe to await inside
finally:
orexcept BaseException/trio.Cancelled
unless you use a shielded cancel scope with a timeout. - TRIO103:
except BaseException
,except trio.Cancelled
or a bareexcept:
with a code path that doesn't re-raise. If you don't want to re-raiseBaseException
, add a separate handler fortrio.Cancelled
before. - TRIO104:
Cancelled
andBaseException
must be re-raised - when a user tries toreturn
orraise
a different exception. - TRIO105: Calling a trio async function without immediately
await
ing it. - TRIO106: trio must be imported with
import trio
for the linter to work. - TRIO107: exit or
return
from async function with no guaranteed checkpoint or exception since function definition. - TRIO108: exit, yield or return from async iterable with no guaranteed checkpoint since possible function entry (yield or function definition)
Checkpoints are
await
,async for
, andasync with
(on one of enter/exit). - TRIO109: Async function definition with a
timeout
parameter - usetrio.[fail/move_on]_[after/at]
instead - TRIO110:
while <condition>: await trio.sleep()
should be replaced by atrio.Event
. - 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. - 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. - TRIO113: using
nursery.start_soon
in__aenter__
doesn't wait for the task to begin. Consider replacing withnursery.start
. - 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. - TRIO115: Replace
trio.sleep(0)
with the more suggestivetrio.lowlevel.checkpoint()
. - TRIO116:
trio.sleep()
with >24 hour interval should usually betrio.sleep_forever()
. - TRIO117: Don't raise or catch
trio.[NonBase]MultiError
, prefer[exceptiongroup.]BaseExceptionGroup
. Even if Trio still raisesMultiError
for legacy code, it can be caught withBaseExceptionGroup
so it's fully redundant. - TRIO200: User-configured error for blocking sync calls in async functions. Does nothing by default, see
trio200-blocking-calls
for how to configure it. - TRIO210: Sync HTTP call in async function, use
httpx.AsyncClient
. - TRIO211: Likely sync HTTP call in async function, use
httpx.AsyncClient
. Looks forurllib3
method calls on pool objects, but only matching on the method signature and not the object. - TRIO212: Blocking sync HTTP call on httpx object, use httpx.AsyncClient.
- TRIO220: Sync process call in async function, use
await nursery.start(trio.run_process, ...)
. - TRIO221: Sync process call in async function, use
await trio.run_process(...)
. - TRIO222: Sync
os.*
call in async function, wrap inawait trio.to_thread.run_sync()
. - TRIO230: Sync IO call in async function, use
trio.open_file(...)
. - TRIO231: Sync IO call in async function, use
trio.wrap_file(...)
. - TRIO232: Blocking sync call on file object, wrap the file object in
trio.wrap_file()
to get an async file object. - TRIO240: Avoid using
os.path
in async functions, prefer usingtrio.Path
objects.
Warnings disabled by default
- TRIO900: Async generator without
@asynccontextmanager
not allowed.
Configuration
You can configure flake8
with command-line options,
but we prefer using a config file. The file needs to start with a section marker [flake8]
and the following options are then parsed using flake8's config parser, and can be used just like any other flake8 options.
no-checkpoint-warning-decorators
Specify a list of decorators to disable checkpointing checks for, turning off TRIO107 and TRIO108 warnings for functions decorated with any decorator matching any in the list. Matching is done with fnmatch. Defaults to disabling for asynccontextmanager
.
Decorators-to-match must be identifiers or dotted names only (not PEP-614 expressions), and will match against the name only - e.g. foo.bar
matches foo.bar
, foo.bar()
, and foo.bar(args, here)
, etc.
For example:
no-checkpoint-warning-decorators =
mydecorator,
mydecoratorpackage.checkpointing_decorators.*,
ign*,
*.ignore,
startable-in-context-manager
Comma-separated list of methods which should be used with .start()
when opening a context manager,
in addition to the default trio.run_process
, trio.serve_tcp
, trio.serve_ssl_over_tcp
, and
trio.serve_listeners
. Names must be valid identifiers as per str.isidentifier()
. For example:
startable-in-context-manager =
myfun,
myfun2,
trio200-blocking-calls
Comma-separated list of pairs of values separated by ->
(optional whitespace stripped), where the first is a pattern for a call that should raise an error if found inside an async function, and the second is what should be suggested to use instead. It uses fnmatch as per no-checkpoint-warning-decorators
for matching. The part after ->
is not used by the checker other than when printing the error, so you could add extra info there if you want.
The format of the error message is User-configured blocking sync call {0} in async function, consider replacing with {1}.
, where {0}
is the pattern the call matches and {1}
is the suggested replacement.
Example:
trio200-blocking-calls =
my_blocking_call -> async.alternative,
module.block_call -> other_function_to_use,
common_error_call -> alternative(). But sometimes you should use other_function(). Ask joe if you're unsure which one,
dangerous_module.* -> corresponding function in safe_module,
*.dangerous_call -> .safe_call()
Specified patterns must not have parentheses, and will only match when the pattern is the name of a call, so given the above configuration
async def my_function():
my_blocking_call() # this would raise an error
x = my_blocking_call(a, b, c) # as would this
y = my_blocking_call # but not this
y() # or this
[my_blocking_call][0]() # nor this
def my_blocking_call(): # it's also safe to use the name in other contexts
...
arbitrary_other_function(my_blocking_call=None)
Changelog
23.2.1
- TRIO103 and TRIO104 no longer triggers when
trio.Cancelled
has been handled in previous except handlers. - Add TRIO117: Reference to deprecated
trio.[NonBase]MultiError
; use[Base]ExceptionGroup
instead. - Add TRIO232: blocking sync call on file object.
- Add TRIO212: blocking sync call on
httpx.Client
object. - Add TRIO222: blocking sync call to
os.wait*
- TRIO221 now also looks for
os.posix_spawn[p]
23.1.4
- TRIO114 avoids a false alarm on posonly args named "task_status"
- TRIO116 will now match on any attribute parameter named
.inf
, not justmath.inf
. - TRIO900 now only checks
@asynccontextmanager
, not other decorators passed with --no-checkpoint-warning-decorators.
23.1.3
- Add TRIO240: usage of
os.path
in async function. - Add TRIO900: ban async generators not decorated with known safe decorator
23.1.2
- Add TRIO230, TRIO231 - sync IO calls in async function
23.1.1
- Add TRIO210, TRIO211 - blocking sync call in async function, using network packages (requests, httpx, urllib3)
- Add TRIO220, TRIO221 - blocking sync call in async function, using subprocess or os.
22.12.5
- The
--startable-in-context-manager
and--trio200-blocking-calls
options now handle spaces and newlines. - Now compatible with flake8-noqa's NQA102 and NQA103 checks.
22.12.4
- TRIO200 no longer warns on directly awaited calls
22.12.3
- Worked around configuration-parsing bug for TRIO200 warning (more to come)
22.12.2
- Add TRIO200: User-configured blocking sync call in async function
22.12.1
- TRIO114 will now trigger on the unqualified name, will now only check the first parameter directly, and parameters to function calls inside that.
- TRIO113 now only supports names that are valid identifiers, rather than fnmatch patterns.
- Add TRIO115: Use
trio.lowlevel.checkpoint()
instead oftrio.sleep(0)
.
22.11.5
- Add TRIO116:
trio.sleep()
with >24 hour interval should usually betrio.sleep_forever()
.
22.11.4
- Add TRIO114 Startable function not in
--startable-in-context-manager
parameter list.
22.11.3
- Add TRIO113, prefer
await nursery.start(...)
tonursery.start_soon()
for compatible functions when opening a context manager
22.11.2
- TRIO105 now also checks that you
await
ednursery.start()
.
22.11.1
- TRIO102 is no longer skipped in (async) context managers, since it's not a missing-checkpoint warning.
22.9.2
- Fix a crash on nontrivial decorator expressions (calls, PEP-614) and document behavior.
22.9.1
- Add
--no-checkpoint-warning-decorators
option, to disable missing-checkpoint warnings for certain decorated functions.
22.8.8
- Fix false alarm on TRIO107 with checkpointing
try
and emptyfinally
- Fix false alarm on TRIO107&108 with infinite loops
22.8.7
- TRIO107+108 now ignores
asynccontextmanager
s, since both__aenter__
and__aexit__
should checkpoint.async with
is also treated as checkpointing on both enter and exit. - TRIO107 now completely ignores any function whose body consists solely of ellipsis, pass, or string constants.
- TRIO103, 107 and 108 now inspects
while
conditions andfor
iterables to avoid false alarms on a couple cases where the loop body is guaranteed to run at least once.
22.8.6
- TRIO103 now correctly handles raises in loops, i.e.
raise
in else is guaranteed to run unless there's abreak
in the body.
22.8.5
- Add 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. - Add TRIO112: this single-task nursery could be replaced by awaiting the function call directly.
22.8.4
- Fix TRIO108 raising errors on yields in some sync code.
- TRIO109 now skips all decorated functions to avoid false alarms
22.8.3
- TRIO108 now gives multiple error messages; one for each path lacking a guaranteed checkpoint
22.8.2
- Merged TRIO108 into TRIO107
- TRIO108 now handles checkpointing in async iterators
22.8.1
- Added TRIO109: Async definitions should not have a
timeout
parameter. Usetrio.[fail/move_on]_[at/after]
- Added TRIO110:
while <condition>: await trio.sleep()
should be replaced by atrio.Event
.
22.7.6
- Extend TRIO102 to also check inside
except BaseException
andexcept trio.Cancelled
- Extend TRIO104 to also check for
yield
- Update error messages on TRIO102 and TRIO103
22.7.5
- Add TRIO103:
except BaseException
orexcept trio.Cancelled
with a code path that doesn't re-raise - Add TRIO104: "Cancelled and BaseException must be re-raised" if user tries to return or raise a different exception.
- Added TRIO107: Async functions must have at least one checkpoint on every code path, unless an exception is raised
- Added TRIO108: Early return from async function must have at least one checkpoint on every code path before it.
22.7.4
- Added TRIO105 check for not immediately
await
ing async trio functions. - Added TRIO106 check that trio is imported in a form that the plugin can easily parse.
22.7.3
- Added TRIO102 check for unsafe checkpoints inside
finally:
blocks
22.7.2
- Avoid
TRIO100
false-alarms on cancel scopes containingasync for
orasync with
.
22.7.1
- Initial release with TRIO100 and TRIO101
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Hashes for flake8_trio-23.2.1-py3-none-any.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | a212522e2f0c3b6cf6f3fad1dbe7af2747af42d41896e845267d61d93914accc |
|
MD5 | 8543f20613d932f5e93f79dff2136386 |
|
BLAKE2b-256 | 3de5c36e4526d88ccda71007a450edda82ff0247e377370a6a9b430eee8c43ef |