Adding a viewtovec trait.

This commit is contained in:
Dessalines 2020-12-10 20:39:42 -05:00
parent eef93440d0
commit afc79ce0e3
6 changed files with 116 additions and 94 deletions

View file

@ -2,6 +2,7 @@ use crate::{
community::{Community, CommunitySafe},
schema::{community, community_follower, user_},
user::{UserSafe, User_},
views::ViewToVec,
ToSafe,
};
use diesel::{result::Error, *};
@ -13,6 +14,8 @@ pub struct CommunityFollowerView {
pub follower: UserSafe,
}
type CommunityFollowerViewTuple = (CommunitySafe, UserSafe);
impl CommunityFollowerView {
pub fn for_community(conn: &PgConnection, for_community_id: i32) -> Result<Vec<Self>, Error> {
let res = community_follower::table
@ -21,9 +24,9 @@ impl CommunityFollowerView {
.select((Community::safe_columns_tuple(), User_::safe_columns_tuple()))
.filter(community_follower::community_id.eq(for_community_id))
.order_by(community_follower::published)
.load::<(CommunitySafe, UserSafe)>(conn)?;
.load::<CommunityFollowerViewTuple>(conn)?;
Ok(to_vec(res))
Ok(Self::to_vec(res))
}
pub fn for_user(conn: &PgConnection, for_user_id: i32) -> Result<Vec<Self>, Error> {
@ -33,18 +36,21 @@ impl CommunityFollowerView {
.select((Community::safe_columns_tuple(), User_::safe_columns_tuple()))
.filter(community_follower::user_id.eq(for_user_id))
.order_by(community_follower::published)
.load::<(CommunitySafe, UserSafe)>(conn)?;
.load::<CommunityFollowerViewTuple>(conn)?;
Ok(to_vec(res))
Ok(Self::to_vec(res))
}
}
fn to_vec(users: Vec<(CommunitySafe, UserSafe)>) -> Vec<CommunityFollowerView> {
impl ViewToVec for CommunityFollowerView {
type DbTuple = CommunityFollowerViewTuple;
fn to_vec(users: Vec<Self::DbTuple>) -> Vec<Self> {
users
.iter()
.map(|a| CommunityFollowerView {
.map(|a| Self {
community: a.0.to_owned(),
follower: a.1.to_owned(),
})
.collect::<Vec<CommunityFollowerView>>()
.collect::<Vec<Self>>()
}
}

View file

@ -2,6 +2,7 @@ use crate::{
community::{Community, CommunitySafe},
schema::{community, community_moderator, user_},
user::{UserSafe, User_},
views::ViewToVec,
ToSafe,
};
use diesel::{result::Error, *};
@ -13,6 +14,8 @@ pub struct CommunityModeratorView {
pub moderator: UserSafe,
}
type CommunityModeratorViewTuple = (CommunitySafe, UserSafe);
impl CommunityModeratorView {
pub fn for_community(conn: &PgConnection, for_community_id: i32) -> Result<Vec<Self>, Error> {
let res = community_moderator::table
@ -21,9 +24,9 @@ impl CommunityModeratorView {
.select((Community::safe_columns_tuple(), User_::safe_columns_tuple()))
.filter(community_moderator::community_id.eq(for_community_id))
.order_by(community_moderator::published)
.load::<(CommunitySafe, UserSafe)>(conn)?;
.load::<CommunityModeratorViewTuple>(conn)?;
Ok(to_vec(res))
Ok(Self::to_vec(res))
}
pub fn for_user(conn: &PgConnection, for_user_id: i32) -> Result<Vec<Self>, Error> {
@ -33,18 +36,21 @@ impl CommunityModeratorView {
.select((Community::safe_columns_tuple(), User_::safe_columns_tuple()))
.filter(community_moderator::user_id.eq(for_user_id))
.order_by(community_moderator::published)
.load::<(CommunitySafe, UserSafe)>(conn)?;
.load::<CommunityModeratorViewTuple>(conn)?;
Ok(to_vec(res))
Ok(Self::to_vec(res))
}
}
fn to_vec(users: Vec<(CommunitySafe, UserSafe)>) -> Vec<CommunityModeratorView> {
users
impl ViewToVec for CommunityModeratorView {
type DbTuple = CommunityModeratorViewTuple;
fn to_vec(community_moderators: Vec<Self::DbTuple>) -> Vec<Self> {
community_moderators
.iter()
.map(|a| CommunityModeratorView {
.map(|a| Self {
community: a.0.to_owned(),
moderator: a.1.to_owned(),
})
.collect::<Vec<CommunityModeratorView>>()
.collect::<Vec<Self>>()
}
}

View file

@ -7,6 +7,7 @@ use crate::{
limit_and_offset,
schema::{category, community, community_aggregates, community_follower, user_},
user::{UserSafe, User_},
views::ViewToVec,
MaybeOptional,
SortType,
ToSafe,
@ -23,6 +24,14 @@ pub struct CommunityView {
pub counts: CommunityAggregates,
}
type CommunityViewTuple = (
CommunitySafe,
UserSafe,
Category,
CommunityAggregates,
Option<CommunityFollower>,
);
impl CommunityView {
pub fn read(
conn: &PgConnection,
@ -51,13 +60,7 @@ impl CommunityView {
community_aggregates::all_columns,
community_follower::all_columns.nullable(),
))
.first::<(
CommunitySafe,
UserSafe,
Category,
CommunityAggregates,
Option<CommunityFollower>,
)>(conn)?;
.first::<CommunityViewTuple>(conn)?;
Ok(CommunityView {
community,
@ -270,35 +273,24 @@ impl<'a> CommunityQueryBuilder<'a> {
.offset(offset)
.filter(community::removed.eq(false))
.filter(community::deleted.eq(false))
.load::<(
CommunitySafe,
UserSafe,
Category,
CommunityAggregates,
Option<CommunityFollower>,
)>(self.conn)?;
.load::<CommunityViewTuple>(self.conn)?;
Ok(to_vec(res))
Ok(CommunityView::to_vec(res))
}
}
fn to_vec(
users: Vec<(
CommunitySafe,
UserSafe,
Category,
CommunityAggregates,
Option<CommunityFollower>,
)>,
) -> Vec<CommunityView> {
users
impl ViewToVec for CommunityView {
type DbTuple = CommunityViewTuple;
fn to_vec(communities: Vec<Self::DbTuple>) -> Vec<Self> {
communities
.iter()
.map(|a| CommunityView {
.map(|a| Self {
community: a.0.to_owned(),
creator: a.1.to_owned(),
category: a.2.to_owned(),
counts: a.3.to_owned(),
subscribed: a.4.is_some(),
})
.collect::<Vec<CommunityView>>()
.collect::<Vec<Self>>()
}
}

View file

@ -5,3 +5,10 @@ pub mod community_view;
pub mod post_view;
pub mod site_view;
pub mod user_view;
pub(crate) trait ViewToVec {
type DbTuple;
fn to_vec(tuple: Vec<Self::DbTuple>) -> Vec<Self>
where
Self: Sized;
}

View file

@ -17,6 +17,7 @@ use crate::{
user_,
},
user::{UserSafe, User_},
views::ViewToVec,
ListingType,
MaybeOptional,
SortType,
@ -38,7 +39,7 @@ pub struct PostView {
pub my_vote: Option<i16>, // Left join to PostLike
}
type OutputTuple = (
type PostViewTuple = (
Post,
UserSafe,
CommunitySafe,
@ -107,7 +108,7 @@ impl PostView {
post_read::all_columns.nullable(),
post_like::score.nullable(),
))
.first::<OutputTuple>(conn)?;
.first::<PostViewTuple>(conn)?;
Ok(PostView {
post,
@ -551,17 +552,18 @@ impl<'a> PostQueryBuilder<'a> {
.filter(post::deleted.eq(false))
.filter(community::removed.eq(false))
.filter(community::deleted.eq(false))
.load::<OutputTuple>(self.conn)?;
.load::<PostViewTuple>(self.conn)?;
Ok(to_vec(res))
Ok(PostView::to_vec(res))
}
}
// TODO turn this into a trait with an associated type
fn to_vec(posts: Vec<OutputTuple>) -> Vec<PostView> {
impl ViewToVec for PostView {
type DbTuple = PostViewTuple;
fn to_vec(posts: Vec<Self::DbTuple>) -> Vec<Self> {
posts
.iter()
.map(|a| PostView {
.map(|a| Self {
post: a.0.to_owned(),
creator: a.1.to_owned(),
community: a.2.to_owned(),
@ -572,5 +574,6 @@ fn to_vec(posts: Vec<OutputTuple>) -> Vec<PostView> {
read: a.7.is_some(),
my_vote: a.8,
})
.collect::<Vec<PostView>>()
.collect::<Vec<Self>>()
}
}

View file

@ -4,6 +4,7 @@ use crate::{
limit_and_offset,
schema::{user_, user_aggregates},
user::{UserSafe, User_},
views::ViewToVec,
MaybeOptional,
SortType,
ToSafe,
@ -17,18 +18,22 @@ pub struct UserViewSafe {
pub counts: UserAggregates,
}
type UserViewSafeTuple = (UserSafe, UserAggregates);
#[derive(Debug, Serialize, Clone)]
pub struct UserViewDangerous {
pub user: User_,
pub counts: UserAggregates,
}
type UserViewDangerousTuple = (User_, UserAggregates);
impl UserViewDangerous {
pub fn read(conn: &PgConnection, id: i32) -> Result<Self, Error> {
let (user, counts) = user_::table
.find(id)
.inner_join(user_aggregates::table)
.first::<(User_, UserAggregates)>(conn)?;
.first::<UserViewDangerousTuple>(conn)?;
Ok(Self { user, counts })
}
}
@ -39,7 +44,7 @@ impl UserViewSafe {
.find(id)
.inner_join(user_aggregates::table)
.select((User_::safe_columns_tuple(), user_aggregates::all_columns))
.first::<(UserSafe, UserAggregates)>(conn)?;
.first::<UserViewSafeTuple>(conn)?;
Ok(Self { user, counts })
}
@ -49,9 +54,9 @@ impl UserViewSafe {
.select((User_::safe_columns_tuple(), user_aggregates::all_columns))
.filter(user_::admin.eq(true))
.order_by(user_::published)
.load::<(UserSafe, UserAggregates)>(conn)?;
.load::<UserViewSafeTuple>(conn)?;
Ok(to_vec(admins))
Ok(Self::to_vec(admins))
}
pub fn banned(conn: &PgConnection) -> Result<Vec<Self>, Error> {
@ -59,9 +64,9 @@ impl UserViewSafe {
.inner_join(user_aggregates::table)
.select((User_::safe_columns_tuple(), user_aggregates::all_columns))
.filter(user_::banned.eq(true))
.load::<(UserSafe, UserAggregates)>(conn)?;
.load::<UserViewSafeTuple>(conn)?;
Ok(to_vec(banned))
Ok(Self::to_vec(banned))
}
}
@ -186,18 +191,21 @@ impl<'a> UserQueryBuilder<'a> {
let (limit, offset) = limit_and_offset(self.page, self.limit);
query = query.limit(limit).offset(offset);
let res = query.load::<(UserSafe, UserAggregates)>(self.conn)?;
let res = query.load::<UserViewSafeTuple>(self.conn)?;
Ok(to_vec(res))
Ok(UserViewSafe::to_vec(res))
}
}
fn to_vec(users: Vec<(UserSafe, UserAggregates)>) -> Vec<UserViewSafe> {
impl ViewToVec for UserViewSafe {
type DbTuple = UserViewSafeTuple;
fn to_vec(users: Vec<Self::DbTuple>) -> Vec<Self> {
users
.iter()
.map(|a| UserViewSafe {
.map(|a| Self {
user: a.0.to_owned(),
counts: a.1.to_owned(),
})
.collect::<Vec<UserViewSafe>>()
.collect::<Vec<Self>>()
}
}