pitchfork_cli/web/
server.rs1use crate::Result;
2use crate::settings::settings;
3use axum::{
4 Router,
5 body::Body,
6 extract::Request,
7 http::StatusCode,
8 middleware::{self, Next},
9 response::{Redirect, Response},
10 routing::{get, post},
11};
12use std::net::SocketAddr;
13
14use super::routes;
15use super::static_files::{set_static_base, set_static_token, static_handler};
16
17async fn token_auth(
20 request: Request<Body>,
21 next: Next,
22 expected_token: String,
23) -> Result<Response, StatusCode> {
24 if expected_token.is_empty() {
25 return Ok(next.run(request).await);
26 }
27 let token = request
28 .headers()
29 .get("X-Pitchfork-Token")
30 .and_then(|v| v.to_str().ok())
31 .unwrap_or("");
32 if token != expected_token {
33 let addr: std::borrow::Cow<'_, str> = request
34 .extensions()
35 .get::<axum::extract::ConnectInfo<SocketAddr>>()
36 .map(|a| a.0.to_string().into())
37 .unwrap_or_else(|| "unknown".into());
38 warn!(
39 "API request rejected: invalid or missing X-Pitchfork-Token from {} to {}",
40 addr,
41 request.uri()
42 );
43 return Err(StatusCode::UNAUTHORIZED);
44 }
45 Ok(next.run(request).await)
46}
47
48fn is_loopback(addr: &str) -> bool {
50 addr.parse::<SocketAddr>()
51 .map(|a| a.ip().is_loopback())
52 .unwrap_or_else(|_| {
53 addr.parse::<std::net::IpAddr>()
55 .map(|ip| ip.is_loopback())
56 .unwrap_or(false)
57 })
58}
59fn generate_token() -> String {
61 let a = uuid::Uuid::new_v4();
62 let b = uuid::Uuid::new_v4();
63 format!("{}{}", a.simple(), b.simple())
64}
65
66fn api_router(token: String) -> Router {
68 let token_clone = token.clone();
69 Router::new()
70 .route("/api/stats", get(routes::api::stats::stats))
71 .route("/api/daemons", get(routes::api::daemons::list))
72 .route("/api/daemons/{id}", get(routes::api::daemons::show))
73 .route("/api/daemons/{id}/start", post(routes::api::daemons::start))
74 .route("/api/daemons/{id}/stop", post(routes::api::daemons::stop))
75 .route(
76 "/api/daemons/{id}/restart",
77 post(routes::api::daemons::restart),
78 )
79 .route(
80 "/api/daemons/{id}/enable",
81 post(routes::api::daemons::enable),
82 )
83 .route(
84 "/api/daemons/{id}/disable",
85 post(routes::api::daemons::disable),
86 )
87 .route("/api/logs/{id}/tail", get(routes::api::logs::tail))
88 .route("/api/namespaces", get(routes::api::namespaces::list))
89 .route("/api/namespaces", post(routes::api::namespaces::register))
90 .route(
91 "/api/namespaces/{name}",
92 axum::routing::delete(routes::api::namespaces::remove),
93 )
94 .route("/api/proxies", get(routes::api::proxies::list))
95 .route(
96 "/api/processes/{id}/tree",
97 get(routes::api::processes::tree),
98 )
99 .route("/logs/{id}/stream", get(routes::logs::stream_sse))
100 .layer(middleware::from_fn(move |req, next| {
101 let t = token_clone.clone();
102 async move { token_auth(req, next, t).await }
103 }))
104}
105
106async fn try_bind(
109 bind_address: &str,
110 port: u16,
111 port_attempts: u16,
112) -> Result<(tokio::net::TcpListener, u16)> {
113 let ip_addr: std::net::IpAddr = bind_address
114 .parse()
115 .map_err(|e| miette::miette!("Invalid bind address '{}': {}", bind_address, e))?;
116
117 let mut last_error = None;
118 for offset in 0..port_attempts {
119 let try_port = port.saturating_add(offset);
120 let addr = SocketAddr::from((ip_addr, try_port));
121
122 match tokio::net::TcpListener::bind(addr).await {
123 Ok(listener) => {
124 let actual_port = listener
125 .local_addr()
126 .map_err(|e| miette::miette!("Failed to inspect bound port: {}", e))?;
127 return Ok((listener, actual_port.port()));
128 }
129 Err(e) => {
130 debug!("Port {try_port} unavailable: {e}");
131 last_error = Some(e);
132 }
133 }
134 }
135
136 Err(miette::miette!(
137 "Failed to bind: tried ports {}-{}, all in use. Last error: {}",
138 port,
139 port.saturating_add(port_attempts - 1),
140 last_error.map(|e| e.to_string()).unwrap_or_default()
141 ))
142}
143
144pub async fn serve(port: u16, web_path: Option<String>) -> Result<()> {
145 let base_path = super::normalize_base_path(web_path.as_deref())?;
146 super::BASE_PATH
147 .set(base_path.clone())
148 .expect("BASE_PATH already set; serve() must only be called once per process");
149 let s = settings();
150 let bind_address = &s.web.bind_address;
151 let port_attempts: u16 = u16::try_from(s.web.port_attempts)
152 .unwrap_or_else(|_| {
153 warn!(
154 "web.port_attempts value {} is out of range (1-65535), clamping to 10",
155 s.web.port_attempts
156 );
157 10
158 })
159 .max(1);
160
161 let mut token = s.api.token.clone();
163 if token.is_empty() && !is_loopback(bind_address) {
164 token = generate_token();
165 info!(
166 "Web UI bound to non-loopback address {}. Auto-generated API token: {}",
167 bind_address, token
168 );
169 eprintln!("pitchfork API security token (auto-generated): {}", token);
171 }
172
173 set_static_token(token.clone());
174 set_static_base(base_path.clone());
175
176 let inner = api_router(token.clone()).fallback(static_handler);
177
178 let app = if base_path.is_empty() {
179 inner
180 } else {
181 let redirect_target = format!("{base_path}/");
182 Router::new()
183 .route(
184 "/",
185 get(move || async move { Redirect::temporary(&redirect_target) }),
186 )
187 .nest(&base_path, inner)
188 };
189
190 let (listener, actual_port) = try_bind(bind_address, port, port_attempts).await?;
191 let _ = super::WEB_PORT.set(actual_port);
192 let actual_addr = listener.local_addr().unwrap();
193
194 info!("Web UI listening on http://{actual_addr}");
195
196 axum::serve(listener, app)
197 .await
198 .map_err(|e| miette::miette!("Web server error: {}", e))
199}
200
201pub async fn serve_api(port: u16, _web_path: Option<String>) -> Result<()> {
204 let s = settings();
205 let bind_address = &s.api.bind_address;
206 let port_attempts: u16 = u16::try_from(s.api.port_attempts)
207 .unwrap_or_else(|_| {
208 warn!(
209 "api.port_attempts value {} is out of range (1-65535), clamping to 10",
210 s.api.port_attempts
211 );
212 10
213 })
214 .max(1);
215
216 let mut token = s.api.token.clone();
218 if token.is_empty() && !is_loopback(bind_address) {
219 token = generate_token();
220 info!(
221 "API server bound to non-loopback address {}. Auto-generated API token: {}",
222 bind_address, token
223 );
224 eprintln!("pitchfork API security token (auto-generated): {}", token);
225 }
226
227 let app = api_router(token);
228
229 let (listener, _actual_port) = try_bind(bind_address, port, port_attempts).await?;
230 let actual_addr = listener.local_addr().unwrap();
231 info!("API server listening on http://{actual_addr}");
232
233 axum::serve(listener, app)
234 .await
235 .map_err(|e| miette::miette!("API server error: {}", e))
236}