1pub mod api_keys;
4pub mod approve_run;
5pub mod audit_logs;
6pub mod auth;
7pub mod cancel_run;
8pub mod create_run;
9pub mod events;
10pub mod get_run;
11pub mod get_stats;
12pub mod get_workflow;
13pub mod health_check;
14mod internal;
15pub mod list_runs;
16pub mod list_workflows;
17#[cfg(feature = "prometheus")]
18pub mod metrics;
19pub mod openapi_spec;
20pub mod retry_run;
21pub mod secrets;
22#[cfg(test)]
23mod test_helpers;
24pub mod users;
25
26use std::path::PathBuf;
27
28use axum::Extension;
29use axum::Router;
30use axum::middleware as axum_mw;
31use axum::routing::{delete, get, patch, post, put};
32use tower_http::limit::RequestBodyLimitLayer;
33use tower_http::services::{ServeDir, ServeFile};
34
35use crate::middleware::{WorkerToken, security_headers, worker_token_auth};
36use crate::rate_limit::{per_minute, rate_limit};
37use crate::state::AppState;
38
39const MAX_BODY_SIZE: usize = 2 * 1024 * 1024;
41
42#[derive(Debug, Clone)]
64pub struct RouterConfig {
65 pub dashboard_dir: Option<PathBuf>,
68 pub rate_limit_auth: Option<u32>,
71 pub rate_limit_general: Option<u32>,
74}
75
76impl Default for RouterConfig {
77 fn default() -> Self {
78 Self {
79 dashboard_dir: None,
80 rate_limit_auth: Some(10),
81 rate_limit_general: Some(60),
82 }
83 }
84}
85
86#[cfg(not(feature = "sign-up"))]
88async fn sign_up_disabled() -> impl axum::response::IntoResponse {
89 crate::error::ApiError::BadRequest("sign-up is disabled".to_string())
90}
91
92pub fn create_router(state: AppState, config: RouterConfig) -> Router {
124 let internal_routes = Router::new()
126 .route("/runs", post(internal::create_run::create_run))
127 .route("/runs/next", get(internal::pick_next_run::pick_next_run))
128 .route(
129 "/runs/{id}",
130 get(internal::get_run::get_run).put(internal::update_run::update_run),
131 )
132 .route(
133 "/runs/{id}/status",
134 put(internal::update_run_status::update_run_status),
135 )
136 .route("/runs/{id}/logs", post(internal::push_logs::push_logs))
137 .route("/steps", post(internal::create_step::create_step))
138 .route("/steps/{id}", put(internal::update_step::update_step))
139 .route(
140 "/step-dependencies",
141 post(internal::create_step_dependencies::create_step_dependencies),
142 )
143 .route("/secrets/{*key}", get(internal::get_secret::get_secret))
144 .layer(axum_mw::from_fn(worker_token_auth))
145 .layer(Extension(WorkerToken(state.worker_token.clone())))
146 .with_state(state.clone());
147
148 #[allow(unused_mut)]
150 let mut auth_credential_routes = Router::new();
151
152 #[cfg(feature = "sign-up")]
153 {
154 auth_credential_routes =
155 auth_credential_routes.route("/sign-up", post(auth::sign_up::sign_up));
156 }
157
158 #[cfg(not(feature = "sign-up"))]
159 {
160 auth_credential_routes = auth_credential_routes.route("/sign-up", post(sign_up_disabled));
161 }
162
163 let mut auth_credential_routes =
164 auth_credential_routes.route("/sign-in", post(auth::sign_in::sign_in));
165
166 if let Some(rpm) = config.rate_limit_auth {
167 auth_credential_routes = auth_credential_routes
168 .layer(axum_mw::from_fn(rate_limit))
169 .layer(Extension(per_minute(rpm)));
170 }
171
172 let auth_session_routes = Router::new()
174 .route("/refresh", post(auth::refresh::refresh))
175 .route("/sign-out", post(auth::sign_out::sign_out))
176 .route("/me", get(auth::me::me));
177
178 #[allow(unused_mut)]
180 let mut api_v1 = Router::new()
181 .route("/health-check", get(health_check::health_check))
182 .route("/openapi.json", get(openapi_spec::openapi_spec))
183 .route(
184 "/runs",
185 get(list_runs::list_runs).post(create_run::create_run),
186 )
187 .route("/runs/{id}", get(get_run::get_run))
188 .route("/runs/{id}/cancel", post(cancel_run::cancel_run))
189 .route("/runs/{id}/approve", post(approve_run::approve_run))
190 .route("/runs/{id}/reject", post(approve_run::reject_run))
191 .route("/runs/{id}/retry", post(retry_run::retry_run))
192 .route("/workflows", get(list_workflows::list_workflows))
193 .route("/workflows/{name}", get(get_workflow::get_workflow))
194 .route("/stats", get(get_stats::get_stats))
195 .route("/audit-logs", get(audit_logs::list_audit_logs))
196 .route("/events", get(events::events))
197 .route(
198 "/api-keys",
199 get(api_keys::list::list_api_keys).post(api_keys::create::create_api_key),
200 )
201 .route(
202 "/api-keys/scopes",
203 get(api_keys::available_scopes::available_scopes),
204 )
205 .route("/api-keys/{id}", delete(api_keys::delete::delete_api_key))
206 .route(
207 "/users",
208 get(users::list::list_users).post(users::create::create_user),
209 )
210 .route("/users/{id}", delete(users::delete::delete_user))
211 .route("/users/{id}/role", patch(users::update_role::update_role))
212 .route(
213 "/secrets",
214 get(secrets::list::list_secrets).post(secrets::create::create_secret),
215 )
216 .route(
217 "/secrets/{*key}",
218 put(secrets::update::update_secret).delete(secrets::delete::delete_secret),
219 );
220
221 #[cfg(feature = "prometheus")]
222 {
223 api_v1 = api_v1.route("/metrics", get(metrics::metrics));
224 }
225
226 let mut api_v1 = api_v1
227 .nest("/auth", auth_credential_routes)
228 .nest("/auth", auth_session_routes);
229
230 if let Some(rpm) = config.rate_limit_general {
231 api_v1 = api_v1
232 .layer(axum_mw::from_fn(rate_limit))
233 .layer(Extension(per_minute(rpm)));
234 }
235
236 let api_v1 = api_v1.with_state(state.clone());
237
238 #[allow(unused_mut)]
239 let mut app = Router::new()
240 .nest("/api/v1/internal", internal_routes)
241 .nest("/api/v1", api_v1)
242 .with_state(state)
243 .layer(RequestBodyLimitLayer::new(MAX_BODY_SIZE))
244 .layer(axum_mw::from_fn(security_headers));
245
246 #[cfg(feature = "prometheus")]
247 {
248 app = app.layer(axum_mw::from_fn(crate::middleware::request_metrics));
249 }
250
251 match config.dashboard_dir {
252 Some(dir) => {
253 let index = dir.join("index.html");
254 let serve = ServeDir::new(dir).fallback(ServeFile::new(index));
255 app.fallback_service(serve)
256 }
257 #[cfg(feature = "dashboard")]
258 None => app.fallback_service(crate::dashboard::EmbeddedDashboard),
259 #[cfg(not(feature = "dashboard"))]
260 None => app,
261 }
262}
263
264#[cfg(test)]
265mod tests {
266 use super::*;
267 use axum::body::Body;
268 use axum::http::{Request, StatusCode};
269 use http_body_util::BodyExt;
270 use ironflow_core::providers::claude::ClaudeCodeProvider;
271 use ironflow_engine::engine::Engine;
272 use ironflow_engine::notify::Event;
273 use ironflow_store::memory::InMemoryStore;
274 use std::sync::Arc;
275 use tokio::sync::broadcast;
276 use tower::ServiceExt;
277 fn test_state() -> AppState {
278 let store = Arc::new(InMemoryStore::new());
279 let provider = Arc::new(ClaudeCodeProvider::new());
280 let engine = Arc::new(Engine::new(store.clone(), provider));
281 let jwt_config = Arc::new(ironflow_auth::jwt::JwtConfig {
282 secret: "test-secret".to_string(),
283 access_token_ttl_secs: 900,
284 refresh_token_ttl_secs: 604800,
285 cookie_domain: None,
286 cookie_secure: false,
287 });
288 let (event_sender, _) = broadcast::channel::<Event>(1);
289 AppState::new(
290 store,
291 engine,
292 jwt_config,
293 "test-worker-token".to_string(),
294 event_sender,
295 )
296 }
297
298 #[tokio::test]
299 async fn health_check_route() {
300 let state = test_state();
301 let app = create_router(state, RouterConfig::default());
302
303 let req = Request::builder()
304 .uri("/api/v1/health-check")
305 .body(Body::empty())
306 .unwrap();
307
308 let resp = app.oneshot(req).await.unwrap();
309 assert_eq!(resp.status(), StatusCode::OK);
310
311 let body = resp.into_body().collect().await.unwrap().to_bytes();
312 assert_eq!(&body[..], b"OK");
313 }
314
315 fn make_auth_header(state: &AppState) -> String {
316 use ironflow_auth::jwt::AccessToken;
317 use uuid::Uuid;
318
319 let user_id = Uuid::now_v7();
320 let token = AccessToken::for_user(user_id, "testuser", false, &state.jwt_config).unwrap();
321 format!("Bearer {}", token.0)
322 }
323
324 #[tokio::test]
325 async fn runs_route_exists() {
326 let state = test_state();
327 let app = create_router(state.clone(), RouterConfig::default());
328 let auth_header = make_auth_header(&state);
329
330 let req = Request::builder()
331 .uri("/api/v1/runs?page=1&per_page=20")
332 .header("authorization", auth_header)
333 .body(Body::empty())
334 .unwrap();
335
336 let resp = app.oneshot(req).await.unwrap();
337 assert_eq!(resp.status(), StatusCode::OK);
338 }
339
340 #[tokio::test]
341 async fn stats_route_exists() {
342 let state = test_state();
343 let app = create_router(state.clone(), RouterConfig::default());
344 let auth_header = make_auth_header(&state);
345
346 let req = Request::builder()
347 .uri("/api/v1/stats")
348 .header("authorization", auth_header)
349 .body(Body::empty())
350 .unwrap();
351
352 let resp = app.oneshot(req).await.unwrap();
353 assert_eq!(resp.status(), StatusCode::OK);
354 }
355
356 #[tokio::test]
357 async fn responses_include_security_headers() {
358 let state = test_state();
359 let app = create_router(state, RouterConfig::default());
360
361 let req = Request::builder()
362 .uri("/api/v1/health-check")
363 .body(Body::empty())
364 .unwrap();
365
366 let resp = app.oneshot(req).await.unwrap();
367
368 assert_eq!(
369 resp.headers().get("x-content-type-options").unwrap(),
370 "nosniff"
371 );
372 assert_eq!(resp.headers().get("x-frame-options").unwrap(), "DENY");
373 assert_eq!(
374 resp.headers().get("x-xss-protection").unwrap(),
375 "1; mode=block"
376 );
377 assert_eq!(
378 resp.headers().get("strict-transport-security").unwrap(),
379 "max-age=63072000; includeSubDomains"
380 );
381 assert!(
382 resp.headers()
383 .get("content-security-policy")
384 .unwrap()
385 .to_str()
386 .unwrap()
387 .contains("default-src 'self'")
388 );
389 }
390
391 #[tokio::test]
392 async fn body_size_limit_rejects_oversized_payload() {
393 let state = test_state();
394 let app = create_router(state.clone(), RouterConfig::default());
395 let auth_header = make_auth_header(&state);
396
397 let oversized = vec![0u8; 3 * 1024 * 1024];
399
400 let req = Request::builder()
401 .method("POST")
402 .uri("/api/v1/runs")
403 .header("content-type", "application/json")
404 .header("authorization", auth_header)
405 .body(Body::from(oversized))
406 .unwrap();
407
408 let resp = app.oneshot(req).await.unwrap();
409 assert_eq!(resp.status(), StatusCode::PAYLOAD_TOO_LARGE);
410 }
411}