1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
use super::super::{GetError, OidcClaims, Store};
use drawbridge_type::{Meta, UserContext, UserRecord};
use async_std::sync::Arc;
use axum::http::StatusCode;
use axum::response::IntoResponse;
use axum::{Extension, Json};
use log::{debug, trace, warn};
pub async fn put(
Extension(ref store): Extension<Arc<Store>>,
claims: OidcClaims,
ref cx: UserContext,
meta: Meta,
Json(ref record): Json<UserRecord>,
) -> impl IntoResponse {
trace!(target: "app::users::put", "called for `{cx}`");
if record.subject != claims.subject().as_str() {
return Err((StatusCode::UNAUTHORIZED, "OpenID Connect subject mismatch").into_response());
}
let subj = claims.subject();
match store.user_by_subject(subj).await {
Err(GetError::NotFound) => (),
Err(e) => {
warn!(target: "app::users::put", "failed to get user by OpenID Connect subject `{}`: {:?}", subj.as_str(), e);
return Err(e.into_response());
}
Ok(_) => {
return Err((
StatusCode::CONFLICT,
format!(
"User already associated with OpenID Connect subject `{}`",
subj.as_str()
),
)
.into_response())
}
}
store
.create_user(cx, meta, record)
.await
.map_err(|e| {
debug!(target: "app::users::put", "failed for `{cx}`: {:?}", e);
e.into_response()
})
.map(|_| StatusCode::CREATED)
}