use crate::middleware::Inner;
use crate::{middleware, PrerenderError, PrerenderMiddleware};
use actix_service::{Service, Transform};
use actix_utils::future;
use actix_utils::future::Ready;
use actix_web::body::{EitherBody, MessageBody};
use actix_web::dev::{ServiceRequest, ServiceResponse};
use actix_web::Error;
use reqwest::header::HeaderMap;
use reqwest::Client;
use std::rc::Rc;
use url::Url;
#[derive(Clone)]
pub struct Prerender {
inner: Rc<Inner>,
}
#[derive()]
pub struct PrerenderBuilder {
pub(crate) forward_headers: bool,
pub(crate) before_render_fn: Option<fn(&ServiceRequest, &mut HeaderMap)>,
}
fn default_client() -> Client {
Client::builder()
.gzip(true)
.timeout(std::time::Duration::new(25, 0))
.build()
.unwrap()
}
impl PrerenderBuilder {
pub fn use_prerender_io(self, token: String) -> Prerender {
let inner = Inner {
before_render_fn: self.before_render_fn,
forward_headers: self.forward_headers,
inner_client: default_client(),
prerender_service_url: middleware::prerender_url(),
prerender_token: Some(token),
};
Prerender { inner: Rc::new(inner) }
}
pub fn use_custom_prerender_url(self, prerender_service_url: &str) -> Result<Prerender, PrerenderError> {
let prerender_service_url = Url::parse(prerender_service_url).map_err(|_| PrerenderError::InvalidUrl)?;
let inner = Inner {
before_render_fn: self.before_render_fn,
forward_headers: self.forward_headers,
inner_client: default_client(),
prerender_service_url,
prerender_token: None,
};
Ok(Prerender { inner: Rc::new(inner) })
}
pub fn set_before_render_fn(mut self, prerender_func: fn(req: &ServiceRequest, headers: &mut HeaderMap)) -> Self {
self.before_render_fn = Some(prerender_func);
self
}
pub const fn forward_headers(mut self) -> Self {
self.forward_headers = true;
self
}
}
impl Prerender {
pub fn build() -> PrerenderBuilder {
PrerenderBuilder {
forward_headers: false,
before_render_fn: None,
}
}
}
impl<S, B> Transform<S, ServiceRequest> for Prerender
where
S: Service<ServiceRequest, Response = ServiceResponse<B>, Error = Error>,
S::Future: 'static,
B: MessageBody + 'static,
{
type Response = ServiceResponse<EitherBody<B>>;
type Error = Error;
type Transform = PrerenderMiddleware<S>;
type InitError = ();
type Future = Ready<Result<Self::Transform, Self::InitError>>;
fn new_transform(&self, service: S) -> Self::Future {
future::ok(PrerenderMiddleware {
service,
inner: Rc::clone(&self.inner),
})
}
}