This introduces `relative-path`, a crate I've written for the specific purpose of providing platform-neutral operations over paths the same way they are used in URLs. This means that `///hello///` == `/hello`, which should do the same as the existing stripping minus the platform-specific path separators causing the [bug being referenced](#1169).
This commit is contained in:
parent
c27f749a86
commit
f9ae897190
8
Cargo.lock
generated
8
Cargo.lock
generated
@ -1900,6 +1900,12 @@ version = "0.6.18"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "26412eb97c6b088a6997e05f69403a802a92d520de2f8e63c2b65f9e0f47c4e8"
|
checksum = "26412eb97c6b088a6997e05f69403a802a92d520de2f8e63c2b65f9e0f47c4e8"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "relative-path"
|
||||||
|
version = "1.3.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "65aff7c83039e88c1c0b4bedf8dfa93d6ec84d5fc2945b37c1fa4186f46c5f94"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "remove_dir_all"
|
name = "remove_dir_all"
|
||||||
version = "0.5.3"
|
version = "0.5.3"
|
||||||
@ -2179,6 +2185,7 @@ dependencies = [
|
|||||||
"link_checker",
|
"link_checker",
|
||||||
"minify-html",
|
"minify-html",
|
||||||
"rayon",
|
"rayon",
|
||||||
|
"relative-path",
|
||||||
"sass-rs",
|
"sass-rs",
|
||||||
"search",
|
"search",
|
||||||
"serde",
|
"serde",
|
||||||
@ -2981,6 +2988,7 @@ dependencies = [
|
|||||||
"lazy_static",
|
"lazy_static",
|
||||||
"notify",
|
"notify",
|
||||||
"open",
|
"open",
|
||||||
|
"relative-path",
|
||||||
"site",
|
"site",
|
||||||
"termcolor",
|
"termcolor",
|
||||||
"tokio",
|
"tokio",
|
||||||
|
@ -35,6 +35,7 @@ ws = "0.9"
|
|||||||
ctrlc = "3"
|
ctrlc = "3"
|
||||||
open = "1.2"
|
open = "1.2"
|
||||||
globset = "0.4"
|
globset = "0.4"
|
||||||
|
relative-path = "1"
|
||||||
|
|
||||||
site = { path = "components/site" }
|
site = { path = "components/site" }
|
||||||
errors = { path = "components/errors" }
|
errors = { path = "components/errors" }
|
||||||
|
@ -14,6 +14,7 @@ serde = "1"
|
|||||||
serde_derive = "1"
|
serde_derive = "1"
|
||||||
sass-rs = "0.2"
|
sass-rs = "0.2"
|
||||||
lazy_static = "1.1"
|
lazy_static = "1.1"
|
||||||
|
relative-path = "1"
|
||||||
|
|
||||||
errors = { path = "../errors" }
|
errors = { path = "../errors" }
|
||||||
config = { path = "../config" }
|
config = { path = "../config" }
|
||||||
|
@ -19,6 +19,7 @@ use config::{get_config, Config};
|
|||||||
use errors::{bail, Error, Result};
|
use errors::{bail, Error, Result};
|
||||||
use front_matter::InsertAnchor;
|
use front_matter::InsertAnchor;
|
||||||
use library::{find_taxonomies, Library, Page, Paginator, Section, Taxonomy};
|
use library::{find_taxonomies, Library, Page, Paginator, Section, Taxonomy};
|
||||||
|
use relative_path::RelativePathBuf;
|
||||||
use templates::render_redirect_template;
|
use templates::render_redirect_template;
|
||||||
use utils::fs::{
|
use utils::fs::{
|
||||||
copy_directory, copy_file_if_needed, create_directory, create_file, ensure_directory_exists,
|
copy_directory, copy_file_if_needed, create_directory, create_file, ensure_directory_exists,
|
||||||
@ -28,7 +29,7 @@ use utils::templates::render_template;
|
|||||||
|
|
||||||
lazy_static! {
|
lazy_static! {
|
||||||
/// The in-memory rendered map content
|
/// The in-memory rendered map content
|
||||||
pub static ref SITE_CONTENT: Arc<RwLock<HashMap<String, String>>> = Arc::new(RwLock::new(HashMap::new()));
|
pub static ref SITE_CONTENT: Arc<RwLock<HashMap<RelativePathBuf, String>>> = Arc::new(RwLock::new(HashMap::new()));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Where are we building the site
|
/// Where are we building the site
|
||||||
@ -513,10 +514,12 @@ impl Site {
|
|||||||
let write_dirs = self.build_mode == BuildMode::Disk || create_dirs;
|
let write_dirs = self.build_mode == BuildMode::Disk || create_dirs;
|
||||||
ensure_directory_exists(&self.output_path)?;
|
ensure_directory_exists(&self.output_path)?;
|
||||||
|
|
||||||
|
let mut site_path = RelativePathBuf::new();
|
||||||
let mut current_path = self.output_path.to_path_buf();
|
let mut current_path = self.output_path.to_path_buf();
|
||||||
|
|
||||||
for component in components {
|
for component in components {
|
||||||
current_path.push(component);
|
current_path.push(component);
|
||||||
|
site_path.push(component);
|
||||||
|
|
||||||
if !current_path.exists() && write_dirs {
|
if !current_path.exists() && write_dirs {
|
||||||
create_directory(¤t_path)?;
|
create_directory(¤t_path)?;
|
||||||
@ -542,17 +545,10 @@ impl Site {
|
|||||||
create_file(&end_path, &final_content)?;
|
create_file(&end_path, &final_content)?;
|
||||||
}
|
}
|
||||||
BuildMode::Memory => {
|
BuildMode::Memory => {
|
||||||
let path = if filename != "index.html" {
|
let site_path =
|
||||||
let p = current_path.join(filename);
|
if filename != "index.html" { site_path.join(filename) } else { site_path };
|
||||||
p.as_os_str().to_string_lossy().replace("public/", "/")
|
|
||||||
} else {
|
&SITE_CONTENT.write().unwrap().insert(site_path, final_content);
|
||||||
// TODO" remove unwrap
|
|
||||||
let p = current_path.strip_prefix(&self.output_path).unwrap();
|
|
||||||
p.as_os_str().to_string_lossy().into_owned()
|
|
||||||
}
|
|
||||||
.trim_end_matches('/')
|
|
||||||
.to_owned();
|
|
||||||
&SITE_CONTENT.write().unwrap().insert(path, final_content);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -568,12 +564,8 @@ impl Site {
|
|||||||
let output = page.render_html(&self.tera, &self.config, &self.library.read().unwrap())?;
|
let output = page.render_html(&self.tera, &self.config, &self.library.read().unwrap())?;
|
||||||
let content = self.inject_livereload(output);
|
let content = self.inject_livereload(output);
|
||||||
let components: Vec<&str> = page.path.split('/').collect();
|
let components: Vec<&str> = page.path.split('/').collect();
|
||||||
let current_path = self.write_content(
|
let current_path =
|
||||||
&components,
|
self.write_content(&components, "index.html", content, !page.assets.is_empty())?;
|
||||||
"index.html",
|
|
||||||
content,
|
|
||||||
!page.assets.is_empty(),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
// Copy any asset we found previously into the same directory as the index.html
|
// Copy any asset we found previously into the same directory as the index.html
|
||||||
for asset in &page.assets {
|
for asset in &page.assets {
|
||||||
@ -932,11 +924,7 @@ impl Site {
|
|||||||
|
|
||||||
if section.meta.generate_feed {
|
if section.meta.generate_feed {
|
||||||
let library = &self.library.read().unwrap();
|
let library = &self.library.read().unwrap();
|
||||||
let pages = section
|
let pages = section.pages.iter().map(|k| library.get_page_by_key(*k)).collect();
|
||||||
.pages
|
|
||||||
.iter()
|
|
||||||
.map(|k| library.get_page_by_key(*k))
|
|
||||||
.collect();
|
|
||||||
self.render_feed(
|
self.render_feed(
|
||||||
pages,
|
pages,
|
||||||
Some(&PathBuf::from(§ion.path[1..])),
|
Some(&PathBuf::from(§ion.path[1..])),
|
||||||
|
@ -38,6 +38,7 @@ use ws::{Message, Sender, WebSocket};
|
|||||||
|
|
||||||
use errors::{Error as ZolaError, Result};
|
use errors::{Error as ZolaError, Result};
|
||||||
use globset::GlobSet;
|
use globset::GlobSet;
|
||||||
|
use relative_path::{RelativePath, RelativePathBuf};
|
||||||
use site::sass::compile_sass;
|
use site::sass::compile_sass;
|
||||||
use site::{Site, SITE_CONTENT};
|
use site::{Site, SITE_CONTENT};
|
||||||
use utils::fs::copy_file;
|
use utils::fs::copy_file;
|
||||||
@ -69,7 +70,12 @@ static NOT_FOUND_TEXT: &[u8] = b"Not Found";
|
|||||||
const LIVE_RELOAD: &str = include_str!("livereload.js");
|
const LIVE_RELOAD: &str = include_str!("livereload.js");
|
||||||
|
|
||||||
async fn handle_request(req: Request<Body>, root: PathBuf) -> Result<Response<Body>> {
|
async fn handle_request(req: Request<Body>, root: PathBuf) -> Result<Response<Body>> {
|
||||||
let path = req.uri().path().trim_end_matches('/').trim_start_matches('/');
|
let mut path = RelativePathBuf::new();
|
||||||
|
|
||||||
|
for c in req.uri().path().split('/') {
|
||||||
|
path.push(c);
|
||||||
|
}
|
||||||
|
|
||||||
// livereload.js is served using the LIVE_RELOAD str, not a file
|
// livereload.js is served using the LIVE_RELOAD str, not a file
|
||||||
if path == "livereload.js" {
|
if path == "livereload.js" {
|
||||||
if req.method() == Method::GET {
|
if req.method() == Method::GET {
|
||||||
@ -79,7 +85,7 @@ async fn handle_request(req: Request<Body>, root: PathBuf) -> Result<Response<Bo
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(content) = SITE_CONTENT.read().unwrap().get(path) {
|
if let Some(content) = SITE_CONTENT.read().unwrap().get(&path) {
|
||||||
return Ok(in_memory_html(content));
|
return Ok(in_memory_html(content));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -87,7 +93,8 @@ async fn handle_request(req: Request<Body>, root: PathBuf) -> Result<Response<Bo
|
|||||||
match result {
|
match result {
|
||||||
ResolveResult::MethodNotMatched => return Ok(method_not_allowed()),
|
ResolveResult::MethodNotMatched => return Ok(method_not_allowed()),
|
||||||
ResolveResult::NotFound | ResolveResult::UriNotMatched => {
|
ResolveResult::NotFound | ResolveResult::UriNotMatched => {
|
||||||
let content_404 = SITE_CONTENT.read().unwrap().get("404.html").map(|x| x.clone());
|
let not_found_path = RelativePath::new("404.html");
|
||||||
|
let content_404 = SITE_CONTENT.read().unwrap().get(not_found_path).map(|x| x.clone());
|
||||||
return Ok(not_found(content_404));
|
return Ok(not_found(content_404));
|
||||||
}
|
}
|
||||||
_ => (),
|
_ => (),
|
||||||
|
Loading…
Reference in New Issue
Block a user