2018-01-29 17:40:12 +00:00
|
|
|
extern crate site;
|
2018-02-22 15:49:35 +00:00
|
|
|
#[macro_use]
|
2018-01-29 17:40:12 +00:00
|
|
|
extern crate errors;
|
|
|
|
extern crate content;
|
|
|
|
extern crate front_matter;
|
|
|
|
|
|
|
|
use std::path::{Path, Component};
|
|
|
|
|
|
|
|
use errors::Result;
|
|
|
|
use site::Site;
|
|
|
|
use content::{Page, Section};
|
|
|
|
use front_matter::{PageFrontMatter, SectionFrontMatter};
|
|
|
|
|
|
|
|
|
|
|
|
/// Finds the section that contains the page given if there is one
|
|
|
|
pub fn find_parent_section<'a>(site: &'a Site, page: &Page) -> Option<&'a Section> {
|
|
|
|
for section in site.sections.values() {
|
|
|
|
if section.is_child_page(&page.file.path) {
|
2018-07-31 13:17:31 +00:00
|
|
|
return Some(section);
|
2018-01-29 17:40:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq)]
|
|
|
|
pub enum PageChangesNeeded {
|
2018-07-16 08:54:05 +00:00
|
|
|
/// Editing `taxonomies`
|
|
|
|
Taxonomies,
|
2018-01-29 17:40:12 +00:00
|
|
|
/// Editing `date`, `order` or `weight`
|
|
|
|
Sort,
|
|
|
|
/// Editing anything causes a re-render of the page
|
|
|
|
Render,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq)]
|
|
|
|
pub enum SectionChangesNeeded {
|
|
|
|
/// Editing `sort_by`
|
|
|
|
Sort,
|
|
|
|
/// Editing `title`, `description`, `extra`, `template` or setting `render` to true
|
|
|
|
Render,
|
|
|
|
/// Editing `paginate_by`, `paginate_path` or `insert_anchor_links`
|
|
|
|
RenderWithPages,
|
|
|
|
/// Setting `render` to false
|
|
|
|
Delete,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Evaluates all the params in the front matter that changed so we can do the smallest
|
|
|
|
/// delta in the serve command
|
|
|
|
/// Order matters as the actions will be done in insertion order
|
|
|
|
fn find_section_front_matter_changes(current: &SectionFrontMatter, new: &SectionFrontMatter) -> Vec<SectionChangesNeeded> {
|
|
|
|
let mut changes_needed = vec![];
|
|
|
|
|
|
|
|
if current.sort_by != new.sort_by {
|
|
|
|
changes_needed.push(SectionChangesNeeded::Sort);
|
|
|
|
}
|
|
|
|
|
|
|
|
// We want to hide the section
|
|
|
|
// TODO: what to do on redirect_path change?
|
2018-03-14 17:22:24 +00:00
|
|
|
if current.render && !new.render {
|
2018-01-29 17:40:12 +00:00
|
|
|
changes_needed.push(SectionChangesNeeded::Delete);
|
|
|
|
// Nothing else we can do
|
|
|
|
return changes_needed;
|
|
|
|
}
|
|
|
|
|
|
|
|
if current.paginate_by != new.paginate_by
|
|
|
|
|| current.paginate_path != new.paginate_path
|
|
|
|
|| current.insert_anchor_links != new.insert_anchor_links {
|
|
|
|
changes_needed.push(SectionChangesNeeded::RenderWithPages);
|
|
|
|
// Nothing else we can do
|
|
|
|
return changes_needed;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Any new change will trigger a re-rendering of the section page only
|
|
|
|
changes_needed.push(SectionChangesNeeded::Render);
|
|
|
|
changes_needed
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Evaluates all the params in the front matter that changed so we can do the smallest
|
|
|
|
/// delta in the serve command
|
|
|
|
/// Order matters as the actions will be done in insertion order
|
|
|
|
fn find_page_front_matter_changes(current: &PageFrontMatter, other: &PageFrontMatter) -> Vec<PageChangesNeeded> {
|
|
|
|
let mut changes_needed = vec![];
|
|
|
|
|
2018-07-16 08:54:05 +00:00
|
|
|
if current.taxonomies != other.taxonomies {
|
|
|
|
changes_needed.push(PageChangesNeeded::Taxonomies);
|
2018-01-29 17:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if current.date != other.date || current.order != other.order || current.weight != other.weight {
|
|
|
|
changes_needed.push(PageChangesNeeded::Sort);
|
|
|
|
}
|
|
|
|
|
|
|
|
changes_needed.push(PageChangesNeeded::Render);
|
|
|
|
changes_needed
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Handles a path deletion: could be a page, a section, a folder
|
|
|
|
fn delete_element(site: &mut Site, path: &Path, is_section: bool) -> Result<()> {
|
|
|
|
// Ignore the event if this path was not known
|
|
|
|
if !site.sections.contains_key(path) && !site.pages.contains_key(path) {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
if is_section {
|
|
|
|
if let Some(s) = site.pages.remove(path) {
|
|
|
|
site.permalinks.remove(&s.file.relative);
|
|
|
|
site.populate_sections();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if let Some(p) = site.pages.remove(path) {
|
|
|
|
site.permalinks.remove(&p.file.relative);
|
|
|
|
|
2018-07-16 08:54:05 +00:00
|
|
|
if !p.meta.taxonomies.is_empty() {
|
2018-07-16 16:14:48 +00:00
|
|
|
site.populate_taxonomies()?;
|
2018-01-29 17:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// if there is a parent section, we will need to re-render it
|
|
|
|
// most likely
|
|
|
|
if find_parent_section(site, &p).is_some() {
|
|
|
|
site.populate_sections();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure we have our fn updated so it doesn't contain the permalink(s)/section/page deleted
|
|
|
|
site.register_tera_global_fns();
|
|
|
|
// Deletion is something that doesn't happen all the time so we
|
|
|
|
// don't need to optimise it too much
|
|
|
|
return site.build();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Handles a `_index.md` (a section) being edited in some ways
|
|
|
|
fn handle_section_editing(site: &mut Site, path: &Path) -> Result<()> {
|
|
|
|
let section = Section::from_file(path, &site.config)?;
|
|
|
|
match site.add_section(section, true)? {
|
|
|
|
// Updating a section
|
|
|
|
Some(prev) => {
|
2018-08-05 08:29:53 +00:00
|
|
|
// Copy the section data so we don't end up with an almost empty object
|
|
|
|
site.sections.get_mut(path).unwrap().pages = prev.pages;
|
|
|
|
site.sections.get_mut(path).unwrap().ignored_pages = prev.ignored_pages;
|
|
|
|
site.sections.get_mut(path).unwrap().subsections = prev.subsections;
|
|
|
|
|
2018-01-29 17:40:12 +00:00
|
|
|
if site.sections[path].meta == prev.meta {
|
|
|
|
// Front matter didn't change, only content did
|
|
|
|
// so we render only the section page, not its pages
|
|
|
|
return site.render_section(&site.sections[path], false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Front matter changed
|
|
|
|
for changes in find_section_front_matter_changes(&site.sections[path].meta, &prev.meta) {
|
|
|
|
// Sort always comes first if present so the rendering will be fine
|
|
|
|
match changes {
|
|
|
|
SectionChangesNeeded::Sort => {
|
|
|
|
site.sort_sections_pages(Some(path));
|
|
|
|
site.register_tera_global_fns();
|
2018-07-31 13:17:31 +00:00
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
SectionChangesNeeded::Render => site.render_section(&site.sections[path], false)?,
|
|
|
|
SectionChangesNeeded::RenderWithPages => site.render_section(&site.sections[path], true)?,
|
|
|
|
// not a common enough operation to make it worth optimizing
|
|
|
|
SectionChangesNeeded::Delete => {
|
|
|
|
site.populate_sections();
|
|
|
|
site.build()?;
|
2018-07-31 13:17:31 +00:00
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
return Ok(());
|
2018-07-31 13:17:31 +00:00
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
// New section, only render that one
|
|
|
|
None => {
|
|
|
|
site.populate_sections();
|
|
|
|
site.register_tera_global_fns();
|
|
|
|
return site.render_section(&site.sections[path], true);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! render_parent_section {
|
|
|
|
($site: expr, $path: expr) => {
|
|
|
|
match find_parent_section($site, &$site.pages[$path]) {
|
|
|
|
Some(s) => {
|
|
|
|
$site.render_section(s, false)?;
|
|
|
|
},
|
|
|
|
None => (),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Handles a page being edited in some ways
|
|
|
|
fn handle_page_editing(site: &mut Site, path: &Path) -> Result<()> {
|
|
|
|
let page = Page::from_file(path, &site.config)?;
|
|
|
|
match site.add_page(page, true)? {
|
|
|
|
// Updating a page
|
|
|
|
Some(prev) => {
|
|
|
|
// Front matter didn't change, only content did
|
|
|
|
if site.pages[path].meta == prev.meta {
|
|
|
|
// Other than the page itself, the summary might be seen
|
|
|
|
// on a paginated list for a blog for example
|
|
|
|
if site.pages[path].summary.is_some() {
|
|
|
|
render_parent_section!(site, path);
|
|
|
|
}
|
|
|
|
// TODO: register_tera_global_fns is expensive as it involves lots of cloning
|
|
|
|
// I can't think of a valid usecase where you would need the content
|
|
|
|
// of a page through a global fn so it's commented out for now
|
|
|
|
// site.register_tera_global_fns();
|
2018-07-31 13:17:31 +00:00
|
|
|
return site.render_page(&site.pages[path]);
|
2018-01-29 17:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Front matter changed
|
|
|
|
let mut sections_populated = false;
|
|
|
|
for changes in find_page_front_matter_changes(&site.pages[path].meta, &prev.meta) {
|
|
|
|
// Sort always comes first if present so the rendering will be fine
|
|
|
|
match changes {
|
2018-07-16 08:54:05 +00:00
|
|
|
PageChangesNeeded::Taxonomies => {
|
2018-07-16 16:14:48 +00:00
|
|
|
site.populate_taxonomies()?;
|
2018-01-29 17:40:12 +00:00
|
|
|
site.register_tera_global_fns();
|
2018-07-16 08:54:05 +00:00
|
|
|
site.render_taxonomies()?;
|
2018-07-31 13:17:31 +00:00
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
PageChangesNeeded::Sort => {
|
|
|
|
let section_path = match find_parent_section(site, &site.pages[path]) {
|
|
|
|
Some(s) => s.file.path.clone(),
|
|
|
|
None => continue // Do nothing if it's an orphan page
|
|
|
|
};
|
|
|
|
if !sections_populated {
|
|
|
|
site.populate_sections();
|
|
|
|
sections_populated = true;
|
|
|
|
}
|
|
|
|
site.sort_sections_pages(Some(§ion_path));
|
|
|
|
site.register_tera_global_fns();
|
|
|
|
site.render_index()?;
|
2018-07-31 13:17:31 +00:00
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
PageChangesNeeded::Render => {
|
|
|
|
if !sections_populated {
|
|
|
|
site.populate_sections();
|
|
|
|
sections_populated = true;
|
|
|
|
}
|
|
|
|
site.register_tera_global_fns();
|
|
|
|
render_parent_section!(site, path);
|
|
|
|
site.render_page(&site.pages[path])?;
|
2018-07-31 13:17:31 +00:00
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
Ok(())
|
2018-07-31 13:17:31 +00:00
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
// It's a new page!
|
|
|
|
None => {
|
|
|
|
site.populate_sections();
|
2018-07-16 16:14:48 +00:00
|
|
|
site.populate_taxonomies()?;
|
2018-01-29 17:40:12 +00:00
|
|
|
site.register_tera_global_fns();
|
|
|
|
// No need to optimise that yet, we can revisit if it becomes an issue
|
|
|
|
site.build()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-22 15:49:35 +00:00
|
|
|
/// What happens when a section or a page is changed
|
2018-01-29 17:40:12 +00:00
|
|
|
pub fn after_content_change(site: &mut Site, path: &Path) -> Result<()> {
|
|
|
|
let is_section = path.file_name().unwrap() == "_index.md";
|
2018-02-22 15:49:35 +00:00
|
|
|
let is_md = path.extension().unwrap() == "md";
|
|
|
|
let index = path.parent().unwrap().join("index.md");
|
|
|
|
|
|
|
|
// A few situations can happen:
|
|
|
|
// 1. Change on .md files
|
|
|
|
// a. Is there an `index.md`? Return an error if it's something other than delete
|
|
|
|
// b. Deleted? remove the element
|
|
|
|
// c. Edited?
|
|
|
|
// 1. filename is `_index.md`, this is a section
|
|
|
|
// 1. it's a page otherwise
|
|
|
|
// 2. Change on non .md files
|
|
|
|
// a. Try to find a corresponding `_index.md`
|
|
|
|
// 1. Nothing? Return Ok
|
|
|
|
// 2. Something? Update the page
|
|
|
|
if is_md {
|
|
|
|
// only delete if it was able to be added in the first place
|
|
|
|
if !index.exists() && !path.exists() {
|
|
|
|
delete_element(site, path, is_section)?;
|
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
|
2018-02-22 15:49:35 +00:00
|
|
|
// Added another .md in a assets directory
|
2018-02-25 18:29:46 +00:00
|
|
|
if index.exists() && path.exists() && path != index {
|
2018-02-22 15:49:35 +00:00
|
|
|
bail!(
|
|
|
|
"Change on {:?} detected but there is already an `index.md` in the same folder",
|
|
|
|
path.display()
|
|
|
|
);
|
|
|
|
} else if index.exists() && !path.exists() {
|
|
|
|
// deleted the wrong .md, do nothing
|
|
|
|
return Ok(());
|
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
|
2018-02-22 15:49:35 +00:00
|
|
|
if is_section {
|
|
|
|
handle_section_editing(site, path)
|
|
|
|
} else {
|
|
|
|
handle_page_editing(site, path)
|
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
} else {
|
2018-07-31 13:17:31 +00:00
|
|
|
if index.exists() {
|
2018-02-22 15:49:35 +00:00
|
|
|
handle_page_editing(site, &index)
|
|
|
|
} else {
|
|
|
|
Ok(())
|
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// What happens when a template is changed
|
|
|
|
pub fn after_template_change(site: &mut Site, path: &Path) -> Result<()> {
|
|
|
|
site.tera.full_reload()?;
|
|
|
|
let filename = path.file_name().unwrap().to_str().unwrap();
|
|
|
|
|
|
|
|
match filename {
|
|
|
|
"sitemap.xml" => site.render_sitemap(),
|
2018-07-16 08:54:05 +00:00
|
|
|
"rss.xml" => site.render_rss_feed(None, None),
|
2018-01-29 17:40:12 +00:00
|
|
|
"robots.txt" => site.render_robots(),
|
2018-07-16 08:54:05 +00:00
|
|
|
"single.html" | "list.html" => site.render_taxonomies(),
|
2018-01-29 17:40:12 +00:00
|
|
|
"page.html" => {
|
|
|
|
site.render_sections()?;
|
|
|
|
site.render_orphan_pages()
|
2018-07-31 13:17:31 +00:00
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
"section.html" => site.render_sections(),
|
|
|
|
// Either the index or some unknown template changed
|
|
|
|
// We can't really know what this change affects so rebuild all
|
|
|
|
// the things
|
|
|
|
_ => {
|
|
|
|
// If we are updating a shortcode, re-render the markdown of all pages/site
|
|
|
|
// because we have no clue which one needs rebuilding
|
|
|
|
// TODO: look if there the shortcode is used in the markdown instead of re-rendering
|
|
|
|
// everything
|
|
|
|
if path.components().collect::<Vec<_>>().contains(&Component::Normal("shortcodes".as_ref())) {
|
|
|
|
site.render_markdown()?;
|
|
|
|
}
|
|
|
|
site.populate_sections();
|
|
|
|
site.render_sections()?;
|
|
|
|
site.render_orphan_pages()?;
|
2018-07-16 08:54:05 +00:00
|
|
|
site.render_taxonomies()
|
2018-07-31 13:17:31 +00:00
|
|
|
}
|
2018-01-29 17:40:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2018-07-16 08:54:05 +00:00
|
|
|
use std::collections::HashMap;
|
|
|
|
|
2018-01-29 17:40:12 +00:00
|
|
|
use front_matter::{PageFrontMatter, SectionFrontMatter, SortBy};
|
|
|
|
use super::{
|
|
|
|
find_page_front_matter_changes, find_section_front_matter_changes,
|
2018-07-31 13:17:31 +00:00
|
|
|
PageChangesNeeded, SectionChangesNeeded,
|
2018-01-29 17:40:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#[test]
|
2018-07-16 08:54:05 +00:00
|
|
|
fn can_find_taxonomy_changes_in_page_frontmatter() {
|
|
|
|
let mut taxonomies = HashMap::new();
|
|
|
|
taxonomies.insert("tags".to_string(), vec!["a tag".to_string()]);
|
|
|
|
let new = PageFrontMatter { taxonomies, ..PageFrontMatter::default() };
|
2018-01-29 17:40:12 +00:00
|
|
|
let changes = find_page_front_matter_changes(&PageFrontMatter::default(), &new);
|
2018-07-16 08:54:05 +00:00
|
|
|
assert_eq!(changes, vec![PageChangesNeeded::Taxonomies, PageChangesNeeded::Render]);
|
2018-01-29 17:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_find_multiple_changes_in_page_frontmatter() {
|
2018-07-16 08:54:05 +00:00
|
|
|
let mut taxonomies = HashMap::new();
|
|
|
|
taxonomies.insert("categories".to_string(), vec!["a category".to_string()]);
|
|
|
|
let current = PageFrontMatter { taxonomies, order: Some(1), ..PageFrontMatter::default() };
|
2018-01-29 17:40:12 +00:00
|
|
|
let changes = find_page_front_matter_changes(¤t, &PageFrontMatter::default());
|
2018-07-16 08:54:05 +00:00
|
|
|
assert_eq!(changes, vec![PageChangesNeeded::Taxonomies, PageChangesNeeded::Sort, PageChangesNeeded::Render]);
|
2018-01-29 17:40:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_find_sort_changes_in_section_frontmatter() {
|
2018-03-14 17:22:24 +00:00
|
|
|
let new = SectionFrontMatter { sort_by: SortBy::Date, ..SectionFrontMatter::default() };
|
2018-01-29 17:40:12 +00:00
|
|
|
let changes = find_section_front_matter_changes(&SectionFrontMatter::default(), &new);
|
|
|
|
assert_eq!(changes, vec![SectionChangesNeeded::Sort, SectionChangesNeeded::Render]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_find_render_changes_in_section_frontmatter() {
|
2018-03-14 17:22:24 +00:00
|
|
|
let new = SectionFrontMatter { render: false, ..SectionFrontMatter::default() };
|
2018-01-29 17:40:12 +00:00
|
|
|
let changes = find_section_front_matter_changes(&SectionFrontMatter::default(), &new);
|
|
|
|
assert_eq!(changes, vec![SectionChangesNeeded::Delete]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_find_paginate_by_changes_in_section_frontmatter() {
|
|
|
|
let new = SectionFrontMatter { paginate_by: Some(10), ..SectionFrontMatter::default() };
|
|
|
|
let changes = find_section_front_matter_changes(&SectionFrontMatter::default(), &new);
|
|
|
|
assert_eq!(changes, vec![SectionChangesNeeded::RenderWithPages]);
|
|
|
|
}
|
|
|
|
}
|