]> Untitled Git - lemmy.git/blob - crates/apub/src/objects/mod.rs
Forbid outgoing requests in activitypub tests (fixes #2289) (#2294)
[lemmy.git] / crates / apub / src / objects / mod.rs
1 use crate::protocol::Source;
2 use activitypub_federation::deser::values::MediaTypeMarkdownOrHtml;
3 use anyhow::anyhow;
4 use html2md::parse_html;
5 use lemmy_utils::{error::LemmyError, settings::structs::Settings};
6 use url::Url;
7
8 pub mod comment;
9 pub mod community;
10 pub mod instance;
11 pub mod person;
12 pub mod post;
13 pub mod private_message;
14
15 pub(crate) fn read_from_string_or_source(
16   content: &str,
17   media_type: &Option<MediaTypeMarkdownOrHtml>,
18   source: &Option<Source>,
19 ) -> String {
20   if let Some(s) = source {
21     // markdown sent by lemmy in source field
22     s.content.clone()
23   } else if media_type == &Some(MediaTypeMarkdownOrHtml::Markdown) {
24     // markdown sent by peertube in content field
25     content.to_string()
26   } else {
27     // otherwise, convert content html to markdown
28     parse_html(content)
29   }
30 }
31
32 pub(crate) fn read_from_string_or_source_opt(
33   content: &Option<String>,
34   media_type: &Option<MediaTypeMarkdownOrHtml>,
35   source: &Option<Source>,
36 ) -> Option<String> {
37   content
38     .as_ref()
39     .map(|content| read_from_string_or_source(content, media_type, source))
40 }
41
42 /// When for example a Post is made in a remote community, the community will send it back,
43 /// wrapped in Announce. If we simply receive this like any other federated object, overwrite the
44 /// existing, local Post. In particular, it will set the field local = false, so that the object
45 /// can't be fetched from the Activitypub HTTP endpoint anymore (which only serves local objects).
46 pub(crate) fn verify_is_remote_object(id: &Url) -> Result<(), LemmyError> {
47   let local_domain = Settings::get().get_hostname_without_port()?;
48   if id.domain() == Some(&local_domain) {
49     Err(anyhow!("cant accept local object from remote instance").into())
50   } else {
51     Ok(())
52   }
53 }
54
55 #[cfg(test)]
56 pub(crate) mod tests {
57   use actix::Actor;
58   use anyhow::anyhow;
59   use diesel::{
60     r2d2::{ConnectionManager, Pool},
61     PgConnection,
62   };
63   use lemmy_api_common::request::build_user_agent;
64   use lemmy_db_schema::{
65     source::secret::Secret,
66     utils::{establish_unpooled_connection, get_database_url_from_env},
67   };
68   use lemmy_utils::{
69     error::LemmyError,
70     rate_limit::{rate_limiter::RateLimiter, RateLimit},
71     settings::structs::Settings,
72   };
73   use lemmy_websocket::{chat_server::ChatServer, LemmyContext};
74   use parking_lot::Mutex;
75   use reqwest::{Client, Request, Response};
76   use reqwest_middleware::{ClientBuilder, Middleware, Next};
77   use std::sync::Arc;
78   use task_local_extensions::Extensions;
79
80   struct BlockedMiddleware;
81
82   /// A reqwest middleware which blocks all requests
83   #[async_trait::async_trait]
84   impl Middleware for BlockedMiddleware {
85     async fn handle(
86       &self,
87       _req: Request,
88       _extensions: &mut Extensions,
89       _next: Next<'_>,
90     ) -> reqwest_middleware::Result<Response> {
91       Err(anyhow!("Network requests not allowed").into())
92     }
93   }
94
95   // TODO: would be nice if we didnt have to use a full context for tests.
96   pub(crate) fn init_context() -> LemmyContext {
97     // call this to run migrations
98     establish_unpooled_connection();
99     let settings = Settings::init().unwrap();
100     let rate_limiter = RateLimit {
101       rate_limiter: Arc::new(Mutex::new(RateLimiter::default())),
102       rate_limit_config: settings.rate_limit.to_owned().unwrap_or_default(),
103     };
104     let client = Client::builder()
105       .user_agent(build_user_agent(&settings))
106       .build()
107       .unwrap();
108
109     let client = ClientBuilder::new(client).with(BlockedMiddleware).build();
110     let secret = Secret {
111       id: 0,
112       jwt_secret: "".to_string(),
113     };
114     let db_url = match get_database_url_from_env() {
115       Ok(url) => url,
116       Err(_) => settings.get_database_url(),
117     };
118     let manager = ConnectionManager::<PgConnection>::new(&db_url);
119     let pool = Pool::builder()
120       .max_size(settings.database.pool_size)
121       .build(manager)
122       .unwrap_or_else(|_| panic!("Error connecting to {}", db_url));
123     async fn x() -> Result<String, LemmyError> {
124       Ok("".to_string())
125     }
126     let chat_server = ChatServer::startup(
127       pool.clone(),
128       rate_limiter,
129       |_, _, _, _| Box::pin(x()),
130       |_, _, _, _| Box::pin(x()),
131       client.clone(),
132       settings.clone(),
133       secret.clone(),
134     )
135     .start();
136     LemmyContext::create(pool, chat_server, client, settings, secret)
137   }
138 }