refactor: eliminate User/UserResponse struct literals, add AP user tests
- Feed/search adapters use #[sqlx(flatten)] UserRow instead of inline User construction — single point of change when User gains fields - User::new_remote constructor replaces struct literal in testing - to_summary_response replaces inline UserResponse in get_users - 5 integration tests for PgApUserRepository (find, count, profile_fields→attachment)
This commit is contained in:
1
crates/adapters/postgres-federation/migrations
Symbolic link
1
crates/adapters/postgres-federation/migrations
Symbolic link
@@ -0,0 +1 @@
|
|||||||
|
../postgres/migrations
|
||||||
@@ -832,3 +832,91 @@ impl ApUserRepository for PgApUserRepository {
|
|||||||
Ok(n as usize)
|
Ok(n as usize)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
use k_ap::ApUserRepository;
|
||||||
|
|
||||||
|
async fn seed_local_user(pool: &PgPool, username: &str) -> uuid::Uuid {
|
||||||
|
let id = uuid::Uuid::new_v4();
|
||||||
|
sqlx::query(
|
||||||
|
"INSERT INTO users (id,username,email,password_hash,local,created_at,updated_at)
|
||||||
|
VALUES ($1,$2,$3,'h',true,NOW(),NOW())",
|
||||||
|
)
|
||||||
|
.bind(id)
|
||||||
|
.bind(username)
|
||||||
|
.bind(format!("{username}@test.com"))
|
||||||
|
.execute(pool)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
id
|
||||||
|
}
|
||||||
|
|
||||||
|
#[sqlx::test(migrations = "./migrations")]
|
||||||
|
async fn find_by_id_returns_local_user(pool: PgPool) {
|
||||||
|
let id = seed_local_user(&pool, "alice").await;
|
||||||
|
let repo = PgApUserRepository::new(pool, "https://example.com".into());
|
||||||
|
|
||||||
|
let user = repo.find_by_id(id).await.unwrap().unwrap();
|
||||||
|
assert_eq!(user.username, "alice");
|
||||||
|
assert!(user.attachment.is_empty());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[sqlx::test(migrations = "./migrations")]
|
||||||
|
async fn find_by_username_returns_local_user(pool: PgPool) {
|
||||||
|
seed_local_user(&pool, "bob").await;
|
||||||
|
let repo = PgApUserRepository::new(pool, "https://example.com".into());
|
||||||
|
|
||||||
|
let user = repo.find_by_username("bob").await.unwrap().unwrap();
|
||||||
|
assert_eq!(user.username, "bob");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[sqlx::test(migrations = "./migrations")]
|
||||||
|
async fn find_by_id_returns_none_for_missing(pool: PgPool) {
|
||||||
|
let repo = PgApUserRepository::new(pool, "https://example.com".into());
|
||||||
|
let result = repo.find_by_id(uuid::Uuid::new_v4()).await.unwrap();
|
||||||
|
assert!(result.is_none());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[sqlx::test(migrations = "./migrations")]
|
||||||
|
async fn profile_fields_map_to_attachment(pool: PgPool) {
|
||||||
|
let id = seed_local_user(&pool, "carol").await;
|
||||||
|
let fields = serde_json::json!([
|
||||||
|
{"name": "Website", "value": "https://carol.dev"},
|
||||||
|
{"name": "Pronouns", "value": "she/her"}
|
||||||
|
]);
|
||||||
|
sqlx::query("UPDATE users SET profile_fields = $2 WHERE id = $1")
|
||||||
|
.bind(id)
|
||||||
|
.bind(&fields)
|
||||||
|
.execute(&pool)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let repo = PgApUserRepository::new(pool, "https://example.com".into());
|
||||||
|
let user = repo.find_by_id(id).await.unwrap().unwrap();
|
||||||
|
|
||||||
|
assert_eq!(user.attachment.len(), 2);
|
||||||
|
assert_eq!(user.attachment[0].name, "Website");
|
||||||
|
assert_eq!(user.attachment[0].value, "https://carol.dev");
|
||||||
|
assert_eq!(user.attachment[1].name, "Pronouns");
|
||||||
|
assert_eq!(user.attachment[1].value, "she/her");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[sqlx::test(migrations = "./migrations")]
|
||||||
|
async fn count_users_counts_local_only(pool: PgPool) {
|
||||||
|
seed_local_user(&pool, "local1").await;
|
||||||
|
seed_local_user(&pool, "local2").await;
|
||||||
|
sqlx::query(
|
||||||
|
"INSERT INTO users (id,username,email,password_hash,local,created_at,updated_at)
|
||||||
|
VALUES ($1,'remote','r@r.com','h',false,NOW(),NOW())",
|
||||||
|
)
|
||||||
|
.bind(uuid::Uuid::new_v4())
|
||||||
|
.execute(&pool)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let repo = PgApUserRepository::new(pool, "https://example.com".into());
|
||||||
|
assert_eq!(repo.count_users().await.unwrap(), 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -9,9 +9,9 @@ use domain::{
|
|||||||
user::User,
|
user::User,
|
||||||
},
|
},
|
||||||
ports::SearchPort,
|
ports::SearchPort,
|
||||||
value_objects::{Content, Email, PasswordHash, ThoughtId, UserId, Username},
|
value_objects::{Content, ThoughtId, UserId},
|
||||||
};
|
};
|
||||||
use postgres::user::{UserRow, USER_SELECT};
|
use postgres::user::USER_SELECT;
|
||||||
use sqlx::PgPool;
|
use sqlx::PgPool;
|
||||||
|
|
||||||
pub struct PgSearchRepository {
|
pub struct PgSearchRepository {
|
||||||
@@ -34,25 +34,15 @@ struct FeedRow {
|
|||||||
sensitive: bool,
|
sensitive: bool,
|
||||||
t_local: bool,
|
t_local: bool,
|
||||||
thought_created_at: DateTime<Utc>,
|
thought_created_at: DateTime<Utc>,
|
||||||
updated_at: Option<DateTime<Utc>>,
|
thought_updated_at: Option<DateTime<Utc>>,
|
||||||
author_id: uuid::Uuid,
|
note_extensions: Option<serde_json::Value>,
|
||||||
username: String,
|
#[sqlx(flatten)]
|
||||||
email: String,
|
author: postgres::user::UserRow,
|
||||||
password_hash: String,
|
|
||||||
display_name: Option<String>,
|
|
||||||
bio: Option<String>,
|
|
||||||
avatar_url: Option<String>,
|
|
||||||
header_url: Option<String>,
|
|
||||||
custom_css: Option<String>,
|
|
||||||
author_local: bool,
|
|
||||||
author_created_at: DateTime<Utc>,
|
|
||||||
author_updated_at: DateTime<Utc>,
|
|
||||||
like_count: i64,
|
like_count: i64,
|
||||||
boost_count: i64,
|
boost_count: i64,
|
||||||
reply_count: i64,
|
reply_count: i64,
|
||||||
liked_by_viewer: bool,
|
liked_by_viewer: bool,
|
||||||
boosted_by_viewer: bool,
|
boosted_by_viewer: bool,
|
||||||
note_extensions: Option<serde_json::Value>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn feed_select(viewer: Option<uuid::Uuid>) -> String {
|
fn feed_select(viewer: Option<uuid::Uuid>) -> String {
|
||||||
@@ -68,11 +58,11 @@ fn feed_select(viewer: Option<uuid::Uuid>) -> String {
|
|||||||
t.id AS thought_id, t.user_id AS t_user_id, t.content,\n\
|
t.id AS thought_id, t.user_id AS t_user_id, t.content,\n\
|
||||||
t.in_reply_to_id,\n\
|
t.in_reply_to_id,\n\
|
||||||
t.visibility, t.content_warning, t.sensitive, t.local AS t_local,\n\
|
t.visibility, t.content_warning, t.sensitive, t.local AS t_local,\n\
|
||||||
t.created_at AS thought_created_at, t.updated_at, t.note_extensions,\n\
|
t.created_at AS thought_created_at, t.updated_at AS thought_updated_at, t.note_extensions,\n\
|
||||||
u.id AS author_id, u.username, u.email, u.password_hash,\n\
|
u.id, u.username, u.email, u.password_hash,\n\
|
||||||
u.display_name, u.bio, u.avatar_url, u.header_url, u.custom_css,\n\
|
u.display_name, u.bio, u.avatar_url, u.header_url, u.custom_css, u.profile_fields,\n\
|
||||||
u.local AS author_local,\n\
|
u.local,\n\
|
||||||
u.created_at AS author_created_at, u.updated_at AS author_updated_at,\n\
|
u.created_at, u.updated_at,\n\
|
||||||
(SELECT COUNT(*) FROM likes l WHERE l.thought_id=t.id) AS like_count,\n\
|
(SELECT COUNT(*) FROM likes l WHERE l.thought_id=t.id) AS like_count,\n\
|
||||||
(SELECT COUNT(*) FROM boosts b WHERE b.thought_id=t.id) AS boost_count,\n\
|
(SELECT COUNT(*) FROM boosts b WHERE b.thought_id=t.id) AS boost_count,\n\
|
||||||
(SELECT COUNT(*) FROM thoughts r WHERE r.in_reply_to_id=t.id) AS reply_count,\n\
|
(SELECT COUNT(*) FROM thoughts r WHERE r.in_reply_to_id=t.id) AS reply_count,\n\
|
||||||
@@ -92,24 +82,10 @@ fn row_to_entry(r: FeedRow, viewer: Option<uuid::Uuid>) -> Result<FeedEntry, Dom
|
|||||||
sensitive: r.sensitive,
|
sensitive: r.sensitive,
|
||||||
local: r.t_local,
|
local: r.t_local,
|
||||||
created_at: r.thought_created_at,
|
created_at: r.thought_created_at,
|
||||||
updated_at: r.updated_at,
|
updated_at: r.thought_updated_at,
|
||||||
note_extensions: r.note_extensions,
|
note_extensions: r.note_extensions,
|
||||||
};
|
};
|
||||||
let author = User {
|
let author = User::from(r.author);
|
||||||
id: UserId::from_uuid(r.author_id),
|
|
||||||
username: Username::from_trusted(r.username),
|
|
||||||
email: Email::from_trusted(r.email),
|
|
||||||
password_hash: PasswordHash(r.password_hash),
|
|
||||||
display_name: r.display_name,
|
|
||||||
bio: r.bio,
|
|
||||||
avatar_url: r.avatar_url,
|
|
||||||
header_url: r.header_url,
|
|
||||||
custom_css: r.custom_css,
|
|
||||||
profile_fields: vec![],
|
|
||||||
local: r.author_local,
|
|
||||||
created_at: r.author_created_at,
|
|
||||||
updated_at: r.author_updated_at,
|
|
||||||
};
|
|
||||||
Ok(FeedEntry {
|
Ok(FeedEntry {
|
||||||
thought,
|
thought,
|
||||||
author,
|
author,
|
||||||
@@ -190,7 +166,7 @@ impl SearchPort for PgSearchRepository {
|
|||||||
ORDER BY similarity(username || ' ' || COALESCE(display_name,''), $1) DESC
|
ORDER BY similarity(username || ' ' || COALESCE(display_name,''), $1) DESC
|
||||||
LIMIT $2 OFFSET $3"
|
LIMIT $2 OFFSET $3"
|
||||||
);
|
);
|
||||||
let rows = sqlx::query_as::<_, UserRow>(&sql)
|
let rows = sqlx::query_as::<_, postgres::user::UserRow>(&sql)
|
||||||
.bind(query)
|
.bind(query)
|
||||||
.bind(page.limit())
|
.bind(page.limit())
|
||||||
.bind(page.offset())
|
.bind(page.offset())
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ use domain::{
|
|||||||
user::User,
|
user::User,
|
||||||
},
|
},
|
||||||
ports::{SearchPort, ThoughtRepository, UserWriter},
|
ports::{SearchPort, ThoughtRepository, UserWriter},
|
||||||
|
value_objects::{Content, Email, PasswordHash, ThoughtId, UserId, Username},
|
||||||
};
|
};
|
||||||
|
|
||||||
async fn seed_thought(pool: &sqlx::PgPool, username: &str, content: &str) -> (User, Thought) {
|
async fn seed_thought(pool: &sqlx::PgPool, username: &str, content: &str) -> (User, Thought) {
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ use domain::{
|
|||||||
user::User,
|
user::User,
|
||||||
},
|
},
|
||||||
ports::{FeedOptions, FeedRepository, FeedRequest, FeedScope, FeedSort},
|
ports::{FeedOptions, FeedRepository, FeedRequest, FeedScope, FeedSort},
|
||||||
value_objects::{Content, Email, PasswordHash, ThoughtId, UserId, Username},
|
value_objects::{Content, ThoughtId, UserId},
|
||||||
};
|
};
|
||||||
use sqlx::PgPool;
|
use sqlx::PgPool;
|
||||||
|
|
||||||
@@ -34,20 +34,10 @@ struct FeedRow {
|
|||||||
sensitive: bool,
|
sensitive: bool,
|
||||||
t_local: bool,
|
t_local: bool,
|
||||||
thought_created_at: DateTime<Utc>,
|
thought_created_at: DateTime<Utc>,
|
||||||
updated_at: Option<DateTime<Utc>>,
|
thought_updated_at: Option<DateTime<Utc>>,
|
||||||
note_extensions: Option<serde_json::Value>,
|
note_extensions: Option<serde_json::Value>,
|
||||||
author_id: uuid::Uuid,
|
#[sqlx(flatten)]
|
||||||
username: String,
|
author: crate::user::UserRow,
|
||||||
email: String,
|
|
||||||
password_hash: String,
|
|
||||||
display_name: Option<String>,
|
|
||||||
bio: Option<String>,
|
|
||||||
avatar_url: Option<String>,
|
|
||||||
header_url: Option<String>,
|
|
||||||
custom_css: Option<String>,
|
|
||||||
author_local: bool,
|
|
||||||
author_created_at: DateTime<Utc>,
|
|
||||||
author_updated_at: DateTime<Utc>,
|
|
||||||
like_count: i64,
|
like_count: i64,
|
||||||
boost_count: i64,
|
boost_count: i64,
|
||||||
reply_count: i64,
|
reply_count: i64,
|
||||||
@@ -66,24 +56,10 @@ fn row_to_entry(r: FeedRow, viewer: Option<uuid::Uuid>) -> Result<FeedEntry, Dom
|
|||||||
sensitive: r.sensitive,
|
sensitive: r.sensitive,
|
||||||
local: r.t_local,
|
local: r.t_local,
|
||||||
created_at: r.thought_created_at,
|
created_at: r.thought_created_at,
|
||||||
updated_at: r.updated_at,
|
updated_at: r.thought_updated_at,
|
||||||
note_extensions: r.note_extensions,
|
note_extensions: r.note_extensions,
|
||||||
};
|
};
|
||||||
let author = User {
|
let author = User::from(r.author);
|
||||||
id: UserId::from_uuid(r.author_id),
|
|
||||||
username: Username::from_trusted(r.username),
|
|
||||||
email: Email::from_trusted(r.email),
|
|
||||||
password_hash: PasswordHash(r.password_hash),
|
|
||||||
display_name: r.display_name,
|
|
||||||
bio: r.bio,
|
|
||||||
avatar_url: r.avatar_url,
|
|
||||||
header_url: r.header_url,
|
|
||||||
custom_css: r.custom_css,
|
|
||||||
profile_fields: vec![],
|
|
||||||
local: r.author_local,
|
|
||||||
created_at: r.author_created_at,
|
|
||||||
updated_at: r.author_updated_at,
|
|
||||||
};
|
|
||||||
Ok(FeedEntry {
|
Ok(FeedEntry {
|
||||||
thought,
|
thought,
|
||||||
author,
|
author,
|
||||||
@@ -135,9 +111,9 @@ impl<'a> FeedSqlBuilder<'a> {
|
|||||||
t.id AS thought_id, t.user_id AS t_user_id, t.content,
|
t.id AS thought_id, t.user_id AS t_user_id, t.content,
|
||||||
t.in_reply_to_id,
|
t.in_reply_to_id,
|
||||||
t.visibility, t.content_warning, t.sensitive, t.local AS t_local,
|
t.visibility, t.content_warning, t.sensitive, t.local AS t_local,
|
||||||
t.created_at AS thought_created_at, t.updated_at,
|
t.created_at AS thought_created_at, t.updated_at AS thought_updated_at,
|
||||||
t.note_extensions,
|
t.note_extensions,
|
||||||
u.id AS author_id,
|
u.id,
|
||||||
CASE WHEN NOT u.local AND ra.handle IS NOT NULL AND ra.handle != ''
|
CASE WHEN NOT u.local AND ra.handle IS NOT NULL AND ra.handle != ''
|
||||||
THEN '@' || ra.handle ||
|
THEN '@' || ra.handle ||
|
||||||
CASE WHEN ra.handle NOT LIKE '%@%'
|
CASE WHEN ra.handle NOT LIKE '%@%'
|
||||||
@@ -148,9 +124,9 @@ impl<'a> FeedSqlBuilder<'a> {
|
|||||||
COALESCE(ra.display_name, u.display_name) AS display_name,
|
COALESCE(ra.display_name, u.display_name) AS display_name,
|
||||||
u.bio,
|
u.bio,
|
||||||
COALESCE(ra.avatar_url, u.avatar_url) AS avatar_url,
|
COALESCE(ra.avatar_url, u.avatar_url) AS avatar_url,
|
||||||
u.header_url, u.custom_css,
|
u.header_url, u.custom_css, u.profile_fields,
|
||||||
u.local AS author_local,
|
u.local,
|
||||||
u.created_at AS author_created_at, u.updated_at AS author_updated_at,
|
u.created_at, u.updated_at,
|
||||||
COALESCE(l_agg.cnt, 0) AS like_count,
|
COALESCE(l_agg.cnt, 0) AS like_count,
|
||||||
COALESCE(b_agg.cnt, 0) AS boost_count,
|
COALESCE(b_agg.cnt, 0) AS boost_count,
|
||||||
COALESCE(r_agg.cnt, 0) AS reply_count,
|
COALESCE(r_agg.cnt, 0) AS reply_count,
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ use domain::{
|
|||||||
user::User,
|
user::User,
|
||||||
},
|
},
|
||||||
ports::{FeedOptions, FeedQuery, FeedRequest, ThoughtRepository, UserWriter},
|
ports::{FeedOptions, FeedQuery, FeedRequest, ThoughtRepository, UserWriter},
|
||||||
|
value_objects::{Content, Email, PasswordHash, ThoughtId, UserId, Username},
|
||||||
};
|
};
|
||||||
|
|
||||||
async fn seed(pool: &sqlx::PgPool, username: &str, content: &str) -> (User, Thought) {
|
async fn seed(pool: &sqlx::PgPool, username: &str, content: &str) -> (User, Thought) {
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ use domain::{
|
|||||||
errors::DomainError,
|
errors::DomainError,
|
||||||
models::user::User,
|
models::user::User,
|
||||||
testing::TestStore,
|
testing::TestStore,
|
||||||
value_objects::{Email, PasswordHash, ThoughtId, UserId, Username},
|
value_objects::{Email, ThoughtId, UserId, Username},
|
||||||
};
|
};
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::sync::{Arc, Mutex};
|
use std::sync::{Arc, Mutex};
|
||||||
@@ -63,21 +63,11 @@ impl ActivityPubRepository for TestApRepo {
|
|||||||
let handle = url::Url::parse(actor_ap_url)
|
let handle = url::Url::parse(actor_ap_url)
|
||||||
.map(|u| u.path().trim_start_matches('/').replace('/', "_"))
|
.map(|u| u.path().trim_start_matches('/').replace('/', "_"))
|
||||||
.unwrap_or_else(|_| format!("remote_{}", &uid.to_string()[..8]));
|
.unwrap_or_else(|_| format!("remote_{}", &uid.to_string()[..8]));
|
||||||
let user = User {
|
let user = User::new_remote(
|
||||||
id: uid.clone(),
|
uid.clone(),
|
||||||
username: Username::from_trusted(handle),
|
Username::from_trusted(handle),
|
||||||
email: Email::from_trusted(format!("{}@remote", uid)),
|
Email::from_trusted(format!("{}@remote", uid)),
|
||||||
password_hash: PasswordHash("".into()),
|
);
|
||||||
display_name: None,
|
|
||||||
bio: None,
|
|
||||||
avatar_url: None,
|
|
||||||
header_url: None,
|
|
||||||
custom_css: None,
|
|
||||||
profile_fields: vec![],
|
|
||||||
local: false,
|
|
||||||
created_at: chrono::Utc::now(),
|
|
||||||
updated_at: chrono::Utc::now(),
|
|
||||||
};
|
|
||||||
self.inner.users.lock().unwrap().push(user);
|
self.inner.users.lock().unwrap().push(user);
|
||||||
self.inner
|
self.inner
|
||||||
.actor_ap_ids
|
.actor_ap_ids
|
||||||
|
|||||||
@@ -52,4 +52,23 @@ impl User {
|
|||||||
updated_at: now,
|
updated_at: now,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn new_remote(id: UserId, username: Username, email: Email) -> Self {
|
||||||
|
let now = Utc::now();
|
||||||
|
Self {
|
||||||
|
id,
|
||||||
|
username,
|
||||||
|
email,
|
||||||
|
password_hash: PasswordHash(String::new()),
|
||||||
|
display_name: None,
|
||||||
|
bio: None,
|
||||||
|
avatar_url: None,
|
||||||
|
header_url: None,
|
||||||
|
custom_css: None,
|
||||||
|
profile_fields: vec![],
|
||||||
|
local: false,
|
||||||
|
created_at: now,
|
||||||
|
updated_at: now,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ use api_types::{
|
|||||||
};
|
};
|
||||||
use application::use_cases::auth::{login, register, LoginInput, RegisterInput};
|
use application::use_cases::auth::{login, register, LoginInput, RegisterInput};
|
||||||
use axum::{http::StatusCode, response::IntoResponse, Json};
|
use axum::{http::StatusCode, response::IntoResponse, Json};
|
||||||
|
use domain::models::feed::UserSummary;
|
||||||
use domain::ports::{AuthService, EventPublisher, PasswordHasher, UserRepository};
|
use domain::ports::{AuthService, EventPublisher, PasswordHasher, UserRepository};
|
||||||
|
|
||||||
deps_struct!(AuthDeps {
|
deps_struct!(AuthDeps {
|
||||||
@@ -37,6 +38,22 @@ pub fn to_user_response(u: &domain::models::user::User) -> UserResponse {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn to_summary_response(u: &UserSummary) -> UserResponse {
|
||||||
|
UserResponse {
|
||||||
|
id: u.id.as_uuid(),
|
||||||
|
username: u.username.clone(),
|
||||||
|
display_name: u.display_name.clone(),
|
||||||
|
bio: u.bio.clone(),
|
||||||
|
avatar_url: u.avatar_url.clone(),
|
||||||
|
header_url: None,
|
||||||
|
custom_css: None,
|
||||||
|
profile_fields: vec![],
|
||||||
|
local: true,
|
||||||
|
is_followed_by_viewer: false,
|
||||||
|
created_at: chrono::Utc::now(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[utoipa::path(
|
#[utoipa::path(
|
||||||
post, path = "/auth/register",
|
post, path = "/auth/register",
|
||||||
request_body = RegisterRequest,
|
request_body = RegisterRequest,
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
errors::ApiError,
|
errors::ApiError,
|
||||||
extractors::{AuthUser, Deps, FromAppState, OptionalAuthUser},
|
extractors::{AuthUser, Deps, FromAppState, OptionalAuthUser},
|
||||||
handlers::auth::to_user_response,
|
handlers::auth::{to_summary_response, to_user_response},
|
||||||
state::AppState,
|
state::AppState,
|
||||||
};
|
};
|
||||||
use api_types::{
|
use api_types::{
|
||||||
@@ -200,23 +200,7 @@ pub async fn get_users(
|
|||||||
}
|
}
|
||||||
|
|
||||||
let result = list_users(&*d.users, page_params).await?;
|
let result = list_users(&*d.users, page_params).await?;
|
||||||
let items: Vec<UserResponse> = result
|
let items: Vec<UserResponse> = result.items.iter().map(to_summary_response).collect();
|
||||||
.items
|
|
||||||
.iter()
|
|
||||||
.map(|u| UserResponse {
|
|
||||||
id: u.id.as_uuid(),
|
|
||||||
username: u.username.clone(),
|
|
||||||
display_name: u.display_name.clone(),
|
|
||||||
bio: u.bio.clone(),
|
|
||||||
avatar_url: u.avatar_url.clone(),
|
|
||||||
header_url: None,
|
|
||||||
custom_css: None,
|
|
||||||
profile_fields: vec![],
|
|
||||||
local: true,
|
|
||||||
is_followed_by_viewer: false,
|
|
||||||
created_at: chrono::Utc::now(),
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
Ok(Json(PagedResponse {
|
Ok(Json(PagedResponse {
|
||||||
items,
|
items,
|
||||||
total: result.total,
|
total: result.total,
|
||||||
|
|||||||
Reference in New Issue
Block a user