Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enabled Event and CapacityLimiter to be instantiated outside an event loop #651

Merged
merged 9 commits into from
Dec 14, 2023

Conversation

agronholm
Copy link
Owner

Relates to #500.

@gschaffner
Copy link
Collaborator

is the motivation for this #500 (comment) / #501 (comment)? if so, @tapetersen's example code has been working as intended since #524 was merged, so a delayed instantiation trick may not be necessary. see also #523 (comment) / python-trio/sniffio#35 (comment).

@agronholm
Copy link
Owner Author

The motivation is to allow users to create events and capacity limiters on the module level, as import side effects, and in the REPL too. You can already do this with asyncio.Event() and trio.Event().

self._event.set()

def is_set(self) -> bool:
return self._internal_event is not None and self._internal_event.is_set()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I understand that if the event is not created then it cannot be set, but should is_set actually create the event?

Suggested change
return self._internal_event is not None and self._internal_event.is_set()
return self._event.is_set()

statistics also doesn't need the event to be created (it could return empty statistics if it's not), yet it creates it.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

perhaps statistics should have a special case (like is_set does) to allow lookup from outside of the event loop context?

Lock.statistics() for instance is accessible from outside of the event loop already.

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added.

def __new__(cls, total_tokens: int) -> CapacityLimiterAdapter:
return object.__new__(cls)

def __init__(self, total_tokens: int) -> None:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Line 519 checks if total_tokens is not math.inf which is a float, so should this be reflected in its type?

Suggested change
def __init__(self, total_tokens: int) -> None:
def __init__(self, total_tokens: int | float) -> None:

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In that case, the annotation should be float, as that already includes integers according to the PEP 484 numeric tower.

Copy link
Collaborator

@smurfix smurfix Dec 13, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well, on the other hand it shouldn't be any float. Only math.inf actually makes sense.

Shouldn't int | math.inf Just Work?

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No, because math.inf is not a type.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Neither is None, but that works just fine.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ugh. They should just have accepted arbitrary singletons. (I do wonder whether any type checkers do …)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it should be int | Literal[math.inf], but mypy doesn't like it. Maybe it's a bug in mypy?

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No, it's not. Floats are not valid literals according to PEP 586.

Copy link
Contributor

@davidbrochart davidbrochart Dec 13, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks.
BTW, why not using None to mean an infinite number of tokens? math.inf carries a direct meaning, but None also means "I don't want to specify a finite number of tokens", which easily translates to an infinite number of tokens IMO.

Copy link
Owner Author

@agronholm agronholm Dec 13, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It couldn't been done, but I was following Trio's API design which uses math.inf instead.

@@ -76,7 +77,7 @@ class SemaphoreStatistics:

class Event:
def __new__(cls) -> Event:
return get_async_backend().create_event()
return EventAdapter()
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

would it be cheaper to avoid using the proxy in the common case?

    def __new__(cls) -> Event:
        try:
            return get_async_backend().create_event()
        except sniffio.AsyncLibraryNotFoundError:
            return EventAdapter()

for the common case (construction under the loop context) this would move the overhead from on-every-attribute-access to on-construction.

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good idea.

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

@@ -76,7 +77,7 @@ class SemaphoreStatistics:

class Event:
def __new__(cls) -> Event:
return get_async_backend().create_event()
return EventAdapter()
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

in the async methods of high-frequency primitives that construct Events (Lock.acquire, Semaphore.acquire, Condition.wait, MemoryObjectReceiveStream.receive, MemoryObjectSendStream.send) would it be good to call get_async_backend().create_event() directly to avoid the overhead from adapters?

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

With my latest change, they won't be getting the adapters, will they?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yep, implementing #651 (comment) addressed this comment too.

