ripress 2.5.1

An Express.js-inspired web framework for Rust
Documentation
use bytes::Bytes;
#[cfg(not(feature = "with-wynd"))]
use http_body_util::Full;
use hyper::Response;

#[cfg(not(feature = "with-wynd"))]
use crate::app::api_error::ApiError;
use crate::res::response_cookie::Cookie;
use crate::res::{HttpResponse, HttpResponseError, ResponseBody};

#[cfg(feature = "with-wynd")]
use crate::app::api_error::ApiError;
use crate::helpers::determine_content_type_response;
use crate::res::{
    response_headers::ResponseHeaders, response_status::StatusCode, ResponseBodyType,
};
use futures::StreamExt;
use http_body_util::BodyExt;
#[cfg(feature = "with-wynd")]
use http_body_util::Full;
use hyper::header::{HeaderName, HeaderValue, CONTENT_LENGTH, SET_COOKIE};
use std::convert::Infallible;

impl HttpResponse {
    #[cfg(feature = "with-wynd")]
    #[doc(hidden)]
    pub async fn from_hyper_response(res: &mut Response<Full<Bytes>>) -> Result<Self, ApiError> {
        let collected = res.body_mut().collect().await?;
        let body_bytes = collected.to_bytes();

        let content_type_hdr = res
            .headers()
            .get(hyper::header::CONTENT_TYPE)
            .and_then(|h| h.to_str().ok());

        let content_type = content_type_hdr
            .map(determine_content_type_response)
            .unwrap_or(ResponseBodyType::BINARY);

        let body = match content_type {
            ResponseBodyType::BINARY => ResponseBody::new_binary(body_bytes),
            ResponseBodyType::TEXT => {
                let text = String::from_utf8(body_bytes.to_vec())
                    .unwrap_or_else(|_| String::from_utf8_lossy(&body_bytes).into_owned());
                ResponseBody::new_text(text)
            }
            ResponseBodyType::JSON => {
                let json_value =
                    serde_json::from_slice(&body_bytes).unwrap_or(serde_json::Value::Null);
                ResponseBody::new_json(json_value)
            }
            ResponseBodyType::HTML => {
                let html = String::from_utf8(body_bytes.to_vec())
                    .unwrap_or_else(|_| String::from_utf8_lossy(&body_bytes).into_owned());
                ResponseBody::new_html(html)
            }
        };

        let status_code = StatusCode::from_u16(res.status().as_u16());
        let mut headers = ResponseHeaders::new();

        for (key, value) in res.headers().iter() {
            if let Ok(v) = value.to_str() {
                headers.insert(key.as_str(), v);
            }
        }
        for value in res.headers().get_all(SET_COOKIE).iter() {
            if let Ok(v) = value.to_str() {
                headers.insert("Set-Cookie", v);
            }
        }

        Ok(HttpResponse {
            body,
            status_code,
            headers,
            cookies: Vec::new(),
            stream: None,
        })
    }
    #[cfg(not(feature = "with-wynd"))]
    #[doc(hidden)]
    pub async fn from_hyper_response(res: &mut Response<Full<Bytes>>) -> Result<Self, ApiError> {
        let collected = res.body_mut().collect().await?;
        let body_bytes = collected.to_bytes();

        let content_type_hdr = res
            .headers()
            .get(hyper::header::CONTENT_TYPE)
            .and_then(|h| h.to_str().ok());

        let content_type = content_type_hdr
            .map(determine_content_type_response)
            .unwrap_or(ResponseBodyType::BINARY);

        let body = match content_type {
            ResponseBodyType::BINARY => ResponseBody::new_binary(body_bytes),
            ResponseBodyType::TEXT => {
                let text = String::from_utf8(body_bytes.to_vec())
                    .unwrap_or_else(|_| String::from_utf8_lossy(&body_bytes).into_owned());
                ResponseBody::new_text(text)
            }
            ResponseBodyType::JSON => {
                let json_value =
                    serde_json::from_slice(&body_bytes).unwrap_or(serde_json::Value::Null);
                ResponseBody::new_json(json_value)
            }
            ResponseBodyType::HTML => {
                let html = String::from_utf8(body_bytes.to_vec())
                    .unwrap_or_else(|_| String::from_utf8_lossy(&body_bytes).into_owned());
                ResponseBody::new_html(html)
            }
        };

        let status_code = StatusCode::from_u16(res.status().as_u16());

        let headers = ResponseHeaders::from(std::mem::take(res.headers_mut()));

        Ok(HttpResponse {
            body,
            status_code,
            headers,
            cookies: Vec::new(),
            stream: None,
        })
    }

