Skip to content

Sending Responses

In Nexios, sending responses is a core part of building web applications. The Response object provides a powerful and flexible way to construct and send HTTP responses to the client. This guide covers the various methods available for sending responses, from simple JSON to complex file downloads.

The Response Object

The Response object is your primary tool for building HTTP responses. It is passed as the second argument (res) to your route handlers. One of the key features of the Response object is that its methods are chainable, allowing you to build a response in a fluent and readable way.

Chainable Responses

You can chain multiple methods together to configure the response before sending it. This makes your code more concise and expressive.

python
from nexios import NexiosApp

app = NexiosApp()

@app.get("/")
async def home(req, res):
    res.status(200).set_cookie("session_id", "123").json({"message": "Hello, World!"})

In this example, we set the status code, add a cookie, and send a JSON response all in a single, chained statement.

Sending Different Types of Responses

Nexios provides several methods for sending different types of responses.

JSON Responses

To send a JSON response, use the .json() method. It automatically sets the Content-Type header to application/json.

python
@app.get("/users")
async def get_users(req, res):
    users = [{"id": 1, "name": "Alice"}, {"id": 2, "name": "Bob"}]
    res.json(users)

HTML Responses

To send an HTML response, use the .html() method. This will set the Content-Type header to text/html.

python
@app.get("/welcome")
async def welcome(req, res):
    html_content = "<h1>Welcome to our website!</h1>"
    res.html(html_content)

Plain Text Responses

For plain text responses, use the .text() method. The Content-Type will be set to text/plain.

python
@app.get("/status")
async def status(req, res):
    res.text("Service is running.")

Redirects

To redirect the client to a different URL, use the .redirect() method.

python
@app.get("/old-path")
async def old_path(req, res):
    res.redirect("/new-path", status_code=301) # Permanent redirect

Customizing the Response

You can customize the response by setting the status code, headers, and cookies.

Setting the Status Code

Use the .status() method to set the HTTP status code.

python
@app.post("/create-user")
async def create_user(req, res):
    # some logic to create a user
    res.status(201).json({"message": "User created successfully"})

Setting Headers

Use the .set_header() method to add or modify HTTP headers.

python
@app.get("/data")
async def get_data(req, res):
    res.set_header("Cache-Control", "no-cache").json({"data": "some data"})

Setting Cookies

Use the .set_cookie() method to set a cookie on the client's browser.

python
@app.post("/login")
async def login(req, res):
    res.set_cookie(
        key="user_token",
        value="secret-token",
        httponly=True,
        max_age=3600 # 1 hour
    ).json({"message": "Logged in"})

File Responses

Nexios allows you to send files as responses using the .file() method. This is useful for serving images, documents, or other static assets.

python
@app.get("/download-report")
async def download_report(req, res):
    file_path = "path/to/your/report.pdf"
    res.file(file_path, content_disposition_type="attachment")

By setting content_disposition_type="attachment", you prompt the browser to download the file instead of displaying it.

Returning Data Directly

For simple cases, you can return a dict, list, or str directly from your handler. Nexios will automatically convert it into a JSON response.

python
@app.get("/simple")
async def simple_response(req, res):
    return {"message": "This is a simple response."}

However, for more control over the response, it is recommended to use the Response object and its methods.

💡 Recommended

For clarity and to leverage the full power of Nexios's response handling, we recommend using the res object to build your responses, especially when you need to set custom headers, cookies, or status codes.

Advanced Usage: Response Classes

For more advanced use cases, Nexios allows you to work directly with Response classes. This gives you the ultimate flexibility to control the response sent to the client. You can either use the built-in response classes or create your own.

Using Built-in Response Classes

Instead of using the res object's methods, you can return an instance of a response class directly from your handler. Nexios provides several built-in response classes in the nexios.http.response module.

  • JSONResponse
  • HTMLResponse
  • TextResponse
  • RedirectResponse
  • FileResponse
  • StreamingResponse

Example:

python
from nexios import NexiosApp
from nexios.http.response import JSONResponse, HTMLResponse

app = NexiosApp()

@app.get("/users-json")
async def get_users_json(req, res):
    users = [{"id": 1, "name": "Alice"}, {"id": 2, "name": "Bob"}]
    return JSONResponse(users, status_code=200)

@app.get("/welcome-html")
async def welcome_html(req, res):
    html_content = "<h1>Welcome from a Response class!</h1>"
    return HTMLResponse(html_content)

Creating Custom Response Classes

You can create your own custom response classes by inheriting from nexios.http.response.Response. This is useful when you need to send responses in a format that is not supported out of the box, such as XML.

Example: Creating an XMLResponse class

python
from nexios import NexiosApp
from nexios.http.response import Response
from dicttoxml import dicttoxml

class XMLResponse(Response):
    media_type = "application/xml"

    def __init__(self, content, *args, **kwargs):
        xml_content = dicttoxml(content)
        super().__init__(content=xml_content, *args, **kwargs)

app = NexiosApp()

@app.get("/data.xml")
async def get_xml_data(req, res):
    data = {"user": {"name": "John Doe", "id": "123"}}
    return XMLResponse(data)

In this example:

  1. We create a new XMLResponse class that inherits from nexios.http.response.Response.
  2. We set the media_type to application/xml.
  3. In the __init__ method, we convert the incoming dict to XML using the dicttoxml library before passing it to the parent class.
  4. Finally, we return an instance of our XMLResponse from the route handler.

By creating custom response classes, you can encapsulate response logic and reuse it across your application, leading to cleaner and more maintainable code.