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

Propose Middleware Solution #33

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

Conversation

drujensen
Copy link
Contributor

This is a proposal and code sample for adding middleware to the light bug project.

Proposal

The chain of responsibility design pattern is used to chain together pieces of middleware that will each handle some aspect of the Request/Response. This allows someone to choose which middleware they want and allow them to remove or replace anything they need. It also supports adding their own custom middleware.

The Context is used to wrap the HTTPRequest and provided a dictionary of parameters. It should support AnyType of value so you can add middleware to handle authentication or parsing cookies, query strings, paths and provide that to other middleware or handlers.

In the middleware package, I've provided 8 example middleware handlers: Error, Compression, Logger, CORS, Basic Authentication, Static, Router, and NotFound.

The Router middleware is special in that it allows for different leaf nodes to be registered for a particular Request Pattern. This can become complex with routing based on any method, path, header combination. Also, it will need to support variable paths. The one provided is very rudimentary. We will want one that uses a high speed tree structure for the best performance.

The HTTPHandler trait is used by the Router for the leaf nodes. We can add a LambdaHandler and support Lambdas when they become available.

The NotFound is the last item in the chain. If no other middleware handles the Request, it will return a 404, not found response. By not hard coding this in the Router, we can support multiple routers in the chain which provides flexibility for mounting other projects or have Authorized vs Unauthorized routes.

The MiddlewareChain handles creating the linked list of Middleware. I added support for the existing HTTPService trait so it can easily be used instead of the existing services.

The lightbug_http.mojo was modified to demonstrate the use of the Middleware.

I also renamed the lightbug_welcome.html to index.html so the StaticMiddleware could be used.

Current Issues:

  1. Traits are not fully supported yet so the next in the middleware and the root in the chain are failing to compile:
/middleware/router.mojo:11:5: error: TODO: dynamic traits not supported yet, please use a compile time generic inste
ad of 'Middleware'
    var next: Middleware
  1. The Dict collection requires the value to support the CollectionElement trait. Both the HTTPHandler and AnyType are traits and not implementations so this causes issues with the Dict in the Router and Dict in the Context that holds the parameters.
/middleware/router.mojo:12:30: error: 'Dict' parameter #1 has 'CollectionElement' type, but value has type 'AnyTrait
[HTTPHandler]'
    var routes: Dict[String, HTTPHandler]

@saviorand
Copy link
Owner

This looks really good! Thank you for this contribution.
For me the most important question is how to distribute this functionality between this library (lightbug_http) and potential future libraries like lightbug_api or lightbug_web which are all supposed to build on each other.

I looked at how this is done in other languages. Seems like middleware infrastructure is normally already included in basic libraries like lightbug_http, e.g. Python's Starlette that I'm heavily using as inspiration for lightbug_http includes the middleware class and two implementations: ServerErrorMiddleware and ExceptionMiddleware.
Deciding where to put the context part is a bit more challenging. I've noticed at least 2-3 different approaches to this.

  1. Starlette does not include context by default in the library, there's a third-party package that can be imported to provide this.
  2. In Golang, context is part of the standard library and usually just imported in addition to a framework, which is nice. Not sure if it will be added to Mojo stdlib, at least it is not there currently.
  3. In Rust, e.g. Actix, which I think lightbug_http is also in spirit of, does include context in the package, but it also has the concept of actors which carry context, which we probably (?) wouldn't want to explore.

About the technical issues you mentioned:

  • For issue 1, I had to deal with this before. The way I did this is to actually cut down on the use of traits so far due to e.g no support for dynamic traits. Same here, we could just remove traits or make them serve purely as a reference. For example, the BasicAuthMiddleware might not inherit Middleware for now. Not pretty, but should allow us to start using/iterating on this early. Maybe there's a better solution for this, not sure...
  • Issue 2 is trickier, because we don't know what kind of handlers the users will write. Maybe we can just pass a fn() instead of a handler trait. I know this is done in systems programming, e.g see this Golang library for middlewares. But if we find a way to make the trait work this is of course preferred

@drujensen
Copy link
Contributor Author

Thanks for taking a look at the proposal.

It looks like Mojo will need to mature a bit before we can build out a proper web framework. I'm hoping they provide a standard for creating middleware built-in to the Mojo standard library. The Crystal language did this and it helped with compatibility between frameworks.

Regarding the issues, I think rust like traits, go like interfaces or OO inheritance will be needed for the duck typing to work. Looking forward to when Mojo properly supports this capability.

@saviorand
Copy link
Owner

Posted a thread on the questions channel in Discord, perhaps the community will have some ideas.
image

@saviorand
Copy link
Owner

@drujensen artemiogr97 from Discord suggested we do this:

@value
struct RouterMiddleware[HTTPHandlerType: HTTPHandler](Middleware):
    var next: Middleware
    var routes: Dict[String, HTTPHandlerType]

    fn __init__(inout self):
        self.routes = Dict[String, HTTPHandlerType]()

I tried it out, seems to work! Compiler error went away. Guess I should've read Mojo docs more closely 😅

@saviorand
Copy link
Owner

@drujensen should we continue to pursue this, since duck typing seems to work?

@drujensen
Copy link
Contributor Author

@savioran I have tried the approach but haven't figured it out yet. I'm a nooby to Mojo and the metaprogramming stuff takes a bit to get used too. The documentation is sparse and copilot is useless. I do have the lsp working in neovim so that should help make the process a bit smoother. I will continue to look into this and see if I can get a working version.

@drujensen drujensen force-pushed the dj/propose-middleware-solution branch from 04187a0 to 393bdfe Compare June 2, 2024 17:42
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
Status: Backlog
Development

Successfully merging this pull request may close these issues.

2 participants