-
Notifications
You must be signed in to change notification settings - Fork 98
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
feat(server): Add support for async server function, module.server, and express module #1842
base: main
Are you sure you want to change the base?
feat(server): Add support for async server function, module.server, and express module #1842
Conversation
Is your use case for setting up data? When are you needing the result of the information? (I guess, can you give a tiny motivation example that chatlas would use?) Why not put it behind an async reactive calculation? Then the value can be used accordingly. Ex: From async def server(input):
data = await get_data() to def server(input):
@reactive.calc
async def data():
return await get_data() |
Inside of chatlas, the server function is defined for you and I want to be able to access the session headers to get the user session token when using chat.app() in chat.app() async def server(input, output, session): # noqa: A002
if session_callback:
await session_callback(session) May be too niche of a problem, but was surprised to see shiny didnt support both. And afaict, no reason that it can't. |
Callable[[Inputs], Awaitable[None] | None] | ||
| Callable[[Inputs, Outputs, Session], Awaitable[None] | None] | ||
| None |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Because this can be defined with an async function, can we use wrap_async
when storing the fn value below?
This will always upgrade the function to async. Then in _session.py
, we will always await
the server execution as the run method is already an async method.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The current cast
calls are lying as to what values are already there. Please update them accordingly.
By wrapping all server functions to be async, we won't need to check if we need to await or not (as we'll always need to await).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sweet. Makes sense to me.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Though, now looking into support server function to be async, would need to go down the rabbit hole of ensure that we tackle other side effects like the module.server() decorator which is not async. Could leave it only supporting sync i suppose.. curious on thoughts from @wch.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Happy to have all server-like functions upgraded to async
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree with @schloerke - it makes sense to make everything async-capable. Hopefully that's not too much more work.
Fair! |
Callable[[Inputs], Awaitable[None] | None] | ||
| Callable[[Inputs, Outputs, Session], Awaitable[None] | None] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Cosmetic split.
Callable[[Inputs], Awaitable[None] | None] | |
| Callable[[Inputs, Outputs, Session], Awaitable[None] | None] | |
Callable[[Inputs], None] | |
| Callable[[Inputs, Outputs, Session], None] | |
| Callable[[Inputs], Awaitable[None]] | |
| Callable[[Inputs, Outputs, Session], Awaitable[None]] |
I remember something from past return types that it's a little better to split Awaitable
from the return type as it can't actually return both, but it is really two independent types.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Agreed, they are different types and need separate entries for correctness.
def not_is_async_callable( | ||
obj: Callable[P, T] | Callable[P, Awaitable[T]], | ||
) -> TypeGuard[Callable[P, T]]: | ||
return not is_async_callable(obj) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Instead of adding this function, can we change is_async_callable()
to a TypeIs
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I just saw the comment below about how TypeIs
didn't work. Can you add a comment explaining the problem? Otherwise we'll keep coming back to this and asking TypeIs
in the future.
@@ -262,15 +262,15 @@ def private_seed() -> Generator[None, None, None]: | |||
|
|||
|
|||
def wrap_async( | |||
fn: Callable[P, R] | Callable[P, Awaitable[R]], | |||
fn: Callable[P, R] | Callable[P, Awaitable[R]] | Callable[P, Awaitable[R] | R], |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think the change on this line shouldn't be necessary, after making the changes above in the signature of __init__
.
I could be wrong, but I don't believe it's possible for a function in Python to have a return value of Awaitable[R] | R
.
) -> Callable[P, Awaitable[R]]: | ||
""" | ||
Given a synchronous function that returns R, return an async function that wraps the | ||
original function. If the input function is already async, then return it unchanged. | ||
""" | ||
|
||
if is_async_callable(fn): | ||
return fn | ||
return cast(Callable[P, Awaitable[R]], fn) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this change necessary?
I wonder if there's a way to define a generic that handles the sync/async function cases. I asked Claude and it gave me something, but it will require a closer look to make sure it's right, and perhaps it should go in a separate PR that puts it in types.py. from typing import TypeVar, Awaitable, Protocol, Callable, Union, ParamSpec, cast, overload
# Define a covariant TypeVar for the return type
R_co = TypeVar('R_co', covariant=True)
# Define a parameter specification for flexible argument handling
P = ParamSpec('P')
# More descriptive type alias
MaybeAwaitableCallable = Union[
Callable[P, R_co],
Callable[P, Awaitable[R_co]]
]
async def wrap(fn: MaybeAwaitableCallable[P, None], *args: P.args, **kwargs: P.kwargs) -> int:
# Implementation remains the same
result = fn(*args, **kwargs)
# Note: We would use is_async_callable() instead of isinstance()
if isinstance(result, Awaitable):
await result
return 42
## Example usage
def sync_function(a: int, b: str) -> None:
print(f"Sync: {a}, {b}")
async def async_function(a: int, b: str) -> None:
print(f"Async: {a}, {b}")
async def test() -> None:
# Both of these are valid
await wrap(sync_function, 1, "test")
await wrap(async_function, 2, "async") |
Coming across a use case using chatlas where I need to be able to await an async function in my server function but shiny does not currently support this.
Simple addition to
_session.py
to handle both sync and async.