implement custom links
This commit is contained in:
parent
e861a31a21
commit
1225f6d313
7 changed files with 286 additions and 111 deletions
|
@ -17,7 +17,9 @@ use tower_http::services::ServeDir;
|
||||||
use tower_livereload::LiveReloadLayer;
|
use tower_livereload::LiveReloadLayer;
|
||||||
use walkdir::WalkDir;
|
use walkdir::WalkDir;
|
||||||
|
|
||||||
use crate::{cli::parse::parse_tree_with_diagnostics, html::tree::branches_to_html};
|
use crate::{
|
||||||
|
cli::parse::parse_tree_with_diagnostics, html::tree::branches_to_html, tree::SemaRoots,
|
||||||
|
};
|
||||||
|
|
||||||
use crate::state::{FileId, Treehouse};
|
use crate::state::{FileId, Treehouse};
|
||||||
|
|
||||||
|
@ -132,6 +134,8 @@ impl Generator {
|
||||||
);
|
);
|
||||||
|
|
||||||
if let Ok(roots) = parse_tree_with_diagnostics(&mut treehouse, file_id) {
|
if let Ok(roots) = parse_tree_with_diagnostics(&mut treehouse, file_id) {
|
||||||
|
let roots = SemaRoots::from_roots(&mut treehouse, file_id, roots);
|
||||||
|
|
||||||
let mut tree = String::new();
|
let mut tree = String::new();
|
||||||
branches_to_html(&mut tree, &mut treehouse, file_id, &roots.branches);
|
branches_to_html(&mut tree, &mut treehouse, file_id, &roots.branches);
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
use std::fmt::{self, Display, Write};
|
use std::fmt::{self, Display, Write};
|
||||||
|
|
||||||
pub mod attributes;
|
|
||||||
mod markdown;
|
mod markdown;
|
||||||
pub mod tree;
|
pub mod tree;
|
||||||
|
|
||||||
|
|
|
@ -1,43 +1,37 @@
|
||||||
use std::fmt::Write;
|
use std::fmt::Write;
|
||||||
|
|
||||||
use codespan_reporting::diagnostic::{Diagnostic, Label, LabelStyle, Severity};
|
use pulldown_cmark::{BrokenLink, LinkType};
|
||||||
use treehouse_format::{ast::Branch, pull::BranchKind};
|
use treehouse_format::pull::BranchKind;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
html::EscapeAttribute,
|
html::EscapeAttribute,
|
||||||
state::{toml_error_to_diagnostic, FileId, TomlError, Treehouse},
|
state::{FileId, Treehouse},
|
||||||
|
tree::{attributes::Content, SemaBranchId},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{
|
use super::{markdown, EscapeHtml};
|
||||||
attributes::{Attributes, Content},
|
|
||||||
markdown, EscapeHtml,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId, branch: &Branch) {
|
|
||||||
let attributes = parse_attributes(treehouse, file_id, branch);
|
|
||||||
|
|
||||||
|
pub fn branch_to_html(
|
||||||
|
s: &mut String,
|
||||||
|
treehouse: &mut Treehouse,
|
||||||
|
file_id: FileId,
|
||||||
|
branch_id: SemaBranchId,
|
||||||
|
) {
|
||||||
// Reborrow because the closure requires unique access (it adds a new diagnostic.)
|
// Reborrow because the closure requires unique access (it adds a new diagnostic.)
|
||||||
let source = treehouse.source(file_id);
|
let source = treehouse.source(file_id);
|
||||||
|
let branch = treehouse.tree.branch(branch_id);
|
||||||
|
|
||||||
let has_children =
|
let has_children =
|
||||||
!branch.children.is_empty() || matches!(attributes.content, Content::Link(_));
|
!branch.children.is_empty() || matches!(branch.attributes.content, Content::Link(_));
|
||||||
|
|
||||||
let id = format!(
|
|
||||||
"{}:{}",
|
|
||||||
treehouse
|
|
||||||
.tree_path(file_id)
|
|
||||||
.expect("file should have a tree path"),
|
|
||||||
attributes.id
|
|
||||||
);
|
|
||||||
|
|
||||||
let class = if has_children { "branch" } else { "leaf" };
|
let class = if has_children { "branch" } else { "leaf" };
|
||||||
let component = if let Content::Link(_) = attributes.content {
|
let component = if let Content::Link(_) = branch.attributes.content {
|
||||||
"th-b-linked"
|
"th-b-linked"
|
||||||
} else {
|
} else {
|
||||||
"th-b"
|
"th-b"
|
||||||
};
|
};
|
||||||
|
|
||||||
let linked_branch = if let Content::Link(link) = &attributes.content {
|
let linked_branch = if let Content::Link(link) = &branch.attributes.content {
|
||||||
format!(" data-th-link=\"{}\"", EscapeHtml(link))
|
format!(" data-th-link=\"{}\"", EscapeHtml(link))
|
||||||
} else {
|
} else {
|
||||||
String::new()
|
String::new()
|
||||||
|
@ -46,7 +40,7 @@ pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId
|
||||||
write!(
|
write!(
|
||||||
s,
|
s,
|
||||||
"<li is=\"{component}\" class=\"{class}\" id=\"{}\"{linked_branch}>",
|
"<li is=\"{component}\" class=\"{class}\" id=\"{}\"{linked_branch}>",
|
||||||
EscapeAttribute(&id)
|
EscapeAttribute(&branch.html_id)
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
{
|
{
|
||||||
|
@ -77,13 +71,38 @@ pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId
|
||||||
unindented_block_content.push('\n');
|
unindented_block_content.push('\n');
|
||||||
}
|
}
|
||||||
|
|
||||||
let markdown_parser = pulldown_cmark::Parser::new_ext(&unindented_block_content, {
|
let broken_link_callback = &mut |broken_link: BrokenLink<'_>| {
|
||||||
use pulldown_cmark::Options;
|
if let LinkType::Reference | LinkType::Shortcut = broken_link.link_type {
|
||||||
Options::ENABLE_STRIKETHROUGH | Options::ENABLE_TABLES
|
broken_link
|
||||||
});
|
.reference
|
||||||
|
.split_once(':')
|
||||||
|
.and_then(|(kind, linked)| match kind {
|
||||||
|
"branch" => treehouse
|
||||||
|
.branches_by_named_id
|
||||||
|
.get(linked)
|
||||||
|
.map(|&branch_id| {
|
||||||
|
(
|
||||||
|
format!("#{}", treehouse.tree.branch(branch_id).html_id).into(),
|
||||||
|
"".into(),
|
||||||
|
)
|
||||||
|
}),
|
||||||
|
_ => None,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let markdown_parser = pulldown_cmark::Parser::new_with_broken_link_callback(
|
||||||
|
&unindented_block_content,
|
||||||
|
{
|
||||||
|
use pulldown_cmark::Options;
|
||||||
|
Options::ENABLE_STRIKETHROUGH | Options::ENABLE_TABLES
|
||||||
|
},
|
||||||
|
Some(broken_link_callback),
|
||||||
|
);
|
||||||
markdown::push_html(s, markdown_parser);
|
markdown::push_html(s, markdown_parser);
|
||||||
|
|
||||||
if let Content::Link(link) = &attributes.content {
|
if let Content::Link(link) = &branch.attributes.content {
|
||||||
write!(
|
write!(
|
||||||
s,
|
s,
|
||||||
"<noscript><a class=\"navigate icon-go\" href=\"{}.html\">Go to linked tree: <code>{}</code></a></noscript>",
|
"<noscript><a class=\"navigate icon-go\" href=\"{}.html\">Go to linked tree: <code>{}</code></a></noscript>",
|
||||||
|
@ -95,7 +114,7 @@ pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId
|
||||||
|
|
||||||
s.push_str("<th-bb>");
|
s.push_str("<th-bb>");
|
||||||
{
|
{
|
||||||
if let Content::Link(link) = &attributes.content {
|
if let Content::Link(link) = &branch.attributes.content {
|
||||||
write!(
|
write!(
|
||||||
s,
|
s,
|
||||||
"<a class=\"icon icon-go\" href=\"{}.html\" title=\"linked tree\"></a>",
|
"<a class=\"icon icon-go\" href=\"{}.html\" title=\"linked tree\"></a>",
|
||||||
|
@ -107,7 +126,7 @@ pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId
|
||||||
write!(
|
write!(
|
||||||
s,
|
s,
|
||||||
"<a class=\"icon icon-permalink\" href=\"#{}\" title=\"permalink\"></a>",
|
"<a class=\"icon icon-permalink\" href=\"#{}\" title=\"permalink\"></a>",
|
||||||
EscapeAttribute(&id)
|
EscapeAttribute(&branch.html_id)
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
@ -115,7 +134,15 @@ pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId
|
||||||
|
|
||||||
if has_children {
|
if has_children {
|
||||||
s.push_str("</summary>");
|
s.push_str("</summary>");
|
||||||
branches_to_html(s, treehouse, file_id, &branch.children);
|
{
|
||||||
|
s.push_str("<ul>");
|
||||||
|
let num_children = branch.children.len();
|
||||||
|
for i in 0..num_children {
|
||||||
|
let child_id = treehouse.tree.branch(branch_id).children[i];
|
||||||
|
branch_to_html(s, treehouse, file_id, child_id);
|
||||||
|
}
|
||||||
|
s.push_str("</ul>");
|
||||||
|
}
|
||||||
s.push_str("</details>");
|
s.push_str("</details>");
|
||||||
} else {
|
} else {
|
||||||
s.push_str("</div>");
|
s.push_str("</div>");
|
||||||
|
@ -124,90 +151,14 @@ pub fn branch_to_html(s: &mut String, treehouse: &mut Treehouse, file_id: FileId
|
||||||
s.push_str("</li>");
|
s.push_str("</li>");
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_attributes(treehouse: &mut Treehouse, file_id: usize, branch: &Branch) -> Attributes {
|
|
||||||
let source = treehouse.source(file_id);
|
|
||||||
|
|
||||||
let mut successfully_parsed = true;
|
|
||||||
let mut attributes = if let Some(attributes) = &branch.attributes {
|
|
||||||
toml_edit::de::from_str(&source[attributes.data.clone()]).unwrap_or_else(|error| {
|
|
||||||
treehouse
|
|
||||||
.diagnostics
|
|
||||||
.push(toml_error_to_diagnostic(TomlError {
|
|
||||||
message: error.message().to_owned(),
|
|
||||||
span: error.span(),
|
|
||||||
file_id,
|
|
||||||
input_range: attributes.data.clone(),
|
|
||||||
}));
|
|
||||||
successfully_parsed = false;
|
|
||||||
Attributes::default()
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
Attributes::default()
|
|
||||||
};
|
|
||||||
let successfully_parsed = successfully_parsed;
|
|
||||||
|
|
||||||
// Only check for attribute validity if the attributes were parsed successfully.
|
|
||||||
if successfully_parsed {
|
|
||||||
let attribute_warning_span = branch
|
|
||||||
.attributes
|
|
||||||
.as_ref()
|
|
||||||
.map(|attributes| attributes.percent.clone())
|
|
||||||
.unwrap_or(branch.kind_span.clone());
|
|
||||||
|
|
||||||
// Check that every block has an ID.
|
|
||||||
if attributes.id.is_empty() {
|
|
||||||
attributes.id = format!("treehouse-missingno-{}", treehouse.next_missingno());
|
|
||||||
treehouse.diagnostics.push(Diagnostic {
|
|
||||||
severity: Severity::Warning,
|
|
||||||
code: Some("attr".into()),
|
|
||||||
message: "branch does not have an `id` attribute".into(),
|
|
||||||
labels: vec![Label {
|
|
||||||
style: LabelStyle::Primary,
|
|
||||||
file_id,
|
|
||||||
range: attribute_warning_span.clone(),
|
|
||||||
message: String::new(),
|
|
||||||
}],
|
|
||||||
notes: vec![
|
|
||||||
format!(
|
|
||||||
"note: a generated id `{}` will be used, but this id is unstable and will not persist across generations",
|
|
||||||
attributes.id
|
|
||||||
),
|
|
||||||
format!("help: run `treehouse fix {}` to add missing ids to branches", treehouse.filename(file_id)),
|
|
||||||
],
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check that link-type blocks are `+`-type to facilitate lazy loading.
|
|
||||||
if let Content::Link(_) = &attributes.content {
|
|
||||||
if branch.kind == BranchKind::Expanded {
|
|
||||||
treehouse.diagnostics.push(Diagnostic {
|
|
||||||
severity: Severity::Warning,
|
|
||||||
code: Some("attr".into()),
|
|
||||||
message: "`content.link` branch is expanded by default".into(),
|
|
||||||
labels: vec![Label {
|
|
||||||
style: LabelStyle::Primary,
|
|
||||||
file_id,
|
|
||||||
range: branch.kind_span.clone(),
|
|
||||||
message: String::new(),
|
|
||||||
}],
|
|
||||||
notes: vec![
|
|
||||||
"note: `content.link` branches should normally be collapsed to allow for lazy loading".into(),
|
|
||||||
],
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
attributes
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn branches_to_html(
|
pub fn branches_to_html(
|
||||||
s: &mut String,
|
s: &mut String,
|
||||||
treehouse: &mut Treehouse,
|
treehouse: &mut Treehouse,
|
||||||
file_id: FileId,
|
file_id: FileId,
|
||||||
branches: &[Branch],
|
branches: &[SemaBranchId],
|
||||||
) {
|
) {
|
||||||
s.push_str("<ul>");
|
s.push_str("<ul>");
|
||||||
for child in branches {
|
for &child in branches {
|
||||||
branch_to_html(s, treehouse, file_id, child);
|
branch_to_html(s, treehouse, file_id, child);
|
||||||
}
|
}
|
||||||
s.push_str("</ul>");
|
s.push_str("</ul>");
|
||||||
|
|
|
@ -12,6 +12,7 @@ mod cli;
|
||||||
mod html;
|
mod html;
|
||||||
mod paths;
|
mod paths;
|
||||||
mod state;
|
mod state;
|
||||||
|
mod tree;
|
||||||
|
|
||||||
async fn fallible_main() -> anyhow::Result<()> {
|
async fn fallible_main() -> anyhow::Result<()> {
|
||||||
let args = ProgramArgs::parse();
|
let args = ProgramArgs::parse();
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use std::ops::Range;
|
use std::{collections::HashMap, ops::Range};
|
||||||
|
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
use codespan_reporting::{
|
use codespan_reporting::{
|
||||||
|
@ -8,6 +8,8 @@ use codespan_reporting::{
|
||||||
};
|
};
|
||||||
use ulid::Ulid;
|
use ulid::Ulid;
|
||||||
|
|
||||||
|
use crate::tree::{SemaBranchId, SemaTree};
|
||||||
|
|
||||||
pub type Files = SimpleFiles<String, String>;
|
pub type Files = SimpleFiles<String, String>;
|
||||||
pub type FileId = <Files as codespan_reporting::files::Files<'static>>::FileId;
|
pub type FileId = <Files as codespan_reporting::files::Files<'static>>::FileId;
|
||||||
|
|
||||||
|
@ -16,18 +18,31 @@ pub struct Treehouse {
|
||||||
pub files: Files,
|
pub files: Files,
|
||||||
pub diagnostics: Vec<Diagnostic<FileId>>,
|
pub diagnostics: Vec<Diagnostic<FileId>>,
|
||||||
|
|
||||||
|
pub tree: SemaTree,
|
||||||
|
pub branches_by_named_id: HashMap<String, SemaBranchId>,
|
||||||
|
|
||||||
// Bit of a hack because I don't wanna write my own `Files`.
|
// Bit of a hack because I don't wanna write my own `Files`.
|
||||||
tree_paths: Vec<Option<String>>,
|
tree_paths: Vec<Option<String>>,
|
||||||
|
|
||||||
missingno_generator: ulid::Generator,
|
missingno_generator: ulid::Generator,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct BranchRef {
|
||||||
|
pub html_id: String,
|
||||||
|
pub file_id: FileId,
|
||||||
|
pub kind_span: Range<usize>,
|
||||||
|
}
|
||||||
|
|
||||||
impl Treehouse {
|
impl Treehouse {
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
Self {
|
Self {
|
||||||
files: Files::new(),
|
files: Files::new(),
|
||||||
diagnostics: vec![],
|
diagnostics: vec![],
|
||||||
|
|
||||||
|
tree: SemaTree::default(),
|
||||||
|
branches_by_named_id: HashMap::new(),
|
||||||
|
|
||||||
tree_paths: vec![],
|
tree_paths: vec![],
|
||||||
|
|
||||||
missingno_generator: ulid::Generator::new(),
|
missingno_generator: ulid::Generator::new(),
|
||||||
|
|
205
crates/treehouse/src/tree/mod.rs
Normal file
205
crates/treehouse/src/tree/mod.rs
Normal file
|
@ -0,0 +1,205 @@
|
||||||
|
pub mod attributes;
|
||||||
|
|
||||||
|
use std::ops::Range;
|
||||||
|
|
||||||
|
use codespan_reporting::diagnostic::{Diagnostic, Label, LabelStyle, Severity};
|
||||||
|
use treehouse_format::{
|
||||||
|
ast::{Branch, Roots},
|
||||||
|
pull::BranchKind,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
state::{toml_error_to_diagnostic, FileId, TomlError, Treehouse},
|
||||||
|
tree::attributes::{Attributes, Content},
|
||||||
|
};
|
||||||
|
|
||||||
|
#[derive(Debug, Default, Clone)]
|
||||||
|
pub struct SemaTree {
|
||||||
|
branches: Vec<SemaBranch>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
|
pub struct SemaBranchId(usize);
|
||||||
|
|
||||||
|
impl SemaTree {
|
||||||
|
pub fn add_branch(&mut self, branch: SemaBranch) -> SemaBranchId {
|
||||||
|
let id = self.branches.len();
|
||||||
|
self.branches.push(branch);
|
||||||
|
SemaBranchId(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn branch(&self, id: SemaBranchId) -> &SemaBranch {
|
||||||
|
&self.branches[id.0]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct SemaRoots {
|
||||||
|
pub branches: Vec<SemaBranchId>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SemaRoots {
|
||||||
|
pub fn from_roots(treehouse: &mut Treehouse, file_id: FileId, roots: Roots) -> Self {
|
||||||
|
Self {
|
||||||
|
branches: roots
|
||||||
|
.branches
|
||||||
|
.into_iter()
|
||||||
|
.map(|branch| SemaBranch::from_branch(treehouse, file_id, branch))
|
||||||
|
.collect(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Analyzed branch.
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct SemaBranch {
|
||||||
|
pub file_id: FileId,
|
||||||
|
|
||||||
|
pub indent_level: usize,
|
||||||
|
pub raw_attributes: Option<treehouse_format::pull::Attributes>,
|
||||||
|
pub kind: BranchKind,
|
||||||
|
pub kind_span: Range<usize>,
|
||||||
|
pub content: Range<usize>,
|
||||||
|
|
||||||
|
pub html_id: String,
|
||||||
|
pub attributes: Attributes,
|
||||||
|
pub children: Vec<SemaBranchId>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SemaBranch {
|
||||||
|
pub fn from_branch(treehouse: &mut Treehouse, file_id: FileId, branch: Branch) -> SemaBranchId {
|
||||||
|
let attributes = Self::parse_attributes(treehouse, file_id, &branch);
|
||||||
|
|
||||||
|
let named_id = attributes.id.clone();
|
||||||
|
let html_id = format!(
|
||||||
|
"{}:{}",
|
||||||
|
treehouse
|
||||||
|
.tree_path(file_id)
|
||||||
|
.expect("file should have a tree path"),
|
||||||
|
attributes.id
|
||||||
|
);
|
||||||
|
|
||||||
|
let branch = Self {
|
||||||
|
file_id,
|
||||||
|
indent_level: branch.indent_level,
|
||||||
|
raw_attributes: branch.attributes,
|
||||||
|
kind: branch.kind,
|
||||||
|
kind_span: branch.kind_span,
|
||||||
|
content: branch.content,
|
||||||
|
html_id,
|
||||||
|
attributes,
|
||||||
|
children: branch
|
||||||
|
.children
|
||||||
|
.into_iter()
|
||||||
|
.map(|child| Self::from_branch(treehouse, file_id, child))
|
||||||
|
.collect(),
|
||||||
|
};
|
||||||
|
let new_branch_id = treehouse.tree.add_branch(branch);
|
||||||
|
|
||||||
|
if let Some(old_branch_id) = treehouse
|
||||||
|
.branches_by_named_id
|
||||||
|
.insert(named_id.clone(), new_branch_id)
|
||||||
|
{
|
||||||
|
let new_branch = treehouse.tree.branch(new_branch_id);
|
||||||
|
let old_branch = treehouse.tree.branch(old_branch_id);
|
||||||
|
|
||||||
|
treehouse.diagnostics.push(
|
||||||
|
Diagnostic::warning()
|
||||||
|
.with_code("sema")
|
||||||
|
.with_message(format!("two branches share the same id `{}`", named_id))
|
||||||
|
.with_labels(vec![
|
||||||
|
Label {
|
||||||
|
style: LabelStyle::Primary,
|
||||||
|
file_id,
|
||||||
|
range: new_branch.kind_span.clone(),
|
||||||
|
message: String::new(),
|
||||||
|
},
|
||||||
|
Label {
|
||||||
|
style: LabelStyle::Primary,
|
||||||
|
file_id: old_branch.file_id,
|
||||||
|
range: old_branch.kind_span.clone(),
|
||||||
|
message: String::new(),
|
||||||
|
},
|
||||||
|
]),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
new_branch_id
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_attributes(treehouse: &mut Treehouse, file_id: FileId, branch: &Branch) -> Attributes {
|
||||||
|
let source = treehouse.source(file_id);
|
||||||
|
|
||||||
|
let mut successfully_parsed = true;
|
||||||
|
let mut attributes = if let Some(attributes) = &branch.attributes {
|
||||||
|
toml_edit::de::from_str(&source[attributes.data.clone()]).unwrap_or_else(|error| {
|
||||||
|
treehouse
|
||||||
|
.diagnostics
|
||||||
|
.push(toml_error_to_diagnostic(TomlError {
|
||||||
|
message: error.message().to_owned(),
|
||||||
|
span: error.span(),
|
||||||
|
file_id,
|
||||||
|
input_range: attributes.data.clone(),
|
||||||
|
}));
|
||||||
|
successfully_parsed = false;
|
||||||
|
Attributes::default()
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
Attributes::default()
|
||||||
|
};
|
||||||
|
let successfully_parsed = successfully_parsed;
|
||||||
|
|
||||||
|
// Only check for attribute validity if the attributes were parsed successfully.
|
||||||
|
if successfully_parsed {
|
||||||
|
let attribute_warning_span = branch
|
||||||
|
.attributes
|
||||||
|
.as_ref()
|
||||||
|
.map(|attributes| attributes.percent.clone())
|
||||||
|
.unwrap_or(branch.kind_span.clone());
|
||||||
|
|
||||||
|
// Check that every block has an ID.
|
||||||
|
if attributes.id.is_empty() {
|
||||||
|
attributes.id = format!("treehouse-missingno-{}", treehouse.next_missingno());
|
||||||
|
treehouse.diagnostics.push(Diagnostic {
|
||||||
|
severity: Severity::Warning,
|
||||||
|
code: Some("attr".into()),
|
||||||
|
message: "branch does not have an `id` attribute".into(),
|
||||||
|
labels: vec![Label {
|
||||||
|
style: LabelStyle::Primary,
|
||||||
|
file_id,
|
||||||
|
range: attribute_warning_span.clone(),
|
||||||
|
message: String::new(),
|
||||||
|
}],
|
||||||
|
notes: vec![
|
||||||
|
format!(
|
||||||
|
"note: a generated id `{}` will be used, but this id is unstable and will not persist across generations",
|
||||||
|
attributes.id
|
||||||
|
),
|
||||||
|
format!("help: run `treehouse fix {}` to add missing ids to branches", treehouse.filename(file_id)),
|
||||||
|
],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that link-type blocks are `+`-type to facilitate lazy loading.
|
||||||
|
if let Content::Link(_) = &attributes.content {
|
||||||
|
if branch.kind == BranchKind::Expanded {
|
||||||
|
treehouse.diagnostics.push(Diagnostic {
|
||||||
|
severity: Severity::Warning,
|
||||||
|
code: Some("attr".into()),
|
||||||
|
message: "`content.link` branch is expanded by default".into(),
|
||||||
|
labels: vec![Label {
|
||||||
|
style: LabelStyle::Primary,
|
||||||
|
file_id,
|
||||||
|
range: branch.kind_span.clone(),
|
||||||
|
message: String::new(),
|
||||||
|
}],
|
||||||
|
notes: vec![
|
||||||
|
"note: `content.link` branches should normally be collapsed to allow for lazy loading".into(),
|
||||||
|
],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
attributes
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue