use std::{cell::UnsafeCell, net::SocketAddr, rc::Rc};
use crate::service::cfg::{CfgContext, Configuration};
use crate::{router::ResourceDef, util::ByteString, util::Extensions};
use super::httprequest::{HttpRequest, HttpRequestInner};
use super::service::{AppServiceFactory, ServiceFactoryWrapper, WebServiceFactory};
use super::{DefaultError, ErrorRenderer, resource::Resource, route::Route};
#[derive(Debug)]
pub struct WebAppConfig {
name: ByteString,
secure: bool,
host: String,
addr: SocketAddr,
config: CfgContext,
pub(super) pool_size: usize,
}
impl Default for WebAppConfig {
fn default() -> Self {
WebAppConfig::new()
}
}
impl Configuration for WebAppConfig {
const NAME: &str = "Web app configuration";
fn ctx(&self) -> &CfgContext {
&self.config
}
fn set_ctx(&mut self, ctx: CfgContext) {
self.config = ctx;
}
}
impl WebAppConfig {
#[must_use]
pub fn new() -> Self {
WebAppConfig::with(
"ntex:web",
false,
"127.0.0.1:8080".parse().unwrap(),
"localhost:8080".to_owned(),
)
}
#[must_use]
pub fn with(name: &str, secure: bool, addr: SocketAddr, host: String) -> Self {
WebAppConfig {
secure,
host,
addr,
name: name.into(),
pool_size: 128,
config: CfgContext::default(),
}
}
pub fn name(&self) -> &ByteString {
&self.name
}
pub fn host(&self) -> &str {
&self.host
}
pub fn secure(&self) -> bool {
self.secure
}
pub fn local_addr(&self) -> SocketAddr {
self.addr
}
#[must_use]
pub fn set_host(mut self, host: String) -> Self {
self.host = host;
self
}
#[must_use]
pub fn set_secure(mut self) -> Self {
self.secure = true;
self
}
#[must_use]
pub fn set_local_addr(mut self, addr: SocketAddr) -> Self {
self.addr = addr;
self
}
#[must_use]
pub fn set_pool_size(mut self, size: usize) -> Self {
self.pool_size = size;
self
}
pub(crate) fn get_request(&self) -> Option<HttpRequest> {
CACHE.with(|cache| {
cache.with(self.config.id(), |cache| cache.pop().map(HttpRequest))
})
}
pub(crate) fn clear_requests(&self) {
CACHE.with(|cache| cache.with(self.config.id(), Vec::clear));
}
}
pub(crate) fn put_request(id: usize, pool_size: usize, req: &mut Rc<HttpRequestInner>) {
CACHE.with(|cache| {
cache.with(id, |cache| {
if cache.len() < pool_size
&& let Some(inner) = Rc::get_mut(req)
{
inner.head.remove_io();
inner.head.extensions.borrow_mut().clear();
cache.push(req.clone());
}
});
});
}
#[derive(derive_more::Debug)]
#[debug("ServiceConfig")]
pub struct ServiceConfig<Err = DefaultError> {
pub(super) services: Vec<Box<dyn AppServiceFactory<Err>>>,
pub(super) state: Extensions,
pub(super) external: Vec<ResourceDef>,
}
impl<Err: ErrorRenderer> ServiceConfig<Err> {
pub fn new() -> Self {
Self {
services: Vec::new(),
state: Extensions::new(),
external: Vec::new(),
}
}
pub fn state<S: 'static>(&mut self, st: S) -> &mut Self {
self.state.insert(st);
self
}
pub fn route(&mut self, path: &str, mut route: Route<Err>) -> &mut Self {
self.service(
Resource::new(path)
.add_guards(route.take_guards())
.route(route),
)
}
pub fn service<F>(&mut self, factory: F) -> &mut Self
where
F: WebServiceFactory<Err> + 'static,
{
self.services
.push(Box::new(ServiceFactoryWrapper::new(factory)));
self
}
pub fn external_resource<N, U>(&mut self, name: N, url: U) -> &mut Self
where
N: AsRef<str>,
U: AsRef<str>,
{
let mut rdef = ResourceDef::new(url.as_ref());
*rdef.name_mut() = name.as_ref().to_string();
self.external.push(rdef);
self
}
}
impl<Err: ErrorRenderer> Default for ServiceConfig<Err> {
fn default() -> Self {
Self::new()
}
}
thread_local! {
static CACHE: LocalCache = LocalCache::new();
}
struct LocalCache {
cache: UnsafeCell<Vec<Vec<Rc<HttpRequestInner>>>>,
}
impl LocalCache {
fn new() -> Self {
Self {
cache: UnsafeCell::new(Vec::with_capacity(16)),
}
}
fn with<F, R>(&self, idx: usize, f: F) -> R
where
F: FnOnce(&mut Vec<Rc<HttpRequestInner>>) -> R,
{
let cache = unsafe { &mut *self.cache.get() };
while cache.len() <= idx {
cache.push(Vec::new());
}
f(&mut cache[idx])
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::http::{Method, StatusCode};
use crate::util::Bytes;
use crate::web::test::{TestRequest, call_service, init_service, read_body};
use crate::web::{self, App, DefaultError, HttpRequest, HttpResponse};
#[crate::rt_test]
async fn test_webappconfig() {
let cfg = WebAppConfig::default()
.set_host("www.example.org".to_string())
.set_local_addr("127.0.0.1:8080".parse().unwrap())
.set_pool_size(256);
assert_eq!(cfg.host(), "www.example.org");
assert_eq!(cfg.local_addr(), "127.0.0.1:8080".parse().unwrap());
assert_eq!(cfg.pool_size, 256);
}
#[crate::rt_test]
async fn test_configure_state() {
let cfg = |cfg: &mut ServiceConfig<_>| {
cfg.state(10usize);
};
let srv = init_service(
App::new().configure(cfg).service(
web::resource("/")
.to(|_: web::types::State<usize>| async { HttpResponse::Ok() }),
),
)
.await;
let req = TestRequest::default().to_request();
let resp = srv.call(req).await.unwrap();
assert_eq!(resp.status(), StatusCode::OK);
}
#[cfg(feature = "url")]
#[crate::rt_test]
async fn test_configure_external_resource() {
let srv = init_service(
App::new()
.configure(|cfg| {
cfg.external_resource(
"youtube",
"https://youtube.com/watch/{video_id}",
);
})
.route(
"/test",
web::get().to(|req: HttpRequest| async move {
HttpResponse::Ok()
.body(format!("{}", req.url_for("youtube", ["12345"]).unwrap()))
}),
),
)
.await;
let req = TestRequest::with_uri("/test").to_request();
let resp = call_service(&srv, req).await;
assert_eq!(resp.status(), StatusCode::OK);
let body = read_body(resp).await;
assert_eq!(body, Bytes::from_static(b"https://youtube.com/watch/12345"));
}
#[crate::rt_test]
async fn test_configure_service() {
let srv = init_service(App::new().configure(|cfg| {
cfg.service(
web::resource("/test")
.route(web::get().to(|| async { HttpResponse::Created() })),
)
.route(
"/index.html",
web::get().to(|| async { HttpResponse::Ok() }),
);
}))
.await;
let req = TestRequest::with_uri("/test")
.method(Method::GET)
.to_request();
let resp = call_service(&srv, req).await;
assert_eq!(resp.status(), StatusCode::CREATED);
let req = TestRequest::with_uri("/index.html")
.method(Method::GET)
.to_request();
let resp = call_service(&srv, req).await;
assert_eq!(resp.status(), StatusCode::OK);
}
#[test]
fn test_new_service_config() {
let cfg: ServiceConfig<DefaultError> = ServiceConfig::default();
assert!(cfg.services.is_empty());
assert!(cfg.external.is_empty());
}
}