add treehouse fix
command that generates ulids for you
This commit is contained in:
parent
7e84005a6b
commit
f16a84f3de
12 changed files with 449 additions and 64 deletions
107
Cargo.lock
generated
107
Cargo.lock
generated
|
@ -312,6 +312,12 @@ dependencies = [
|
||||||
"termcolor",
|
"termcolor",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "equivalent"
|
||||||
|
version = "1.0.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "errno"
|
name = "errno"
|
||||||
version = "0.3.2"
|
version = "0.3.2"
|
||||||
|
@ -397,6 +403,17 @@ dependencies = [
|
||||||
"version_check",
|
"version_check",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "getrandom"
|
||||||
|
version = "0.2.10"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "be4136b2a15dd319360be1c07d9933517ccf0be8f16bf62a3bee4f0d618df427"
|
||||||
|
dependencies = [
|
||||||
|
"cfg-if",
|
||||||
|
"libc",
|
||||||
|
"wasi",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "gimli"
|
name = "gimli"
|
||||||
version = "0.27.3"
|
version = "0.27.3"
|
||||||
|
@ -417,6 +434,12 @@ dependencies = [
|
||||||
"thiserror",
|
"thiserror",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "hashbrown"
|
||||||
|
version = "0.14.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "2c6201b9ff9fd90a5a3bac2e56a830d0caa509576f0e503818ee82c181b3437a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "heck"
|
name = "heck"
|
||||||
version = "0.4.1"
|
version = "0.4.1"
|
||||||
|
@ -498,6 +521,16 @@ dependencies = [
|
||||||
"want",
|
"want",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "indexmap"
|
||||||
|
version = "2.0.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d5477fe2230a79769d8dc68e0eabf5437907c0457a5614a9e8dddb67f65eb65d"
|
||||||
|
dependencies = [
|
||||||
|
"equivalent",
|
||||||
|
"hashbrown",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "is-terminal"
|
name = "is-terminal"
|
||||||
version = "0.4.9"
|
version = "0.4.9"
|
||||||
|
@ -721,6 +754,12 @@ version = "0.1.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184"
|
checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "ppv-lite86"
|
||||||
|
version = "0.2.17"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "5b40af805b3121feab8a3c29f04d8ad262fa8e0561883e7653e024ae4479e6de"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "proc-macro2"
|
name = "proc-macro2"
|
||||||
version = "1.0.66"
|
version = "1.0.66"
|
||||||
|
@ -750,6 +789,36 @@ dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rand"
|
||||||
|
version = "0.8.5"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
|
||||||
|
dependencies = [
|
||||||
|
"libc",
|
||||||
|
"rand_chacha",
|
||||||
|
"rand_core",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rand_chacha"
|
||||||
|
version = "0.3.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
|
||||||
|
dependencies = [
|
||||||
|
"ppv-lite86",
|
||||||
|
"rand_core",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rand_core"
|
||||||
|
version = "0.6.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c"
|
||||||
|
dependencies = [
|
||||||
|
"getrandom",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "redox_syscall"
|
name = "redox_syscall"
|
||||||
version = "0.3.5"
|
version = "0.3.5"
|
||||||
|
@ -1028,6 +1097,23 @@ dependencies = [
|
||||||
"tokio",
|
"tokio",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "toml_datetime"
|
||||||
|
version = "0.6.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7cda73e2f1397b1262d6dfdcef8aafae14d1de7748d66822d3bfeeb6d03e5e4b"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "toml_edit"
|
||||||
|
version = "0.19.14"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f8123f27e969974a3dfba720fdb560be359f57b44302d280ba72e76a74480e8a"
|
||||||
|
dependencies = [
|
||||||
|
"indexmap",
|
||||||
|
"toml_datetime",
|
||||||
|
"winnow",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "tower"
|
name = "tower"
|
||||||
version = "0.4.13"
|
version = "0.4.13"
|
||||||
|
@ -1129,11 +1215,14 @@ dependencies = [
|
||||||
"handlebars",
|
"handlebars",
|
||||||
"log",
|
"log",
|
||||||
"pulldown-cmark",
|
"pulldown-cmark",
|
||||||
|
"rand",
|
||||||
"serde",
|
"serde",
|
||||||
"tokio",
|
"tokio",
|
||||||
|
"toml_edit",
|
||||||
"tower-http",
|
"tower-http",
|
||||||
"tower-livereload",
|
"tower-livereload",
|
||||||
"treehouse-format",
|
"treehouse-format",
|
||||||
|
"ulid",
|
||||||
"walkdir",
|
"walkdir",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -1162,6 +1251,15 @@ version = "0.1.6"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "ed646292ffc8188ef8ea4d1e0e0150fb15a5c2e12ad9b8fc191ae7a8a7f3c4b9"
|
checksum = "ed646292ffc8188ef8ea4d1e0e0150fb15a5c2e12ad9b8fc191ae7a8a7f3c4b9"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "ulid"
|
||||||
|
version = "1.0.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "13a3aaa69b04e5b66cc27309710a569ea23593612387d67daaf102e73aa974fd"
|
||||||
|
dependencies = [
|
||||||
|
"rand",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "unicase"
|
name = "unicase"
|
||||||
version = "2.6.0"
|
version = "2.6.0"
|
||||||
|
@ -1316,3 +1414,12 @@ name = "windows_x86_64_msvc"
|
||||||
version = "0.48.3"
|
version = "0.48.3"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "b6ade45bc8bf02ae2aa34a9d54ba660a1a58204da34ba793c00d83ca3730b5f1"
|
checksum = "b6ade45bc8bf02ae2aa34a9d54ba660a1a58204da34ba793c00d83ca3730b5f1"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "winnow"
|
||||||
|
version = "0.5.14"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d09770118a7eb1ccaf4a594a221334119a44a814fcb0d31c5b85e83e97227a97"
|
||||||
|
dependencies = [
|
||||||
|
"memchr",
|
||||||
|
]
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
id = "treehouse"
|
id = "treehouse"
|
||||||
- # treehouse
|
- # treehouse
|
||||||
|
|
||||||
|
% # example
|
||||||
- welcome to the treehouse!
|
- welcome to the treehouse!
|
||||||
|
|
||||||
- treehouse is a brand new static website generator, inspired by the likes of Jekyll and Hugo, but offering a writing experience more close to Logseq
|
- treehouse is a brand new static website generator, inspired by the likes of Jekyll and Hugo, but offering a writing experience more close to Logseq
|
||||||
|
|
|
@ -1 +1,2 @@
|
||||||
|
% id = "01H87RB4MB7RM6V2K1DR8XBV8N"
|
||||||
- He is behind the tree.
|
- He is behind the tree.
|
||||||
|
|
35
content/ulid-test.tree
Normal file
35
content/ulid-test.tree
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
% always_expanded = true
|
||||||
|
id = "treehouse" # this one has a custom id
|
||||||
|
- root
|
||||||
|
|
||||||
|
% id = "01H87R133DNH9JRGS4ZWN586M4"
|
||||||
|
# this one is missing an ULID
|
||||||
|
- child
|
||||||
|
|
||||||
|
% id = "01H87R133D75W92HRESGAXBTC5"
|
||||||
|
|
||||||
|
# this one is formatted a bit weirdly
|
||||||
|
- another child
|
||||||
|
|
||||||
|
% id = "custom-id"
|
||||||
|
- intertwined with one that has a custom ID
|
||||||
|
|
||||||
|
% id = "01H87R133D4MXM01VGTEXHPVAR"
|
||||||
|
|
||||||
|
# this is on a separate line and lacks indentation
|
||||||
|
- and another child
|
||||||
|
|
||||||
|
% always_expanded = true
|
||||||
|
id = "01H87R133D9GDXRD7WA14ECDES"
|
||||||
|
- this one has a custom attribute but not an ID
|
||||||
|
|
||||||
|
% id = "01H87R7PVRJ7FTWKC1SXHGQXDC"
|
||||||
|
- hellow
|
||||||
|
|
||||||
|
% id = "01H87R133DHQPZWJVTMWPKVGP0"
|
||||||
|
- <!-- and that one is plain empty, including the content. or conten't, for that matter. -->
|
||||||
|
|
||||||
|
% id = "01H87R133DRJ7VKEMBSJKHYRJN"
|
||||||
|
|
||||||
|
# comment
|
||||||
|
-
|
|
@ -108,6 +108,7 @@ impl<'a> Parser<'a> {
|
||||||
self.advance();
|
self.advance();
|
||||||
let after_percent = self.position;
|
let after_percent = self.position;
|
||||||
self.eat_indented_lines_until(indent_level, |c| c == '-' || c == '+')?;
|
self.eat_indented_lines_until(indent_level, |c| c == '-' || c == '+')?;
|
||||||
|
self.eat_as_long_as(' ');
|
||||||
let end = self.position;
|
let end = self.position;
|
||||||
Some(Attributes {
|
Some(Attributes {
|
||||||
percent: start..after_percent,
|
percent: start..after_percent,
|
||||||
|
|
|
@ -18,8 +18,11 @@ handlebars = "4.3.7"
|
||||||
pulldown-cmark = { version = "0.9.3", default-features = false }
|
pulldown-cmark = { version = "0.9.3", default-features = false }
|
||||||
serde = { version = "1.0.183", features = ["derive"] }
|
serde = { version = "1.0.183", features = ["derive"] }
|
||||||
tokio = { version = "1.32.0", features = ["full"] }
|
tokio = { version = "1.32.0", features = ["full"] }
|
||||||
|
toml_edit = "0.19.14"
|
||||||
tower-http = { version = "0.4.3", features = ["fs"] }
|
tower-http = { version = "0.4.3", features = ["fs"] }
|
||||||
tower-livereload = "0.8.0"
|
tower-livereload = "0.8.0"
|
||||||
walkdir = "2.3.3"
|
walkdir = "2.3.3"
|
||||||
|
ulid = "1.0.0"
|
||||||
|
rand = "0.8.5"
|
||||||
|
|
||||||
|
|
||||||
|
|
42
crates/treehouse/src/cli/diagnostics.rs
Normal file
42
crates/treehouse/src/cli/diagnostics.rs
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
use anyhow::Context;
|
||||||
|
use codespan_reporting::{
|
||||||
|
diagnostic::Diagnostic,
|
||||||
|
files::SimpleFiles,
|
||||||
|
term::termcolor::{ColorChoice, StandardStream},
|
||||||
|
};
|
||||||
|
|
||||||
|
pub type Files = SimpleFiles<String, String>;
|
||||||
|
pub type FileId = <Files as codespan_reporting::files::Files<'static>>::FileId;
|
||||||
|
|
||||||
|
pub struct Diagnosis {
|
||||||
|
pub files: Files,
|
||||||
|
pub diagnostics: Vec<Diagnostic<FileId>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Diagnosis {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
files: Files::new(),
|
||||||
|
diagnostics: vec![],
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get the source code of a file, assuming it was previously registered.
|
||||||
|
pub fn get_source(&self, file_id: FileId) -> &str {
|
||||||
|
self.files
|
||||||
|
.get(file_id)
|
||||||
|
.expect("file should have been registered previously")
|
||||||
|
.source()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn report(&self) -> anyhow::Result<()> {
|
||||||
|
let writer = StandardStream::stderr(ColorChoice::Auto);
|
||||||
|
let config = codespan_reporting::term::Config::default();
|
||||||
|
for diagnostic in &self.diagnostics {
|
||||||
|
codespan_reporting::term::emit(&mut writer.lock(), &config, &self.files, diagnostic)
|
||||||
|
.context("could not emit diagnostic")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
150
crates/treehouse/src/cli/fix.rs
Normal file
150
crates/treehouse/src/cli/fix.rs
Normal file
|
@ -0,0 +1,150 @@
|
||||||
|
use std::ops::Range;
|
||||||
|
|
||||||
|
use anyhow::Context;
|
||||||
|
use treehouse_format::ast::Branch;
|
||||||
|
|
||||||
|
use super::{
|
||||||
|
diagnostics::{Diagnosis, FileId},
|
||||||
|
parse::{self, parse_toml_with_diagnostics, parse_tree_with_diagnostics},
|
||||||
|
FixArgs,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Fix {
|
||||||
|
range: Range<usize>,
|
||||||
|
replacement: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
struct State {
|
||||||
|
fixes: Vec<Fix>,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn dfs_fix_branch(diagnosis: &mut Diagnosis, file_id: FileId, state: &mut State, branch: &Branch) {
|
||||||
|
let mut rng = rand::thread_rng();
|
||||||
|
let ulid = ulid::Generator::new()
|
||||||
|
.generate_with_source(&mut rng)
|
||||||
|
.expect("failed to generate ulid for block"); // (wtf moment. you know how big the 80-bit combination space is?)
|
||||||
|
|
||||||
|
let indent = " ".repeat(branch.indent_level);
|
||||||
|
if let Some(attributes) = branch.attributes.clone() {
|
||||||
|
// Scenario: Attributes need to be parsed as TOML and the id attribute has to be added into
|
||||||
|
// the top-level table. Then we also need to pretty-print everything to match the right
|
||||||
|
// indentation level.
|
||||||
|
if let Ok(mut toml) =
|
||||||
|
parse_toml_with_diagnostics(diagnosis, file_id, attributes.data.clone())
|
||||||
|
{
|
||||||
|
if !toml.contains_key("id") {
|
||||||
|
toml["id"] = toml_edit::value(ulid.to_string());
|
||||||
|
toml.key_decor_mut("id")
|
||||||
|
.unwrap()
|
||||||
|
.set_prefix(" ".repeat(branch.indent_level + 2));
|
||||||
|
}
|
||||||
|
let mut toml_string = toml.to_string();
|
||||||
|
|
||||||
|
// This is incredibly janky and barely works.
|
||||||
|
let leading_spaces: usize = toml_string.chars().take_while(|&c| c == ' ').count();
|
||||||
|
match leading_spaces {
|
||||||
|
0 => toml_string.insert(0, ' '),
|
||||||
|
1 => (),
|
||||||
|
_ => toml_string.replace_range(0..leading_spaces - 1, ""),
|
||||||
|
}
|
||||||
|
|
||||||
|
let toml_string = fix_indent_in_generated_toml(&toml_string, branch.indent_level);
|
||||||
|
|
||||||
|
state.fixes.push(Fix {
|
||||||
|
range: attributes.data.clone(),
|
||||||
|
replacement: toml_string,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Scenario: No attributes at all.
|
||||||
|
// In this case we can do a fast path where we generate the `% id = "whatever"` string
|
||||||
|
// directly, not going through toml_edit.
|
||||||
|
state.fixes.push(Fix {
|
||||||
|
range: branch.kind_span.start..branch.kind_span.start,
|
||||||
|
replacement: format!("% id = \"{ulid}\"\n{indent}"),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Then we fix child branches.
|
||||||
|
for child in &branch.children {
|
||||||
|
dfs_fix_branch(diagnosis, file_id, state, child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fix_indent_in_generated_toml(toml: &str, min_indent_level: usize) -> String {
|
||||||
|
let toml = toml.trim_end();
|
||||||
|
|
||||||
|
let mut result = String::with_capacity(toml.len());
|
||||||
|
|
||||||
|
for (i, line) in toml.lines().enumerate() {
|
||||||
|
if line.is_empty() {
|
||||||
|
result.push('\n');
|
||||||
|
} else {
|
||||||
|
let desired_line_indent_level = if i == 0 { 1 } else { min_indent_level + 2 };
|
||||||
|
let leading_spaces: usize = line.chars().take_while(|&c| c == ' ').count();
|
||||||
|
let needed_indentation = desired_line_indent_level.saturating_sub(leading_spaces);
|
||||||
|
for _ in 0..needed_indentation {
|
||||||
|
result.push(' ');
|
||||||
|
}
|
||||||
|
result.push_str(line);
|
||||||
|
result.push('\n');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _ in 0..min_indent_level {
|
||||||
|
result.push(' ');
|
||||||
|
}
|
||||||
|
|
||||||
|
result
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fix_file(
|
||||||
|
diagnosis: &mut Diagnosis,
|
||||||
|
file_id: FileId,
|
||||||
|
) -> Result<String, parse::ErrorsEmitted> {
|
||||||
|
parse_tree_with_diagnostics(diagnosis, file_id).map(|roots| {
|
||||||
|
let mut source = diagnosis.get_source(file_id).to_owned();
|
||||||
|
let mut state = State::default();
|
||||||
|
|
||||||
|
for branch in &roots.branches {
|
||||||
|
dfs_fix_branch(diagnosis, file_id, &mut state, branch);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Doing a depth-first search of the branches yields fixes from the beginning of the file
|
||||||
|
// to its end. The most efficient way to apply all the fixes then is to reverse their order,
|
||||||
|
// which lets us modify the source string in place because the fix ranges always stay
|
||||||
|
// correct.
|
||||||
|
for fix in state.fixes.iter().rev() {
|
||||||
|
source.replace_range(fix.range.clone(), &fix.replacement);
|
||||||
|
}
|
||||||
|
|
||||||
|
source
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fix_file_cli(fix_args: FixArgs) -> anyhow::Result<()> {
|
||||||
|
let utf8_filename = fix_args.file.to_string_lossy().into_owned();
|
||||||
|
let file = std::fs::read_to_string(&fix_args.file).context("cannot read file to fix")?;
|
||||||
|
|
||||||
|
let mut diagnosis = Diagnosis::new();
|
||||||
|
let file_id = diagnosis.files.add(utf8_filename, file);
|
||||||
|
|
||||||
|
if let Ok(fixed) = fix_file(&mut diagnosis, file_id) {
|
||||||
|
if fix_args.apply {
|
||||||
|
// Try to write the backup first. If writing that fails, bail out without overwriting
|
||||||
|
// the source file.
|
||||||
|
if let Some(backup_path) = fix_args.backup {
|
||||||
|
std::fs::write(backup_path, diagnosis.get_source(file_id))
|
||||||
|
.context("cannot write backup; original file will not be overwritten")?;
|
||||||
|
}
|
||||||
|
std::fs::write(&fix_args.file, fixed).context("cannot overwrite original file")?;
|
||||||
|
} else {
|
||||||
|
println!("{fixed}");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
diagnosis.report()?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
|
@ -1,5 +1,10 @@
|
||||||
|
pub mod diagnostics;
|
||||||
|
pub mod fix;
|
||||||
|
mod parse;
|
||||||
pub mod regenerate;
|
pub mod regenerate;
|
||||||
|
|
||||||
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use clap::{Args, Parser, Subcommand};
|
use clap::{Args, Parser, Subcommand};
|
||||||
|
|
||||||
#[derive(Parser)]
|
#[derive(Parser)]
|
||||||
|
@ -12,6 +17,9 @@ pub struct ProgramArgs {
|
||||||
pub enum Command {
|
pub enum Command {
|
||||||
/// Regenerate the website.
|
/// Regenerate the website.
|
||||||
Regenerate(#[clap(flatten)] RegenerateArgs),
|
Regenerate(#[clap(flatten)] RegenerateArgs),
|
||||||
|
|
||||||
|
/// Populate missing metadata in blocks.
|
||||||
|
Fix(#[clap(flatten)] FixArgs),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Args)]
|
#[derive(Args)]
|
||||||
|
@ -20,3 +28,19 @@ pub struct RegenerateArgs {
|
||||||
#[clap(short, long)]
|
#[clap(short, long)]
|
||||||
pub serve: bool,
|
pub serve: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Args)]
|
||||||
|
pub struct FixArgs {
|
||||||
|
/// Which file to fix. The fixed file will be printed into stdout so that you have a chance to
|
||||||
|
/// see the changes.
|
||||||
|
pub file: PathBuf,
|
||||||
|
|
||||||
|
/// If you're happy with the suggested changes, specifying this will apply them to the file
|
||||||
|
/// (overwrite it in place.)
|
||||||
|
#[clap(long)]
|
||||||
|
pub apply: bool,
|
||||||
|
|
||||||
|
/// Write the previous version back to the specified path.
|
||||||
|
#[clap(long)]
|
||||||
|
pub backup: Option<PathBuf>,
|
||||||
|
}
|
||||||
|
|
57
crates/treehouse/src/cli/parse.rs
Normal file
57
crates/treehouse/src/cli/parse.rs
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
use std::{ops::Range, str::FromStr};
|
||||||
|
|
||||||
|
use codespan_reporting::diagnostic::{Diagnostic, Label, LabelStyle, Severity};
|
||||||
|
use treehouse_format::ast::Roots;
|
||||||
|
|
||||||
|
use super::diagnostics::{Diagnosis, FileId};
|
||||||
|
|
||||||
|
pub struct ErrorsEmitted;
|
||||||
|
|
||||||
|
pub fn parse_tree_with_diagnostics(
|
||||||
|
diagnosis: &mut Diagnosis,
|
||||||
|
file_id: FileId,
|
||||||
|
) -> Result<Roots, ErrorsEmitted> {
|
||||||
|
let input = diagnosis.get_source(file_id);
|
||||||
|
Roots::parse(&mut treehouse_format::pull::Parser { input, position: 0 }).map_err(|error| {
|
||||||
|
diagnosis.diagnostics.push(Diagnostic {
|
||||||
|
severity: Severity::Error,
|
||||||
|
code: Some("tree".into()),
|
||||||
|
message: error.kind.to_string(),
|
||||||
|
labels: vec![Label {
|
||||||
|
style: LabelStyle::Primary,
|
||||||
|
file_id,
|
||||||
|
range: error.range,
|
||||||
|
message: String::new(),
|
||||||
|
}],
|
||||||
|
notes: vec![],
|
||||||
|
});
|
||||||
|
ErrorsEmitted
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn parse_toml_with_diagnostics(
|
||||||
|
diagnosis: &mut Diagnosis,
|
||||||
|
file_id: FileId,
|
||||||
|
range: Range<usize>,
|
||||||
|
) -> Result<toml_edit::Document, ErrorsEmitted> {
|
||||||
|
let input = &diagnosis.get_source(file_id)[range.clone()];
|
||||||
|
toml_edit::Document::from_str(input).map_err(|error| {
|
||||||
|
diagnosis.diagnostics.push(Diagnostic {
|
||||||
|
severity: Severity::Error,
|
||||||
|
code: Some("toml".into()),
|
||||||
|
message: error.message().to_owned(),
|
||||||
|
labels: error
|
||||||
|
.span()
|
||||||
|
.map(|span| Label {
|
||||||
|
style: LabelStyle::Primary,
|
||||||
|
file_id,
|
||||||
|
range: range.start + span.start..range.start + span.end,
|
||||||
|
message: String::new(),
|
||||||
|
})
|
||||||
|
.into_iter()
|
||||||
|
.collect(),
|
||||||
|
notes: vec![],
|
||||||
|
});
|
||||||
|
ErrorsEmitted
|
||||||
|
})
|
||||||
|
}
|
|
@ -7,8 +7,7 @@ use anyhow::{bail, Context};
|
||||||
use axum::Router;
|
use axum::Router;
|
||||||
use codespan_reporting::{
|
use codespan_reporting::{
|
||||||
diagnostic::{Diagnostic, Label, LabelStyle, Severity},
|
diagnostic::{Diagnostic, Label, LabelStyle, Severity},
|
||||||
files::{Files as _, SimpleFiles},
|
files::Files as _,
|
||||||
term::termcolor::{ColorChoice, StandardStream},
|
|
||||||
};
|
};
|
||||||
use copy_dir::copy_dir;
|
use copy_dir::copy_dir;
|
||||||
use handlebars::Handlebars;
|
use handlebars::Handlebars;
|
||||||
|
@ -16,24 +15,17 @@ use log::{debug, info};
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use tower_http::services::ServeDir;
|
use tower_http::services::ServeDir;
|
||||||
use tower_livereload::LiveReloadLayer;
|
use tower_livereload::LiveReloadLayer;
|
||||||
use treehouse_format::ast::Roots;
|
|
||||||
use walkdir::WalkDir;
|
use walkdir::WalkDir;
|
||||||
|
|
||||||
use crate::html::tree::branches_to_html;
|
use crate::{cli::parse::parse_tree_with_diagnostics, html::tree::branches_to_html};
|
||||||
|
|
||||||
|
use super::diagnostics::{Diagnosis, FileId};
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
struct Generator {
|
struct Generator {
|
||||||
tree_files: Vec<PathBuf>,
|
tree_files: Vec<PathBuf>,
|
||||||
}
|
}
|
||||||
|
|
||||||
type Files = SimpleFiles<String, String>;
|
|
||||||
type FileId = <Files as codespan_reporting::files::Files<'static>>::FileId;
|
|
||||||
|
|
||||||
pub struct Diagnosis {
|
|
||||||
pub files: Files,
|
|
||||||
pub diagnostics: Vec<Diagnostic<FileId>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Generator {
|
impl Generator {
|
||||||
fn add_directory_rec(&mut self, directory: &Path) -> anyhow::Result<()> {
|
fn add_directory_rec(&mut self, directory: &Path) -> anyhow::Result<()> {
|
||||||
for entry in WalkDir::new(directory) {
|
for entry in WalkDir::new(directory) {
|
||||||
|
@ -108,10 +100,7 @@ impl Generator {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn generate_all_files(&self, dirs: &Dirs<'_>) -> anyhow::Result<Diagnosis> {
|
fn generate_all_files(&self, dirs: &Dirs<'_>) -> anyhow::Result<Diagnosis> {
|
||||||
let mut diagnosis = Diagnosis {
|
let mut diagnosis = Diagnosis::new();
|
||||||
files: Files::new(),
|
|
||||||
diagnostics: vec![],
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut handlebars = Handlebars::new();
|
let mut handlebars = Handlebars::new();
|
||||||
let tree_template = Self::register_template(
|
let tree_template = Self::register_template(
|
||||||
|
@ -145,51 +134,28 @@ impl Generator {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
let file_id = diagnosis.files.add(utf8_filename.into_owned(), source);
|
let file_id = diagnosis.files.add(utf8_filename.into_owned(), source);
|
||||||
let source = diagnosis
|
|
||||||
.files
|
|
||||||
.get(file_id)
|
|
||||||
.expect("file was just added to the list")
|
|
||||||
.source();
|
|
||||||
|
|
||||||
let parse_result = Roots::parse(&mut treehouse_format::pull::Parser {
|
if let Ok(roots) = parse_tree_with_diagnostics(&mut diagnosis, file_id) {
|
||||||
input: source,
|
let mut tree = String::new();
|
||||||
position: 0,
|
let source = diagnosis.get_source(file_id);
|
||||||
});
|
branches_to_html(&mut tree, &roots.branches, source);
|
||||||
|
|
||||||
match parse_result {
|
let template_data = TemplateData { tree };
|
||||||
Ok(roots) => {
|
let templated_html = match handlebars.render("tree", &template_data) {
|
||||||
let mut tree = String::new();
|
Ok(html) => html,
|
||||||
branches_to_html(&mut tree, &roots.branches, source);
|
Err(error) => {
|
||||||
|
Self::wrangle_handlebars_error_into_diagnostic(
|
||||||
|
&mut diagnosis,
|
||||||
|
tree_template,
|
||||||
|
error.line_no,
|
||||||
|
error.column_no,
|
||||||
|
error.desc,
|
||||||
|
)?;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let template_data = TemplateData { tree };
|
std::fs::write(target_path, templated_html)?;
|
||||||
let templated_html = match handlebars.render("tree", &template_data) {
|
|
||||||
Ok(html) => html,
|
|
||||||
Err(error) => {
|
|
||||||
Self::wrangle_handlebars_error_into_diagnostic(
|
|
||||||
&mut diagnosis,
|
|
||||||
tree_template,
|
|
||||||
error.line_no,
|
|
||||||
error.column_no,
|
|
||||||
error.desc,
|
|
||||||
)?;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
std::fs::write(target_path, templated_html)?;
|
|
||||||
}
|
|
||||||
Err(error) => diagnosis.diagnostics.push(Diagnostic {
|
|
||||||
severity: Severity::Error,
|
|
||||||
code: Some("tree".into()),
|
|
||||||
message: error.kind.to_string(),
|
|
||||||
labels: vec![Label {
|
|
||||||
style: LabelStyle::Primary,
|
|
||||||
file_id,
|
|
||||||
range: error.range,
|
|
||||||
message: String::new(),
|
|
||||||
}],
|
|
||||||
notes: vec![],
|
|
||||||
}),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -223,12 +189,7 @@ pub fn regenerate(dirs: &Dirs<'_>) -> anyhow::Result<()> {
|
||||||
generator.add_directory_rec(dirs.content_dir)?;
|
generator.add_directory_rec(dirs.content_dir)?;
|
||||||
let diagnosis = generator.generate_all_files(dirs)?;
|
let diagnosis = generator.generate_all_files(dirs)?;
|
||||||
|
|
||||||
let writer = StandardStream::stderr(ColorChoice::Auto);
|
diagnosis.report()?;
|
||||||
let config = codespan_reporting::term::Config::default();
|
|
||||||
for diagnostic in &diagnosis.diagnostics {
|
|
||||||
codespan_reporting::term::emit(&mut writer.lock(), &config, &diagnosis.files, diagnostic)
|
|
||||||
.context("could not emit diagnostic")?;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@ use std::path::Path;
|
||||||
|
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
use cli::{
|
use cli::{
|
||||||
|
fix::fix_file_cli,
|
||||||
regenerate::{self, regenerate_or_report_error, Dirs},
|
regenerate::{self, regenerate_or_report_error, Dirs},
|
||||||
Command, ProgramArgs,
|
Command, ProgramArgs,
|
||||||
};
|
};
|
||||||
|
@ -32,6 +33,8 @@ async fn fallible_main() -> anyhow::Result<()> {
|
||||||
regenerate::web_server().await?;
|
regenerate::web_server().await?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Command::Fix(fix_args) => fix_file_cli(fix_args)?,
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|
Loading…
Reference in a new issue