1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
use std::collections::BTreeMap;
use std::fmt;
use crate::compiler::tokens::Span;
use crate::error::ErrorKind;
use crate::value::Value;
#[cfg_attr(docsrs, doc(cfg(feature = "debug")))]
#[derive(Default)]
pub(crate) struct DebugInfo {
pub(crate) template_source: Option<String>,
pub(crate) referenced_locals: BTreeMap<String, Value>,
}
struct VarPrinter<'x>(&'x BTreeMap<String, Value>);
impl<'x> fmt::Debug for VarPrinter<'x> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
if self.0.is_empty() {
return write!(f, "No referenced variables");
}
let mut m = f.debug_struct("Referenced variables:");
let mut vars = self.0.iter().collect::<Vec<_>>();
vars.sort_by_key(|x| x.0);
for (key, value) in vars {
m.field(key, value);
}
m.finish()
}
}
impl DebugInfo {
pub fn source(&self) -> Option<&str> {
self.template_source.as_deref()
}
}
pub(super) fn render_debug_info(
f: &mut fmt::Formatter,
name: Option<&str>,
kind: ErrorKind,
line: Option<usize>,
span: Option<Span>,
info: &DebugInfo,
) -> fmt::Result {
if let Some(source) = info.source() {
let title = format!(
" {} ",
name.unwrap_or_default()
.rsplit(&['/', '\\'])
.next()
.unwrap_or("Template Source")
);
ok!(writeln!(f));
writeln!(f, "{:-^1$}", title, 79).unwrap();
let lines: Vec<_> = source.lines().enumerate().collect();
let idx = line.unwrap_or(1).saturating_sub(1);
let skip = idx.saturating_sub(3);
let pre = lines.iter().skip(skip).take(3.min(idx)).collect::<Vec<_>>();
let post = lines.iter().skip(idx + 1).take(3).collect::<Vec<_>>();
for (idx, line) in pre {
writeln!(f, "{:>4} | {}", idx + 1, line).unwrap();
}
writeln!(f, "{:>4} > {}", idx + 1, lines[idx].1).unwrap();
if let Some(span) = span {
if span.start_line == span.end_line {
ok!(writeln!(
f,
" i {}{} {}",
" ".repeat(span.start_col),
"^".repeat(span.end_col - span.start_col),
kind,
));
}
}
for (idx, line) in post {
writeln!(f, "{:>4} | {}", idx + 1, line).unwrap();
}
write!(f, "{:~^1$}", "", 79).unwrap();
}
ok!(writeln!(f));
ok!(writeln!(f, "{:#?}", VarPrinter(&info.referenced_locals)));
write!(f, "{:-^1$}", "", 79).unwrap();
Ok(())
}