Skip to content

Commit

Permalink
Fix preferred encoding computation (#1355)
Browse files Browse the repository at this point in the history
* Decides which encoding to use depending on both `Accept-Encode` and
the preferred encoding (chooses the preferred one in case brotli and
gzip have the same q value)
* Uses `gzip` by default

This will fix #1315
  • Loading branch information
nyurik authored May 29, 2024
1 parent 73909f2 commit e4228fb
Show file tree
Hide file tree
Showing 2 changed files with 58 additions and 21 deletions.
69 changes: 53 additions & 16 deletions martin/src/srv/tiles.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
use actix_http::header::Quality;
use actix_http::ContentEncoding;
use actix_web::error::{ErrorBadRequest, ErrorNotFound};
use actix_web::error::{ErrorBadRequest, ErrorNotAcceptable, ErrorNotFound};
use actix_web::http::header::{
AcceptEncoding, Encoding as HeaderEnc, Preference, CONTENT_ENCODING,
};
Expand All @@ -21,13 +22,7 @@ use crate::utils::{
};
use crate::{Tile, TileCoord, TileData};

static PREFER_BROTLI_ENC: &[HeaderEnc] = &[
HeaderEnc::brotli(),
HeaderEnc::gzip(),
HeaderEnc::identity(),
];

static PREFER_GZIP_ENC: &[HeaderEnc] = &[
static SUPPORTED_ENC: &[HeaderEnc] = &[
HeaderEnc::gzip(),
HeaderEnc::brotli(),
HeaderEnc::identity(),
Expand Down Expand Up @@ -180,6 +175,49 @@ impl<'a> DynTileSource<'a> {
self.recompress(data)
}

/// Decide which encoding to use for the uncompressed tile data, based on the client's Accept-Encoding header
fn decide_encoding(&self, accept_enc: &AcceptEncoding) -> ActixResult<Option<ContentEncoding>> {
let mut q_gzip = None;
let mut q_brotli = None;
for enc in accept_enc.iter() {
if let Preference::Specific(HeaderEnc::Known(e)) = enc.item {
match e {
ContentEncoding::Gzip => q_gzip = Some(enc.quality),
ContentEncoding::Brotli => q_brotli = Some(enc.quality),
_ => {}
}
} else if let Preference::Any = enc.item {
q_gzip.get_or_insert(enc.quality);
q_brotli.get_or_insert(enc.quality);
}
}
Ok(match (q_gzip, q_brotli) {
(Some(q_gzip), Some(q_brotli)) if q_gzip == q_brotli => {
if q_gzip > Quality::ZERO {
Some(self.get_preferred_enc())
} else {
None
}
}
(Some(q_gzip), Some(q_brotli)) if q_brotli > q_gzip => Some(ContentEncoding::Brotli),
(Some(_), Some(_)) => Some(ContentEncoding::Gzip),
_ => {
if let Some(HeaderEnc::Known(enc)) = accept_enc.negotiate(SUPPORTED_ENC.iter()) {
Some(enc)
} else {
return Err(ErrorNotAcceptable("No supported encoding found"));
}
}
})
}

fn get_preferred_enc(&self) -> ContentEncoding {
match self.preferred_enc {
None | Some(PreferredEncoding::Gzip) => ContentEncoding::Gzip,
Some(PreferredEncoding::Brotli) => ContentEncoding::Brotli,
}
}

fn recompress(&self, tile: TileData) -> ActixResult<Tile> {
let mut tile = Tile::new(tile, self.info);
if let Some(accept_enc) = &self.accept_enc {
Expand All @@ -198,18 +236,12 @@ impl<'a> DynTileSource<'a> {
}

if tile.info.encoding == Encoding::Uncompressed {
let ordered_encodings = match self.preferred_enc {
Some(PreferredEncoding::Gzip) | None => PREFER_GZIP_ENC,
Some(PreferredEncoding::Brotli) => PREFER_BROTLI_ENC,
};

// only apply compression if the content supports it
if let Some(HeaderEnc::Known(enc)) = accept_enc.negotiate(ordered_encodings.iter())
{
if let Some(enc) = self.decide_encoding(accept_enc)? {
// (re-)compress the tile into the preferred encoding
tile = encode(tile, enc)?;
}
}

Ok(tile)
} else {
// no accepted-encoding header, decode the tile if compressed
Expand Down Expand Up @@ -270,6 +302,11 @@ mod tests {
use super::*;
use crate::srv::server::tests::TestSource;

#[actix_rt::test]
async fn test_deleteme() {
test_enc_preference(&["gzip", "deflate", "br", "zstd"], None, Encoding::Gzip).await;
}

#[rstest]
#[trace]
#[case(&["gzip", "deflate", "br", "zstd"], None, Encoding::Gzip)]
Expand Down
10 changes: 5 additions & 5 deletions martin/tests/mb_server_test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,8 @@ use actix_web::test::{call_service, read_body, read_body_json, TestRequest};
use ctor::ctor;
use indoc::indoc;
use insta::assert_yaml_snapshot;
use martin::decode_gzip;
use martin::srv::SrvConfig;
use martin::{decode_brotli, decode_gzip};
use tilejson::TileJSON;

pub mod utils;
Expand Down Expand Up @@ -211,7 +211,7 @@ async fn mbt_get_mvt_brotli() {
assert_eq!(response.headers().get(CONTENT_ENCODING).unwrap(), "br");
let body = read_body(response).await;
assert_eq!(body.len(), 871); // this number could change if compression gets more optimized
let body = martin::decode_brotli(&body).unwrap();
let body = decode_brotli(&body).unwrap();
assert_eq!(body.len(), 1828);
}

Expand Down Expand Up @@ -267,10 +267,10 @@ async fn mbt_get_raw_mvt_gzip_br() {
response.headers().get(CONTENT_TYPE).unwrap(),
"application/x-protobuf"
);
assert_eq!(response.headers().get(CONTENT_ENCODING).unwrap(), "br");
assert_eq!(response.headers().get(CONTENT_ENCODING).unwrap(), "gzip");
let body = read_body(response).await;
assert_eq!(body.len(), 871); // this number could change if compression gets more optimized
let body = martin::decode_brotli(&body).unwrap();
assert_eq!(body.len(), 1107); // this number could change if compression gets more optimized
let body = decode_gzip(&body).unwrap();
assert_eq!(body.len(), 1828);
}

Expand Down

0 comments on commit e4228fb

Please sign in to comment.