mod application; mod attachment; mod card; mod poll; use super::account::Account; use super::emoji::Emoji; use super::id::Id; use super::mention::Mention; use super::tag::Tag; use super::visibility::Visibility; use application::Application; use attachment::Attachment; use card::Card; use poll::Poll; use crate::request::Blocks; use hashbrown::HashSet; use serde::{Deserialize, Serialize}; use std::boxed::Box; use std::string::String; #[serde(deny_unknown_fields)] #[derive(Serialize, Deserialize, Debug, Clone, PartialEq)] pub struct Status { id: Id, uri: String, created_at: String, account: Account, content: String, visibility: Visibility, sensitive: bool, spoiler_text: String, media_attachments: Vec, application: Option, // Should be non-optional? mentions: Vec, tags: Vec, emojis: Vec, reblogs_count: i64, favourites_count: i64, replies_count: i64, url: Option, in_reply_to_id: Option, in_reply_to_account_id: Option, reblog: Option>, poll: Option, card: Option, language: Option, text: Option, // ↓↓↓ Only for authorized users favourited: Option, reblogged: Option, muted: Option, bookmarked: Option, pinned: Option, } impl Status { /// Returns `true` if the status is filtered out based on its language pub(crate) fn language_not(&self, allowed_langs: &HashSet) -> bool { const ALLOW: bool = false; const REJECT: bool = true; let reject_and_maybe_log = |toot_language| { log::info!("Filtering out toot from `{}`", &self.account.acct); log::info!("Toot language: `{}`", toot_language); log::info!("Recipient's allowed languages: `{:?}`", allowed_langs); REJECT }; if allowed_langs.is_empty() { return ALLOW; // listing no allowed_langs results in allowing all languages } match self.language.as_ref() { Some(toot_language) if allowed_langs.contains(toot_language) => ALLOW, None => ALLOW, // If toot language is unknown, toot is always allowed Some(empty) if empty == &String::new() => ALLOW, Some(toot_language) => reject_and_maybe_log(toot_language), } } /// Returns `true` if the Status originated from a blocked domain, is from an account /// that has blocked the current user, or if the User's list of blocked/muted users /// includes a user involved in the Status. /// /// A user is involved in the Status/toot if they: /// * Are mentioned in this toot /// * Wrote this toot /// * Wrote a toot that this toot is replying to (if any) /// * Wrote the toot that this toot is boosting (if any) pub(crate) fn involves_any(&self, blocks: &Blocks) -> bool { const ALLOW: bool = false; const REJECT: bool = true; let Blocks { blocked_users, blocking_users, blocked_domains, } = blocks; let user_id = &Id(self.account.id.0); if blocking_users.contains(user_id) || self.involves(blocked_users) { REJECT } else { let full_username = &self.account.acct; match full_username.split('@').nth(1) { Some(originating_domain) if blocked_domains.contains(originating_domain) => REJECT, Some(_) | None => ALLOW, // None means the local instance, which can't be blocked } } } fn involves(&self, blocked_users: &HashSet) -> bool { // involved_users = mentioned_users + author + replied-to user + boosted user let mut involved_users: HashSet = self .mentions .iter() .map(|mention| Id(mention.id.0)) .collect(); // author involved_users.insert(Id(self.account.id.0)); // replied-to user if let Some(user_id) = self.in_reply_to_account_id { involved_users.insert(Id(user_id.0)); } // boosted user if let Some(boosted_status) = self.reblog.clone() { involved_users.insert(Id(boosted_status.account.id.0)); } !involved_users.is_disjoint(blocked_users) } }