shoutdotdev/src/router.rs

232 lines
6.7 KiB
Rust
Raw Normal View History

2025-02-26 13:10:47 -08:00
use anyhow::Context;
2025-02-26 13:10:50 -08:00
use askama_axum::Template;
use axum::{
extract::{Path, State},
response::{Html, IntoResponse, Redirect},
routing::{get, post},
Form, Router,
};
2025-02-26 13:10:48 -08:00
use diesel::{dsl::insert_into, prelude::*};
2025-02-26 13:10:50 -08:00
use serde::Deserialize;
use tower::ServiceBuilder;
use tower_http::{
compression::CompressionLayer,
services::{ServeDir, ServeFile},
trace::TraceLayer,
};
use uuid::Uuid;
use crate::{
api_keys::ApiKey,
app_error::AppError,
2025-02-26 13:10:48 -08:00
app_state::{AppState, DbConn},
auth,
csrf::generate_csrf_token,
guards,
2025-02-26 13:10:50 -08:00
projects::Project,
schema,
2025-02-26 13:10:48 -08:00
settings::Settings,
team_memberships::TeamMembership,
teams::Team,
2025-02-26 13:10:50 -08:00
users::{CurrentUser, User},
2025-02-26 13:10:47 -08:00
v0_router,
2025-02-26 13:10:50 -08:00
};
pub fn new_router(state: AppState) -> Router<()> {
let base_path = state.settings.base_path.clone();
Router::new().nest(
format!("{}", base_path).as_str(),
Router::new()
.route("/", get(landing_page))
2025-02-26 13:10:47 -08:00
.nest("/v0", v0_router::new_router(state.clone()))
2025-02-26 13:10:50 -08:00
.route("/teams", get(teams_page))
.route("/teams/{team_id}", get(team_page))
.route("/teams/{team_id}/projects", get(projects_page))
.route("/teams/{team_id}/new-api-key", post(post_new_api_key))
.route("/new-team", get(new_team_page))
.route("/new-team", post(post_new_team))
.nest("/auth", auth::new_router())
.fallback_service(
ServeDir::new("static").not_found_service(ServeFile::new("static/_404.html")),
)
.layer(
ServiceBuilder::new()
.layer(TraceLayer::new_for_http())
.layer(CompressionLayer::new()),
)
.with_state(state),
)
}
async fn landing_page(State(state): State<AppState>) -> impl IntoResponse {
Redirect::to(&format!("{}/teams", state.settings.base_path))
}
async fn teams_page(
2025-02-26 13:10:48 -08:00
State(Settings { base_path, .. }): State<Settings>,
DbConn(conn): DbConn,
2025-02-26 13:10:50 -08:00
CurrentUser(current_user): CurrentUser,
) -> Result<impl IntoResponse, AppError> {
2025-02-26 13:10:47 -08:00
let team_memberships_query = current_user.clone().team_memberships();
let teams: Vec<Team> = conn
.interact(move |conn| team_memberships_query.load(conn))
2025-02-26 13:10:50 -08:00
.await
2025-02-26 13:10:47 -08:00
.unwrap()
.context("failed to load team memberships")
.map(|memberships| memberships.into_iter().map(|(_, team)| team).collect())?;
2025-02-26 13:10:50 -08:00
#[derive(Template)]
#[template(path = "teams.html")]
struct ResponseTemplate {
base_path: String,
teams: Vec<Team>,
current_user: User,
}
Ok(Html(
ResponseTemplate {
2025-02-26 13:10:48 -08:00
base_path,
2025-02-26 13:10:50 -08:00
current_user,
2025-02-26 13:10:47 -08:00
teams,
2025-02-26 13:10:50 -08:00
}
.render()?,
2025-02-26 13:10:48 -08:00
))
2025-02-26 13:10:50 -08:00
}
async fn team_page(State(state): State<AppState>, Path(team_id): Path<Uuid>) -> impl IntoResponse {
Redirect::to(&format!(
"{}/teams/{}/projects",
state.settings.base_path, team_id
))
}
#[derive(Deserialize)]
struct PostNewApiKeyForm {
csrf_token: String,
}
async fn post_new_api_key(
2025-02-26 13:10:48 -08:00
State(Settings { base_path, .. }): State<Settings>,
DbConn(db_conn): DbConn,
2025-02-26 13:10:50 -08:00
Path(team_id): Path<Uuid>,
2025-02-26 13:10:48 -08:00
CurrentUser(current_user): CurrentUser,
2025-02-26 13:10:50 -08:00
Form(form): Form<PostNewApiKeyForm>,
) -> Result<impl IntoResponse, AppError> {
guards::require_valid_csrf_token(&form.csrf_token, &current_user, &db_conn).await?;
let team = guards::require_team_membership(&current_user, &team_id, &db_conn).await?;
2025-02-26 13:10:48 -08:00
ApiKey::generate_for_team(&db_conn, team.id.clone()).await?;
2025-02-26 13:10:50 -08:00
Ok(Redirect::to(&format!(
"{}/teams/{}/projects",
2025-02-26 13:10:48 -08:00
base_path,
team.id.hyphenated().to_string()
2025-02-26 13:10:50 -08:00
))
.into_response())
}
async fn new_team_page(
2025-02-26 13:10:48 -08:00
State(Settings { base_path, .. }): State<Settings>,
DbConn(db_conn): DbConn,
2025-02-26 13:10:50 -08:00
CurrentUser(current_user): CurrentUser,
) -> Result<impl IntoResponse, AppError> {
2025-02-26 13:10:48 -08:00
let csrf_token = generate_csrf_token(&db_conn, Some(current_user.id)).await?;
2025-02-26 13:10:47 -08:00
2025-02-26 13:10:50 -08:00
#[derive(Template)]
#[template(path = "new-team.html")]
struct ResponseTemplate {
base_path: String,
csrf_token: String,
current_user: User,
}
Ok(Html(
ResponseTemplate {
2025-02-26 13:10:48 -08:00
base_path,
2025-02-26 13:10:50 -08:00
csrf_token,
current_user,
}
.render()?,
))
}
#[derive(Deserialize)]
struct PostNewTeamForm {
name: String,
csrf_token: String,
}
async fn post_new_team(
2025-02-26 13:10:48 -08:00
DbConn(db_conn): DbConn,
State(Settings { base_path, .. }): State<Settings>,
CurrentUser(current_user): CurrentUser,
2025-02-26 13:10:50 -08:00
Form(form): Form<PostNewTeamForm>,
) -> Result<impl IntoResponse, AppError> {
guards::require_valid_csrf_token(&form.csrf_token, &current_user, &db_conn).await?;
2025-02-26 13:10:48 -08:00
2025-02-26 13:10:50 -08:00
let team_id = Uuid::now_v7();
let team = Team {
id: team_id.clone(),
name: form.name,
};
let team_membership = TeamMembership {
team_id: team_id.clone(),
user_id: current_user.id,
roles: vec![Some("OWNER".to_string())],
};
2025-02-26 13:10:48 -08:00
db_conn
2025-02-26 13:10:50 -08:00
.interact(move |conn| {
conn.transaction(move |conn| {
insert_into(schema::teams::table)
.values(team)
.execute(conn)?;
insert_into(schema::team_memberships::table)
.values(team_membership)
.execute(conn)?;
diesel::QueryResult::Ok(())
})
})
.await
.unwrap()
.unwrap();
2025-02-26 13:10:48 -08:00
ApiKey::generate_for_team(&db_conn, team_id.clone()).await?;
Ok(Redirect::to(&format!("{}/teams/{}/projects", base_path, team_id)).into_response())
2025-02-26 13:10:50 -08:00
}
async fn projects_page(
2025-02-26 13:10:48 -08:00
State(Settings { base_path, .. }): State<Settings>,
DbConn(db_conn): DbConn,
2025-02-26 13:10:50 -08:00
Path(team_id): Path<Uuid>,
CurrentUser(current_user): CurrentUser,
) -> Result<impl IntoResponse, AppError> {
let team = guards::require_team_membership(&current_user, &team_id, &db_conn).await?;
2025-02-26 13:10:48 -08:00
2025-02-26 13:10:47 -08:00
let api_keys_query = team.clone().api_keys();
let (api_keys, projects) = db_conn
2025-02-26 13:10:50 -08:00
.interact(move |conn| {
2025-02-26 13:10:47 -08:00
diesel::QueryResult::Ok((api_keys_query.load(conn)?, Project::all().load(conn)?))
2025-02-26 13:10:50 -08:00
})
.await
2025-02-26 13:10:48 -08:00
.unwrap()?;
2025-02-26 13:10:47 -08:00
2025-02-26 13:10:50 -08:00
#[derive(Template)]
#[template(path = "projects.html")]
struct ResponseTemplate {
base_path: String,
csrf_token: String,
keys: Vec<ApiKey>,
projects: Vec<Project>,
team: Team,
current_user: User,
}
2025-02-26 13:10:48 -08:00
let csrf_token = generate_csrf_token(&db_conn, Some(current_user.id.clone())).await?;
2025-02-26 13:10:50 -08:00
Ok(Html(
ResponseTemplate {
2025-02-26 13:10:48 -08:00
base_path,
2025-02-26 13:10:50 -08:00
csrf_token,
current_user,
2025-02-26 13:10:47 -08:00
projects,
2025-02-26 13:10:50 -08:00
team,
keys: api_keys,
}
.render()?,
)
.into_response())
}