2019-03-30 18:04:38 +01:00
|
|
|
//! Essentials helper functions and types for application registration.
|
2019-04-22 23:22:08 +02:00
|
|
|
use actix_http::http::Method;
|
2019-12-25 17:13:52 +01:00
|
|
|
use actix_router::IntoPattern;
|
2020-05-18 04:47:20 +02:00
|
|
|
use std::future::Future;
|
2019-03-30 18:04:38 +01:00
|
|
|
|
|
|
|
pub use actix_http::Response as HttpResponse;
|
2020-10-24 21:31:23 +02:00
|
|
|
pub use bytes::{Buf, BufMut, Bytes, BytesMut};
|
2019-03-30 18:04:38 +01:00
|
|
|
|
2019-12-02 18:33:39 +01:00
|
|
|
use crate::error::BlockingError;
|
2019-03-30 18:04:38 +01:00
|
|
|
use crate::extract::FromRequest;
|
2020-12-26 22:46:19 +01:00
|
|
|
use crate::handler::Handler;
|
2019-03-30 18:04:38 +01:00
|
|
|
use crate::resource::Resource;
|
|
|
|
use crate::responder::Responder;
|
|
|
|
use crate::route::Route;
|
|
|
|
use crate::scope::Scope;
|
2019-04-25 00:29:15 +02:00
|
|
|
use crate::service::WebService;
|
2019-03-30 18:04:38 +01:00
|
|
|
|
2019-04-15 16:32:49 +02:00
|
|
|
pub use crate::config::ServiceConfig;
|
2019-05-05 04:43:49 +02:00
|
|
|
pub use crate::data::Data;
|
2019-03-30 18:04:38 +01:00
|
|
|
pub use crate::request::HttpRequest;
|
2020-10-24 19:49:50 +02:00
|
|
|
pub use crate::request_data::ReqData;
|
2019-03-30 18:04:38 +01:00
|
|
|
pub use crate::types::*;
|
|
|
|
|
|
|
|
/// Create resource for a specific path.
|
|
|
|
///
|
|
|
|
/// Resources may have variable path segments. For example, a
|
|
|
|
/// resource with the path `/a/{name}/c` would match all incoming
|
|
|
|
/// requests with paths such as `/a/b/c`, `/a/1/c`, or `/a/etc/c`.
|
|
|
|
///
|
|
|
|
/// A variable segment is specified in the form `{identifier}`,
|
|
|
|
/// where the identifier can be used later in a request handler to
|
|
|
|
/// access the matched value for that segment. This is done by
|
|
|
|
/// looking up the identifier in the `Params` object returned by
|
|
|
|
/// `HttpRequest.match_info()` method.
|
|
|
|
///
|
|
|
|
/// By default, each segment matches the regular expression `[^{}/]+`.
|
|
|
|
///
|
|
|
|
/// You can also specify a custom regex in the form `{identifier:regex}`:
|
|
|
|
///
|
|
|
|
/// For instance, to route `GET`-requests on any route matching
|
|
|
|
/// `/users/{userid}/{friend}` and store `userid` and `friend` in
|
|
|
|
/// the exposed `Params` object:
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// # extern crate actix_web;
|
|
|
|
/// use actix_web::{web, App, HttpResponse};
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::resource("/users/{userid}/{friend}")
|
|
|
|
/// .route(web::get().to(|| HttpResponse::Ok()))
|
|
|
|
/// .route(web::head().to(|| HttpResponse::MethodNotAllowed()))
|
|
|
|
/// );
|
2019-03-30 18:04:38 +01:00
|
|
|
/// ```
|
2019-12-25 17:13:52 +01:00
|
|
|
pub fn resource<T: IntoPattern>(path: T) -> Resource {
|
2019-03-30 18:04:38 +01:00
|
|
|
Resource::new(path)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Configure scope for common root path.
|
|
|
|
///
|
|
|
|
/// Scopes collect multiple paths under a common path prefix.
|
|
|
|
/// Scope path can contain variable path segments as resources.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use actix_web::{web, App, HttpResponse};
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::scope("/{project_id}")
|
|
|
|
/// .service(web::resource("/path1").to(|| HttpResponse::Ok()))
|
|
|
|
/// .service(web::resource("/path2").to(|| HttpResponse::Ok()))
|
|
|
|
/// .service(web::resource("/path3").to(|| HttpResponse::MethodNotAllowed()))
|
|
|
|
/// );
|
2019-03-30 18:04:38 +01:00
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// In the above example, three routes get added:
|
|
|
|
/// * /{project_id}/path1
|
|
|
|
/// * /{project_id}/path2
|
|
|
|
/// * /{project_id}/path3
|
|
|
|
///
|
2019-04-13 23:50:54 +02:00
|
|
|
pub fn scope(path: &str) -> Scope {
|
2019-03-30 18:04:38 +01:00
|
|
|
Scope::new(path)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create *route* without configuration.
|
2019-04-13 23:50:54 +02:00
|
|
|
pub fn route() -> Route {
|
2019-03-30 18:04:38 +01:00
|
|
|
Route::new()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create *route* with `GET` method guard.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use actix_web::{web, App, HttpResponse};
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::resource("/{project_id}")
|
|
|
|
/// .route(web::get().to(|| HttpResponse::Ok()))
|
|
|
|
/// );
|
2019-03-30 18:04:38 +01:00
|
|
|
/// ```
|
|
|
|
///
|
2020-02-06 19:00:22 +01:00
|
|
|
/// In the above example, one `GET` route gets added:
|
2019-03-30 18:04:38 +01:00
|
|
|
/// * /{project_id}
|
|
|
|
///
|
2019-04-13 23:50:54 +02:00
|
|
|
pub fn get() -> Route {
|
2019-04-15 04:52:12 +02:00
|
|
|
method(Method::GET)
|
2019-03-30 18:04:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create *route* with `POST` method guard.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use actix_web::{web, App, HttpResponse};
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::resource("/{project_id}")
|
|
|
|
/// .route(web::post().to(|| HttpResponse::Ok()))
|
|
|
|
/// );
|
2019-03-30 18:04:38 +01:00
|
|
|
/// ```
|
|
|
|
///
|
2020-02-06 19:00:22 +01:00
|
|
|
/// In the above example, one `POST` route gets added:
|
2019-03-30 18:04:38 +01:00
|
|
|
/// * /{project_id}
|
|
|
|
///
|
2019-04-13 23:50:54 +02:00
|
|
|
pub fn post() -> Route {
|
2019-04-15 04:52:12 +02:00
|
|
|
method(Method::POST)
|
2019-03-30 18:04:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create *route* with `PUT` method guard.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use actix_web::{web, App, HttpResponse};
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::resource("/{project_id}")
|
|
|
|
/// .route(web::put().to(|| HttpResponse::Ok()))
|
|
|
|
/// );
|
2019-03-30 18:04:38 +01:00
|
|
|
/// ```
|
|
|
|
///
|
2020-02-06 19:00:22 +01:00
|
|
|
/// In the above example, one `PUT` route gets added:
|
2019-03-30 18:04:38 +01:00
|
|
|
/// * /{project_id}
|
|
|
|
///
|
2019-04-13 23:50:54 +02:00
|
|
|
pub fn put() -> Route {
|
2019-04-15 04:52:12 +02:00
|
|
|
method(Method::PUT)
|
2019-03-30 18:04:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create *route* with `PATCH` method guard.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use actix_web::{web, App, HttpResponse};
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::resource("/{project_id}")
|
|
|
|
/// .route(web::patch().to(|| HttpResponse::Ok()))
|
|
|
|
/// );
|
2019-03-30 18:04:38 +01:00
|
|
|
/// ```
|
|
|
|
///
|
2020-02-06 19:00:22 +01:00
|
|
|
/// In the above example, one `PATCH` route gets added:
|
2019-03-30 18:04:38 +01:00
|
|
|
/// * /{project_id}
|
|
|
|
///
|
2019-04-13 23:50:54 +02:00
|
|
|
pub fn patch() -> Route {
|
2019-04-15 04:52:12 +02:00
|
|
|
method(Method::PATCH)
|
2019-03-30 18:04:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create *route* with `DELETE` method guard.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use actix_web::{web, App, HttpResponse};
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::resource("/{project_id}")
|
|
|
|
/// .route(web::delete().to(|| HttpResponse::Ok()))
|
|
|
|
/// );
|
2019-03-30 18:04:38 +01:00
|
|
|
/// ```
|
|
|
|
///
|
2020-02-06 19:00:22 +01:00
|
|
|
/// In the above example, one `DELETE` route gets added:
|
2019-03-30 18:04:38 +01:00
|
|
|
/// * /{project_id}
|
|
|
|
///
|
2019-04-13 23:50:54 +02:00
|
|
|
pub fn delete() -> Route {
|
2019-04-15 04:52:12 +02:00
|
|
|
method(Method::DELETE)
|
2019-03-30 18:04:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Create *route* with `HEAD` method guard.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use actix_web::{web, App, HttpResponse};
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::resource("/{project_id}")
|
|
|
|
/// .route(web::head().to(|| HttpResponse::Ok()))
|
|
|
|
/// );
|
2019-03-30 18:04:38 +01:00
|
|
|
/// ```
|
|
|
|
///
|
2020-02-06 19:00:22 +01:00
|
|
|
/// In the above example, one `HEAD` route gets added:
|
2019-03-30 18:04:38 +01:00
|
|
|
/// * /{project_id}
|
|
|
|
///
|
2019-04-13 23:50:54 +02:00
|
|
|
pub fn head() -> Route {
|
2019-04-15 04:52:12 +02:00
|
|
|
method(Method::HEAD)
|
2019-03-30 18:04:38 +01:00
|
|
|
}
|
|
|
|
|
2020-02-22 16:19:29 +01:00
|
|
|
/// Create *route* with `TRACE` method guard.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use actix_web::{web, App, HttpResponse};
|
|
|
|
///
|
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::resource("/{project_id}")
|
|
|
|
/// .route(web::trace().to(|| HttpResponse::Ok()))
|
|
|
|
/// );
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// In the above example, one `HEAD` route gets added:
|
|
|
|
/// * /{project_id}
|
|
|
|
///
|
|
|
|
pub fn trace() -> Route {
|
|
|
|
method(Method::TRACE)
|
|
|
|
}
|
|
|
|
|
2019-03-30 18:04:38 +01:00
|
|
|
/// Create *route* and add method guard.
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use actix_web::{web, http, App, HttpResponse};
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::resource("/{project_id}")
|
|
|
|
/// .route(web::method(http::Method::GET).to(|| HttpResponse::Ok()))
|
|
|
|
/// );
|
2019-03-30 18:04:38 +01:00
|
|
|
/// ```
|
|
|
|
///
|
2020-02-06 19:00:22 +01:00
|
|
|
/// In the above example, one `GET` route gets added:
|
2019-03-30 18:04:38 +01:00
|
|
|
/// * /{project_id}
|
|
|
|
///
|
2019-04-13 23:50:54 +02:00
|
|
|
pub fn method(method: Method) -> Route {
|
2019-03-30 18:04:38 +01:00
|
|
|
Route::new().method(method)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a new route and add handler.
|
|
|
|
///
|
|
|
|
/// ```rust
|
2019-11-21 16:34:04 +01:00
|
|
|
/// use actix_web::{web, App, HttpResponse, Responder};
|
2019-03-30 18:04:38 +01:00
|
|
|
///
|
2019-11-21 16:34:04 +01:00
|
|
|
/// async fn index() -> impl Responder {
|
|
|
|
/// HttpResponse::Ok()
|
2019-03-30 18:04:38 +01:00
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// App::new().service(
|
|
|
|
/// web::resource("/").route(
|
|
|
|
/// web::to(index))
|
|
|
|
/// );
|
|
|
|
/// ```
|
2020-12-26 22:46:19 +01:00
|
|
|
pub fn to<F, I, R>(handler: F) -> Route
|
2019-03-30 18:04:38 +01:00
|
|
|
where
|
2020-12-26 22:46:19 +01:00
|
|
|
F: Handler<I, R>,
|
2019-04-13 23:50:54 +02:00
|
|
|
I: FromRequest + 'static,
|
2020-12-26 22:46:19 +01:00
|
|
|
R: Future + 'static,
|
|
|
|
R::Output: Responder + 'static,
|
2019-03-30 18:04:38 +01:00
|
|
|
{
|
2019-11-21 16:34:04 +01:00
|
|
|
Route::new().to(handler)
|
2019-03-30 18:04:38 +01:00
|
|
|
}
|
|
|
|
|
2019-04-25 00:29:15 +02:00
|
|
|
/// Create raw service for a specific path.
|
|
|
|
///
|
|
|
|
/// ```rust
|
2019-11-20 18:33:22 +01:00
|
|
|
/// use actix_web::{dev, web, guard, App, Error, HttpResponse};
|
2019-04-25 00:29:15 +02:00
|
|
|
///
|
2019-11-21 10:56:49 +01:00
|
|
|
/// async fn my_service(req: dev::ServiceRequest) -> Result<dev::ServiceResponse, Error> {
|
|
|
|
/// Ok(req.into_response(HttpResponse::Ok().finish()))
|
2019-04-25 00:29:15 +02:00
|
|
|
/// }
|
|
|
|
///
|
2019-12-08 07:31:16 +01:00
|
|
|
/// let app = App::new().service(
|
|
|
|
/// web::service("/users/*")
|
|
|
|
/// .guard(guard::Header("content-type", "text/plain"))
|
|
|
|
/// .finish(my_service)
|
|
|
|
/// );
|
2019-04-25 00:29:15 +02:00
|
|
|
/// ```
|
2019-12-25 17:13:52 +01:00
|
|
|
pub fn service<T: IntoPattern>(path: T) -> WebService {
|
2019-04-25 00:29:15 +02:00
|
|
|
WebService::new(path)
|
|
|
|
}
|
|
|
|
|
2019-03-30 18:04:38 +01:00
|
|
|
/// Execute blocking function on a thread pool, returns future that resolves
|
|
|
|
/// to result of the function execution.
|
2019-12-02 18:33:39 +01:00
|
|
|
pub async fn block<F, I, E>(f: F) -> Result<I, BlockingError<E>>
|
2019-03-30 18:04:38 +01:00
|
|
|
where
|
2019-12-02 18:33:39 +01:00
|
|
|
F: FnOnce() -> Result<I, E> + Send + 'static,
|
|
|
|
I: Send + 'static,
|
|
|
|
E: Send + std::fmt::Debug + 'static,
|
2019-03-30 18:04:38 +01:00
|
|
|
{
|
2021-01-09 17:04:19 +01:00
|
|
|
match actix_rt::task::spawn_blocking(f).await {
|
|
|
|
Ok(res) => res.map_err(BlockingError::Error),
|
|
|
|
Err(_) => Err(BlockingError::Canceled),
|
|
|
|
}
|
2019-03-30 18:04:38 +01:00
|
|
|
}
|