use chrono::NaiveDateTime; use diesel::{self, ExpressionMethods, QueryDsl, RunQueryDsl}; use std::iter::Iterator; use ap_url; use plume_common::utils::md_to_html; use safe_string::SafeString; use schema::{instances, users}; use users::User; use Connection; #[derive(Clone, Identifiable, Queryable, Serialize)] pub struct Instance { pub id: i32, pub public_domain: String, pub name: String, pub local: bool, pub blocked: bool, pub creation_date: NaiveDateTime, pub open_registrations: bool, pub short_description: SafeString, pub long_description: SafeString, pub default_license: String, pub long_description_html: String, pub short_description_html: String, } #[derive(Clone, Insertable)] #[table_name = "instances"] pub struct NewInstance { pub public_domain: String, pub name: String, pub local: bool, pub open_registrations: bool, pub short_description: SafeString, pub long_description: SafeString, pub default_license: String, pub long_description_html: String, pub short_description_html: String, } impl Instance { pub fn get_local(conn: &Connection) -> Option { instances::table .filter(instances::local.eq(true)) .limit(1) .load::(conn) .expect("Instance::get_local: loading error") .into_iter() .nth(0) } pub fn get_remotes(conn: &Connection) -> Vec { instances::table .filter(instances::local.eq(false)) .load::(conn) .expect("Instance::get_remotes: loading error") } pub fn page(conn: &Connection, (min, max): (i32, i32)) -> Vec { instances::table .order(instances::public_domain.asc()) .offset(min.into()) .limit((max - min).into()) .load::(conn) .expect("Instance::page: loading error") } pub fn local_id(conn: &Connection) -> i32 { Instance::get_local(conn) .expect("Instance::local_id: local instance not found error") .id } insert!(instances, NewInstance); get!(instances); find_by!(instances, find_by_domain, public_domain as String); pub fn toggle_block(&self, conn: &Connection) { diesel::update(self) .set(instances::blocked.eq(!self.blocked)) .execute(conn) .expect("Instance::toggle_block: update error"); } /// id: AP object id pub fn is_blocked(conn: &Connection, id: String) -> bool { for block in instances::table .filter(instances::blocked.eq(true)) .get_results::(conn) .expect("Instance::is_blocked: loading error") { if id.starts_with(format!("https://{}/", block.public_domain).as_str()) { return true; } } false } pub fn has_admin(&self, conn: &Connection) -> bool { users::table .filter(users::instance_id.eq(self.id)) .filter(users::is_admin.eq(true)) .load::(conn) .expect("Instance::has_admin: loading error") .len() > 0 } pub fn main_admin(&self, conn: &Connection) -> User { users::table .filter(users::instance_id.eq(self.id)) .filter(users::is_admin.eq(true)) .limit(1) .get_result::(conn) .expect("Instance::main_admin: loading error") } pub fn compute_box( &self, prefix: &'static str, name: String, box_name: &'static str, ) -> String { ap_url(format!( "{instance}/{prefix}/{name}/{box_name}", instance = self.public_domain, prefix = prefix, name = name, box_name = box_name )) } pub fn update( &self, conn: &Connection, name: String, open_registrations: bool, short_description: SafeString, long_description: SafeString, ) { let (sd, _, _) = md_to_html(short_description.as_ref()); let (ld, _, _) = md_to_html(long_description.as_ref()); diesel::update(self) .set(( instances::name.eq(name), instances::open_registrations.eq(open_registrations), instances::short_description.eq(short_description), instances::long_description.eq(long_description), instances::short_description_html.eq(sd), instances::long_description_html.eq(ld), )) .execute(conn) .expect("Instance::update: update error"); } pub fn count(conn: &Connection) -> i64 { instances::table .count() .get_result(conn) .expect("Instance::count: counting error") } } #[cfg(test)] pub(crate) mod tests { use super::*; use diesel::Connection; use tests::db; use Connection as Conn; pub(crate) fn fill_database(conn: &Conn) -> Vec<(NewInstance, Instance)> { vec![ NewInstance { default_license: "WTFPL".to_string(), local: true, long_description: SafeString::new("This is my instance."), long_description_html: "

This is my instance

".to_string(), short_description: SafeString::new("My instance."), short_description_html: "

My instance

".to_string(), name: "My instance".to_string(), open_registrations: true, public_domain: "plu.me".to_string(), }, NewInstance { default_license: "WTFPL".to_string(), local: false, long_description: SafeString::new("This is an instance."), long_description_html: "

This is an instance

".to_string(), short_description: SafeString::new("An instance."), short_description_html: "

An instance

".to_string(), name: "An instance".to_string(), open_registrations: true, public_domain: "1plu.me".to_string(), }, NewInstance { default_license: "CC-0".to_string(), local: false, long_description: SafeString::new("This is the instance of someone."), long_description_html: "

This is the instance of someone

