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

[RFC] feat: scoped requests #3229

Open
wants to merge 2 commits into
base: main
Choose a base branch
from

Conversation

jgranstrom
Copy link
Contributor

@jgranstrom jgranstrom commented Dec 3, 2024

/fixes #3197
/claim #3197

RFC Adds support for per-request Scopes at server level.

Adds the ability to run a server that provides a scope for each request. This is to not add any overhead or overly complicate the existing path that does not provide request-bound scopes.

I have not added tests or similar yet as I need some feedback on the idea of this as a solution first.
(Edit: also only implemented with the Netty driver for the above reason)

Copy link

algora-pbc bot commented Dec 3, 2024

💵 To receive payouts, sign up on Algora, link your Github account and connect with Stripe.

@jgranstrom jgranstrom force-pushed the feat-scoped-requests branch from 375bc68 to a8df79e Compare December 3, 2024 16:26
@jgranstrom jgranstrom changed the title feat: scoped requests [RFC] feat: scoped requests Dec 3, 2024
@jgranstrom jgranstrom force-pushed the feat-scoped-requests branch from a8df79e to 87329b7 Compare December 4, 2024 06:12
handler.apply
case Right(scopedHandler) =>
val handler = scopedHandler.toHandler
(req: Request) => ZIO.scoped(handler(req))
Copy link
Contributor Author

Choose a reason for hiding this comment

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

@guersam
Copy link
Contributor

guersam commented Dec 9, 2024

Thanks for tackling this issue! My two cents:

I think the Scope env could be resolved by the handler via something like Handler.scoped. It reflects the locality of each scope more precisely by not leaking the requirement, and it's more consistent with ZLayer.scoped.

@jgranstrom
Copy link
Contributor Author

Thanks for tackling this issue! My two cents:

I think the Scope env could be resolved by the handler via something like Handler.scoped. It reflects the locality of each scope more precisely by not leaking the requirement, and it's more consistent with ZLayer.scoped.

Yeah I went with that initially but it gets fairly convoluted to thread that through everything, so I thought of a less invasive option to just fork it top-level, but also agree that it would be cleaner for the outside to do that.

@jgranstrom
Copy link
Contributor Author

jgranstrom commented Jan 5, 2025

Thanks for tackling this issue! My two cents:

I think the Scope env could be resolved by the handler via something like Handler.scoped. It reflects the locality of each scope more precisely by not leaking the requirement, and it's more consistent with ZLayer.scoped.

I went over this one more time, with that approach. And I landed on what is originally in this PR being the most sensible approach to me still. It's a balance between introducing overhead everywhere to determine if a Scope is needed, vs just always creating the Scope.

For something like Handler.scoped it's not known until routes are traversed and matched whether a Scope is needed, meaning it must be created lazily when invoking the handler. However, that Scope must then leak outside of the Handler so that it can be held open and later closed by the Driver when the request-response fully completes.

Server.installScoped could be named better, or the syntax could be changed to something like Server.install(app.scoped), or some variation that is more proper.

However the option of Handler.scoped is really messy. It must compose correctly, all over the place, to retain it being Scoped with other handlers, and also onto Routes. Also applying Handlers must then be able to receive a Scope pre-apply, that is lazily created by the Driver, that is also not closed as part of running the handler itself, but by the Driver. I think that is just too much complexity, and most likely overhead, versus what I proposed here.

It also fits other scoped operators fairly well still, if moving it to Server.scoped.install(routes), as that similarly to ZIO.scoped etc allows Routes to have Scope in R, and erases it by providing it via the server. That seems reasonably ZIO-y IMO? I just threw it on installScoped for the RFC.

WDYT? @guersam @987Nabil 🙏

Edit: Also with this I was thinking to have a completely separate inbound handler when it should provide a scope. Since I must add some overhead in it to manage the scope lifetime that is not needed otherwise. That also becomes fairly straight-forward when it is decided at server-level to provide request-Scopes or not.

@guersam
Copy link
Contributor

guersam commented Jan 11, 2025

I understood your concern about Handler.scoped due to its very nature of the executable encoding.

Then what about Route(s).scoped instead of Handler.scoped that

  1. eliminates the Scope requirement from the handler, and
  2. enables a flag that the underlying handler is scoped, or create another sealed trait variant ScopedRoute so that the runtime can branch the scoping logic in a similar way to what you've done in this PR?

The implementation can be much easier because Route is more declarative than Handler is, and this approach can resolve the issue that a single leaked scope affects the signature of the entire application.

Optionally we can define a scoped route forming method as an alternative of RoutePattern#-> such as RoutePattern#-!->, but I'm also worried about that adding arbitrary symbolic operator might make the API unnecessarily cryptic.

What do you think? @jgranstrom @987Nabil

@jgranstrom
Copy link
Contributor Author

I understood your concern about Handler.scoped due to its very nature of the executable encoding.

Then what about Route(s).scoped instead of Handler.scoped that

  1. eliminates the Scope requirement from the handler, and
  2. enables a flag that the underlying handler is scoped, or create another sealed trait variant ScopedRoute so that the runtime can branch the scoping logic in a similar way to what you've done in this PR?

The implementation can be much easier because Route is more declarative than Handler is, and this approach can resolve the issue that a single leaked scope affects the signature of the entire application.

Optionally we can define a scoped route forming method as an alternative of RoutePattern#-> such as RoutePattern#-!->, but I'm also worried about that adding arbitrary symbolic operator might make the API unnecessarily cryptic.

What do you think? @jgranstrom @987Nabil

Yeah, that is what I explored with Handler as well. But I think the main difference is that either we have an API to tell the Server to provide Scopes for request, or an API where something we give to the Server affect its behavior. That's true even if we just do it on Routes.

I think my thoughts are:

  1. Do scope at Handler level, meaning applying handlers may cause the Server to create a Scope
  2. Do scope at Server level, meaning take Routes that need a Scope and have the Server provide one per request
  3. (yours): Do scope at Route level, determine in the server if the Routes need a Scope and provide one

With the concern that:

  1. will add overhead to everything because we have to pipe this through and make runtime decisions
  2. we may create Scopes that are not needed
  3. this will be similar to 1) but already so coarse that it is likely also similar to 2) especially if the server behavior is based on the composed routes anyways

I'm still on 2) but I'm open to more arguments!

@frekw
Copy link
Contributor

frekw commented Jan 14, 2025

As a user of (and sporadic contributor to) zio-http, I think it would make sense to just make the default be to always provide a Scope to every request without having to make the choice if Handler.scoped isn't feasible (which it sounds like it isn't).

Having a Scope at the request level seems like a very natural pattern for an HTTP server, and since Scopes are pervasive within all of ZIO I doubt it would add that much overhead (even if I haven't benchmarked it), especially in real-world applications.

@guersam
Copy link
Contributor

guersam commented Jan 15, 2025

I agree with @frekw . A handler might always add a scope requirement in R type like ZIOApp#run does:

sealed trait Handler[-R, +Err, -In, +Out] {
  def apply(in: In): ZIO[Scope & R, Err, Out]
}

I think the performance impact by unused scope is negligible in most real world use cases, it might affect minimal benchmarks. If it's a real concern, there could be a special variant of Route that removes Scope requirement for unscoped handlers.

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