Merge branch 'teromene-config_dif_addr' into config_dif_addr_merge
This commit is contained in:
commit
b75abcdbc7
7 changed files with 697 additions and 278 deletions
|
@ -238,21 +238,16 @@ impl Perform<GetPostsResponse> for Oper<GetPosts> {
|
||||||
let type_ = ListingType::from_str(&data.type_)?;
|
let type_ = ListingType::from_str(&data.type_)?;
|
||||||
let sort = SortType::from_str(&data.sort)?;
|
let sort = SortType::from_str(&data.sort)?;
|
||||||
|
|
||||||
let posts = match PostView::list(
|
let posts = match PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.listing_type(type_)
|
||||||
type_,
|
.sort(&sort)
|
||||||
&sort,
|
.show_nsfw(show_nsfw)
|
||||||
data.community_id,
|
.for_community_id_optional(data.community_id)
|
||||||
None,
|
.my_user_id_optional(user_id)
|
||||||
None,
|
.page_optional(data.page)
|
||||||
None,
|
.limit_optional(data.limit)
|
||||||
user_id,
|
.list()
|
||||||
show_nsfw,
|
{
|
||||||
false,
|
|
||||||
false,
|
|
||||||
data.page,
|
|
||||||
data.limit,
|
|
||||||
) {
|
|
||||||
Ok(posts) => posts,
|
Ok(posts) => posts,
|
||||||
Err(_e) => return Err(APIError::err(&self.op, "couldnt_get_posts"))?,
|
Err(_e) => return Err(APIError::err(&self.op, "couldnt_get_posts"))?,
|
||||||
};
|
};
|
||||||
|
|
|
@ -319,21 +319,14 @@ impl Perform<SearchResponse> for Oper<Search> {
|
||||||
|
|
||||||
match type_ {
|
match type_ {
|
||||||
SearchType::Posts => {
|
SearchType::Posts => {
|
||||||
posts = PostView::list(
|
posts = PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.sort(&sort)
|
||||||
ListingType::All,
|
.show_nsfw(true)
|
||||||
&sort,
|
.for_community_id_optional(data.community_id)
|
||||||
data.community_id,
|
.search_term(data.q.to_owned())
|
||||||
None,
|
.page_optional(data.page)
|
||||||
Some(data.q.to_owned()),
|
.limit_optional(data.limit)
|
||||||
None,
|
.list()?;
|
||||||
None,
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
data.page,
|
|
||||||
data.limit,
|
|
||||||
)?;
|
|
||||||
}
|
}
|
||||||
SearchType::Comments => {
|
SearchType::Comments => {
|
||||||
comments = CommentView::list(
|
comments = CommentView::list(
|
||||||
|
@ -363,21 +356,15 @@ impl Perform<SearchResponse> for Oper<Search> {
|
||||||
users = UserView::list(&conn, &sort, Some(data.q.to_owned()), data.page, data.limit)?;
|
users = UserView::list(&conn, &sort, Some(data.q.to_owned()), data.page, data.limit)?;
|
||||||
}
|
}
|
||||||
SearchType::All => {
|
SearchType::All => {
|
||||||
posts = PostView::list(
|
posts = PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.sort(&sort)
|
||||||
ListingType::All,
|
.show_nsfw(true)
|
||||||
&sort,
|
.for_community_id_optional(data.community_id)
|
||||||
data.community_id,
|
.search_term(data.q.to_owned())
|
||||||
None,
|
.page_optional(data.page)
|
||||||
Some(data.q.to_owned()),
|
.limit_optional(data.limit)
|
||||||
None,
|
.list()?;
|
||||||
None,
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
data.page,
|
|
||||||
data.limit,
|
|
||||||
)?;
|
|
||||||
comments = CommentView::list(
|
comments = CommentView::list(
|
||||||
&conn,
|
&conn,
|
||||||
&sort,
|
&sort,
|
||||||
|
@ -401,21 +388,14 @@ impl Perform<SearchResponse> for Oper<Search> {
|
||||||
users = UserView::list(&conn, &sort, Some(data.q.to_owned()), data.page, data.limit)?;
|
users = UserView::list(&conn, &sort, Some(data.q.to_owned()), data.page, data.limit)?;
|
||||||
}
|
}
|
||||||
SearchType::Url => {
|
SearchType::Url => {
|
||||||
posts = PostView::list(
|
posts = PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.sort(&sort)
|
||||||
ListingType::All,
|
.show_nsfw(true)
|
||||||
&sort,
|
.for_community_id_optional(data.community_id)
|
||||||
data.community_id,
|
.url_search(data.q.to_owned())
|
||||||
None,
|
.page_optional(data.page)
|
||||||
None,
|
.limit_optional(data.limit)
|
||||||
Some(data.q.to_owned()),
|
.list()?;
|
||||||
None,
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
data.page,
|
|
||||||
data.limit,
|
|
||||||
)?;
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -366,40 +366,22 @@ impl Perform<GetUserDetailsResponse> for Oper<GetUserDetails> {
|
||||||
|
|
||||||
let user_view = UserView::read(&conn, user_details_id)?;
|
let user_view = UserView::read(&conn, user_details_id)?;
|
||||||
|
|
||||||
|
let mut posts_query = PostQueryBuilder::create(&conn)
|
||||||
|
.sort(&sort)
|
||||||
|
.show_nsfw(show_nsfw)
|
||||||
|
.saved_only(data.saved_only)
|
||||||
|
.for_community_id_optional(data.community_id)
|
||||||
|
.my_user_id_optional(user_id)
|
||||||
|
.page_optional(data.page)
|
||||||
|
.limit_optional(data.limit);
|
||||||
|
|
||||||
// If its saved only, you don't care what creator it was
|
// If its saved only, you don't care what creator it was
|
||||||
let posts = if data.saved_only {
|
if !data.saved_only {
|
||||||
PostView::list(
|
posts_query = posts_query.for_creator_id(user_details_id);
|
||||||
&conn,
|
}
|
||||||
ListingType::All,
|
|
||||||
&sort,
|
let posts = posts_query.list()?;
|
||||||
data.community_id,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
Some(user_details_id),
|
|
||||||
show_nsfw,
|
|
||||||
data.saved_only,
|
|
||||||
false,
|
|
||||||
data.page,
|
|
||||||
data.limit,
|
|
||||||
)?
|
|
||||||
} else {
|
|
||||||
PostView::list(
|
|
||||||
&conn,
|
|
||||||
ListingType::All,
|
|
||||||
&sort,
|
|
||||||
data.community_id,
|
|
||||||
Some(user_details_id),
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
user_id,
|
|
||||||
show_nsfw,
|
|
||||||
data.saved_only,
|
|
||||||
false,
|
|
||||||
data.page,
|
|
||||||
data.limit,
|
|
||||||
)?
|
|
||||||
};
|
|
||||||
let comments = if data.saved_only {
|
let comments = if data.saved_only {
|
||||||
CommentView::list(
|
CommentView::list(
|
||||||
&conn,
|
&conn,
|
||||||
|
@ -777,21 +759,12 @@ impl Perform<LoginResponse> for Oper<DeleteAccount> {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Posts
|
// Posts
|
||||||
let posts = PostView::list(
|
let posts = PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.sort(&SortType::New)
|
||||||
ListingType::All,
|
.show_nsfw(true)
|
||||||
&SortType::New,
|
.for_creator_id(user_id)
|
||||||
None,
|
.limit(std::i64::MAX)
|
||||||
Some(user_id),
|
.list()?;
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
None,
|
|
||||||
Some(std::i64::MAX),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
for post in &posts {
|
for post in &posts {
|
||||||
let post_form = PostForm {
|
let post_form = PostForm {
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
|
use super::post_view::post_view::BoxedQuery;
|
||||||
use super::*;
|
use super::*;
|
||||||
|
use diesel::pg::Pg;
|
||||||
|
|
||||||
// The faked schema since diesel doesn't do views
|
// The faked schema since diesel doesn't do views
|
||||||
table! {
|
table! {
|
||||||
|
@ -73,82 +75,161 @@ pub struct PostView {
|
||||||
pub saved: Option<bool>,
|
pub saved: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PostView {
|
pub struct PostQueryBuilder<'a> {
|
||||||
pub fn list(
|
conn: &'a PgConnection,
|
||||||
conn: &PgConnection,
|
query: BoxedQuery<'a, Pg>,
|
||||||
type_: ListingType,
|
listing_type: ListingType,
|
||||||
sort: &SortType,
|
sort: &'a SortType,
|
||||||
for_community_id: Option<i32>,
|
my_user_id: Option<i32>,
|
||||||
for_creator_id: Option<i32>,
|
for_creator_id: Option<i32>,
|
||||||
search_term: Option<String>,
|
show_nsfw: bool,
|
||||||
url_search: Option<String>,
|
saved_only: bool,
|
||||||
my_user_id: Option<i32>,
|
unread_only: bool,
|
||||||
show_nsfw: bool,
|
page: Option<i64>,
|
||||||
saved_only: bool,
|
limit: Option<i64>,
|
||||||
unread_only: bool,
|
}
|
||||||
page: Option<i64>,
|
|
||||||
limit: Option<i64>,
|
impl<'a> PostQueryBuilder<'a> {
|
||||||
) -> Result<Vec<Self>, Error> {
|
pub fn create(conn: &'a PgConnection) -> Self {
|
||||||
use super::post_view::post_view::dsl::*;
|
use super::post_view::post_view::dsl::*;
|
||||||
|
|
||||||
let (limit, offset) = limit_and_offset(page, limit);
|
let query = post_view.into_boxed();
|
||||||
|
|
||||||
let mut query = post_view.into_boxed();
|
PostQueryBuilder {
|
||||||
|
conn,
|
||||||
|
query,
|
||||||
|
my_user_id: None,
|
||||||
|
for_creator_id: None,
|
||||||
|
listing_type: ListingType::All,
|
||||||
|
sort: &SortType::Hot,
|
||||||
|
show_nsfw: true,
|
||||||
|
saved_only: false,
|
||||||
|
unread_only: false,
|
||||||
|
page: None,
|
||||||
|
limit: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// If its for a specific user, show the removed / deleted
|
pub fn listing_type(mut self, listing_type: ListingType) -> Self {
|
||||||
if let Some(for_creator_id) = for_creator_id {
|
self.listing_type = listing_type;
|
||||||
query = query.filter(creator_id.eq(for_creator_id));
|
self
|
||||||
} else {
|
}
|
||||||
query = query
|
|
||||||
.filter(removed.eq(false))
|
|
||||||
.filter(deleted.eq(false))
|
|
||||||
.filter(community_removed.eq(false))
|
|
||||||
.filter(community_deleted.eq(false));
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Some(search_term) = search_term {
|
pub fn sort(mut self, sort: &'a SortType) -> Self {
|
||||||
query = query.filter(name.ilike(fuzzy_search(&search_term)));
|
self.sort = sort;
|
||||||
};
|
self
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(url_search) = url_search {
|
pub fn for_community_id(mut self, for_community_id: i32) -> Self {
|
||||||
query = query.filter(url.eq(url_search));
|
use super::post_view::post_view::dsl::*;
|
||||||
};
|
self.query = self.query.filter(community_id.eq(for_community_id));
|
||||||
|
self.query = self.query.then_order_by(stickied.desc());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(for_community_id) = for_community_id {
|
pub fn for_community_id_optional(self, for_community_id: Option<i32>) -> Self {
|
||||||
query = query.filter(community_id.eq(for_community_id));
|
match for_community_id {
|
||||||
query = query.then_order_by(stickied.desc());
|
Some(for_community_id) => self.for_community_id(for_community_id),
|
||||||
};
|
None => self,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// TODO these are wrong, bc they'll only show saved for your logged in user, not theirs
|
pub fn for_creator_id(mut self, for_creator_id: i32) -> Self {
|
||||||
if saved_only {
|
self.for_creator_id = Some(for_creator_id);
|
||||||
query = query.filter(saved.eq(true));
|
self
|
||||||
};
|
}
|
||||||
|
|
||||||
if unread_only {
|
pub fn for_creator_id_optional(self, for_creator_id: Option<i32>) -> Self {
|
||||||
query = query.filter(read.eq(false));
|
match for_creator_id {
|
||||||
};
|
Some(for_creator_id) => self.for_creator_id(for_creator_id),
|
||||||
|
None => self,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
match type_ {
|
pub fn search_term(mut self, search_term: String) -> Self {
|
||||||
|
use super::post_view::post_view::dsl::*;
|
||||||
|
self.query = self.query.filter(name.ilike(fuzzy_search(&search_term)));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn search_term_optional(self, search_term: Option<String>) -> Self {
|
||||||
|
match search_term {
|
||||||
|
Some(search_term) => self.search_term(search_term),
|
||||||
|
None => self,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn url_search(mut self, url_search: String) -> Self {
|
||||||
|
use super::post_view::post_view::dsl::*;
|
||||||
|
self.query = self.query.filter(url.eq(url_search));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn url_search_optional(self, url_search: Option<String>) -> Self {
|
||||||
|
match url_search {
|
||||||
|
Some(url_search) => self.url_search(url_search),
|
||||||
|
None => self,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn my_user_id(mut self, my_user_id: i32) -> Self {
|
||||||
|
self.my_user_id = Some(my_user_id);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn my_user_id_optional(mut self, my_user_id: Option<i32>) -> Self {
|
||||||
|
self.my_user_id = my_user_id;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn show_nsfw(mut self, show_nsfw: bool) -> Self {
|
||||||
|
self.show_nsfw = show_nsfw;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn saved_only(mut self, saved_only: bool) -> Self {
|
||||||
|
self.saved_only = saved_only;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn unread_only(mut self, unread_only: bool) -> Self {
|
||||||
|
self.unread_only = unread_only;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn page(mut self, page: i64) -> Self {
|
||||||
|
self.page = Some(page);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn page_optional(mut self, page: Option<i64>) -> Self {
|
||||||
|
self.page = page;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn limit(mut self, limit: i64) -> Self {
|
||||||
|
self.limit = Some(limit);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn limit_optional(mut self, limit: Option<i64>) -> Self {
|
||||||
|
self.limit = limit;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn list(self) -> Result<Vec<PostView>, Error> {
|
||||||
|
use super::post_view::post_view::dsl::*;
|
||||||
|
|
||||||
|
let mut query = self.query;
|
||||||
|
|
||||||
|
match self.listing_type {
|
||||||
ListingType::Subscribed => {
|
ListingType::Subscribed => {
|
||||||
query = query.filter(subscribed.eq(true));
|
query = query.filter(subscribed.eq(true));
|
||||||
}
|
}
|
||||||
_ => {}
|
_ => {}
|
||||||
};
|
};
|
||||||
|
|
||||||
// The view lets you pass a null user_id, if you're not logged in
|
query = match self.sort {
|
||||||
if let Some(my_user_id) = my_user_id {
|
|
||||||
query = query.filter(user_id.eq(my_user_id));
|
|
||||||
} else {
|
|
||||||
query = query.filter(user_id.is_null());
|
|
||||||
}
|
|
||||||
|
|
||||||
if !show_nsfw {
|
|
||||||
query = query
|
|
||||||
.filter(nsfw.eq(false))
|
|
||||||
.filter(community_nsfw.eq(false));
|
|
||||||
};
|
|
||||||
|
|
||||||
query = match sort {
|
|
||||||
SortType::Hot => query
|
SortType::Hot => query
|
||||||
.then_order_by(hot_rank.desc())
|
.then_order_by(hot_rank.desc())
|
||||||
.then_order_by(published.desc()),
|
.then_order_by(published.desc()),
|
||||||
|
@ -168,11 +249,53 @@ impl PostView {
|
||||||
.then_order_by(score.desc()),
|
.then_order_by(score.desc()),
|
||||||
};
|
};
|
||||||
|
|
||||||
query = query.limit(limit).offset(offset);
|
// The view lets you pass a null user_id, if you're not logged in
|
||||||
|
query = if let Some(my_user_id) = self.my_user_id {
|
||||||
|
query.filter(user_id.eq(my_user_id))
|
||||||
|
} else {
|
||||||
|
query.filter(user_id.is_null())
|
||||||
|
};
|
||||||
|
|
||||||
query.load::<Self>(conn)
|
// If its for a specific user, show the removed / deleted
|
||||||
|
if let Some(for_creator_id) = self.for_creator_id {
|
||||||
|
query = query.filter(creator_id.eq(for_creator_id));
|
||||||
|
} else {
|
||||||
|
query = query
|
||||||
|
.filter(removed.eq(false))
|
||||||
|
.filter(deleted.eq(false))
|
||||||
|
.filter(community_removed.eq(false))
|
||||||
|
.filter(community_deleted.eq(false));
|
||||||
|
}
|
||||||
|
|
||||||
|
if !self.show_nsfw {
|
||||||
|
query = query
|
||||||
|
.filter(nsfw.eq(false))
|
||||||
|
.filter(community_nsfw.eq(false));
|
||||||
|
};
|
||||||
|
|
||||||
|
// TODO these are wrong, bc they'll only show saved for your logged in user, not theirs
|
||||||
|
if self.saved_only {
|
||||||
|
query = query.filter(saved.eq(true));
|
||||||
|
};
|
||||||
|
|
||||||
|
if self.unread_only {
|
||||||
|
query = query.filter(read.eq(false));
|
||||||
|
};
|
||||||
|
|
||||||
|
let (limit, offset) = limit_and_offset(self.page, self.limit);
|
||||||
|
query = query
|
||||||
|
.limit(limit)
|
||||||
|
.offset(offset)
|
||||||
|
.filter(removed.eq(false))
|
||||||
|
.filter(deleted.eq(false))
|
||||||
|
.filter(community_removed.eq(false))
|
||||||
|
.filter(community_deleted.eq(false));
|
||||||
|
|
||||||
|
query.load::<PostView>(self.conn)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PostView {
|
||||||
pub fn read(
|
pub fn read(
|
||||||
conn: &PgConnection,
|
conn: &PgConnection,
|
||||||
from_post_id: i32,
|
from_post_id: i32,
|
||||||
|
@ -345,38 +468,21 @@ mod tests {
|
||||||
nsfw: false,
|
nsfw: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
let read_post_listings_with_user = PostView::list(
|
let read_post_listings_with_user = PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.listing_type(ListingType::Community)
|
||||||
ListingType::Community,
|
.sort(&SortType::New)
|
||||||
&SortType::New,
|
.for_community_id(inserted_community.id)
|
||||||
Some(inserted_community.id),
|
.my_user_id(inserted_user.id)
|
||||||
None,
|
.list()
|
||||||
None,
|
.unwrap();
|
||||||
None,
|
|
||||||
Some(inserted_user.id),
|
let read_post_listings_no_user = PostQueryBuilder::create(&conn)
|
||||||
false,
|
.listing_type(ListingType::Community)
|
||||||
false,
|
.sort(&SortType::New)
|
||||||
false,
|
.for_community_id(inserted_community.id)
|
||||||
None,
|
.list()
|
||||||
None,
|
.unwrap();
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
let read_post_listings_no_user = PostView::list(
|
|
||||||
&conn,
|
|
||||||
ListingType::Community,
|
|
||||||
&SortType::New,
|
|
||||||
Some(inserted_community.id),
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
let read_post_listing_no_user = PostView::read(&conn, inserted_post.id, None).unwrap();
|
let read_post_listing_no_user = PostView::read(&conn, inserted_post.id, None).unwrap();
|
||||||
let read_post_listing_with_user =
|
let read_post_listing_with_user =
|
||||||
PostView::read(&conn, inserted_post.id, Some(inserted_user.id)).unwrap();
|
PostView::read(&conn, inserted_post.id, Some(inserted_user.id)).unwrap();
|
||||||
|
|
|
@ -4,7 +4,7 @@ use super::*;
|
||||||
use crate::db::comment_view::ReplyView;
|
use crate::db::comment_view::ReplyView;
|
||||||
use crate::db::community::Community;
|
use crate::db::community::Community;
|
||||||
use crate::db::community_view::SiteView;
|
use crate::db::community_view::SiteView;
|
||||||
use crate::db::post_view::PostView;
|
use crate::db::post_view::{PostView, PostQueryBuilder};
|
||||||
use crate::db::user::User_;
|
use crate::db::user::User_;
|
||||||
use crate::db::user_mention_view::UserMentionView;
|
use crate::db::user_mention_view::UserMentionView;
|
||||||
use crate::db::{establish_connection, ListingType, SortType};
|
use crate::db::{establish_connection, ListingType, SortType};
|
||||||
|
@ -86,21 +86,10 @@ fn get_feed_all_data(sort_type: &SortType) -> Result<String, Error> {
|
||||||
|
|
||||||
let site_view = SiteView::read(&conn)?;
|
let site_view = SiteView::read(&conn)?;
|
||||||
|
|
||||||
let posts = PostView::list(
|
let posts = PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.listing_type(ListingType::All)
|
||||||
ListingType::All,
|
.sort(sort_type)
|
||||||
sort_type,
|
.list()?;
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let items = create_post_items(posts);
|
let items = create_post_items(posts);
|
||||||
|
|
||||||
|
@ -124,21 +113,11 @@ fn get_feed_user(sort_type: &SortType, user_name: String) -> Result<String, Erro
|
||||||
let user = User_::find_by_email_or_username(&conn, &user_name)?;
|
let user = User_::find_by_email_or_username(&conn, &user_name)?;
|
||||||
let user_url = format!("https://{}/u/{}", Settings::get().hostname, user.name);
|
let user_url = format!("https://{}/u/{}", Settings::get().hostname, user.name);
|
||||||
|
|
||||||
let posts = PostView::list(
|
let posts = PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.listing_type(ListingType::All)
|
||||||
ListingType::All,
|
.sort(sort_type)
|
||||||
sort_type,
|
.for_creator_id(user.id)
|
||||||
None,
|
.list()?;
|
||||||
Some(user.id),
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let items = create_post_items(posts);
|
let items = create_post_items(posts);
|
||||||
|
|
||||||
|
@ -158,21 +137,11 @@ fn get_feed_community(sort_type: &SortType, community_name: String) -> Result<St
|
||||||
let community = Community::read_from_name(&conn, community_name)?;
|
let community = Community::read_from_name(&conn, community_name)?;
|
||||||
let community_url = format!("https://{}/c/{}", Settings::get().hostname, community.name);
|
let community_url = format!("https://{}/c/{}", Settings::get().hostname, community.name);
|
||||||
|
|
||||||
let posts = PostView::list(
|
let posts = PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.listing_type(ListingType::All)
|
||||||
ListingType::All,
|
.sort(sort_type)
|
||||||
sort_type,
|
.for_community_id(community.id)
|
||||||
Some(community.id),
|
.list()?;
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let items = create_post_items(posts);
|
let items = create_post_items(posts);
|
||||||
|
|
||||||
|
@ -195,21 +164,11 @@ fn get_feed_front(sort_type: &SortType, jwt: String) -> Result<String, Error> {
|
||||||
let site_view = SiteView::read(&conn)?;
|
let site_view = SiteView::read(&conn)?;
|
||||||
let user_id = db::user::Claims::decode(&jwt)?.claims.id;
|
let user_id = db::user::Claims::decode(&jwt)?.claims.id;
|
||||||
|
|
||||||
let posts = PostView::list(
|
let posts = PostQueryBuilder::create(&conn)
|
||||||
&conn,
|
.listing_type(ListingType::Subscribed)
|
||||||
ListingType::Subscribed,
|
.sort(sort_type)
|
||||||
sort_type,
|
.my_user_id(user_id)
|
||||||
None,
|
.list()?;
|
||||||
None,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
Some(user_id),
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
None,
|
|
||||||
None,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let items = create_post_items(posts);
|
let items = create_post_items(posts);
|
||||||
|
|
||||||
|
|
413
server/src/feeds.rs.orig
vendored
Normal file
413
server/src/feeds.rs.orig
vendored
Normal file
|
@ -0,0 +1,413 @@
|
||||||
|
extern crate rss;
|
||||||
|
|
||||||
|
use super::*;
|
||||||
|
use crate::db::comment_view::ReplyView;
|
||||||
|
use crate::db::community::Community;
|
||||||
|
use crate::db::community_view::SiteView;
|
||||||
|
use crate::db::post_view::PostQueryBuilder;
|
||||||
|
use crate::db::user::User_;
|
||||||
|
<<<<<<< HEAD
|
||||||
|
use crate::db::user_mention_view::UserMentionView;
|
||||||
|
use crate::db::{establish_connection, ListingType, SortType};
|
||||||
|
=======
|
||||||
|
use crate::db::{establish_connection, SortType};
|
||||||
|
>>>>>>> teromene-config_dif_addr
|
||||||
|
use crate::Settings;
|
||||||
|
use actix_web::body::Body;
|
||||||
|
use actix_web::{web, HttpResponse, Result};
|
||||||
|
use failure::Error;
|
||||||
|
use rss::{CategoryBuilder, ChannelBuilder, GuidBuilder, Item, ItemBuilder};
|
||||||
|
use serde::Deserialize;
|
||||||
|
use std::str::FromStr;
|
||||||
|
use strum::ParseError;
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
pub struct Params {
|
||||||
|
sort: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
enum RequestType {
|
||||||
|
Community,
|
||||||
|
User,
|
||||||
|
Front,
|
||||||
|
Inbox,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_all_feed(info: web::Query<Params>) -> HttpResponse<Body> {
|
||||||
|
let sort_type = match get_sort_type(info) {
|
||||||
|
Ok(sort_type) => sort_type,
|
||||||
|
Err(_) => return HttpResponse::BadRequest().finish(),
|
||||||
|
};
|
||||||
|
|
||||||
|
let feed_result = get_feed_all_data(&sort_type);
|
||||||
|
|
||||||
|
match feed_result {
|
||||||
|
Ok(rss) => HttpResponse::Ok()
|
||||||
|
.content_type("application/rss+xml")
|
||||||
|
.body(rss),
|
||||||
|
Err(_) => HttpResponse::NotFound().finish(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_feed(path: web::Path<(String, String)>, info: web::Query<Params>) -> HttpResponse<Body> {
|
||||||
|
let sort_type = match get_sort_type(info) {
|
||||||
|
Ok(sort_type) => sort_type,
|
||||||
|
Err(_) => return HttpResponse::BadRequest().finish(),
|
||||||
|
};
|
||||||
|
|
||||||
|
let request_type = match path.0.as_ref() {
|
||||||
|
"u" => RequestType::User,
|
||||||
|
"c" => RequestType::Community,
|
||||||
|
"front" => RequestType::Front,
|
||||||
|
"inbox" => RequestType::Inbox,
|
||||||
|
_ => return HttpResponse::NotFound().finish(),
|
||||||
|
};
|
||||||
|
|
||||||
|
let param = path.1.to_owned();
|
||||||
|
|
||||||
|
let feed_result = match request_type {
|
||||||
|
RequestType::User => get_feed_user(&sort_type, param),
|
||||||
|
RequestType::Community => get_feed_community(&sort_type, param),
|
||||||
|
RequestType::Front => get_feed_front(&sort_type, param),
|
||||||
|
RequestType::Inbox => get_feed_inbox(param),
|
||||||
|
};
|
||||||
|
|
||||||
|
match feed_result {
|
||||||
|
Ok(rss) => HttpResponse::Ok()
|
||||||
|
.content_type("application/rss+xml")
|
||||||
|
.body(rss),
|
||||||
|
Err(_) => HttpResponse::NotFound().finish(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_sort_type(info: web::Query<Params>) -> Result<SortType, ParseError> {
|
||||||
|
let sort_query = info.sort.to_owned().unwrap_or(SortType::Hot.to_string());
|
||||||
|
SortType::from_str(&sort_query)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_feed_all_data(sort_type: &SortType) -> Result<String, Error> {
|
||||||
|
let conn = establish_connection();
|
||||||
|
|
||||||
|
let site_view = SiteView::read(&conn)?;
|
||||||
|
|
||||||
|
let posts = PostView::list(
|
||||||
|
&conn,
|
||||||
|
ListingType::All,
|
||||||
|
sort_type,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
true,
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
let items = create_post_items(posts);
|
||||||
|
|
||||||
|
let mut channel_builder = ChannelBuilder::default();
|
||||||
|
channel_builder
|
||||||
|
.title(&format!("{} - All", site_view.name))
|
||||||
|
.link(format!("https://{}", Settings::get().hostname))
|
||||||
|
.items(items);
|
||||||
|
|
||||||
|
if let Some(site_desc) = site_view.description {
|
||||||
|
channel_builder.description(&site_desc);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(channel_builder.build().unwrap().to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_feed_user(sort_type: &SortType, user_name: String) -> Result<String, Error> {
|
||||||
|
let conn = establish_connection();
|
||||||
|
|
||||||
|
let site_view = SiteView::read(&conn)?;
|
||||||
|
let user = User_::find_by_email_or_username(&conn, &user_name)?;
|
||||||
|
let user_url = format!("https://{}/u/{}", Settings::get().hostname, user.name);
|
||||||
|
|
||||||
|
let posts = PostView::list(
|
||||||
|
&conn,
|
||||||
|
ListingType::All,
|
||||||
|
sort_type,
|
||||||
|
None,
|
||||||
|
Some(user.id),
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
true,
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
let items = create_post_items(posts);
|
||||||
|
|
||||||
|
let mut channel_builder = ChannelBuilder::default();
|
||||||
|
channel_builder
|
||||||
|
.title(&format!("{} - {}", site_view.name, user.name))
|
||||||
|
.link(user_url)
|
||||||
|
.items(items);
|
||||||
|
|
||||||
|
Ok(channel_builder.build().unwrap().to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_feed_community(sort_type: &SortType, community_name: String) -> Result<String, Error> {
|
||||||
|
let conn = establish_connection();
|
||||||
|
|
||||||
|
let site_view = SiteView::read(&conn)?;
|
||||||
|
let community = Community::read_from_name(&conn, community_name)?;
|
||||||
|
let community_url = format!("https://{}/c/{}", Settings::get().hostname, community.name);
|
||||||
|
|
||||||
|
<<<<<<< HEAD
|
||||||
|
let posts = PostView::list(
|
||||||
|
&conn,
|
||||||
|
ListingType::All,
|
||||||
|
sort_type,
|
||||||
|
Some(community.id),
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
true,
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
)?;
|
||||||
|
=======
|
||||||
|
let posts = PostQueryBuilder::create(&conn)
|
||||||
|
.sort(sort_type)
|
||||||
|
.show_nsfw(true)
|
||||||
|
.for_community_id_optional(community_id)
|
||||||
|
.for_creator_id_optional(creator_id)
|
||||||
|
.list()?;
|
||||||
|
>>>>>>> teromene-config_dif_addr
|
||||||
|
|
||||||
|
let items = create_post_items(posts);
|
||||||
|
|
||||||
|
let mut channel_builder = ChannelBuilder::default();
|
||||||
|
channel_builder
|
||||||
|
.title(&format!("{} - {}", site_view.name, community.name))
|
||||||
|
.link(community_url)
|
||||||
|
.items(items);
|
||||||
|
|
||||||
|
if let Some(community_desc) = community.description {
|
||||||
|
channel_builder.description(&community_desc);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(channel_builder.build().unwrap().to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_feed_front(sort_type: &SortType, jwt: String) -> Result<String, Error> {
|
||||||
|
let conn = establish_connection();
|
||||||
|
|
||||||
|
let site_view = SiteView::read(&conn)?;
|
||||||
|
let user_id = db::user::Claims::decode(&jwt)?.claims.id;
|
||||||
|
|
||||||
|
let posts = PostView::list(
|
||||||
|
&conn,
|
||||||
|
ListingType::Subscribed,
|
||||||
|
sort_type,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
Some(user_id),
|
||||||
|
true,
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
let items = create_post_items(posts);
|
||||||
|
|
||||||
|
let mut channel_builder = ChannelBuilder::default();
|
||||||
|
channel_builder
|
||||||
|
.title(&format!("{} - Subscribed", site_view.name))
|
||||||
|
.link(format!("https://{}", Settings::get().hostname))
|
||||||
|
.items(items);
|
||||||
|
|
||||||
|
if let Some(site_desc) = site_view.description {
|
||||||
|
channel_builder.description(&site_desc);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(channel_builder.build().unwrap().to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_feed_inbox(jwt: String) -> Result<String, Error> {
|
||||||
|
let conn = establish_connection();
|
||||||
|
|
||||||
|
let site_view = SiteView::read(&conn)?;
|
||||||
|
let user_id = db::user::Claims::decode(&jwt)?.claims.id;
|
||||||
|
|
||||||
|
let sort = SortType::New;
|
||||||
|
|
||||||
|
let replies = ReplyView::get_replies(&conn, user_id, &sort, false, None, None)?;
|
||||||
|
|
||||||
|
let mentions = UserMentionView::get_mentions(&conn, user_id, &sort, false, None, None)?;
|
||||||
|
|
||||||
|
let items = create_reply_and_mention_items(replies, mentions);
|
||||||
|
|
||||||
|
let mut channel_builder = ChannelBuilder::default();
|
||||||
|
channel_builder
|
||||||
|
.title(&format!("{} - Inbox", site_view.name))
|
||||||
|
.link(format!("https://{}/inbox", Settings::get().hostname))
|
||||||
|
.items(items);
|
||||||
|
|
||||||
|
if let Some(site_desc) = site_view.description {
|
||||||
|
channel_builder.description(&site_desc);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(channel_builder.build().unwrap().to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_reply_and_mention_items(
|
||||||
|
replies: Vec<ReplyView>,
|
||||||
|
mentions: Vec<UserMentionView>,
|
||||||
|
) -> Vec<Item> {
|
||||||
|
let mut items: Vec<Item> = Vec::new();
|
||||||
|
|
||||||
|
for r in replies {
|
||||||
|
let mut i = ItemBuilder::default();
|
||||||
|
|
||||||
|
i.title(format!("Reply from {}", r.creator_name));
|
||||||
|
|
||||||
|
let author_url = format!("https://{}/u/{}", Settings::get().hostname, r.creator_name);
|
||||||
|
i.author(format!(
|
||||||
|
"/u/{} <a href=\"{}\">(link)</a>",
|
||||||
|
r.creator_name, author_url
|
||||||
|
));
|
||||||
|
|
||||||
|
let dt = DateTime::<Utc>::from_utc(r.published, Utc);
|
||||||
|
i.pub_date(dt.to_rfc2822());
|
||||||
|
|
||||||
|
let reply_url = format!(
|
||||||
|
"https://{}/post/{}/comment/{}",
|
||||||
|
Settings::get().hostname,
|
||||||
|
r.post_id,
|
||||||
|
r.id
|
||||||
|
);
|
||||||
|
i.comments(reply_url.to_owned());
|
||||||
|
let guid = GuidBuilder::default()
|
||||||
|
.permalink(true)
|
||||||
|
.value(&reply_url)
|
||||||
|
.build();
|
||||||
|
i.guid(guid.unwrap());
|
||||||
|
|
||||||
|
i.link(reply_url);
|
||||||
|
|
||||||
|
// TODO find a markdown to html parser here, do images, etc
|
||||||
|
i.description(r.content);
|
||||||
|
|
||||||
|
items.push(i.build().unwrap());
|
||||||
|
}
|
||||||
|
|
||||||
|
for m in mentions {
|
||||||
|
let mut i = ItemBuilder::default();
|
||||||
|
|
||||||
|
i.title(format!("Mention from {}", m.creator_name));
|
||||||
|
|
||||||
|
let author_url = format!("https://{}/u/{}", Settings::get().hostname, m.creator_name);
|
||||||
|
i.author(format!(
|
||||||
|
"/u/{} <a href=\"{}\">(link)</a>",
|
||||||
|
m.creator_name, author_url
|
||||||
|
));
|
||||||
|
|
||||||
|
let dt = DateTime::<Utc>::from_utc(m.published, Utc);
|
||||||
|
i.pub_date(dt.to_rfc2822());
|
||||||
|
|
||||||
|
let mention_url = format!(
|
||||||
|
"https://{}/post/{}/comment/{}",
|
||||||
|
Settings::get().hostname,
|
||||||
|
m.post_id,
|
||||||
|
m.id
|
||||||
|
);
|
||||||
|
i.comments(mention_url.to_owned());
|
||||||
|
let guid = GuidBuilder::default()
|
||||||
|
.permalink(true)
|
||||||
|
.value(&mention_url)
|
||||||
|
.build();
|
||||||
|
i.guid(guid.unwrap());
|
||||||
|
|
||||||
|
i.link(mention_url);
|
||||||
|
|
||||||
|
// TODO find a markdown to html parser here, do images, etc
|
||||||
|
i.description(m.content);
|
||||||
|
|
||||||
|
items.push(i.build().unwrap());
|
||||||
|
}
|
||||||
|
|
||||||
|
items
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_post_items(posts: Vec<PostView>) -> Vec<Item> {
|
||||||
|
let mut items: Vec<Item> = Vec::new();
|
||||||
|
|
||||||
|
for p in posts {
|
||||||
|
let mut i = ItemBuilder::default();
|
||||||
|
|
||||||
|
i.title(p.name);
|
||||||
|
|
||||||
|
let author_url = format!("https://{}/u/{}", Settings::get().hostname, p.creator_name);
|
||||||
|
i.author(format!(
|
||||||
|
"/u/{} <a href=\"{}\">(link)</a>",
|
||||||
|
p.creator_name, author_url
|
||||||
|
));
|
||||||
|
|
||||||
|
let dt = DateTime::<Utc>::from_utc(p.published, Utc);
|
||||||
|
i.pub_date(dt.to_rfc2822());
|
||||||
|
|
||||||
|
let post_url = format!("https://{}/post/{}", Settings::get().hostname, p.id);
|
||||||
|
i.comments(post_url.to_owned());
|
||||||
|
let guid = GuidBuilder::default()
|
||||||
|
.permalink(true)
|
||||||
|
.value(&post_url)
|
||||||
|
.build();
|
||||||
|
i.guid(guid.unwrap());
|
||||||
|
|
||||||
|
let community_url = format!(
|
||||||
|
"https://{}/c/{}",
|
||||||
|
Settings::get().hostname,
|
||||||
|
p.community_name
|
||||||
|
);
|
||||||
|
|
||||||
|
let category = CategoryBuilder::default()
|
||||||
|
.name(format!(
|
||||||
|
"/c/{} <a href=\"{}\">(link)</a>",
|
||||||
|
p.community_name, community_url
|
||||||
|
))
|
||||||
|
.domain(Settings::get().hostname)
|
||||||
|
.build();
|
||||||
|
i.categories(vec![category.unwrap()]);
|
||||||
|
|
||||||
|
if let Some(url) = p.url {
|
||||||
|
i.link(url);
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO find a markdown to html parser here, do images, etc
|
||||||
|
let mut description = format!("
|
||||||
|
submitted by <a href=\"{}\">{}</a> to <a href=\"{}\">{}</a><br>{} points | <a href=\"{}\">{} comments</a>",
|
||||||
|
author_url,
|
||||||
|
p.creator_name,
|
||||||
|
community_url,
|
||||||
|
p.community_name,
|
||||||
|
p.score,
|
||||||
|
post_url,
|
||||||
|
p.number_of_comments);
|
||||||
|
|
||||||
|
if let Some(body) = p.body {
|
||||||
|
description.push_str(&format!("<br><br>{}", body));
|
||||||
|
}
|
||||||
|
|
||||||
|
i.description(description);
|
||||||
|
|
||||||
|
items.push(i.build().unwrap());
|
||||||
|
}
|
||||||
|
|
||||||
|
items
|
||||||
|
}
|
|
@ -134,21 +134,14 @@ impl ChatServer {
|
||||||
use crate::db::post_view::*;
|
use crate::db::post_view::*;
|
||||||
use crate::db::*;
|
use crate::db::*;
|
||||||
let conn = establish_connection();
|
let conn = establish_connection();
|
||||||
let posts = PostView::list(
|
|
||||||
&conn,
|
let posts = PostQueryBuilder::create(&conn)
|
||||||
ListingType::Community,
|
.listing_type(ListingType::Community)
|
||||||
&SortType::New,
|
.sort(&SortType::New)
|
||||||
Some(*community_id),
|
.for_community_id(*community_id)
|
||||||
None,
|
.limit(9999)
|
||||||
None,
|
.list()?;
|
||||||
None,
|
|
||||||
None,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
None,
|
|
||||||
Some(9999),
|
|
||||||
)?;
|
|
||||||
for post in posts {
|
for post in posts {
|
||||||
self.send_room_message(&post.id, message, skip_id);
|
self.send_room_message(&post.id, message, skip_id);
|
||||||
}
|
}
|
||||||
|
|
Reference in a new issue