skyhook 0.5.51

Application server for Ordinary
Documentation
use crate::server::{GMT_FORMAT, OrdinaryAppServerState};
use axum::extract::{Path, State};
use axum::http::StatusCode;
use axum::http::header;
use axum::response::{IntoResponse, Response};
use hyper::HeaderMap;
use hyper::http::HeaderValue;
use ordinary_config::CompressionAlgorithm;
use ordinary_utils::middleware::check_if_none_match;
use std::sync::Arc;
use time::{Duration, OffsetDateTime, UtcDateTime};

pub async fn get(
    State(state): State<Arc<OrdinaryAppServerState>>,
    Path(path): Path<String>,
    headers: HeaderMap,
) -> impl IntoResponse {
    get_asset_to_res(state, path, headers)
}

#[allow(clippy::needless_pass_by_value)]
pub fn get_asset_to_res(
    state: Arc<OrdinaryAppServerState>,
    path: String,
    headers: HeaderMap,
) -> impl IntoResponse {
    let span = tracing::info_span!("assets");

    span.in_scope(|| {
        let span = tracing::info_span!("storage");

        span.in_scope(|| {
            let no_compress = if let Some(ext) = path.split('.').next_back() {
                matches!(
                    ext,
                    "png"
                        | "apng"
                        | "gif"
                        | "jpg"
                        | "jpeg"
                        | "bmp"
                        | "tif"
                        | "tiff"
                        | "webp"
                        | "avif"
                        | "ico"
                        | "pdf"
                )
            } else {
                false
            };

            if !no_compress
                && let Some(assets_config) = &state.config.assets
                && let Some(precompression) = &assets_config.precompression
                && let Some(compressions) = headers.get(header::ACCEPT_ENCODING)
                && let Ok(compressions_str) = compressions.to_str()
            {
                for alg in precompression {
                    if compressions_str.contains(alg.as_str())
                        && let Some(value) =
                            get_asset_with_compression(&state, &path, &headers, Some(alg))
                    {
                        return value;
                    }
                }
            }

            if let Some(value) = get_asset_with_compression(&state, &path, &headers, None) {
                return value;
            }

            StatusCode::NOT_FOUND.into_response()
        })
    })
}

fn get_asset_with_compression(
    state: &Arc<OrdinaryAppServerState>,
    path: &str,
    headers: &HeaderMap,
    compression: Option<&CompressionAlgorithm>,
) -> Option<Response> {
    let span = tracing::info_span!("asset");

    span.in_scope(|| {
        if let Ok(asset) = state.storage.asset.get(path, compression)
            && let Ok(reader) = flexbuffers::Reader::get_root(asset.as_ref())
        {
            let vec = reader.as_vector();

            let etag = vec.idx(2).as_str();
            let last_modified = vec.idx(3).as_str();

            let mut header_map = HeaderMap::with_capacity(9);
            header_map.insert(
                header::VARY,
                HeaderValue::from_static(header::ACCEPT_ENCODING.as_str()),
            );

            if let Ok(etag) = HeaderValue::from_str(etag) {
                header_map.insert(header::ETAG, etag);
            }
            if let Ok(last_modified) = HeaderValue::from_str(last_modified) {
                header_map.insert(header::LAST_MODIFIED, last_modified);
            }
            if let Some(assets) = &state.config.assets {
                if let Some(http_cache_control) = &assets.internal_cache_control_header_value
                    && let Ok(cache_control) = HeaderValue::from_str(http_cache_control.as_str())
                {
                    header_map.insert(header::CACHE_CONTROL, cache_control);
                }

                if let Some(http_cache) = &assets.http
                    && let Some(expires_s) = http_cache.expires
                {
                    let future = UtcDateTime::now() + Duration::seconds(expires_s.cast_signed());

                    if let Ok(formatted) = future.format(&GMT_FORMAT)
                        && let Ok(expires) = HeaderValue::from_str(formatted.as_str())
                    {
                        header_map.insert(header::EXPIRES, expires);
                    }
                }
            }

            if let Some(etag) = check_if_none_match(headers, etag)
                && let Ok(etag_header) = HeaderValue::from_str(etag)
            {
                header_map.insert(header::ETAG, etag_header);

                return Some((StatusCode::NOT_MODIFIED, header_map).into_response());
            } else if let Some(if_modified_since) = headers.get(header::IF_MODIFIED_SINCE)
                && let Ok(if_modified_since_str) = if_modified_since.to_str()
                && let Ok(if_modified_since) =
                    OffsetDateTime::parse(if_modified_since_str, &GMT_FORMAT)
                && let Ok(last_modified) = OffsetDateTime::parse(last_modified, &GMT_FORMAT)
                && if_modified_since >= last_modified
            {
                return Some((StatusCode::NOT_MODIFIED, header_map).into_response());
            }

            let mime = vec.idx(0).as_str();

            if let Ok(mime) = HeaderValue::from_str(mime) {
                header_map.insert(header::CONTENT_TYPE, mime);
            }

            if let Some(compression) = compression {
                header_map.insert(
                    header::CONTENT_ENCODING,
                    HeaderValue::from_static(compression.as_str()),
                );
            }

            return Some(
                (
                    StatusCode::OK,
                    header_map,
                    bytes::Bytes::copy_from_slice(vec.idx(1).as_blob().0),
                )
                    .into_response(),
            );
        }

        None
    })
}