Writing middleware ================== Middleware allows you to insert code that runs before and after your handlers. It is useful for cross‑cutting concerns such as logging, authentication, rate limiting or explicit routing. MicroPie defines separate middleware classes for HTTP and WebSocket connections. HTTP middleware --------------- To create HTTP middleware, subclass :class:`~micropie.HttpMiddleware` and implement two asynchronous methods: * :meth:`~micropie.HttpMiddleware.before_request` – called before the handler is executed. If this method returns a dictionary with ``status_code`` and ``body`` keys, MicroPie immediately sends that response and skips calling the handler. You can also provide additional headers via a ``headers`` key. * :meth:`~micropie.HttpMiddleware.after_request` – called after the handler has returned a response but before it is sent to the client. You can modify the status code, body or headers by returning a dictionary with updated values. Register your middleware by appending an instance to :attr:`~micropie.App.middlewares` on your application: .. code-block:: python from micropie import App, HttpMiddleware class LoggingMiddleware(HttpMiddleware): async def before_request(self, request): print(f"Incoming request: {request.method} {request.scope['path']}") # returning None continues processing async def after_request(self, request, status_code, body, headers): print(f"Response status: {status_code}") # returning None uses the original response class MyApp(App): async def index(self): return "Hello, world!" app = MyApp() app.middlewares.append(LoggingMiddleware()) WebSocket middleware -------------------- WebSocket middleware must subclass :class:`~micropie.WebSocketMiddleware` and implement two methods: * :meth:`~micropie.WebSocketMiddleware.before_websocket` – called before a WebSocket handler runs. If this method returns a dictionary with ``code`` and ``reason``, MicroPie closes the connection with the given code and reason. * :meth:`~micropie.WebSocketMiddleware.after_websocket` – called after the WebSocket handler completes. Use this to perform cleanup. Example: .. code-block:: python from micropie import App, WebSocketMiddleware class RejectAnonymous(WebSocketMiddleware): async def before_websocket(self, request): # Reject connections without a "user" query parameter if "user" not in request.query_params: return {"code": 1008, "reason": "User name required"} async def after_websocket(self, request): print("WebSocket closed") class MyApp(App): async def ws_chat(self, ws, user): await ws.accept() await ws.send_text(f"Welcome, {user}!") await ws.close() app = MyApp() app.ws_middlewares.append(RejectAnonymous()) Explicit routing and other patterns ----------------------------------- You can implement custom routing schemes by writing middleware that parses the incoming path and sets ``request._route_handler`` or ``request._ws_route_handler`` accordingly. See the examples in the ``examples/middleware`` directory for a complete implementation.