1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
/*
* Copyright 2024 G-Core Innovations SARL
*/
use proc_macro::TokenStream;

use quote::quote;
use syn::{parse_macro_input, ItemFn};

/// Main function attribute for a FastEdge application.
///
/// ## Usage
///
/// The `main` function takes a request and returns a response or an error. For example:
///
/// ```rust,no_run
/// use anyhow::Result;
/// use fastedge::http::{Request, Response, StatusCode};
/// use fastedge::body::Body;
///
/// #[fastedge::http]
/// fn main(req: Request<Body>) -> Result<Response<Body>> {
///     Response::builder().status(StatusCode::OK).body(Body::empty())
/// }
#[proc_macro_attribute]
pub fn http(_attr: TokenStream, item: TokenStream) -> TokenStream {
    let func = parse_macro_input!(item as ItemFn);
    let func_name = &func.sig.ident;

    quote!(
        use fastedge::http_handler::Guest;
        struct Component;

        #[inline(always)]
        fn internal_error(body: &str) -> ::fastedge::http_handler::Response {
            ::fastedge::http_handler::Response {
                status: ::fastedge::http::StatusCode::INTERNAL_SERVER_ERROR.as_u16(),
                headers: Some(vec![]),
                body: Some(body.as_bytes().to_vec()),
            }
        }

        #[inline(always)]
        #[no_mangle]
        #func

        impl Guest for Component {
            #[no_mangle]
            fn process(req: ::fastedge::http_handler::Request) -> ::fastedge::http_handler::Response {

                let Ok(request) = req.try_into() else {
                    return internal_error("http request decode error")
                };

                let res = match #func_name(request) {
                    Ok(res) => res,
                    Err(error) => {
                        return internal_error(error.to_string().as_str());
                    }
                };

                let Ok(response) = ::fastedge::http_handler::Response::try_from(res) else {
                    return internal_error("http response encode error")
                };
                response
            }
        }

        fastedge::export!(Component with_types_in fastedge);


    ).into()
}