2019-12-21 21:52:39 +00:00
|
|
|
use lazy_static::lazy_static;
|
|
|
|
use serde_derive::{Deserialize, Serialize};
|
2017-05-13 04:01:38 +00:00
|
|
|
|
2019-12-21 21:52:39 +00:00
|
|
|
use errors::{bail, Error, Result};
|
2018-10-31 07:18:57 +00:00
|
|
|
use regex::Regex;
|
|
|
|
use std::path::Path;
|
2017-05-13 04:01:38 +00:00
|
|
|
|
|
|
|
mod page;
|
|
|
|
mod section;
|
|
|
|
|
2017-07-01 07:47:41 +00:00
|
|
|
pub use page::PageFrontMatter;
|
|
|
|
pub use section::SectionFrontMatter;
|
2017-05-13 04:01:38 +00:00
|
|
|
|
|
|
|
lazy_static! {
|
2018-10-31 07:18:57 +00:00
|
|
|
static ref PAGE_RE: Regex =
|
2020-08-18 11:14:27 +00:00
|
|
|
Regex::new(r"^[[:space:]]*\+\+\+(\r?\n(?s).*?(?-s))\+\+\+\r?\n?((?s).*(?-s))$").unwrap();
|
2017-05-13 04:01:38 +00:00
|
|
|
}
|
|
|
|
|
2017-07-01 07:47:41 +00:00
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, Serialize, Deserialize)]
|
|
|
|
#[serde(rename_all = "lowercase")]
|
|
|
|
pub enum SortBy {
|
2017-09-27 14:37:17 +00:00
|
|
|
/// Most recent to oldest
|
2017-07-01 07:47:41 +00:00
|
|
|
Date,
|
2017-09-27 14:37:17 +00:00
|
|
|
/// Lower weight comes first
|
2017-07-01 07:47:41 +00:00
|
|
|
Weight,
|
2017-09-27 14:37:17 +00:00
|
|
|
/// No sorting
|
2017-07-01 07:47:41 +00:00
|
|
|
None,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, Serialize, Deserialize)]
|
|
|
|
#[serde(rename_all = "lowercase")]
|
|
|
|
pub enum InsertAnchor {
|
|
|
|
Left,
|
|
|
|
Right,
|
|
|
|
None,
|
|
|
|
}
|
|
|
|
|
2017-05-13 04:01:38 +00:00
|
|
|
/// Split a file between the front matter and its content
|
|
|
|
/// Will return an error if the front matter wasn't found
|
2020-08-18 11:14:27 +00:00
|
|
|
fn split_content<'c>(file_path: &Path, content: &'c str) -> Result<(&'c str, &'c str)> {
|
2017-05-13 04:01:38 +00:00
|
|
|
if !PAGE_RE.is_match(content) {
|
2018-10-31 07:18:57 +00:00
|
|
|
bail!(
|
|
|
|
"Couldn't find front matter in `{}`. Did you forget to add `+++`?",
|
|
|
|
file_path.to_string_lossy()
|
|
|
|
);
|
2017-05-13 04:01:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 2. extract the front matter and the content
|
|
|
|
let caps = PAGE_RE.captures(content).unwrap();
|
|
|
|
// caps[0] is the full match
|
|
|
|
// caps[1] => front matter
|
|
|
|
// caps[2] => content
|
2020-08-18 11:14:27 +00:00
|
|
|
Ok((caps.get(1).unwrap().as_str(), caps.get(2).unwrap().as_str()))
|
2017-05-13 04:01:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Split a file between the front matter and its content.
|
2017-05-15 03:23:19 +00:00
|
|
|
/// Returns a parsed `SectionFrontMatter` and the rest of the content
|
2020-08-18 11:14:27 +00:00
|
|
|
pub fn split_section_content<'c>(
|
2018-10-31 07:18:57 +00:00
|
|
|
file_path: &Path,
|
2020-08-18 11:14:27 +00:00
|
|
|
content: &'c str,
|
|
|
|
) -> Result<(SectionFrontMatter, &'c str)> {
|
2017-05-13 04:01:38 +00:00
|
|
|
let (front_matter, content) = split_content(file_path, content)?;
|
2019-01-11 19:29:46 +00:00
|
|
|
let meta = SectionFrontMatter::parse(&front_matter).map_err(|e| {
|
2019-02-09 18:54:46 +00:00
|
|
|
Error::chain(
|
|
|
|
format!("Error when parsing front matter of section `{}`", file_path.to_string_lossy()),
|
|
|
|
e,
|
|
|
|
)
|
2018-10-31 07:18:57 +00:00
|
|
|
})?;
|
2017-05-13 04:01:38 +00:00
|
|
|
Ok((meta, content))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Split a file between the front matter and its content
|
2017-05-15 03:23:19 +00:00
|
|
|
/// Returns a parsed `PageFrontMatter` and the rest of the content
|
2020-08-18 11:14:27 +00:00
|
|
|
pub fn split_page_content<'c>(file_path: &Path, content: &'c str) -> Result<(PageFrontMatter, &'c str)> {
|
2017-05-13 04:01:38 +00:00
|
|
|
let (front_matter, content) = split_content(file_path, content)?;
|
2019-01-11 19:29:46 +00:00
|
|
|
let meta = PageFrontMatter::parse(&front_matter).map_err(|e| {
|
2019-02-09 18:54:46 +00:00
|
|
|
Error::chain(
|
|
|
|
format!("Error when parsing front matter of page `{}`", file_path.to_string_lossy()),
|
|
|
|
e,
|
|
|
|
)
|
2018-10-31 07:18:57 +00:00
|
|
|
})?;
|
2017-05-13 04:01:38 +00:00
|
|
|
Ok((meta, content))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use std::path::Path;
|
|
|
|
|
2018-10-31 07:18:57 +00:00
|
|
|
use super::{split_page_content, split_section_content};
|
2017-05-13 04:01:38 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_split_page_content_valid() {
|
|
|
|
let content = r#"
|
|
|
|
+++
|
|
|
|
title = "Title"
|
|
|
|
description = "hey there"
|
2018-01-14 17:03:57 +00:00
|
|
|
date = 2002-10-12
|
2017-05-13 04:01:38 +00:00
|
|
|
+++
|
|
|
|
Hello
|
|
|
|
"#;
|
|
|
|
let (front_matter, content) = split_page_content(Path::new(""), content).unwrap();
|
|
|
|
assert_eq!(content, "Hello\n");
|
|
|
|
assert_eq!(front_matter.title.unwrap(), "Title");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_split_section_content_valid() {
|
|
|
|
let content = r#"
|
|
|
|
+++
|
|
|
|
paginate_by = 10
|
|
|
|
+++
|
|
|
|
Hello
|
|
|
|
"#;
|
|
|
|
let (front_matter, content) = split_section_content(Path::new(""), content).unwrap();
|
|
|
|
assert_eq!(content, "Hello\n");
|
|
|
|
assert!(front_matter.is_paginated());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_split_content_with_only_frontmatter_valid() {
|
|
|
|
let content = r#"
|
|
|
|
+++
|
|
|
|
title = "Title"
|
|
|
|
description = "hey there"
|
2018-01-14 17:03:57 +00:00
|
|
|
date = 2002-10-12
|
2017-05-13 04:01:38 +00:00
|
|
|
+++"#;
|
|
|
|
let (front_matter, content) = split_page_content(Path::new(""), content).unwrap();
|
|
|
|
assert_eq!(content, "");
|
|
|
|
assert_eq!(front_matter.title.unwrap(), "Title");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_split_content_lazily() {
|
|
|
|
let content = r#"
|
|
|
|
+++
|
|
|
|
title = "Title"
|
|
|
|
description = "hey there"
|
2018-01-14 17:03:57 +00:00
|
|
|
date = 2002-10-02T15:00:00Z
|
2017-05-13 04:01:38 +00:00
|
|
|
+++
|
|
|
|
+++"#;
|
|
|
|
let (front_matter, content) = split_page_content(Path::new(""), content).unwrap();
|
|
|
|
assert_eq!(content, "+++");
|
|
|
|
assert_eq!(front_matter.title.unwrap(), "Title");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn errors_if_cannot_locate_frontmatter() {
|
|
|
|
let content = r#"
|
|
|
|
+++
|
|
|
|
title = "Title"
|
|
|
|
description = "hey there"
|
2018-01-14 17:03:57 +00:00
|
|
|
date = 2002-10-12"#;
|
2017-05-13 04:01:38 +00:00
|
|
|
let res = split_page_content(Path::new(""), content);
|
|
|
|
assert!(res.is_err());
|
|
|
|
}
|
|
|
|
}
|