mas_handlers/admin/v1/users/
list.rs1use aide::{OperationIo, transform::TransformOperation};
8use axum::{
9 Json,
10 extract::{Query, rejection::QueryRejection},
11 response::IntoResponse,
12};
13use axum_macros::FromRequestParts;
14use hyper::StatusCode;
15use mas_axum_utils::record_error;
16use mas_storage::{Page, user::UserFilter};
17use schemars::JsonSchema;
18use serde::Deserialize;
19
20use crate::{
21 admin::{
22 call_context::CallContext,
23 model::{Resource, User},
24 params::Pagination,
25 response::{ErrorResponse, PaginatedResponse},
26 },
27 impl_from_error_for_route,
28};
29
30#[derive(Deserialize, JsonSchema, Clone, Copy)]
31#[serde(rename_all = "snake_case")]
32enum UserStatus {
33 Active,
34 Locked,
35 Deactivated,
36}
37
38impl std::fmt::Display for UserStatus {
39 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
40 match self {
41 Self::Active => write!(f, "active"),
42 Self::Locked => write!(f, "locked"),
43 Self::Deactivated => write!(f, "deactivated"),
44 }
45 }
46}
47
48#[derive(FromRequestParts, Deserialize, JsonSchema, OperationIo)]
49#[serde(rename = "UserFilter")]
50#[aide(input_with = "Query<FilterParams>")]
51#[from_request(via(Query), rejection(RouteError))]
52pub struct FilterParams {
53 #[serde(rename = "filter[admin]")]
55 admin: Option<bool>,
56
57 #[serde(rename = "filter[status]")]
67 status: Option<UserStatus>,
68}
69
70impl std::fmt::Display for FilterParams {
71 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
72 let mut sep = '?';
73
74 if let Some(admin) = self.admin {
75 write!(f, "{sep}filter[admin]={admin}")?;
76 sep = '&';
77 }
78 if let Some(status) = self.status {
79 write!(f, "{sep}filter[status]={status}")?;
80 sep = '&';
81 }
82
83 let _ = sep;
84 Ok(())
85 }
86}
87
88#[derive(Debug, thiserror::Error, OperationIo)]
89#[aide(output_with = "Json<ErrorResponse>")]
90pub enum RouteError {
91 #[error(transparent)]
92 Internal(Box<dyn std::error::Error + Send + Sync + 'static>),
93
94 #[error("Invalid filter parameters")]
95 InvalidFilter(#[from] QueryRejection),
96}
97
98impl_from_error_for_route!(mas_storage::RepositoryError);
99
100impl IntoResponse for RouteError {
101 fn into_response(self) -> axum::response::Response {
102 let error = ErrorResponse::from_error(&self);
103 let sentry_event_id = record_error!(self, Self::Internal(_));
104 let status = match self {
105 Self::Internal(_) => StatusCode::INTERNAL_SERVER_ERROR,
106 Self::InvalidFilter(_) => StatusCode::BAD_REQUEST,
107 };
108 (status, sentry_event_id, Json(error)).into_response()
109 }
110}
111
112pub fn doc(operation: TransformOperation) -> TransformOperation {
113 operation
114 .id("listUsers")
115 .summary("List users")
116 .tag("user")
117 .response_with::<200, Json<PaginatedResponse<User>>, _>(|t| {
118 let users = User::samples();
119 let pagination = mas_storage::Pagination::first(users.len());
120 let page = Page {
121 edges: users.into(),
122 has_next_page: true,
123 has_previous_page: false,
124 };
125
126 t.description("Paginated response of users")
127 .example(PaginatedResponse::new(page, pagination, 42, User::PATH))
128 })
129}
130
131#[tracing::instrument(name = "handler.admin.v1.users.list", skip_all)]
132pub async fn handler(
133 CallContext { mut repo, .. }: CallContext,
134 Pagination(pagination): Pagination,
135 params: FilterParams,
136) -> Result<Json<PaginatedResponse<User>>, RouteError> {
137 let base = format!("{path}{params}", path = User::PATH);
138 let filter = UserFilter::default();
139
140 let filter = match params.admin {
141 Some(true) => filter.can_request_admin_only(),
142 Some(false) => filter.cannot_request_admin_only(),
143 None => filter,
144 };
145
146 let filter = match params.status {
147 Some(UserStatus::Active) => filter.active_only(),
148 Some(UserStatus::Locked) => filter.locked_only(),
149 Some(UserStatus::Deactivated) => filter.deactivated_only(),
150 None => filter,
151 };
152
153 let page = repo.user().list(filter, pagination).await?;
154 let count = repo.user().count(filter).await?;
155
156 Ok(Json(PaginatedResponse::new(
157 page.map(User::from),
158 pagination,
159 count,
160 &base,
161 )))
162}