Skip to main content

ironflow_api/routes/
mod.rs

1//! Router assembly — one module per route.
2
3pub 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
39/// Maximum request body size: 2 MiB.
40const MAX_BODY_SIZE: usize = 2 * 1024 * 1024;
41
42/// Router-level configuration with sensible defaults.
43///
44/// Controls dashboard serving, rate limiting, and other router behaviors.
45/// Use [`Default::default()`] for production-ready defaults, then override
46/// individual fields as needed.
47///
48/// # Examples
49///
50/// ```
51/// use ironflow_api::routes::RouterConfig;
52///
53/// // All defaults: rate limiting enabled, no custom dashboard dir
54/// let config = RouterConfig::default();
55/// assert_eq!(config.rate_limit_auth, Some(10));
56///
57/// // Disable auth rate limiting, custom dashboard
58/// let config = RouterConfig {
59///     rate_limit_auth: None,
60///     ..RouterConfig::default()
61/// };
62/// ```
63#[derive(Debug, Clone)]
64pub struct RouterConfig {
65    /// Filesystem path to dashboard assets. When set, serves the SPA
66    /// from this directory instead of the embedded build.
67    pub dashboard_dir: Option<PathBuf>,
68    /// Rate limit for auth credential routes (sign-in, sign-up) in
69    /// requests per minute per IP. `None` disables the limiter.
70    pub rate_limit_auth: Option<u32>,
71    /// Rate limit for general public API routes in requests per minute
72    /// per IP. `None` disables the limiter.
73    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/// Handler that returns a JSON 404 when the `sign-up` feature is disabled.
87#[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
92/// Create the main application router.
93///
94/// # Examples
95///
96/// ```no_run
97/// use ironflow_api::routes::{RouterConfig, create_router};
98/// use ironflow_api::state::AppState;
99/// use ironflow_auth::jwt::JwtConfig;
100/// use ironflow_store::prelude::*;
101/// use ironflow_engine::engine::Engine;
102/// use ironflow_core::providers::claude::ClaudeCodeProvider;
103/// use std::sync::Arc;
104/// use tokio::sync::broadcast;
105/// use ironflow_engine::notify::Event;
106///
107/// # async fn example() {
108/// let store: Arc<dyn ironflow_store::store::Store> = Arc::new(InMemoryStore::new());
109/// let provider = Arc::new(ClaudeCodeProvider::new());
110/// let engine = Arc::new(Engine::new(store.clone(), provider));
111/// let jwt_config = Arc::new(JwtConfig {
112///     secret: "secret".to_string(),
113///     access_token_ttl_secs: 900,
114///     refresh_token_ttl_secs: 604800,
115///     cookie_domain: None,
116///     cookie_secure: false,
117/// });
118/// let (event_sender, _) = broadcast::channel::<Event>(1);
119/// let state = AppState::new(store, engine, jwt_config, "token".to_string(), event_sender);
120/// let router = create_router(state, RouterConfig::default());
121/// # }
122/// ```
123pub fn create_router(state: AppState, config: RouterConfig) -> Router {
124    // Internal routes (worker-to-API, protected by WORKER_TOKEN)
125    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    // Auth credential routes (rate-limited when configured)
149    #[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    // Auth session routes (no strict rate limiting, covered by general limiter)
173    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    // Public + user-authenticated routes (rate-limited when configured)
179    #[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        // 3 MiB payload — exceeds the 2 MiB limit
398        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}