".to_string(), short_description: SafeString::new("Someone instance."), short_description_html: "

Someone instance

".to_string(), name: "Someone instance".to_string(), open_registrations: false, public_domain: "2plu.me".to_string(), }, NewInstance { default_license: "CC-0-BY-SA".to_string(), local: false, long_description: SafeString::new("Good morning"), long_description_html: "

Good morning

".to_string(), short_description: SafeString::new("Hello"), short_description_html: "

Hello

".to_string(), name: "Nice day".to_string(), open_registrations: true, public_domain: "3plu.me".to_string(), }, ].into_iter() .map(|inst| { ( inst.clone(), Instance::find_by_domain(conn, inst.public_domain.clone()) .unwrap_or_else(|| Instance::insert(conn, inst)), ) }) .collect() } #[test] fn local_instance() { let conn = &db(); conn.test_transaction::<_, (), _>(|| { let inserted = fill_database(conn) .into_iter() .map(|(inserted, _)| inserted) .find(|inst| inst.local) .unwrap(); let res = Instance::get_local(conn).unwrap(); part_eq!( res, inserted, [ default_license, local, long_description, long_description_html, short_description, short_description_html, name, open_registrations, public_domain ] ); assert_eq!(Instance::local_id(conn), res.id); Ok(()) }); } #[test] fn remote_instance() { let conn = &db(); conn.test_transaction::<_, (), _>(|| { let inserted = fill_database(conn); assert_eq!(Instance::count(conn), inserted.len() as i64); let res = Instance::get_remotes(conn); assert_eq!( res.len(), inserted.iter().filter(|(inst, _)| !inst.local).count() ); inserted .iter() .filter(|(newinst, _)| !newinst.local) .map(|(newinst, inst)| (newinst, res.iter().find(|res| res.id == inst.id).unwrap())) .for_each(|(newinst, inst)| { part_eq!( newinst, inst, [ default_license, local, long_description, long_description_html, short_description, short_description_html, name, open_registrations, public_domain ] ) }); let page = Instance::page(conn, (0, 2)); assert_eq!(page.len(), 2); let page1 = &page[0]; let page2 = &page[1]; assert!(page1.public_domain <= page2.public_domain); let mut last_domaine: String = Instance::page(conn, (0, 1))[0].public_domain.clone(); for i in 1..inserted.len() as i32 { let page = Instance::page(conn, (i, i + 1)); assert_eq!(page.len(), 1); assert!(last_domaine <= page[0].public_domain); last_domaine = page[0].public_domain.clone(); } Ok(()) }); } #[test] fn blocked() { let conn = &db(); conn.test_transaction::<_, (), _>(|| { let inst_list = fill_database(conn); let inst = &inst_list[0].1; let inst_list = &inst_list[1..]; let blocked = inst.blocked; inst.toggle_block(conn); let inst = Instance::get(conn, inst.id).unwrap(); assert_eq!(inst.blocked, !blocked); assert_eq!( inst_list .iter() .filter( |(_, inst)| inst.blocked != Instance::get(conn, inst.id).unwrap().blocked ) .count(), 0 ); assert_eq!( Instance::is_blocked(conn, format!("https://{}/something", inst.public_domain)), inst.blocked ); assert_eq!( Instance::is_blocked(conn, format!("https://{}a/something", inst.public_domain)), Instance::find_by_domain(conn, format!("{}a", inst.public_domain)) .map(|inst| inst.blocked) .unwrap_or(false) ); inst.toggle_block(conn); let inst = Instance::get(conn, inst.id).unwrap(); assert_eq!(inst.blocked, blocked); assert_eq!( Instance::is_blocked(conn, format!("https://{}/something", inst.public_domain)), inst.blocked ); assert_eq!( Instance::is_blocked(conn, format!("https://{}a/something", inst.public_domain)), Instance::find_by_domain(conn, format!("{}a", inst.public_domain)) .map(|inst| inst.blocked) .unwrap_or(false) ); assert_eq!( inst_list .iter() .filter( |(_, inst)| inst.blocked != Instance::get(conn, inst.id).unwrap().blocked ) .count(), 0 ); Ok(()) }); } #[test] fn update() { let conn = &db(); conn.test_transaction::<_, (), _>(|| { let inst = &fill_database(conn)[0].1; inst.update( conn, "NewName".to_owned(), false, SafeString::new("[short](#link)"), SafeString::new("[long_description](/with_link)"), ); let inst = Instance::get(conn, inst.id).unwrap(); assert_eq!(inst.name, "NewName".to_owned()); assert_eq!(inst.open_registrations, false); assert_eq!( inst.long_description.get(), "[long_description](/with_link)" ); assert_eq!( inst.long_description_html, "

long_description

\n" ); assert_eq!(inst.short_description.get(), "[short](#link)"); assert_eq!( inst.short_description_html, "

short

\n" ); Ok(()) }); } }