diff --git a/Cargo.toml b/Cargo.toml
index 2b5ccf84..33790d96 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -1,4 +1,5 @@
[workspace]
members = [
"ntex",
+ "ntex-web-macros",
]
diff --git a/README.md b/README.md
index 0b55c5ba..22929267 100644
--- a/README.md
+++ b/README.md
@@ -1,106 +1,9 @@
-
Actix web
-
Actix web is a small, pragmatic, and extremely fast rust web framework
+
ntex
+
This is personal project, please, do not use it
-
[](https://travis-ci.org/actix/actix-web)
[](https://codecov.io/gh/actix/actix-web)
[](https://crates.io/crates/actix-web)
-[](https://gitter.im/actix/actix?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
-[](https://docs.rs/actix-web)
-[](https://crates.io/crates/actix-web)
-[](https://blog.rust-lang.org/2019/11/07/Rust-1.39.0.html)
-
-
-
-
-
-
-Actix web is a simple, pragmatic and extremely fast web framework for Rust.
-
-* Supported *HTTP/1.x* and *HTTP/2.0* protocols
-* Streaming and pipelining
-* Keep-alive and slow requests handling
-* Client/server [WebSockets](https://actix.rs/docs/websockets/) support
-* Transparent content compression/decompression (br, gzip, deflate)
-* Configurable [request routing](https://actix.rs/docs/url-dispatch/)
-* Multipart streams
-* Static assets
-* SSL support with OpenSSL or Rustls
-* Middlewares ([Logger, Session, CORS, etc](https://actix.rs/docs/middleware/))
-* Includes an asynchronous [HTTP client](https://actix.rs/actix-web/actix_web/client/index.html)
-* Supports [Actix actor framework](https://github.com/actix/actix)
-
-## Example
-
-Dependencies:
-
-```toml
-[dependencies]
-actix-web = "2"
-actix-rt = "1"
-```
-
-Code:
-
-```rust
-use actix_web::{get, web, App, HttpServer, Responder};
-
-#[get("/{id}/{name}/index.html")]
-async fn index(info: web::Path<(u32, String)>) -> impl Responder {
- format!("Hello {}! id:{}", info.1, info.0)
-}
-
-#[actix_rt::main]
-async fn main() -> std::io::Result<()> {
- HttpServer::new(|| App::new().service(index))
- .bind("127.0.0.1:8080")?
- .run()
- .await
-}
-```
-
-### More examples
-
-* [Basics](https://github.com/actix/examples/tree/master/basics/)
-* [Stateful](https://github.com/actix/examples/tree/master/state/)
-* [Multipart streams](https://github.com/actix/examples/tree/master/multipart/)
-* [Simple websocket](https://github.com/actix/examples/tree/master/websocket/)
-* [Tera](https://github.com/actix/examples/tree/master/template_tera/) /
-* [Askama](https://github.com/actix/examples/tree/master/template_askama/) templates
-* [Diesel integration](https://github.com/actix/examples/tree/master/diesel/)
-* [r2d2](https://github.com/actix/examples/tree/master/r2d2/)
-* [OpenSSL](https://github.com/actix/examples/tree/master/openssl/)
-* [Rustls](https://github.com/actix/examples/tree/master/rustls/)
-* [Tcp/Websocket chat](https://github.com/actix/examples/tree/master/websocket-chat/)
-* [Json](https://github.com/actix/examples/tree/master/json/)
-
-You may consider checking out
-[this directory](https://github.com/actix/examples/tree/master/) for more examples.
-
-## Benchmarks
-
-* [TechEmpower Framework Benchmark](https://www.techempower.com/benchmarks/#section=data-r18)
-
-## License
-
-This project is licensed under either of
-
-* Apache License, Version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or [http://www.apache.org/licenses/LICENSE-2.0](http://www.apache.org/licenses/LICENSE-2.0))
-* MIT license ([LICENSE-MIT](LICENSE-MIT) or [http://opensource.org/licenses/MIT](http://opensource.org/licenses/MIT))
-
-at your option.
-
-## Code of Conduct
-
-Contribution to the actix-web crate is organized under the terms of the
-Contributor Covenant, the maintainer of actix-web, @fafhrd91, promises to
-intervene to uphold that code of conduct.
diff --git a/examples/echo.rs b/examples/echo.rs
deleted file mode 100644
index 3d57a472..00000000
--- a/examples/echo.rs
+++ /dev/null
@@ -1,42 +0,0 @@
-use std::{env, io};
-
-use actix_http::{Error, HttpService, Request, Response};
-use actix_server::Server;
-use bytes::BytesMut;
-use futures::StreamExt;
-use http::header::HeaderValue;
-use log::info;
-
-#[actix_rt::main]
-async fn main() -> io::Result<()> {
- env::set_var("RUST_LOG", "echo=info");
- env_logger::init();
-
- Server::build()
- .bind("echo", "127.0.0.1:8080", || {
- HttpService::build()
- .client_timeout(1000)
- .client_disconnect(1000)
- .finish(|mut req: Request| {
- async move {
- let mut body = BytesMut::new();
- while let Some(item) = req.payload().next().await {
- body.extend_from_slice(&item?);
- }
-
- info!("request body: {:?}", body);
- Ok::<_, Error>(
- Response::Ok()
- .header(
- "x-head",
- HeaderValue::from_static("dummy value!"),
- )
- .body(body),
- )
- }
- })
- .tcp()
- })?
- .run()
- .await
-}
diff --git a/ntex-web-macros/CHANGES.md b/ntex-web-macros/CHANGES.md
new file mode 100644
index 00000000..54c13042
--- /dev/null
+++ b/ntex-web-macros/CHANGES.md
@@ -0,0 +1,5 @@
+# Changes
+
+## [0.1.0] - 2020-xx-xx
+
+* Fork
diff --git a/ntex-web-macros/Cargo.toml b/ntex-web-macros/Cargo.toml
new file mode 100644
index 00000000..6ee53230
--- /dev/null
+++ b/ntex-web-macros/Cargo.toml
@@ -0,0 +1,21 @@
+[package]
+name = "ntex-web-macros"
+version = "0.1.0"
+description = "ntex web proc macros"
+readme = "README.md"
+authors = ["Nikolay Kim "]
+license = "MIT/Apache-2.0"
+edition = "2018"
+
+[lib]
+proc-macro = true
+
+[dependencies]
+quote = "^1"
+syn = { version = "^1", features = ["full", "parsing"] }
+proc-macro2 = "^1"
+
+[dev-dependencies]
+actix-rt = "1.0.0"
+ntex = { path = "../ntex/" }
+futures = { version = "0.3.1" }
diff --git a/ntex-web-macros/README.md b/ntex-web-macros/README.md
new file mode 100644
index 00000000..a75d091d
--- /dev/null
+++ b/ntex-web-macros/README.md
@@ -0,0 +1 @@
+# Macros for ntex::web framework [](https://travis-ci.org/fafhrd91/ntex)
diff --git a/ntex-web-macros/src/lib.rs b/ntex-web-macros/src/lib.rs
new file mode 100644
index 00000000..8a03a3f5
--- /dev/null
+++ b/ntex-web-macros/src/lib.rs
@@ -0,0 +1,185 @@
+#![recursion_limit = "512"]
+//! web macros module
+//!
+//! Generators for routes and scopes
+//!
+//! ## Route
+//!
+//! Macros:
+//!
+//! - [get](attr.get.html)
+//! - [post](attr.post.html)
+//! - [put](attr.put.html)
+//! - [delete](attr.delete.html)
+//! - [head](attr.head.html)
+//! - [connect](attr.connect.html)
+//! - [options](attr.options.html)
+//! - [trace](attr.trace.html)
+//! - [patch](attr.patch.html)
+//!
+//! ### Attributes:
+//!
+//! - `"path"` - Raw literal string with path for which to register handle. Mandatory.
+//! - `guard="function_name"` - Registers function as guard using `ntex::web::guard::fn_guard`
+//!
+//! ## Notes
+//!
+//! Function name can be specified as any expression that is going to be accessible to the generate
+//! code (e.g `my_guard` or `my_module::my_guard`)
+//!
+//! ## Example:
+//!
+//! ```rust
+//! use ntex::web::{get, HttpResponse};
+//! use futures::{future, Future};
+//!
+//! #[get("/test")]
+//! async fn async_test() -> Result {
+//! Ok(HttpResponse::Ok().finish())
+//! }
+//! ```
+
+extern crate proc_macro;
+
+mod route;
+
+use proc_macro::TokenStream;
+use syn::parse_macro_input;
+
+/// Creates route handler with `GET` method guard.
+///
+/// Syntax: `#[get("path"[, attributes])]`
+///
+/// ## Attributes:
+///
+/// - `"path"` - Raw literal string with path for which to register handler. Mandatory.
+/// - `guard="function_name"` - Registers function as guard using `ntex::web::guard::fn_guard`
+#[proc_macro_attribute]
+pub fn get(args: TokenStream, input: TokenStream) -> TokenStream {
+ let args = parse_macro_input!(args as syn::AttributeArgs);
+ let gen = match route::Route::new(args, input, route::GuardType::Get) {
+ Ok(gen) => gen,
+ Err(err) => return err.to_compile_error().into(),
+ };
+ gen.generate()
+}
+
+/// Creates route handler with `POST` method guard.
+///
+/// Syntax: `#[post("path"[, attributes])]`
+///
+/// Attributes are the same as in [get](attr.get.html)
+#[proc_macro_attribute]
+pub fn post(args: TokenStream, input: TokenStream) -> TokenStream {
+ let args = parse_macro_input!(args as syn::AttributeArgs);
+ let gen = match route::Route::new(args, input, route::GuardType::Post) {
+ Ok(gen) => gen,
+ Err(err) => return err.to_compile_error().into(),
+ };
+ gen.generate()
+}
+
+/// Creates route handler with `PUT` method guard.
+///
+/// Syntax: `#[put("path"[, attributes])]`
+///
+/// Attributes are the same as in [get](attr.get.html)
+#[proc_macro_attribute]
+pub fn put(args: TokenStream, input: TokenStream) -> TokenStream {
+ let args = parse_macro_input!(args as syn::AttributeArgs);
+ let gen = match route::Route::new(args, input, route::GuardType::Put) {
+ Ok(gen) => gen,
+ Err(err) => return err.to_compile_error().into(),
+ };
+ gen.generate()
+}
+
+/// Creates route handler with `DELETE` method guard.
+///
+/// Syntax: `#[delete("path"[, attributes])]`
+///
+/// Attributes are the same as in [get](attr.get.html)
+#[proc_macro_attribute]
+pub fn delete(args: TokenStream, input: TokenStream) -> TokenStream {
+ let args = parse_macro_input!(args as syn::AttributeArgs);
+ let gen = match route::Route::new(args, input, route::GuardType::Delete) {
+ Ok(gen) => gen,
+ Err(err) => return err.to_compile_error().into(),
+ };
+ gen.generate()
+}
+
+/// Creates route handler with `HEAD` method guard.
+///
+/// Syntax: `#[head("path"[, attributes])]`
+///
+/// Attributes are the same as in [head](attr.head.html)
+#[proc_macro_attribute]
+pub fn head(args: TokenStream, input: TokenStream) -> TokenStream {
+ let args = parse_macro_input!(args as syn::AttributeArgs);
+ let gen = match route::Route::new(args, input, route::GuardType::Head) {
+ Ok(gen) => gen,
+ Err(err) => return err.to_compile_error().into(),
+ };
+ gen.generate()
+}
+
+/// Creates route handler with `CONNECT` method guard.
+///
+/// Syntax: `#[connect("path"[, attributes])]`
+///
+/// Attributes are the same as in [connect](attr.connect.html)
+#[proc_macro_attribute]
+pub fn connect(args: TokenStream, input: TokenStream) -> TokenStream {
+ let args = parse_macro_input!(args as syn::AttributeArgs);
+ let gen = match route::Route::new(args, input, route::GuardType::Connect) {
+ Ok(gen) => gen,
+ Err(err) => return err.to_compile_error().into(),
+ };
+ gen.generate()
+}
+
+/// Creates route handler with `OPTIONS` method guard.
+///
+/// Syntax: `#[options("path"[, attributes])]`
+///
+/// Attributes are the same as in [options](attr.options.html)
+#[proc_macro_attribute]
+pub fn options(args: TokenStream, input: TokenStream) -> TokenStream {
+ let args = parse_macro_input!(args as syn::AttributeArgs);
+ let gen = match route::Route::new(args, input, route::GuardType::Options) {
+ Ok(gen) => gen,
+ Err(err) => return err.to_compile_error().into(),
+ };
+ gen.generate()
+}
+
+/// Creates route handler with `TRACE` method guard.
+///
+/// Syntax: `#[trace("path"[, attributes])]`
+///
+/// Attributes are the same as in [trace](attr.trace.html)
+#[proc_macro_attribute]
+pub fn trace(args: TokenStream, input: TokenStream) -> TokenStream {
+ let args = parse_macro_input!(args as syn::AttributeArgs);
+ let gen = match route::Route::new(args, input, route::GuardType::Trace) {
+ Ok(gen) => gen,
+ Err(err) => return err.to_compile_error().into(),
+ };
+ gen.generate()
+}
+
+/// Creates route handler with `PATCH` method guard.
+///
+/// Syntax: `#[patch("path"[, attributes])]`
+///
+/// Attributes are the same as in [patch](attr.patch.html)
+#[proc_macro_attribute]
+pub fn patch(args: TokenStream, input: TokenStream) -> TokenStream {
+ let args = parse_macro_input!(args as syn::AttributeArgs);
+ let gen = match route::Route::new(args, input, route::GuardType::Patch) {
+ Ok(gen) => gen,
+ Err(err) => return err.to_compile_error().into(),
+ };
+ gen.generate()
+}
diff --git a/ntex-web-macros/src/route.rs b/ntex-web-macros/src/route.rs
new file mode 100644
index 00000000..9f8ceb6e
--- /dev/null
+++ b/ntex-web-macros/src/route.rs
@@ -0,0 +1,212 @@
+extern crate proc_macro;
+
+use proc_macro::TokenStream;
+use proc_macro2::{Span, TokenStream as TokenStream2};
+use quote::{quote, ToTokens, TokenStreamExt};
+use syn::{AttributeArgs, Ident, NestedMeta};
+
+enum ResourceType {
+ Async,
+ Sync,
+}
+
+impl ToTokens for ResourceType {
+ fn to_tokens(&self, stream: &mut TokenStream2) {
+ let ident = match self {
+ ResourceType::Async => "to",
+ ResourceType::Sync => "to",
+ };
+ let ident = Ident::new(ident, Span::call_site());
+ stream.append(ident);
+ }
+}
+
+#[derive(PartialEq)]
+pub enum GuardType {
+ Get,
+ Post,
+ Put,
+ Delete,
+ Head,
+ Connect,
+ Options,
+ Trace,
+ Patch,
+}
+
+impl GuardType {
+ fn as_str(&self) -> &'static str {
+ match self {
+ GuardType::Get => "Get",
+ GuardType::Post => "Post",
+ GuardType::Put => "Put",
+ GuardType::Delete => "Delete",
+ GuardType::Head => "Head",
+ GuardType::Connect => "Connect",
+ GuardType::Options => "Options",
+ GuardType::Trace => "Trace",
+ GuardType::Patch => "Patch",
+ }
+ }
+}
+
+impl ToTokens for GuardType {
+ fn to_tokens(&self, stream: &mut TokenStream2) {
+ let ident = self.as_str();
+ let ident = Ident::new(ident, Span::call_site());
+ stream.append(ident);
+ }
+}
+
+struct Args {
+ path: syn::LitStr,
+ guards: Vec,
+}
+
+impl Args {
+ fn new(args: AttributeArgs) -> syn::Result {
+ let mut path = None;
+ let mut guards = Vec::new();
+ for arg in args {
+ match arg {
+ NestedMeta::Lit(syn::Lit::Str(lit)) => match path {
+ None => {
+ path = Some(lit);
+ }
+ _ => {
+ return Err(syn::Error::new_spanned(
+ lit,
+ "Multiple paths specified! Should be only one!",
+ ));
+ }
+ },
+ NestedMeta::Meta(syn::Meta::NameValue(nv)) => {
+ if nv.path.is_ident("guard") {
+ if let syn::Lit::Str(lit) = nv.lit {
+ guards.push(Ident::new(&lit.value(), Span::call_site()));
+ } else {
+ return Err(syn::Error::new_spanned(
+ nv.lit,
+ "Attribute guard expects literal string!",
+ ));
+ }
+ } else {
+ return Err(syn::Error::new_spanned(
+ nv.path,
+ "Unknown attribute key is specified. Allowed: guard",
+ ));
+ }
+ }
+ arg => {
+ return Err(syn::Error::new_spanned(arg, "Unknown attribute"));
+ }
+ }
+ }
+ Ok(Args {
+ path: path.unwrap(),
+ guards,
+ })
+ }
+}
+
+pub struct Route {
+ name: syn::Ident,
+ args: Args,
+ ast: syn::ItemFn,
+ resource_type: ResourceType,
+ guard: GuardType,
+}
+
+fn guess_resource_type(typ: &syn::Type) -> ResourceType {
+ let mut guess = ResourceType::Sync;
+
+ if let syn::Type::ImplTrait(typ) = typ {
+ for bound in typ.bounds.iter() {
+ if let syn::TypeParamBound::Trait(bound) = bound {
+ for bound in bound.path.segments.iter() {
+ if bound.ident == "Future" {
+ guess = ResourceType::Async;
+ break;
+ } else if bound.ident == "Responder" {
+ guess = ResourceType::Sync;
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ guess
+}
+
+impl Route {
+ pub fn new(
+ args: AttributeArgs,
+ input: TokenStream,
+ guard: GuardType,
+ ) -> syn::Result {
+ if args.is_empty() {
+ return Err(syn::Error::new(
+ Span::call_site(),
+ format!(
+ r#"invalid server definition, expected #[{}("")]"#,
+ guard.as_str().to_ascii_lowercase()
+ ),
+ ));
+ }
+ let ast: syn::ItemFn = syn::parse(input)?;
+ let name = ast.sig.ident.clone();
+
+ let args = Args::new(args)?;
+
+ let resource_type = if ast.sig.asyncness.is_some() {
+ ResourceType::Async
+ } else {
+ match ast.sig.output {
+ syn::ReturnType::Default => {
+ return Err(syn::Error::new_spanned(
+ ast,
+ "Function has no return type. Cannot be used as handler",
+ ));
+ }
+ syn::ReturnType::Type(_, ref typ) => guess_resource_type(typ.as_ref()),
+ }
+ };
+
+ Ok(Self {
+ name,
+ args,
+ ast,
+ resource_type,
+ guard,
+ })
+ }
+
+ pub fn generate(&self) -> TokenStream {
+ let name = &self.name;
+ let resource_name = name.to_string();
+ let guard = &self.guard;
+ let ast = &self.ast;
+ let path = &self.args.path;
+ let extra_guards = &self.args.guards;
+ let resource_type = &self.resource_type;
+ let stream = quote! {
+ #[allow(non_camel_case_types)]
+ pub struct #name;
+
+ impl ntex::web::dev::HttpServiceFactory for #name {
+ fn register(self, __config: &mut ntex::web::dev::AppService) {
+ #ast
+ let __resource = ntex::web::Resource::new(#path)
+ .name(#resource_name)
+ .guard(ntex::web::guard::#guard())
+ #(.guard(ntex::web::guard::fn_guard(#extra_guards)))*
+ .#resource_type(#name);
+
+ ntex::web::dev::HttpServiceFactory::register(__resource, __config)
+ }
+ }
+ };
+ stream.into()
+ }
+}
diff --git a/ntex-web-macros/tests/test_macro.rs b/ntex-web-macros/tests/test_macro.rs
new file mode 100644
index 00000000..833dc4d6
--- /dev/null
+++ b/ntex-web-macros/tests/test_macro.rs
@@ -0,0 +1,158 @@
+use ntex::http::{Error, StatusCode, Method};
+use ntex::web::{test, types::Path, App, HttpResponse, Responder};
+use ntex_web_macros::{connect, delete, get, head, options, patch, post, put, trace};
+use futures::{future, Future};
+
+// Make sure that we can name function as 'config'
+#[get("/config")]
+async fn config() -> impl Responder {
+ HttpResponse::Ok()
+}
+
+#[get("/test")]
+async fn test_handler() -> impl Responder {
+ HttpResponse::Ok()
+}
+
+#[put("/test")]
+async fn put_test() -> impl Responder {
+ HttpResponse::Created()
+}
+
+#[patch("/test")]
+async fn patch_test() -> impl Responder {
+ HttpResponse::Ok()
+}
+
+#[post("/test")]
+async fn post_test() -> impl Responder {
+ HttpResponse::NoContent()
+}
+
+#[head("/test")]
+async fn head_test() -> impl Responder {
+ HttpResponse::Ok()
+}
+
+#[connect("/test")]
+async fn connect_test() -> impl Responder {
+ HttpResponse::Ok()
+}
+
+#[options("/test")]
+async fn options_test() -> impl Responder {
+ HttpResponse::Ok()
+}
+
+#[trace("/test")]
+async fn trace_test() -> impl Responder {
+ HttpResponse::Ok()
+}
+
+#[get("/test")]
+fn auto_async() -> impl Future