1 use actix_web::{error::ErrorBadRequest, *};
3 use chrono::{DateTime, NaiveDateTime, Utc};
4 use diesel::PgConnection;
5 use lemmy_api::claims::Claims;
7 comment_view::{ReplyQueryBuilder, ReplyView},
9 post_view::{PostQueryBuilder, PostView},
12 user_mention_view::{UserMentionQueryBuilder, UserMentionView},
16 use lemmy_structs::blocking;
17 use lemmy_utils::{settings::Settings, utils::markdown_to_html, LemmyError};
18 use lemmy_websocket::LemmyContext;
20 extension::dublincore::DublinCoreExtensionBuilder,
26 use serde::Deserialize;
27 use std::{collections::HashMap, str::FromStr};
28 use strum::ParseError;
30 #[derive(Deserialize)]
42 pub fn config(cfg: &mut web::ServiceConfig) {
44 .route("/feeds/{type}/{name}.xml", web::get().to(get_feed))
45 .route("/feeds/all.xml", web::get().to(get_all_feed));
49 static ref RSS_NAMESPACE: HashMap<String, String> = {
50 let mut h = HashMap::new();
53 rss::extension::dublincore::NAMESPACE.to_string(),
59 async fn get_all_feed(
60 info: web::Query<Params>,
61 context: web::Data<LemmyContext>,
62 ) -> Result<HttpResponse, Error> {
63 let sort_type = get_sort_type(info).map_err(ErrorBadRequest)?;
65 let rss = blocking(context.pool(), move |conn| {
66 get_feed_all_data(conn, &sort_type)
69 .map_err(ErrorBadRequest)?;
73 .content_type("application/rss+xml")
78 fn get_feed_all_data(conn: &PgConnection, sort_type: &SortType) -> Result<String, LemmyError> {
79 let site_view = SiteView::read(&conn)?;
81 let posts = PostQueryBuilder::create(&conn)
82 .listing_type(ListingType::All)
86 let items = create_post_items(posts)?;
88 let mut channel_builder = ChannelBuilder::default();
90 .namespaces(RSS_NAMESPACE.to_owned())
91 .title(&format!("{} - All", site_view.name))
92 .link(Settings::get().get_protocol_and_hostname())
95 if let Some(site_desc) = site_view.description {
96 channel_builder.description(&site_desc);
99 Ok(channel_builder.build().map_err(|e| anyhow!(e))?.to_string())
103 web::Path((req_type, param)): web::Path<(String, String)>,
104 info: web::Query<Params>,
105 context: web::Data<LemmyContext>,
106 ) -> Result<HttpResponse, Error> {
107 let sort_type = get_sort_type(info).map_err(ErrorBadRequest)?;
109 let request_type = match req_type.as_str() {
110 "u" => RequestType::User,
111 "c" => RequestType::Community,
112 "front" => RequestType::Front,
113 "inbox" => RequestType::Inbox,
114 _ => return Err(ErrorBadRequest(LemmyError::from(anyhow!("wrong_type")))),
117 let builder = blocking(context.pool(), move |conn| match request_type {
118 RequestType::User => get_feed_user(conn, &sort_type, param),
119 RequestType::Community => get_feed_community(conn, &sort_type, param),
120 RequestType::Front => get_feed_front(conn, &sort_type, param),
121 RequestType::Inbox => get_feed_inbox(conn, param),
124 .map_err(ErrorBadRequest)?;
126 let rss = builder.build().map_err(ErrorBadRequest)?.to_string();
130 .content_type("application/rss+xml")
135 fn get_sort_type(info: web::Query<Params>) -> Result<SortType, ParseError> {
136 let sort_query = info
139 .unwrap_or_else(|| SortType::Hot.to_string());
140 SortType::from_str(&sort_query)
145 sort_type: &SortType,
147 ) -> Result<ChannelBuilder, LemmyError> {
148 let site_view = SiteView::read(&conn)?;
149 let user = User_::find_by_username(&conn, &user_name)?;
150 let user_url = user.get_profile_url(&Settings::get().hostname);
152 let posts = PostQueryBuilder::create(&conn)
153 .listing_type(ListingType::All)
155 .for_creator_id(user.id)
158 let items = create_post_items(posts)?;
160 let mut channel_builder = ChannelBuilder::default();
162 .namespaces(RSS_NAMESPACE.to_owned())
163 .title(&format!("{} - {}", site_view.name, user.name))
170 fn get_feed_community(
172 sort_type: &SortType,
173 community_name: String,
174 ) -> Result<ChannelBuilder, LemmyError> {
175 let site_view = SiteView::read(&conn)?;
176 let community = Community::read_from_name(&conn, &community_name)?;
178 let posts = PostQueryBuilder::create(&conn)
179 .listing_type(ListingType::All)
181 .for_community_id(community.id)
184 let items = create_post_items(posts)?;
186 let mut channel_builder = ChannelBuilder::default();
188 .namespaces(RSS_NAMESPACE.to_owned())
189 .title(&format!("{} - {}", site_view.name, community.name))
190 .link(community.actor_id)
193 if let Some(community_desc) = community.description {
194 channel_builder.description(&community_desc);
202 sort_type: &SortType,
204 ) -> Result<ChannelBuilder, LemmyError> {
205 let site_view = SiteView::read(&conn)?;
206 let user_id = Claims::decode(&jwt)?.claims.id;
208 let posts = PostQueryBuilder::create(&conn)
209 .listing_type(ListingType::Subscribed)
214 let items = create_post_items(posts)?;
216 let mut channel_builder = ChannelBuilder::default();
218 .namespaces(RSS_NAMESPACE.to_owned())
219 .title(&format!("{} - Subscribed", site_view.name))
220 .link(Settings::get().get_protocol_and_hostname())
223 if let Some(site_desc) = site_view.description {
224 channel_builder.description(&site_desc);
230 fn get_feed_inbox(conn: &PgConnection, jwt: String) -> Result<ChannelBuilder, LemmyError> {
231 let site_view = SiteView::read(&conn)?;
232 let user_id = Claims::decode(&jwt)?.claims.id;
234 let sort = SortType::New;
236 let replies = ReplyQueryBuilder::create(&conn, user_id)
240 let mentions = UserMentionQueryBuilder::create(&conn, user_id)
244 let items = create_reply_and_mention_items(replies, mentions)?;
246 let mut channel_builder = ChannelBuilder::default();
248 .namespaces(RSS_NAMESPACE.to_owned())
249 .title(&format!("{} - Inbox", site_view.name))
252 Settings::get().get_protocol_and_hostname()
256 if let Some(site_desc) = site_view.description {
257 channel_builder.description(&site_desc);
263 fn create_reply_and_mention_items(
264 replies: Vec<ReplyView>,
265 mentions: Vec<UserMentionView>,
266 ) -> Result<Vec<Item>, LemmyError> {
267 let mut reply_items: Vec<Item> = replies
270 let reply_url = format!(
271 "{}/post/{}/comment/{}",
272 Settings::get().get_protocol_and_hostname(),
276 build_item(&r.creator_name, &r.published, &reply_url, &r.content)
278 .collect::<Result<Vec<Item>, LemmyError>>()?;
280 let mut mention_items: Vec<Item> = mentions
283 let mention_url = format!(
284 "{}/post/{}/comment/{}",
285 Settings::get().get_protocol_and_hostname(),
289 build_item(&m.creator_name, &m.published, &mention_url, &m.content)
291 .collect::<Result<Vec<Item>, LemmyError>>()?;
293 reply_items.append(&mut mention_items);
299 published: &NaiveDateTime,
302 ) -> Result<Item, LemmyError> {
303 let mut i = ItemBuilder::default();
304 i.title(format!("Reply from {}", creator_name));
305 let author_url = format!(
307 Settings::get().get_protocol_and_hostname(),
311 "/u/{} <a href=\"{}\">(link)</a>",
312 creator_name, author_url
314 let dt = DateTime::<Utc>::from_utc(*published, Utc);
315 i.pub_date(dt.to_rfc2822());
316 i.comments(url.to_owned());
317 let guid = GuidBuilder::default()
321 .map_err(|e| anyhow!(e))?;
323 i.link(url.to_owned());
325 let html = markdown_to_html(&content.to_string());
327 Ok(i.build().map_err(|e| anyhow!(e))?)
330 fn create_post_items(posts: Vec<PostView>) -> Result<Vec<Item>, LemmyError> {
331 let mut items: Vec<Item> = Vec::new();
334 let mut i = ItemBuilder::default();
335 let mut dc_extension = DublinCoreExtensionBuilder::default();
339 dc_extension.creators(vec![p.creator_actor_id.to_owned()]);
341 let dt = DateTime::<Utc>::from_utc(p.published, Utc);
342 i.pub_date(dt.to_rfc2822());
344 let post_url = format!(
346 Settings::get().get_protocol_and_hostname(),
349 i.comments(post_url.to_owned());
350 let guid = GuidBuilder::default()
354 .map_err(|e| anyhow!(e))?;
357 let community_url = format!(
359 Settings::get().get_protocol_and_hostname(),
363 // TODO: for category we should just put the name of the category, but then we would have
364 // to read each community from the db
366 if let Some(url) = p.url {
371 let mut description = format!("submitted by <a href=\"{}\">{}</a> to <a href=\"{}\">{}</a><br>{} points | <a href=\"{}\">{} comments</a>",
378 p.number_of_comments);
380 if let Some(body) = p.body {
381 let html = markdown_to_html(&body);
382 description.push_str(&html);
385 i.description(description);
387 i.dublin_core_ext(dc_extension.build().map_err(|e| anyhow!(e))?);
388 items.push(i.build().map_err(|e| anyhow!(e))?);