zola/components/front_matter/src/page.rs

290 lines
8.1 KiB
Rust
Raw Normal View History

2018-07-16 08:54:05 +00:00
use std::collections::HashMap;
use chrono::prelude::*;
2019-12-21 21:52:39 +00:00
use serde_derive::Deserialize;
2018-10-31 07:18:57 +00:00
use tera::{Map, Value};
use toml;
2019-12-21 21:52:39 +00:00
use errors::{bail, Result};
2018-11-05 21:22:15 +00:00
use utils::de::{fix_toml_dates, from_toml_datetime};
/// The front matter of every page
2018-09-21 07:40:52 +00:00
#[derive(Debug, Clone, PartialEq, Deserialize)]
2018-03-15 17:58:32 +00:00
#[serde(default)]
pub struct PageFrontMatter {
/// <title> of the page
pub title: Option<String>,
/// Description in <meta> that appears when linked, e.g. on twitter
pub description: Option<String>,
/// Updated date
#[serde(default, deserialize_with = "from_toml_datetime")]
pub updated: Option<String>,
/// Date if we want to order pages (ie blog post)
2018-11-05 21:22:15 +00:00
#[serde(default, deserialize_with = "from_toml_datetime")]
pub date: Option<String>,
2018-09-21 07:40:52 +00:00
/// Chrono converted datetime
#[serde(default, skip_deserializing)]
pub datetime: Option<NaiveDateTime>,
/// The converted date into a (year, month, day) tuple
#[serde(default, skip_deserializing)]
pub datetime_tuple: Option<(i32, u32, u32)>,
2019-08-24 20:23:08 +00:00
/// Whether this page is a draft
2018-03-21 15:18:24 +00:00
pub draft: bool,
/// The page slug. Will be used instead of the filename if present
/// Can't be an empty string if present
pub slug: Option<String>,
2017-10-04 00:35:37 +00:00
/// The path the page appears at, overrides the slug if set in the front-matter
/// otherwise is set after parsing front matter and sections
/// Can't be an empty string if present
2017-10-04 00:35:37 +00:00
pub path: Option<String>,
2018-07-16 08:54:05 +00:00
pub taxonomies: HashMap<String, Vec<String>>,
/// Integer to use to order content. Lowest is at the bottom, highest first
pub order: Option<usize>,
2017-07-01 10:13:21 +00:00
/// Integer to use to order content. Highest is at the bottom, lowest first
pub weight: Option<usize>,
/// All aliases for that page. Zola will create HTML templates that will
/// redirect to this
2017-06-16 14:09:01 +00:00
#[serde(skip_serializing)]
2018-03-21 15:18:24 +00:00
pub aliases: Vec<String>,
/// Specify a template different from `page.html` to use for that page
#[serde(skip_serializing)]
pub template: Option<String>,
2018-03-14 21:03:06 +00:00
/// Whether the page is included in the search index
/// Defaults to `true` but is only used if search if explicitly enabled in the config.
2018-03-15 17:58:32 +00:00
#[serde(skip_serializing)]
2018-03-14 21:03:06 +00:00
pub in_search_index: bool,
/// Any extra parameter present in the front matter
pub extra: Map<String, Value>,
}
impl PageFrontMatter {
pub fn parse(toml: &str) -> Result<PageFrontMatter> {
let mut f: PageFrontMatter = match toml::from_str(toml) {
Ok(d) => d,
Err(e) => bail!(e),
};
if let Some(ref slug) = f.slug {
if slug == "" {
bail!("`slug` can't be empty if present")
}
}
2017-10-04 00:35:37 +00:00
if let Some(ref path) = f.path {
if path == "" {
bail!("`path` can't be empty if present")
}
}
f.extra = match fix_toml_dates(f.extra) {
Value::Object(o) => o,
_ => unreachable!("Got something other than a table in page extra"),
};
2018-09-21 07:40:52 +00:00
f.date_to_datetime();
Ok(f)
}
/// Converts the TOML datetime to a Chrono naive datetime
2018-09-21 07:40:52 +00:00
/// Also grabs the year/month/day tuple that will be used in serialization
pub fn date_to_datetime(&mut self) {
self.datetime = if let Some(ref d) = self.date {
if d.contains('T') {
DateTime::parse_from_rfc3339(&d).ok().map(|s| s.naive_local())
} else {
2019-12-23 08:21:51 +00:00
NaiveDate::parse_from_str(&d, "%Y-%m-%d").ok().map(|s| s.and_hms(0, 0, 0))
}
} else {
None
2018-09-21 07:40:52 +00:00
};
self.datetime_tuple = if let Some(ref dt) = self.datetime {
Some((dt.year(), dt.month(), dt.day()))
} else {
None
};
}
pub fn order(&self) -> usize {
self.order.unwrap()
}
2017-07-01 10:13:21 +00:00
pub fn weight(&self) -> usize {
self.weight.unwrap()
}
}
impl Default for PageFrontMatter {
fn default() -> PageFrontMatter {
PageFrontMatter {
title: None,
description: None,
updated: None,
date: None,
2018-09-21 07:40:52 +00:00
datetime: None,
datetime_tuple: None,
2018-03-21 15:18:24 +00:00
draft: false,
slug: None,
2017-10-04 00:35:37 +00:00
path: None,
2018-07-16 08:54:05 +00:00
taxonomies: HashMap::new(),
order: None,
2017-07-01 10:13:21 +00:00
weight: None,
2018-03-21 15:18:24 +00:00
aliases: Vec::new(),
2018-03-14 21:03:06 +00:00
in_search_index: true,
template: None,
extra: Map::new(),
}
}
}
#[cfg(test)]
mod tests {
use super::PageFrontMatter;
2018-10-31 07:18:57 +00:00
use tera::to_value;
#[test]
fn can_have_empty_front_matter() {
let content = r#" "#;
let res = PageFrontMatter::parse(content);
println!("{:?}", res);
assert!(res.is_ok());
}
#[test]
fn can_parse_valid_front_matter() {
let content = r#"
title = "Hello"
description = "hey there""#;
let res = PageFrontMatter::parse(content);
assert!(res.is_ok());
let res = res.unwrap();
assert_eq!(res.title.unwrap(), "Hello".to_string());
assert_eq!(res.description.unwrap(), "hey there".to_string())
}
#[test]
fn errors_with_invalid_front_matter() {
let content = r#"title = 1\n"#;
let res = PageFrontMatter::parse(content);
assert!(res.is_err());
}
#[test]
fn errors_on_present_but_empty_slug() {
let content = r#"
title = "Hello"
description = "hey there"
slug = """#;
let res = PageFrontMatter::parse(content);
assert!(res.is_err());
}
#[test]
2017-10-04 00:35:37 +00:00
fn errors_on_present_but_empty_path() {
let content = r#"
title = "Hello"
description = "hey there"
2017-10-04 00:35:37 +00:00
path = """#;
let res = PageFrontMatter::parse(content);
assert!(res.is_err());
}
#[test]
fn can_parse_date_yyyy_mm_dd() {
let content = r#"
title = "Hello"
description = "hey there"
date = 2016-10-10
"#;
let res = PageFrontMatter::parse(content).unwrap();
assert!(res.date.is_some());
}
#[test]
fn can_parse_date_rfc3339() {
let content = r#"
title = "Hello"
description = "hey there"
date = 2002-10-02T15:00:00Z
"#;
let res = PageFrontMatter::parse(content).unwrap();
assert!(res.date.is_some());
}
#[test]
fn cannot_parse_random_date_format() {
let content = r#"
title = "Hello"
description = "hey there"
date = 2002/10/12"#;
let res = PageFrontMatter::parse(content);
assert!(res.is_err());
}
#[test]
fn cannot_parse_invalid_date_format() {
let content = r#"
title = "Hello"
description = "hey there"
date = 2002-14-01"#;
let res = PageFrontMatter::parse(content);
assert!(res.is_err());
}
2018-01-22 12:46:36 +00:00
#[test]
fn cannot_parse_date_as_string() {
let content = r#"
title = "Hello"
description = "hey there"
date = "2002-14-01""#;
let res = PageFrontMatter::parse(content);
assert!(res.is_err());
}
#[test]
fn can_parse_dates_in_extra() {
let content = r#"
title = "Hello"
description = "hey there"
[extra]
some-date = 2002-14-01"#;
let res = PageFrontMatter::parse(content);
println!("{:?}", res);
assert!(res.is_ok());
assert_eq!(res.unwrap().extra["some-date"], to_value("2002-14-01").unwrap());
}
#[test]
fn can_parse_nested_dates_in_extra() {
let content = r#"
title = "Hello"
description = "hey there"
[extra.something]
some-date = 2002-14-01"#;
let res = PageFrontMatter::parse(content);
println!("{:?}", res);
assert!(res.is_ok());
assert_eq!(res.unwrap().extra["something"]["some-date"], to_value("2002-14-01").unwrap());
}
2018-07-16 08:54:05 +00:00
#[test]
fn can_parse_taxonomies() {
let content = r#"
title = "Hello World"
[taxonomies]
tags = ["Rust", "JavaScript"]
categories = ["Dev"]
"#;
let res = PageFrontMatter::parse(content);
println!("{:?}", res);
assert!(res.is_ok());
let res2 = res.unwrap();
assert_eq!(res2.taxonomies["categories"], vec!["Dev"]);
assert_eq!(res2.taxonomies["tags"], vec!["Rust", "JavaScript"]);
}
}