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
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
use crate::context::PuffContext;
use crate::errors::Result;
use crate::program::{Runnable, RunnableCommand};
use crate::python::asgi::{create_server_context, AsyncFn};
use crate::web::server::Router;
use clap::{ArgMatches, Command};
use std::process::ExitCode;
use crate::types::Text;
use futures_util::future::LocalBoxFuture;
use futures_util::FutureExt;
use pyo3::prelude::*;
use tokio::sync::oneshot::Receiver;
use crate::program::commands::HttpServerConfig;
use crate::python::asgi::handler::AsgiHandler;
struct ASGIConstructor {
config: HttpServerConfig,
router: Router,
puff_context: PuffContext,
}
impl AsyncFn for ASGIConstructor {
fn call(self, handler: AsgiHandler, rx: Receiver<()>) -> LocalBoxFuture<'static, ()> {
start(self.config, self.router, self.puff_context, handler, rx).boxed_local()
}
}
pub struct ASGIServerCommand {
router_fn: Option<Box<dyn FnOnce() -> Router + 'static>>,
app_path: Text,
}
impl ASGIServerCommand {
pub fn new<M: Into<Text>>(app_path: M) -> Self {
Self {
router_fn: Some(Box::new(|| Router::new())),
app_path: app_path.into(),
}
}
pub fn new_with_router<M: Into<Text>>(app_path: M, r: Router) -> Self {
Self {
router_fn: Some(Box::new(|| r)),
app_path: app_path.into(),
}
}
pub fn new_with_router_init<M: Into<Text>, F: FnOnce() -> Router + 'static>(
app_path: M,
f: F,
) -> Self {
Self {
router_fn: Some(Box::new(f)),
app_path: app_path.into(),
}
}
}
impl RunnableCommand for ASGIServerCommand {
fn cli_parser(&self) -> Command {
HttpServerConfig::add_command_options(Command::new("runserver"))
}
fn make_runnable(&mut self, args: &ArgMatches, context: PuffContext) -> Result<Runnable> {
let wsgi_app = Python::with_gil(|py| {
let puff_mod = py.import("puff")?;
let puff_asyncio_mod = py.import("puff.asyncio_support")?;
let app =
puff_mod.call_method1("import_string", (self.app_path.clone().into_py(py),))?;
let new_app = puff_asyncio_mod.call_method1("wrap_asgi", (app,))?;
PyResult::Ok(new_app.into_py(py))
})?;
let config = HttpServerConfig::new_from_args(args);
let router_fn = self.router_fn.take().expect("Already ran.");
let fut = async move {
let router = router_fn();
let mut ctx = create_server_context(
wsgi_app,
ASGIConstructor {
config,
router,
puff_context: context.clone(),
},
);
ctx.start()?.await?;
Ok(ExitCode::SUCCESS)
};
Ok(Runnable::new(fut))
}
}
async fn start(
http_configuration: HttpServerConfig,
router: Router,
puff_context: PuffContext,
asgi: AsgiHandler,
rx: Receiver<()>,
) {
let app = router.into_axum_router(puff_context).fallback(asgi);
if let Err(err) = http_configuration
.server_builder()
.serve(app.into_make_service())
.with_graceful_shutdown(async move {
if let Err(e) = rx.await {
eprintln!("{e}");
}
})
.await
{
eprintln!("error running server: {err}");
};
}