✨ Add support for Response parameters to set headers, cookies, and status codes (#294)
* ✨ Add support for declaring a Response parameter to set headers and cookies
* ✅ Add source for docs and tests
* 📝 Add docs for setting headers, cookies and status code
* 📝 Add attribution to Hug for inspiring response parameters
@ -240,7 +240,7 @@ It was one of the first extremely fast Python frameworks based on `asyncio`. It
Falcon is another high performance Python framework, it is designed to be minimal, and work as the foundation of other frameworks like Hug.
Falcon is another high performance Python framework, it is designed to be minimal, and work as the foundation of other frameworks like Hug.
It uses the previous standard for Python web frameworks (WSGI) which is synchronous, so it can't handle Websockets and other use cases. Nevertheless, it also has a very good performance.
It uses the previous standard for Python web frameworks (WSGI) which is synchronous, so it can't handle WebSockets and other use cases. Nevertheless, it also has a very good performance.
It is designed to have functions that receive two parameters, one "request" and one "response". Then you "read" parts from the request, and "write" parts to the response. Because of this design, it is not possible to declare request parameters and bodies with standard Python type hints as function parameters.
It is designed to have functions that receive two parameters, one "request" and one "response". Then you "read" parts from the request, and "write" parts to the response. Because of this design, it is not possible to declare request parameters and bodies with standard Python type hints as function parameters.
@ -249,6 +249,10 @@ So, data validation, serialization, and documentation, have to be done in code,
!!! check "Inspired **FastAPI** to"
!!! check "Inspired **FastAPI** to"
Find ways to get great performance.
Find ways to get great performance.
Along with Hug (as Hug is based on Falcon) inspired **FastAPI** to declare a `response` parameter in functions.
Although in FastAPI it's optional, and is used mainly to set headers, cookies, and alternative status codes.
You probably read before that you can set a <ahref="https://fastapi.tiangolo.com/tutorial/response-status-code/"target="_blank">default Response Status Code</a>.
But in some cases you need to return a different status code than the default.
## Use case
For example, imagine that you want to return an HTTP status code of "OK" `200` by default.
But if the data didn't exist, you want to create it, and return an HTTP status code of "CREATED" `201`.
But you still want to be able to filter and convert the data you return with a `response_model`.
For those cases, you can use a `Response` parameter.
## Use a `Response` parameter
You can declare a parameter of type `Response` in your *path operation function* (as you can do for cookies and headers).
And then you can set the `status_code` in that *temporal* response object.
And then you can return any object you need, as you normally would (a `dict`, a database model, etc).
And if you declared a `response_model`, it will still be used to filter and convert the object you returned.
**FastAPI** will use that *temporal* response to extract the status code (also cookies and headers), and will put them in the final response that contains the value you returned, filtered by any `response_model`.
You can also declare the `Response` parameter in dependencies, and set the status code in them. But have in mind that the last one to be set will win.
You can declare a parameter of type `Response` in your *path operation function*, the same way you can declare a `Request` parameter.
And then you can set headers in that *temporal* response object.
```Python hl_lines="2 8 9"
{!./src/response_cookies/tutorial002.py!}
```
And then you can return any object you need, as you normally would (a `dict`, a database model, etc).
And if you declared a `response_model`, it will still be used to filter and convert the object you returned.
**FastAPI** will use that *temporal* response to extract the cookies (also headers and status code), and will put them in the final response that contains the value you returned, filtered by any `response_model`.
You can also declare the `Response` parameter in dependencies, and set cookies (and headers) in them.
## Return a `Response` directly
You can also create cookies when returning a `Response` directly in your code.
To do that, you can create a response as described in <ahref="https://fastapi.tiangolo.com/tutorial/response-directly/"target="_blank">Return a Response directly</a>.
To do that, you can create a response as described in <ahref="https://fastapi.tiangolo.com/tutorial/response-directly/"target="_blank">Return a Response directly</a>.
@ -8,6 +28,13 @@ Then set Cookies in it, and then return it:
{!./src/response_cookies/tutorial001.py!}
{!./src/response_cookies/tutorial001.py!}
```
```
## More info
!!! tip
Have in mind that if you return a response directly instead of using the `Response` parameter, FastAPI will return it directly.
So, you will have to make sure your data is of the correct type. E.g. it is compatible with JSON, if you are returning a `JSONResponse`.
And also that you are not sending any data that should have been filtered by a `response_model`.
### More info
To see all the available parameters and options, check the <ahref="https://www.starlette.io/responses/#set-cookie"target="_blank">documentation in Starlette</a>.
To see all the available parameters and options, check the <ahref="https://www.starlette.io/responses/#set-cookie"target="_blank">documentation in Starlette</a>.
You can declare a parameter of type `Response` in your *path operation function* (as you can do for cookies), the same way you can declare a `Request` parameter.
And then you can set headers in that *temporal* response object.
```Python hl_lines="2 8 9"
{!./src/response_headers/tutorial002.py!}
```
And then you can return any object you need, as you normally would (a `dict`, a database model, etc).
And if you declared a `response_model`, it will still be used to filter and convert the object you returned.
**FastAPI** will use that *temporal* response to extract the headers (also cookies and status code), and will put them in the final response that contains the value you returned, filtered by any `response_model`.
You can also declare the `Response` parameter in dependencies, and set headers (and cookies) in them.
## Return a `Response` directly
You can also add headers when you return a `Response` directly.
Create a response as described in <ahref="https://fastapi.tiangolo.com/tutorial/response-directly/"target="_blank">Return a Response directly</a> and pass the headers as an additional parameter:
Create a response as described in <ahref="https://fastapi.tiangolo.com/tutorial/response-directly/"target="_blank">Return a Response directly</a> and pass the headers as an additional parameter:
@ -6,7 +26,8 @@ Create a response as described in <a href="https://fastapi.tiangolo.com/tutorial
{!./src/response_headers/tutorial001.py!}
{!./src/response_headers/tutorial001.py!}
```
```
!!! tip
## Custom Headers
Have in mind that custom proprietary headers can be added <ahref="https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers"target="_blank">using the 'X-' prefix</a>.
Have in mind that custom proprietary headers can be added <ahref="https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers"target="_blank">using the 'X-' prefix</a>.
But if you have custom headers that you want a client in a browser to be able to see, you need to add them to your <ahref="https://fastapi.tiangolo.com/tutorial/cors/"target="_blank">CORS configurations</a>, using the parameter `expose_headers` documented in <ahref="https://www.starlette.io/middleware/#corsmiddleware"target="_blank">Starlette's CORS docs</a>.
But if you have custom headers that you want a client in a browser to be able to see, you need to add them to your <ahref="https://fastapi.tiangolo.com/tutorial/cors/"target="_blank">CORS configurations</a>, using the parameter `expose_headers` documented in <ahref="https://www.starlette.io/middleware/#corsmiddleware"target="_blank">Starlette's CORS docs</a>.
To know more about each status code and which code is for what, check the <ahref="https://developer.mozilla.org/en-US/docs/Web/HTTP/Status"target="_blank"><abbrtitle="Mozilla Developer Network">MDN</abbr> documentation about HTTP status codes</a>.
To know more about each status code and which code is for what, check the <ahref="https://developer.mozilla.org/en-US/docs/Web/HTTP/Status"target="_blank"><abbrtitle="Mozilla Developer Network">MDN</abbr> documentation about HTTP status codes</a>.
## Shortcut to remember the names
## Shortcut to remember the names
Let's see the previous example again:
Let's see the previous example again:
@ -69,3 +68,7 @@ You can use the convenience variables from `starlette.status`.
They are just a convenience, they hold the same number, but that way you can use the editor's autocomplete to find them:
They are just a convenience, they hold the same number, but that way you can use the editor's autocomplete to find them:
Later, in a more advanced part of the tutorial/user guide, you will see how to <ahref="https://fastapi.tiangolo.com/tutorial/response-change-status-code/"target="_blank">return a different status code than the default</a> you are declaring here.