mod templates; pub use templates::{ErrorPage, MainPage, Template}; mod onboarding; pub use onboarding::OnboardingPage; mod indieauth; pub use indieauth::AuthorizationRequestPage; mod login; pub use login::LoginPage; mod mf2; pub use mf2::{Entry, VCard, Feed, Food, POSTS_PER_PAGE}; #[cfg(test)] mod tests { use faker_rand::en_us::internet::Domain; use faker_rand::lorem::Word; use microformats::types::{Document, Item, PropertyValue, Url}; use serde_json::json; use std::cell::RefCell; use std::rc::Rc; enum PostType { Note, Article, ReplyTo(serde_json::Value), ReplyToLink(String), LikeOf(serde_json::Value), LikeOfLink(String), } fn gen_hcard(domain: &str) -> serde_json::Value { use faker_rand::en_us::names::FirstName; json!({ "type": ["h-card"], "properties": { "name": [rand::random::<FirstName>().to_string()], "photo": [format!("https://{domain}/media/me.png")], "uid": [format!("https://{domain}/")], "url": [format!("https://{domain}/")] } }) } fn gen_random_post(domain: &str, kind: PostType) -> serde_json::Value { use faker_rand::lorem::{Paragraph, Sentence}; fn html(content: Paragraph) -> serde_json::Value { json!({ "html": format!("<p>{}</p>", content), "value": content.to_string() }) } let uid = format!( "https://{domain}/posts/{}-{}-{}", rand::random::<Word>(), rand::random::<Word>(), rand::random::<Word>() ); let dt = chrono::offset::Local::now().to_rfc3339_opts(chrono::SecondsFormat::Secs, true); match kind { PostType::Note => { let content = rand::random::<Paragraph>(); json!({ "type": ["h-entry"], "properties": { "content": [html(content)], "published": [dt], "uid": [&uid], "url": [&uid], "author": [gen_hcard(domain)] } }) } PostType::Article => { let content = rand::random::<Paragraph>(); let name = rand::random::<Sentence>(); json!({ "type": ["h-entry"], "properties": { "content": [html(content)], "published": [dt], "uid": [&uid], "url": [&uid], "author": [gen_hcard(domain)], "name": [name.to_string()] } }) } PostType::ReplyTo(ctx) => { let content = rand::random::<Paragraph>(); json!({ "type": ["h-entry"], "properties": { "content": [html(content)], "published": [dt], "uid": [&uid], "url": [&uid], "author": [gen_hcard(domain)], "in-reply-to": [{ "type": ["h-cite"], "properties": ctx["properties"] }] } }) } PostType::ReplyToLink(link) => { let content = rand::random::<Paragraph>(); json!({ "type": ["h-entry"], "properties": { "content": [html(content)], "published": [dt], "uid": [&uid], "url": [&uid], "author": [gen_hcard(domain)], "in-reply-to": [link] } }) } PostType::LikeOf(ctx) => { json!({ "type": ["h-entry"], "properties": { "published": [dt], "author": [gen_hcard(domain)], "uid": [&uid], "url": [&uid], "like-of": [{ "type": ["h-cite"], "properties": ctx["properties"] }] } }) } PostType::LikeOfLink(link) => { json!({ "type": ["h-entry"], "properties": { "published": [dt], "author": [gen_hcard(domain)], "uid": [&uid], "url": [&uid], "like-of": [link] } }) } } } fn check_dt_published(mf2: &serde_json::Value, item: &Rc<RefCell<Item>>) { use microformats::types::temporal::Value as TemporalValue; let _item = item.borrow(); let props = _item.properties.borrow(); assert!(props.contains_key("published")); if let Some(PropertyValue::Temporal(TemporalValue::Timestamp(item))) = props.get("published").and_then(|v| v.first()) { use chrono::{DateTime, FixedOffset, NaiveDateTime}; // Faithfully reconstruct the original datetime // I wonder why not just have an Enum that would // get you either date, time or a datetime, // potentially with an offset? let offset = item.as_offset().unwrap().data; let ndt: NaiveDateTime = item.as_date().unwrap().data .and_time(item.as_time().unwrap().data) // subtract the offset here, since we will add it back - offset; let dt = DateTime::<FixedOffset>::from_utc(ndt, offset); let expected: DateTime<FixedOffset> = chrono::DateTime::parse_from_rfc3339( mf2["properties"]["published"][0].as_str().unwrap(), ) .unwrap(); assert_eq!(dt, expected); } else { unreachable!() } } fn check_e_content(mf2: &serde_json::Value, item: &Rc<RefCell<Item>>) { let _item = item.borrow(); let props = _item.properties.borrow(); assert!(props.contains_key("content")); if let Some(PropertyValue::Fragment(content)) = props.get("content").and_then(|v| v.first()) { assert_eq!( content.html, mf2["properties"]["content"][0]["html"].as_str().unwrap() ); } else { unreachable!() } } #[test] #[ignore = "see https://gitlab.com/maxburon/microformats-parser/-/issues/7"] fn test_note() { let mf2 = gen_random_post(&rand::random::<Domain>().to_string(), PostType::Note); let html = crate::mf2::Entry { post: &mf2 }.to_string(); let url: Url = mf2 .pointer("/properties/uid/0") .and_then(|i| i.as_str()) .and_then(|u| u.parse().ok()) .unwrap(); let parsed: Document = microformats::from_html(&html, url.clone()).unwrap(); if let Some(PropertyValue::Item(item)) = parsed.get_item_by_url(&url) { let _item = item.borrow(); let props = _item.properties.borrow(); check_e_content(&mf2, &item); check_dt_published(&mf2, &item); assert!(props.contains_key("uid")); assert!(props.contains_key("url")); assert!(props .get("url") .unwrap() .iter() .any(|i| i == props.get("uid").and_then(|v| v.first()).unwrap())); // XXX: fails because of https://gitlab.com/maxburon/microformats-parser/-/issues/7 assert!(!props.contains_key("name")); } else { unreachable!() } } #[test] fn test_article() { let mf2 = gen_random_post(&rand::random::<Domain>().to_string(), PostType::Article); let html = crate::mf2::Entry { post: &mf2 }.to_string(); let url: Url = mf2 .pointer("/properties/uid/0") .and_then(|i| i.as_str()) .and_then(|u| u.parse().ok()) .unwrap(); let parsed: Document = microformats::from_html(&html, url.clone()).unwrap(); if let Some(PropertyValue::Item(item)) = parsed.get_item_by_url(&url) { let _item = item.borrow(); let props = _item.properties.borrow(); check_e_content(&mf2, &item); check_dt_published(&mf2, &item); assert!(props.contains_key("uid")); assert!(props.contains_key("url")); assert!(props .get("url") .unwrap() .iter() .any(|i| i == props.get("uid").and_then(|v| v.first()).unwrap())); assert!(props.contains_key("name")); if let Some(PropertyValue::Plain(name)) = props.get("name").and_then(|v| v.first()) { assert_eq!( name, mf2.pointer("/properties/name/0") .and_then(|v| v.as_str()) .unwrap() ); } else { panic!("Name wasn't a plain property!"); } } else { unreachable!() } } #[test] fn test_like_of() { for likeof in [ PostType::LikeOf(gen_random_post( &rand::random::<Domain>().to_string(), PostType::Note, )), PostType::LikeOfLink(format!( "https://{}/posts/{}-{}-{}", &rand::random::<Domain>(), &rand::random::<Word>(), &rand::random::<Word>(), &rand::random::<Word>(), )), ] { let mf2 = gen_random_post(&rand::random::<Domain>().to_string(), likeof); let url: Url = mf2 .pointer("/properties/uid/0") .and_then(|i| i.as_str()) .and_then(|u| u.parse().ok()) .unwrap(); let html = crate::mf2::Entry { post: &mf2 }.to_string(); let parsed: Document = microformats::from_html(&html, url.clone()).unwrap(); if let Some(item) = parsed.items.get(0) { let _item = item.borrow(); let props = _item.properties.borrow(); check_dt_published(&mf2, item); assert!(props.contains_key("like-of")); match props.get("like-of").and_then(|v| v.first()) { Some(PropertyValue::Url(url)) => { assert_eq!( url, &mf2.pointer("/properties/like-of/0") .and_then(|i| i.as_str()) .or_else(|| mf2 .pointer("/properties/like-of/0/properties/uid/0") .and_then(|i| i.as_str())) .and_then(|u| u.parse::<Url>().ok()) .unwrap() ); } Some(PropertyValue::Item(_cite)) => { todo!() } other => panic!("Unexpected value in like-of: {:?}", other), } } else { unreachable!() } } } }