use activitypub_federation::{ axum::json::FederationJson, config::Data, protocol::context::WithContext, traits::Object, }; use axum::{debug_handler, extract::Path, http::StatusCode, response::IntoResponse}; use tracing::trace; use crate::{error::AppError, state::AppHandle}; #[debug_handler] pub async fn http_get_user( Path(name): Path, data: Data, ) -> Result { if let Some(a) = read_user(&name, &data).await? { let json_user = a.into_json(&data).await?; Ok(( StatusCode::OK, FederationJson(WithContext::new_default(json_user)), ) .into_response()) } else { Ok((StatusCode::NOT_FOUND, "").into_response()) } } pub async fn read_user( name: &str, data: &Data, ) -> Result, AppError> { trace!(username = name, "getting user"); let read = sqlx::query_as!( crate::entity::user::DbUser, "select * from account where username = $1 and local = $2", name, true ) .fetch_optional(&data.services.postgres) .await?; let user = read.into_iter().find(|value| value.username.eq(&name)); let user = match user { Some(user) => Some(crate::entity::user::User::try_from(user)?), None => None, }; Ok(user) } #[cfg(test)] mod tests { use axum::{ body::Body, http::{Request, StatusCode}, }; use sqlx::PgPool; use stack_up::Services; use tower::ServiceExt; use crate::{ server::{self, test_config}, state::AppState, }; #[sqlx::test] async fn get_user(pool: PgPool) { let services = Services { postgres: pool }; let state = AppState::new(services, &test_config()).await.unwrap(); let app = server::router(state); let response = app .oneshot( Request::builder() .uri("/users/sellershut") .body(Body::empty()) .unwrap(), ) .await .unwrap(); assert_eq!(response.status(), StatusCode::OK); } #[sqlx::test] async fn get_user_not_found(pool: PgPool) { let services = Services { postgres: pool }; let state = AppState::new(services, &test_config()).await.unwrap(); let app = server::router(state); let response = app .oneshot( Request::builder() .uri("/users/selut") .body(Body::empty()) .unwrap(), ) .await .unwrap(); assert_eq!(response.status(), StatusCode::NOT_FOUND); } }