    #[doc(hidden)]
    pub async fn to_hyper_response(self) -> Result<Response<Full<Bytes>>, Infallible> {
        let body = self.body;

        if self.stream.is_some() {
            let response = Response::builder()
                .status(self.status_code.as_u16())
                .header("Content-Type", "text/event-stream")
                .header("Connection", "keep-alive");

            let mut header_map = self.headers.into_header_map();

            header_map.remove(hyper::header::CONTENT_TYPE);
            header_map.remove(hyper::header::CONNECTION);

            for c in self.cookies.iter() {
                match c {
                    Cookie::AddCookie(c) => {
                        let mut cookie_builder = cookie::Cookie::build((c.name, c.value))
                            .http_only(c.options.http_only)
                            .same_site(match c.options.same_site {
                                crate::res::CookieSameSiteOptions::Lax => cookie::SameSite::Lax,
                                crate::res::CookieSameSiteOptions::Strict => {
                                    cookie::SameSite::Strict
                                }
                                crate::res::CookieSameSiteOptions::None => cookie::SameSite::None,
                            })
                            .secure(c.options.secure)
                            .path(c.options.path.as_deref().unwrap_or("/"));
                        if let Some(domain) = c.options.domain.as_deref() {
                            cookie_builder = cookie_builder.domain(domain);
                        }
                        if let Some(max_age_secs) = c.options.max_age {
                            cookie_builder = cookie_builder
                                .max_age(cookie::time::Duration::seconds(max_age_secs));
                        }
                        if let Some(expires_unix) = c.options.expires {
                            if let Ok(odt) =
                                cookie::time::OffsetDateTime::from_unix_timestamp(expires_unix)
                            {
                                cookie_builder = cookie_builder.expires(odt);
                            }
                        }

                        if let Ok(cookie_value) =
                            HeaderValue::from_bytes(cookie_builder.to_string().as_bytes())
                        {
                            header_map.append(SET_COOKIE, cookie_value);
                        }
                    }
                    Cookie::RemoveCookie(cookie_name) => {
                        let expired_cookie = cookie::Cookie::build((cookie_name.to_string(), ""))
                            .path("/")
                            .max_age(cookie::time::Duration::seconds(0));

                        if let Ok(cookie_value) =
                            HeaderValue::from_bytes(expired_cookie.to_string().as_bytes())
                        {
                            header_map.append(SET_COOKIE, cookie_value);
                        }
                    }
                }
            }

            let collected_results: Vec<Result<Bytes, HttpResponseError>> =
                self.stream.unwrap().collect().await;

            let bytes = collected_results
                .into_iter()
                .collect::<Result<Vec<Bytes>, _>>()
                .map(|chunks| chunks.concat().into())
                .unwrap_or_else(|_| Bytes::new());

            let mut hyper_response = response.body(Full::from(bytes)).unwrap();

            hyper_response.headers_mut().extend(header_map);

            hyper_response.headers_mut().remove(CONTENT_LENGTH);
            let header_value = HeaderValue::from_static("chunked");
            hyper_response
                .headers_mut()
                .insert(HeaderName::from_static("transfer-encoding"), header_value);

            return Ok(hyper_response);
        } else {
            let mut response = match body {
                ResponseBody::JSON(json) => {
                    let json_bytes = serde_json::to_vec(&json).unwrap_or_else(|e| {
                        println!("JSON serialization error: {:?}", e);
                        Vec::from(b"{}")
                    });

                    Response::builder()
                        .status(self.status_code.as_u16())
                        .header("Content-Type", "application/json")
                        .body(Full::from(Bytes::from(json_bytes)))
                }
                ResponseBody::TEXT(text) => Response::builder()
                    .status(self.status_code.as_u16())
                    .header("Content-Type", "text/plain")
                    .body(Full::from(Bytes::from(text))),
                ResponseBody::HTML(html) => Response::builder()
                    .status(self.status_code.as_u16())
                    .header("Content-Type", "text/html")
                    .body(Full::from(Bytes::from(html))),
                ResponseBody::BINARY(bytes) => Response::builder()
                    .status(self.status_code.as_u16())
                    .header("Content-Type", "application/octet-stream")
                    .body(Full::from(Bytes::from(bytes))),
            }
            .unwrap();

            let mut header_map = self.headers.into_header_map();

            header_map.remove(hyper::header::CONTENT_TYPE);

            for c in self.cookies {
                match c {
                    Cookie::AddCookie(c) => {
                        let mut cookie_builder = cookie::Cookie::build((c.name, c.value))
                            .http_only(c.options.http_only)
                            .same_site(match c.options.same_site {
                                crate::res::CookieSameSiteOptions::Lax => cookie::SameSite::Lax,
                                crate::res::CookieSameSiteOptions::Strict => {
                                    cookie::SameSite::Strict
                                }
                                crate::res::CookieSameSiteOptions::None => cookie::SameSite::None,
                            })
                            .secure(c.options.secure)
                            .path(c.options.path.as_deref().unwrap_or("/"));

                        if let Some(domain) = c.options.domain.as_deref() {
                            cookie_builder = cookie_builder.domain(domain);
                        }
                        if let Some(max_age_secs) = c.options.max_age {
                            cookie_builder = cookie_builder
                                .max_age(cookie::time::Duration::seconds(max_age_secs));
                        }
                        if let Some(expires_unix) = c.options.expires {
                            if let Ok(odt) =
                                cookie::time::OffsetDateTime::from_unix_timestamp(expires_unix)
                            {
                                cookie_builder = cookie_builder.expires(odt);
                            }
                        }

                        if let Ok(cookie_value) =
                            HeaderValue::from_bytes(cookie_builder.to_string().as_bytes())
                        {
                            header_map.append(SET_COOKIE, cookie_value);
                        }
                    }

                    Cookie::RemoveCookie(cookie_name) => {
                        let expired_cookie = cookie::Cookie::build((cookie_name, ""))
                            .path("/")
                            .max_age(cookie::time::Duration::seconds(0));

                        if let Ok(cookie_value) =
                            HeaderValue::from_bytes(expired_cookie.to_string().as_bytes())
                        {
                            header_map.append(SET_COOKIE, cookie_value);
                        }
                    }
                }
            }

            response.headers_mut().extend(header_map);

            return Ok(response);
        }
    }
}