ASGI Request Duration is a middleware for ASGI applications that measures the duration of HTTP requests and integrates this information into response headers and log records. This middleware is designed to be easy to integrate and configure, providing valuable insights into the performance of your ASGI application.
Note: If you find this project useful, please consider giving it a star ⭐ on GitHub. This helps prioritize its maintenance and development. If you encounter any typos, bugs 🐛, or have new feature requests, feel free to open an issue. I will be happy to address them.
- Measure the duration of each HTTP request.
- Add the request duration to response headers.
- Integrate the request duration into log records.
- Configurable header name and precision.
- Exclude specific paths from timing.
You can install the package using pip:
pip install asgi-request-duration
To use the middleware, add it to your ASGI application:
from asgi_request_duration.middleware import RequestDurationMiddleware
from starlette.applications import Starlette
app = Starlette()
app.add_middleware(RequestDurationMiddleware)
To use the logging filter, configure your logger to use the RequestDurationFilter
:
import logging
from asgi_request_duration.filters import RequestDurationFilter
logging.basicConfig(level=logging.INFO)
logger = logging.getLogger("myapp")
logger.addFilter(RequestDurationFilter())
You can configure the middleware by passing parameters to the RequestDurationMiddleware
:
excluded_paths
: List of paths to exclude from timing.header_name
: The name of the header to store the request duration.precision
: The precision of the recorded duration.skip_validate_header_name
: Flag to skip header name validation.skip_validate_precision
: Flag to skip precision validation.
Example:
app.add_middleware(
RequestDurationMiddleware,
excluded_paths=["^/health/?$"],
header_name="x-request-duration",
precision=3,
skip_validate_header_name=False,
skip_validate_precision=False,
)
You can configure the logging filter by passing parameters to the RequestDurationFilter
:
context_key
: The key to retrieve the request duration context value.default_value
: The default value if the request duration context key is not found.
Example:
logger.addFilter(RequestDurationFilter(context_key="request_duration", default_value="-"))
Here is a complete example of how to use the middleware with the Starlette framework. For more examples and detailed usage, please refer to the examples folder in the repository.
from asgi_request_duration import RequestDurationMiddleware
from starlette.applications import Starlette
from starlette.requests import Request
from starlette.responses import JSONResponse
from starlette.routing import Route
from uvicorn import run
async def info_endpoint(request: Request) -> JSONResponse:
return JSONResponse({"message": "info"})
async def excluded_endpoint(request: Request) -> JSONResponse:
return JSONResponse({"message": "excluded"})
routes = [
Route("/info", info_endpoint, methods=["GET"]),
Route("/excluded", excluded_endpoint, methods=["GET"]),
]
app = Starlette(routes=routes)
app.add_middleware(
RequestDurationMiddleware,
excluded_paths=["/excluded"],
header_name="x-request-duration",
precision=4,
skip_validate_header_name=False,
skip_validate_precision=False,
)
if __name__ == "__main__":
run(app, host='127.0.0.1', port=8000)
Contributions are welcome! Please refer to the CONTRIBUTING.md file for guidelines on how to contribute to this project.
This project is licensed under the GNU GPLv3 License. See the LICENSE file for more details.