Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Content-Length is not set when using compression #3035

Closed
1 task done
mladedav opened this issue Nov 17, 2024 · 8 comments
Closed
1 task done

Content-Length is not set when using compression #3035

mladedav opened this issue Nov 17, 2024 · 8 comments
Labels

Comments

@mladedav
Copy link
Collaborator

mladedav commented Nov 17, 2024

  • I have looked for existing issues (including closed) about this

Bug Report

Version

0.7.9 and main

Platform

All

Crates

axum

Description

When the following code is run and a request is made against / with no accept-encoding header, we used to send a response with content-length but we're instead sending transfer-encoding: chunked now. If compression is applied (e.g. accept-encoding: gzip is set for the request to the example code here), we use transfer-encoding: chunked in the current and previous versions.

use std::future::ready;

use axum::{routing::get, Router};
use tower_http::compression::CompressionLayer;

#[tokio::main]
async fn main() {
    let app = Router::new()
        .route("/", get(|| ready("\n")).layer(CompressionLayer::new()));

    let listener = tokio::net::TcpListener::bind("127.0.0.1:3000")
        .await
        .unwrap();
    axum::serve(listener, app).await.unwrap();
}

This was flagged by crates-io. I would expect most clients to handle this change without any issues but there theoretically could be some that would reject any response without content-length set so I'm not sure if the upgrade would be completely safe. If we assume that all clients should be able to handle this, we can close this as not a bug.

The most likely change that had an impact on this behavior would be #2897. I did not look closer into the content-length setting code yet.

This change is most likely not just about compression but possibly any layer that wraps the body.

As a follow up, I think we should consider (in a separate issue if it doesn't turn out to be the easiest solution) whether we could set content-length for compressed responses where the whole body is already present.

@mladedav
Copy link
Collaborator Author

@SabrinaJewson pinging since you made the patch regarding content length and layers. If you have any idea how to make this work, that would be appreciated, if not, I'll try to dig into it when I get some time.

@jplatte
Copy link
Member

jplatte commented Nov 17, 2024

I thought I had read that content-length referred to the un-encoded form when the response body is compressed, however I found this sentence on MDN:

When the Content-Encoding header is present, other metadata (e.g., Content-Length) refer to the encoded form of the data, not the original resource, unless explicitly stated.

Actually looking at previous tower-http issues, I found that CompressionService should already have been removing the Content-Length previously, because with the streaming compression implemented there, it's impossible to know up-front how big the content is going to be.

So maybe this actually fixed a bug for crates.io? It seems weird, do HTTP clients not raise an error when they receive fewer body bytes than the Content-Length indicates? More questions than answers here really. cc @Turbo87

@jplatte jplatte added the A-axum label Nov 17, 2024
@mladedav
Copy link
Collaborator Author

There never was a scenario when less bytes were sent than declared in the content length header.

When compression occurred, it used chunked encoding. As far as I can tell the only difference is when there is a compression layer that does nothing because of a missing accept encoding header in the request. In that case, we used to send the content length, but we do not anymore.

@jplatte
Copy link
Member

jplatte commented Nov 17, 2024

Oh, I see. Interesting! That sounds like it may be fixable by adding a size hint to tower-http's compression body if it's in no-compression mode?

@SabrinaJewson
Copy link
Contributor

I agree with jplatte, so I opened tower-rs/tower-http#531

@mladedav
Copy link
Collaborator Author

So the fix for anyone looking for the original behavior should be to just wait for new tower-http release and upgrade that along with axum.

I'll probably try this with crates.io when I get some time to see if there are any other layers there that have similar effect and also need the size hint.

@jplatte
Copy link
Member

jplatte commented Nov 18, 2024

New version of tower-http is out (thanks to Sean). Does that fix the issue?

@Turbo87
Copy link
Collaborator

Turbo87 commented Nov 18, 2024

rust-lang/crates.io#9994 updated crates.io to the new tower-http release, and with that rust-lang/crates.io#9969 (the axum update) is now green too. big thank you to everyone involved! :)

@jplatte jplatte closed this as not planned Won't fix, can't repro, duplicate, stale Nov 18, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

No branches or pull requests

4 participants