X-Git-Url: https://git.r.bdr.sh/rbdr/blog/blobdiff_plain/50f53dc480fda8b3daab7a34454c2dd9f3f5f991..795d79afdbe5bfe5fd80902f08afdb6b9fa4db03:/src/generator/rss.rs diff --git a/src/generator/rss.rs b/src/generator/rss.rs index 6a13bb8..c4b8bdb 100644 --- a/src/generator/rss.rs +++ b/src/generator/rss.rs @@ -1,20 +1,194 @@ +use crate::template::{find, parse, Context}; use std::fs::write; use std::io::{Error, ErrorKind::Other, Result}; -use std::path::PathBuf; -use crate::template::{find, parse, TemplateContext}; +use std::path::Path; const FILENAME: &str = "feed.xml"; -pub fn generate(_: &PathBuf, template_directory: &PathBuf, target: &PathBuf, context: &TemplateContext) -> Result<()> { - match find(template_directory, FILENAME) { - Some(template) => { - let parsed_template = parse(&template) - .ok_or_else(|| Error::new(Other, "Unable to parse RSS template"))?; - let rendered_template = parsed_template.render(context)?; - let location = target.join(FILENAME); - write(location, rendered_template)?; - }, - None => {} +pub fn generate( + _: &Path, + template_directory: &Path, + target: &Path, + context: &Context, +) -> Result<()> { + if let Some(template) = find(template_directory, FILENAME) { + let parsed_template = + parse(&template).ok_or_else(|| Error::new(Other, "Unable to parse RSS template"))?; + let rendered_template = parsed_template.render(context)?; + let location = target.join(FILENAME); + write(location, rendered_template)?; } Ok(()) } + +#[cfg(test)] +mod tests { + use std::collections::HashMap; + use std::fs::create_dir_all; + + use super::*; + + use crate::template::Value; + + use test_utilities::*; + + #[test] + fn test_generates_rss_with_default_layout() { + let test_dir = setup_test_dir(); + let static_dir = test_dir.join("static"); + let template_dir = test_dir.join("templates"); + let output_dir = test_dir.join("output"); + + create_dir_all(&static_dir).expect("Could not create static directory"); + create_dir_all(&template_dir).expect("Could not create template directory"); + create_dir_all(&output_dir).expect("Could not create output directory"); + + let context = HashMap::from([ + ("has_posts".to_string(), Value::Bool(true)), + ( + "posts".to_string(), + Value::Collection(vec![ + HashMap::from([ + ("id".to_string(), Value::String("123".to_string())), + ( + "created_on_utc".to_string(), + Value::String("Last Saturday".to_string()), + ), + ( + "title".to_string(), + Value::String("Big words I know.".to_string()), + ), + ( + "escaped_html".to_string(), + Value::String("

Contextualization

".to_string()), + ), + ]), + HashMap::from([ + ("id".to_string(), Value::String("456".to_string())), + ( + "created_on_utc".to_string(), + Value::String("This Saturday".to_string()), + ), + ( + "title".to_string(), + Value::String("Big words I don't know.".to_string()), + ), + ( + "escaped_html".to_string(), + Value::String("

Contexternalization

".to_string()), + ), + ]), + ]), + ), + ]); + + generate(&static_dir, &template_dir, &output_dir, &context).expect("Generate failed"); + + assert_file_contains(&output_dir.join("feed.xml"), "

Contextualization

"); + assert_file_contains(&output_dir.join("feed.xml"), "

Contexternalization

"); + } + + #[test] + fn test_uses_custom_layout_if_available() { + let test_dir = setup_test_dir(); + let static_dir = test_dir.join("static"); + let template_dir = test_dir.join("templates"); + let output_dir = test_dir.join("output"); + + create_dir_all(&static_dir).expect("Could not create static directory"); + create_dir_all(&template_dir).expect("Could not create template directory"); + create_dir_all(&output_dir).expect("Could not create output directory"); + create_test_file( + &template_dir.join("feed.xml"), + "\ +{{~ posts:post }} + {{= post.html }} +{{~}} +", + ); + let context = HashMap::from([( + "posts".to_string(), + Value::Collection(vec![ + HashMap::from([( + "html".to_string(), + Value::String("

Recontextualization

".to_string()), + )]), + HashMap::from([( + "html".to_string(), + Value::String("

Recontexternalization

".to_string()), + )]), + ]), + )]); + + generate(&static_dir, &template_dir, &output_dir, &context).expect("Generate failed"); + + assert_file_contents( + &output_dir.join("feed.xml"), + "\ +

Recontextualization

+ +

Recontexternalization

+", + ); + } + + #[test] + fn test_fails_if_rss_is_malformed() { + let test_dir = setup_test_dir(); + let static_dir = test_dir.join("static"); + let template_dir = test_dir.join("templates"); + let output_dir = test_dir.join("output"); + + create_dir_all(&static_dir).expect("Could not create static directory"); + create_dir_all(&template_dir).expect("Could not create template directory"); + create_dir_all(&output_dir).expect("Could not create output directory"); + create_test_file( + &template_dir.join("feed.xml"), + "\ +{{~ posts:post }} + {{ post.html }} +{{~}} +", + ); + let context = HashMap::new(); + + let result = generate(&static_dir, &template_dir, &output_dir, &context); + + assert!(result.is_err()); + } + + #[test] + fn test_fails_if_output_is_not_writable() { + let test_dir = setup_test_dir(); + let static_dir = test_dir.join("static"); + let template_dir = test_dir.join("templates"); + let output_dir = test_dir.join("output"); + + create_dir_all(&template_dir).expect("Could not create template directory"); + create_test_file( + &template_dir.join("feed.xml"), + "\ +{{~ posts:post }} + {{= post.html }} +{{~}} +", + ); + let context = HashMap::from([( + "posts".to_string(), + Value::Collection(vec![ + HashMap::from([( + "html".to_string(), + Value::String("

Recontextualization

".to_string()), + )]), + HashMap::from([( + "html".to_string(), + Value::String("

Recontexternalization

".to_string()), + )]), + ]), + )]); + + let result = generate(&static_dir, &template_dir, &output_dir, &context); + + assert!(result.is_err()); + } +}