(technically Event() still has some overhead (get_async_backend().create_event() vs. try: get_async_backend().create_event()) but it should be very cheap https://docs.python.org/3/faq/design.html#how-fast-are-exceptions.)

self._event.set()

def is_set(self) -> bool:
return self._internal_event is not None and self._internal_event.is_set()
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

perhaps statistics should have a special case (like is_set does) to allow lookup from outside of the event loop context?

Lock.statistics() for instance is accessible from outside of the event loop already.

@@ -208,6 +210,18 @@ async def waiter() -> None:

assert event.statistics().tasks_waiting == 0

def test_instantiate_outside_event_loop(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

to prevent the other synchronization primitives from regressing on this issue, it could make sense to add an analogous test_instantiate_outside_event_loop for each of them.

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

Copy link
Collaborator

@gschaffner gschaffner left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

one minor comment, at your discretion. otherwise LGTM!

@@ -76,7 +77,7 @@ class SemaphoreStatistics:

class Event:
def __new__(cls) -> Event:
return get_async_backend().create_event()
return EventAdapter()
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yep, implementing #651 (comment) addressed this comment too.

(technically Event() still has some overhead (get_async_backend().create_event() vs. try: get_async_backend().create_event()) but it should be very cheap https://docs.python.org/3/faq/design.html#how-fast-are-exceptions.)


@total_tokens.setter
def total_tokens(self, value: float) -> None:
self._limiter.total_tokens = value
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(obscure/minor comment) could this be delayed too?

Suggested change
self._limiter.total_tokens = value
if self._internal_limiter is None:
self._total_tokens = value
else:
self._limiter.total_tokens = value

Copy link
Collaborator

@gschaffner gschaffner Dec 14, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

mhm, i suppose that if total_tokens.setter is delayed then this adapter might not be thread-safe. is it intended to be thread-safe?

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Frankly, the idea that someone would change the adapter's total tokens from a worker thread had not even occurred to me. The whole idea seems pretty bizarre to me.

Copy link
Collaborator

@gschaffner gschaffner Dec 14, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yeah, i think that it's reasonable to require them to do the mutation in the event loop thread. CancelScope.cancel for example is not safe to call from a non-main thread or a signal handler. Event.set is not thread-safe either. it's extra not thread safe after adding EventAdapter, but it wasn't thread-safe before either.

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll revert my latest commit then?

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The more I think about it, the more I'm leaning towards allowing users to mutate total_tokens before the initialization.

Copy link
Collaborator

@gschaffner gschaffner Dec 14, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

huh, does Trio document that?

i'm peeking at the Trio code and total_tokens.setter looks thread-unsafe—there's no threading.Lock, so if two threads mutate total_tokens in parallel then two _wake_waiters will run, which can cause too many waiters to get woken up.

(anyio.CapacityLimiter.)total_tokens.setter is currently thread-unsafe, though, so i think that d8ba6de is in alignment with the status quo.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The more I think about it, the more I'm leaning towards allowing users to mutate total_tokens before the initialization.

i'm leaning this way too.

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, generally the API should be considered thread-unsafe, save for that parts that were explicitly designed to allow access from worker threads (the anyio.from_thread module).

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

oops—i misinterpreted

It might be reasonable to allow the tokens to be set from outside the event loop, just like trio.CapacityLimiter allows.

as

It might be reasonable to allow the tokens to be set from other threads, just like trio.CapacityLimiter allows.

and was replying to something you didn't say. :p

@agronholm agronholm merged commit 28516e2 into master Dec 14, 2023
16 checks passed
@agronholm agronholm deleted the lazy-event-instantiation branch December 14, 2023 10:04
@agronholm
Copy link
Owner Author

Thank you for the reviews!

@tapetersen
Copy link

is the motivation for this #500 (comment) / #501 (comment)? if so, @tapetersen's example code has been working as intended since #524 was merged, so a delayed instantiation trick may not be necessary. see also #523 (comment) / python-trio/sniffio#35 (comment).

Oh i unfortunately hadn't noticed before. Like mentioned it wasn't really hard to workaround in our case. Nice to see it getting solved outside a loop at all as well as that was the case of asyncio and why I was surprised it didn't work.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

None yet

6 participants