mozilla-services / python-dockerflow

A Python package to implement tools and helpers for Mozilla Dockerflow
https://python-dockerflow.readthedocs.io
Mozilla Public License 2.0
38 stars 21 forks source link

(fastapi) Log request ID when set in headers #100

Closed leplatrem closed 6 months ago

leplatrem commented 6 months ago

I realized that tools like asgi-correlation-id for FastAPI add this header.

https://github.com/snok/asgi-correlation-id/blob/7665c31af175de194a8b1f051b71d25ea18c5187/asgi_correlation_id/middleware.py#L83-L86

However I don't really like it to be hard-coded, and would welcome any suggestion.

Option 0

This PR, hard-coded. But in the end, the header name X-Request-ID is almost like a defacto standard, and very unlikely to be customized.

Option 1

Include asgi-correlation-id by default as part of fastapi extras dependencies, and mention middleware in docs or introduce a dockerflow.setup(app) function that does all the cabling

Option 2

Introspect app.user_middleware list, and look for AsgiCorrelationMiddleware to read its header_name attribute https://github.com/encode/starlette/blob/a4cd0b5506ac2fd3fa0ce1e967f019b94df87b97/starlette/applications.py#L142C14-L142C29

Option 3

Read header name from a constant in app.state dict... (and let consumer app be responsible of setting it / aligning it with AsgiCorrelationMiddleware header_name param, etc.)

app.state.DOCKER_REQUEST_ID_HEADER_NAME = "X-Request-ID"
app.add_middleware(AsgiCorrelationMiddleware(header_name=app.state.DOCKER_REQUEST_ID_HEADER_NAME)

Option 4

?

grahamalama commented 6 months ago

X-Request-ID is almost like a defacto standard.

Almost, but it seems like Correlation-Id is common enough that it warrants a customization option here.

I like the idea of Option 1, whether we actually use asgi-correlation-id or maybe even roll our own version and ship it directly with python-dockerflow. It's really not that much code. I also wonder if it would end up being useful in the other integrations.

Options 2 feels brittle.

Option 3 looks pretty good too.

leplatrem commented 6 months ago

I also wonder if it would end up being useful in the other integrations.

There's already something in the other integrations: https://github.com/mozilla-services/python-dockerflow/blob/c93963707645fb7ebdf353a76f13f4b204796c94/src/dockerflow/sanic/app.py#L133 https://github.com/mozilla-services/python-dockerflow/blob/c93963707645fb7ebdf353a76f13f4b204796c94/src/dockerflow/flask/app.py#L191 https://github.com/mozilla-services/python-dockerflow/blob/c93963707645fb7ebdf353a76f13f4b204796c94/src/dockerflow/django/middleware.py#L36 but does not read it from a header that would be set by the reverse proxy (which is the ideal since it would be tracked along the whole stack)

I like the idea of Option 1

I also have a preference for Option 1, because in practice we will use this package in our services. And in the end, Option 3 consists in maintaining (and documenting) an abstraction that does not seem to bring much value

codecov-commenter commented 6 months ago

Codecov Report

All modified and coverable lines are covered by tests :white_check_mark:

Project coverage is 98.02%. Comparing base (2ba0254) to head (a0efa7a). Report is 3 commits behind head on main.

Additional details and impacted files ```diff @@ Coverage Diff @@ ## main #100 +/- ## ========================================== + Coverage 97.97% 98.02% +0.05% ========================================== Files 22 22 Lines 692 710 +18 Branches 92 96 +4 ========================================== + Hits 678 696 +18 Misses 8 8 Partials 6 6 ```

:umbrella: View full report in Codecov by Sentry.
:loudspeaker: Have feedback on the report? Share it here.