mirror of
https://github.com/askama-rs/askama.git
synced 2025-10-02 07:20:55 +00:00

I ran `cargo +nightly clippy --all-targets --fix -- -D warnings` and made only tiny manual improvements.
1700 lines
65 KiB
Rust
1700 lines
65 KiB
Rust
use std::borrow::Cow;
|
|
use std::collections::hash_map::{Entry, HashMap};
|
|
use std::fmt::{self, Debug, Write};
|
|
use std::mem;
|
|
|
|
use parser::expr::BinOp;
|
|
use parser::node::{
|
|
Call, Comment, Cond, CondTest, FilterBlock, If, Include, Let, Lit, Loop, Macro, Match,
|
|
Whitespace, Ws,
|
|
};
|
|
use parser::{Expr, Node, Span, Target, WithSpan};
|
|
use rustc_hash::FxBuildHasher;
|
|
|
|
use super::{
|
|
DisplayWrap, FILTER_SOURCE, Generator, LocalMeta, MapChain, compile_time_escape, is_copyable,
|
|
normalize_identifier,
|
|
};
|
|
use crate::generator::{LocalCallerMeta, Writable};
|
|
use crate::heritage::{Context, Heritage};
|
|
use crate::integration::Buffer;
|
|
use crate::{CompileError, FileInfo, fmt_left, fmt_right};
|
|
|
|
impl<'a> Generator<'a, '_> {
|
|
pub(super) fn impl_template_inner(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
buf: &mut Buffer,
|
|
) -> Result<usize, CompileError> {
|
|
buf.set_discard(self.buf_writable.discard);
|
|
let size_hint = if let Some(heritage) = self.heritage {
|
|
self.handle(heritage.root, heritage.root.nodes, buf, AstLevel::Top)
|
|
} else {
|
|
self.handle(ctx, ctx.nodes, buf, AstLevel::Top)
|
|
}?;
|
|
self.flush_ws(Ws(None, None));
|
|
buf.set_discard(false);
|
|
Ok(size_hint)
|
|
}
|
|
|
|
fn push_locals<T, F>(&mut self, callback: F) -> Result<T, CompileError>
|
|
where
|
|
F: FnOnce(&mut Self) -> Result<T, CompileError>,
|
|
{
|
|
self.locals.stack_push();
|
|
let res = callback(self);
|
|
self.locals.stack_pop();
|
|
res
|
|
}
|
|
|
|
fn with_child<'b, T, F>(
|
|
&mut self,
|
|
heritage: Option<&'b Heritage<'a, 'b>>,
|
|
callback: F,
|
|
) -> Result<T, CompileError>
|
|
where
|
|
F: FnOnce(&mut Generator<'a, 'b>) -> Result<T, CompileError>,
|
|
{
|
|
self.locals.stack_push();
|
|
|
|
let buf_writable = mem::take(&mut self.buf_writable);
|
|
let locals = mem::replace(&mut self.locals, MapChain::new_empty());
|
|
|
|
let mut child = Generator::new(
|
|
self.input,
|
|
self.contexts,
|
|
heritage,
|
|
locals,
|
|
self.buf_writable.discard,
|
|
self.is_in_filter_block,
|
|
);
|
|
child.buf_writable = buf_writable;
|
|
let res = callback(&mut child);
|
|
Generator {
|
|
locals: self.locals,
|
|
buf_writable: self.buf_writable,
|
|
..
|
|
} = child;
|
|
|
|
self.locals.stack_pop();
|
|
res
|
|
}
|
|
|
|
fn handle(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
nodes: &'a [Node<'_>],
|
|
buf: &mut Buffer,
|
|
level: AstLevel,
|
|
) -> Result<usize, CompileError> {
|
|
let mut size_hint = 0;
|
|
for n in nodes {
|
|
match *n {
|
|
Node::Lit(ref lit) => {
|
|
self.write_lit(lit);
|
|
}
|
|
Node::Comment(ref comment) => {
|
|
self.write_comment(comment);
|
|
}
|
|
Node::Expr(ws, ref val) => {
|
|
size_hint += self.write_expr(ctx, buf, ws, val)?;
|
|
}
|
|
Node::Let(ref l) => {
|
|
self.write_let(ctx, buf, l)?;
|
|
}
|
|
Node::If(ref i) => {
|
|
size_hint += self.write_if(ctx, buf, i)?;
|
|
}
|
|
Node::Match(ref m) => {
|
|
size_hint += self.write_match(ctx, buf, m)?;
|
|
}
|
|
Node::Loop(ref loop_block) => {
|
|
size_hint += self.write_loop(ctx, buf, loop_block)?;
|
|
}
|
|
Node::BlockDef(ref b) => {
|
|
size_hint +=
|
|
self.write_block(ctx, buf, Some(b.name), Ws(b.ws1.0, b.ws2.1), b.span())?;
|
|
}
|
|
Node::Include(ref i) => {
|
|
size_hint += self.handle_include(ctx, buf, i)?;
|
|
}
|
|
Node::Call(ref call) => {
|
|
size_hint += self.write_call(ctx, buf, call)?;
|
|
}
|
|
Node::FilterBlock(ref filter) => {
|
|
size_hint += self.write_filter_block(ctx, buf, filter)?;
|
|
}
|
|
Node::Macro(ref m) => {
|
|
if level != AstLevel::Top {
|
|
return Err(ctx.generate_error(
|
|
"macro blocks only allowed at the top level",
|
|
m.span(),
|
|
));
|
|
}
|
|
self.flush_ws(m.ws1);
|
|
self.prepare_ws(m.ws2);
|
|
}
|
|
Node::Raw(ref raw) => {
|
|
self.handle_ws(raw.ws1);
|
|
self.write_lit(&raw.lit);
|
|
self.handle_ws(raw.ws2);
|
|
}
|
|
Node::Import(ref i) => {
|
|
if level != AstLevel::Top {
|
|
return Err(ctx.generate_error(
|
|
"import blocks only allowed at the top level",
|
|
i.span(),
|
|
));
|
|
}
|
|
self.handle_ws(i.ws);
|
|
}
|
|
Node::Extends(ref e) => {
|
|
if level != AstLevel::Top {
|
|
return Err(ctx.generate_error(
|
|
"extend blocks only allowed at the top level",
|
|
e.span(),
|
|
));
|
|
}
|
|
// No whitespace handling: child template top-level is not used,
|
|
// except for the blocks defined in it.
|
|
}
|
|
Node::Break(ref ws) => {
|
|
self.handle_ws(**ws);
|
|
self.write_buf_writable(ctx, buf)?;
|
|
buf.write("break;");
|
|
}
|
|
Node::Continue(ref ws) => {
|
|
self.handle_ws(**ws);
|
|
self.write_buf_writable(ctx, buf)?;
|
|
buf.write("continue;");
|
|
}
|
|
}
|
|
}
|
|
|
|
if AstLevel::Top == level {
|
|
// Handle any pending whitespace.
|
|
if self.next_ws.is_some() {
|
|
self.flush_ws(Ws(Some(self.skip_ws), None));
|
|
}
|
|
|
|
size_hint += self.write_buf_writable(ctx, buf)?;
|
|
}
|
|
Ok(size_hint)
|
|
}
|
|
|
|
fn evaluate_condition(
|
|
&self,
|
|
expr: WithSpan<'a, Expr<'a>>,
|
|
only_contains_is_defined: &mut bool,
|
|
) -> (EvaluatedResult, WithSpan<'a, Expr<'a>>) {
|
|
let (expr, span) = expr.deconstruct();
|
|
|
|
match expr {
|
|
Expr::NumLit(_, _)
|
|
| Expr::StrLit(_)
|
|
| Expr::CharLit(_)
|
|
| Expr::Var(_)
|
|
| Expr::Path(_)
|
|
| Expr::Array(_)
|
|
| Expr::AssociatedItem(_, _)
|
|
| Expr::Index(_, _)
|
|
| Expr::Filter(_)
|
|
| Expr::Range(_)
|
|
| Expr::Call { .. }
|
|
| Expr::RustMacro(_, _)
|
|
| Expr::Try(_)
|
|
| Expr::Tuple(_)
|
|
| Expr::NamedArgument(_, _)
|
|
| Expr::FilterSource
|
|
| Expr::As(_, _)
|
|
| Expr::Concat(_)
|
|
| Expr::LetCond(_)
|
|
| Expr::ArgumentPlaceholder => {
|
|
*only_contains_is_defined = false;
|
|
(EvaluatedResult::Unknown, WithSpan::new(expr, span))
|
|
}
|
|
Expr::BoolLit(true) => (EvaluatedResult::AlwaysTrue, WithSpan::new(expr, span)),
|
|
Expr::BoolLit(false) => (EvaluatedResult::AlwaysFalse, WithSpan::new(expr, span)),
|
|
Expr::Unary("!", inner) => {
|
|
let (result, expr) = self.evaluate_condition(*inner, only_contains_is_defined);
|
|
match result {
|
|
EvaluatedResult::AlwaysTrue => (
|
|
EvaluatedResult::AlwaysFalse,
|
|
WithSpan::new(Expr::BoolLit(false), ""),
|
|
),
|
|
EvaluatedResult::AlwaysFalse => (
|
|
EvaluatedResult::AlwaysTrue,
|
|
WithSpan::new(Expr::BoolLit(true), ""),
|
|
),
|
|
EvaluatedResult::Unknown => (
|
|
EvaluatedResult::Unknown,
|
|
WithSpan::new(Expr::Unary("!", Box::new(expr)), span),
|
|
),
|
|
}
|
|
}
|
|
Expr::Unary(_, _) => (EvaluatedResult::Unknown, WithSpan::new(expr, span)),
|
|
Expr::BinOp(v) if v.op == "&&" => {
|
|
let (result_left, expr_left) =
|
|
self.evaluate_condition(v.lhs, only_contains_is_defined);
|
|
if result_left == EvaluatedResult::AlwaysFalse {
|
|
// The right side of the `&&` won't be evaluated, no need to go any further.
|
|
return (result_left, WithSpan::new(Expr::BoolLit(false), ""));
|
|
}
|
|
let (result_right, expr_right) =
|
|
self.evaluate_condition(v.rhs, only_contains_is_defined);
|
|
match (result_left, result_right) {
|
|
(EvaluatedResult::AlwaysTrue, EvaluatedResult::AlwaysTrue) => (
|
|
EvaluatedResult::AlwaysTrue,
|
|
WithSpan::new(Expr::BoolLit(true), ""),
|
|
),
|
|
(_, EvaluatedResult::AlwaysFalse) => (
|
|
EvaluatedResult::AlwaysFalse,
|
|
bin_op(span, "&&", expr_left, expr_right),
|
|
),
|
|
(EvaluatedResult::AlwaysTrue, _) => (result_right, expr_right),
|
|
(_, EvaluatedResult::AlwaysTrue) => (result_left, expr_left),
|
|
_ => (
|
|
EvaluatedResult::Unknown,
|
|
bin_op(span, "&&", expr_left, expr_right),
|
|
),
|
|
}
|
|
}
|
|
Expr::BinOp(v) if v.op == "||" => {
|
|
let (result_left, expr_left) =
|
|
self.evaluate_condition(v.lhs, only_contains_is_defined);
|
|
if result_left == EvaluatedResult::AlwaysTrue {
|
|
// The right side of the `||` won't be evaluated, no need to go any further.
|
|
return (result_left, WithSpan::new(Expr::BoolLit(true), ""));
|
|
}
|
|
let (result_right, expr_right) =
|
|
self.evaluate_condition(v.rhs, only_contains_is_defined);
|
|
match (result_left, result_right) {
|
|
(EvaluatedResult::AlwaysFalse, EvaluatedResult::AlwaysFalse) => (
|
|
EvaluatedResult::AlwaysFalse,
|
|
WithSpan::new(Expr::BoolLit(false), ""),
|
|
),
|
|
(_, EvaluatedResult::AlwaysTrue) => (
|
|
EvaluatedResult::AlwaysTrue,
|
|
bin_op(span, "||", expr_left, expr_right),
|
|
),
|
|
(EvaluatedResult::AlwaysFalse, _) => (result_right, expr_right),
|
|
(_, EvaluatedResult::AlwaysFalse) => (result_left, expr_left),
|
|
_ => (
|
|
EvaluatedResult::Unknown,
|
|
bin_op(span, "||", expr_left, expr_right),
|
|
),
|
|
}
|
|
}
|
|
Expr::BinOp(_) => {
|
|
*only_contains_is_defined = false;
|
|
(EvaluatedResult::Unknown, WithSpan::new(expr, span))
|
|
}
|
|
Expr::Group(inner) => {
|
|
let (result, expr) = self.evaluate_condition(*inner, only_contains_is_defined);
|
|
(result, WithSpan::new(Expr::Group(Box::new(expr)), span))
|
|
}
|
|
Expr::IsDefined(left) => {
|
|
// Variable is defined so we want to keep the condition.
|
|
if self.is_var_defined(left) {
|
|
(
|
|
EvaluatedResult::AlwaysTrue,
|
|
WithSpan::new(Expr::BoolLit(true), ""),
|
|
)
|
|
} else {
|
|
(
|
|
EvaluatedResult::AlwaysFalse,
|
|
WithSpan::new(Expr::BoolLit(false), ""),
|
|
)
|
|
}
|
|
}
|
|
Expr::IsNotDefined(left) => {
|
|
// Variable is defined so we don't want to keep the condition.
|
|
if self.is_var_defined(left) {
|
|
(
|
|
EvaluatedResult::AlwaysFalse,
|
|
WithSpan::new(Expr::BoolLit(false), ""),
|
|
)
|
|
} else {
|
|
(
|
|
EvaluatedResult::AlwaysTrue,
|
|
WithSpan::new(Expr::BoolLit(true), ""),
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn write_if(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
buf: &mut Buffer,
|
|
if_: &'a If<'_>,
|
|
) -> Result<usize, CompileError> {
|
|
let mut flushed = 0;
|
|
let mut arm_sizes = Vec::new();
|
|
let mut has_else = false;
|
|
|
|
let conds = Conds::compute_branches(self, if_);
|
|
|
|
if let Some(ws_before) = conds.ws_before {
|
|
self.handle_ws(ws_before);
|
|
}
|
|
|
|
let mut iter = conds.conds.iter().enumerate().peekable();
|
|
while let Some((pos, cond_info)) = iter.next() {
|
|
let cond = cond_info.cond;
|
|
|
|
if pos == 0 {
|
|
self.handle_ws(cond.ws);
|
|
flushed += self.write_buf_writable(ctx, buf)?;
|
|
}
|
|
|
|
self.push_locals(|this| {
|
|
let mut arm_size = 0;
|
|
|
|
if let Some(CondTest { target, expr, .. }) = &cond.cond {
|
|
let expr = cond_info.cond_expr.as_ref().unwrap_or(expr);
|
|
|
|
if pos == 0 {
|
|
if cond_info.generate_condition {
|
|
buf.write("if ");
|
|
}
|
|
// Otherwise it means it will be the only condition generated,
|
|
// so nothing to be added here.
|
|
} else if cond_info.generate_condition {
|
|
buf.write("} else if ");
|
|
} else {
|
|
buf.write("} else {");
|
|
has_else = true;
|
|
}
|
|
|
|
if let Some(target) = target {
|
|
let mut expr_buf = Buffer::new();
|
|
buf.write("let ");
|
|
// If this is a chain condition, then we need to declare the variable after the
|
|
// left expression has been handled but before the right expression is handled
|
|
// but this one should have access to the let-bound variable.
|
|
match &**expr {
|
|
Expr::BinOp(v) if matches!(v.op, "||" | "&&") => {
|
|
let display_wrap =
|
|
this.visit_expr_first(ctx, &mut expr_buf, &v.lhs)?;
|
|
this.visit_target(buf, true, true, target);
|
|
this.visit_expr_not_first(
|
|
ctx,
|
|
&mut expr_buf,
|
|
&v.lhs,
|
|
display_wrap,
|
|
)?;
|
|
buf.write(format_args!("= &{expr_buf} {} ", v.op));
|
|
this.visit_condition(ctx, buf, &v.rhs)?;
|
|
}
|
|
_ => {
|
|
let display_wrap =
|
|
this.visit_expr_first(ctx, &mut expr_buf, expr)?;
|
|
this.visit_target(buf, true, true, target);
|
|
this.visit_expr_not_first(ctx, &mut expr_buf, expr, display_wrap)?;
|
|
buf.write(format_args!("= &{expr_buf}"));
|
|
}
|
|
}
|
|
buf.write("{");
|
|
} else if cond_info.generate_condition {
|
|
this.visit_condition(ctx, buf, expr)?;
|
|
buf.write('{');
|
|
}
|
|
} else if pos != 0 {
|
|
buf.write("} else {");
|
|
has_else = true;
|
|
}
|
|
|
|
if cond_info.generate_content {
|
|
arm_size += this.handle(ctx, &cond.nodes, buf, AstLevel::Nested)?;
|
|
}
|
|
arm_sizes.push(arm_size);
|
|
|
|
if let Some((_, cond_info)) = iter.peek() {
|
|
let cond = cond_info.cond;
|
|
|
|
this.handle_ws(cond.ws);
|
|
flushed += this.write_buf_writable(ctx, buf)?;
|
|
} else {
|
|
if let Some(ws_after) = conds.ws_after {
|
|
this.handle_ws(ws_after);
|
|
}
|
|
this.handle_ws(if_.ws);
|
|
flushed += this.write_buf_writable(ctx, buf)?;
|
|
}
|
|
Ok(0)
|
|
})?;
|
|
}
|
|
|
|
if conds.nb_conds > 0 {
|
|
buf.write('}');
|
|
}
|
|
|
|
if !has_else && !conds.conds.is_empty() {
|
|
arm_sizes.push(0);
|
|
}
|
|
Ok(flushed + median(&mut arm_sizes))
|
|
}
|
|
|
|
#[allow(clippy::too_many_arguments)]
|
|
fn write_match(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
buf: &mut Buffer,
|
|
m: &'a Match<'a>,
|
|
) -> Result<usize, CompileError> {
|
|
let Match {
|
|
ws1,
|
|
ref expr,
|
|
ref arms,
|
|
ws2,
|
|
} = *m;
|
|
|
|
self.flush_ws(ws1);
|
|
let flushed = self.write_buf_writable(ctx, buf)?;
|
|
let mut arm_sizes = Vec::new();
|
|
|
|
let expr_code = self.visit_expr_root(ctx, expr)?;
|
|
buf.write(format_args!("match &{expr_code} {{"));
|
|
|
|
let mut arm_size = 0;
|
|
let mut iter = arms.iter().enumerate().peekable();
|
|
while let Some((i, arm)) = iter.next() {
|
|
if i == 0 {
|
|
self.handle_ws(arm.ws);
|
|
}
|
|
|
|
self.push_locals(|this| {
|
|
for (index, target) in arm.target.iter().enumerate() {
|
|
if index != 0 {
|
|
buf.write('|');
|
|
}
|
|
this.visit_target(buf, true, true, target);
|
|
}
|
|
buf.write(" => {");
|
|
|
|
arm_size = this.handle(ctx, &arm.nodes, buf, AstLevel::Nested)?;
|
|
|
|
if let Some((_, arm)) = iter.peek() {
|
|
this.handle_ws(arm.ws);
|
|
arm_sizes.push(arm_size + this.write_buf_writable(ctx, buf)?);
|
|
|
|
buf.write('}');
|
|
} else {
|
|
this.handle_ws(ws2);
|
|
arm_sizes.push(arm_size + this.write_buf_writable(ctx, buf)?);
|
|
buf.write('}');
|
|
}
|
|
Ok(0)
|
|
})?;
|
|
}
|
|
|
|
buf.write('}');
|
|
|
|
Ok(flushed + median(&mut arm_sizes))
|
|
}
|
|
|
|
fn write_loop(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
buf: &mut Buffer,
|
|
loop_block: &'a WithSpan<'a, Loop<'_>>,
|
|
) -> Result<usize, CompileError> {
|
|
self.handle_ws(loop_block.ws1);
|
|
self.push_locals(|this| {
|
|
let has_else_nodes = !loop_block.else_nodes.is_empty();
|
|
|
|
let flushed = this.write_buf_writable(ctx, buf)?;
|
|
buf.write('{');
|
|
if has_else_nodes {
|
|
buf.write("let mut __askama_did_loop = false;");
|
|
}
|
|
|
|
buf.write("let __askama_iter =");
|
|
this.visit_loop_iter(ctx, buf, &loop_block.iter)?;
|
|
buf.write(';');
|
|
if let Some(cond) = &loop_block.cond {
|
|
this.push_locals(|this| {
|
|
buf.write("let __askama_iter = __askama_iter.filter(|");
|
|
this.visit_target(buf, true, true, &loop_block.var);
|
|
buf.write("| -> askama::helpers::core::primitive::bool {");
|
|
this.visit_expr(ctx, buf, cond)?;
|
|
buf.write("});");
|
|
Ok(0)
|
|
})?;
|
|
}
|
|
|
|
let size_hint1 = this.push_locals(|this| {
|
|
buf.write("for (");
|
|
this.visit_target(buf, true, true, &loop_block.var);
|
|
buf.write(
|
|
", __askama_item) in askama::helpers::TemplateLoop::new(__askama_iter) {",
|
|
);
|
|
|
|
if has_else_nodes {
|
|
buf.write("__askama_did_loop = true;");
|
|
}
|
|
let mut size_hint1 = this.handle(ctx, &loop_block.body, buf, AstLevel::Nested)?;
|
|
this.handle_ws(loop_block.ws2);
|
|
size_hint1 += this.write_buf_writable(ctx, buf)?;
|
|
Ok(size_hint1)
|
|
})?;
|
|
buf.write('}');
|
|
|
|
let size_hint2;
|
|
if has_else_nodes {
|
|
buf.write("if !__askama_did_loop {");
|
|
size_hint2 = this.push_locals(|this| {
|
|
let mut size_hint =
|
|
this.handle(ctx, &loop_block.else_nodes, buf, AstLevel::Nested)?;
|
|
this.handle_ws(loop_block.ws3);
|
|
size_hint += this.write_buf_writable(ctx, buf)?;
|
|
Ok(size_hint)
|
|
})?;
|
|
buf.write('}');
|
|
} else {
|
|
this.handle_ws(loop_block.ws3);
|
|
size_hint2 = this.write_buf_writable(ctx, buf)?;
|
|
}
|
|
|
|
buf.write('}');
|
|
Ok(flushed + ((size_hint1 * 3) + size_hint2) / 2)
|
|
})
|
|
}
|
|
|
|
fn write_call(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
buf: &mut Buffer,
|
|
call: &'a WithSpan<'a, Call<'_>>,
|
|
) -> Result<usize, CompileError> {
|
|
let Call {
|
|
ws1,
|
|
scope,
|
|
name,
|
|
ref args,
|
|
ws2,
|
|
..
|
|
} = **call;
|
|
|
|
let (def, own_ctx) = if let Some(s) = scope {
|
|
let path = ctx.imports.get(s).ok_or_else(|| {
|
|
ctx.generate_error(format_args!("no import found for scope {s:?}"), call.span())
|
|
})?;
|
|
let mctx = self.contexts.get(path).ok_or_else(|| {
|
|
ctx.generate_error(format_args!("context for {path:?} not found"), call.span())
|
|
})?;
|
|
let def = mctx.macros.get(name).ok_or_else(|| {
|
|
ctx.generate_error(
|
|
format_args!("macro {name:?} not found in scope {s:?}"),
|
|
call.span(),
|
|
)
|
|
})?;
|
|
(*def, mctx)
|
|
} else {
|
|
let def = ctx.macros.get(name).ok_or_else(|| {
|
|
ctx.generate_error(format_args!("macro {name:?} not found"), call.span())
|
|
})?;
|
|
(*def, ctx)
|
|
};
|
|
|
|
if self
|
|
.seen_callers
|
|
.iter()
|
|
.any(|(_, s, _)| std::ptr::eq(*s, def))
|
|
{
|
|
let mut message = "Found recursion in macro calls:".to_owned();
|
|
for (_, m, f) in &self.seen_callers {
|
|
if let Some(f) = f {
|
|
write!(message, "{f}").unwrap();
|
|
} else {
|
|
write!(message, "\n`{}`", m.name.escape_debug()).unwrap();
|
|
}
|
|
}
|
|
return Err(ctx.generate_error(message, call.span()));
|
|
} else {
|
|
self.seen_callers
|
|
.push((call, def, ctx.file_info_of(call.span())));
|
|
}
|
|
self.flush_ws(ws1); // Cannot handle_ws() here: whitespace from macro definition comes first
|
|
let size_hint = self.push_locals(|this| {
|
|
this.locals.insert("caller".into(), LocalMeta::caller(call, ctx.clone()));
|
|
|
|
macro_call_ensure_arg_count(call, def, ctx)?;
|
|
|
|
this.write_buf_writable(ctx, buf)?;
|
|
buf.write('{');
|
|
this.prepare_ws(def.ws1);
|
|
|
|
let mut named_arguments: HashMap<&str, _, FxBuildHasher> = HashMap::default();
|
|
// Since named arguments can only be passed last, we only need to check if the last argument
|
|
// is a named one.
|
|
if let Some(Expr::NamedArgument(_, _)) = args.last().map(|expr| &**expr) {
|
|
// First we check that all named arguments actually exist in the called item.
|
|
for (index, arg) in args.iter().enumerate().rev() {
|
|
let Expr::NamedArgument(arg_name, _) = &**arg else {
|
|
break;
|
|
};
|
|
if !def.args.iter().any(|(arg, _)| arg == arg_name) {
|
|
return Err(ctx.generate_error(
|
|
format_args!("no argument named `{arg_name}` in macro {name:?}"),
|
|
call.span(),
|
|
));
|
|
}
|
|
named_arguments.insert(arg_name, (index, arg));
|
|
}
|
|
}
|
|
|
|
let mut value = Buffer::new();
|
|
|
|
// Handling both named and unnamed arguments requires to be careful of the named arguments
|
|
// order. To do so, we iterate through the macro defined arguments and then check if we have
|
|
// a named argument with this name:
|
|
//
|
|
// * If there is one, we add it and move to the next argument.
|
|
// * If there isn't one, then we pick the next argument (we can do it without checking
|
|
// anything since named arguments are always last).
|
|
let mut allow_positional = true;
|
|
let mut used_named_args = vec![false; args.len()];
|
|
for (index, (arg, default_value)) in def.args.iter().enumerate() {
|
|
let expr = if let Some((index, expr)) = named_arguments.get(arg) {
|
|
used_named_args[*index] = true;
|
|
allow_positional = false;
|
|
expr
|
|
} else {
|
|
match args.get(index) {
|
|
Some(arg_expr) if !matches!(**arg_expr, Expr::NamedArgument(_, _)) => {
|
|
// If there is already at least one named argument, then it's not allowed
|
|
// to use unnamed ones at this point anymore.
|
|
if !allow_positional {
|
|
return Err(ctx.generate_error(
|
|
format_args!(
|
|
"cannot have unnamed argument (`{arg}`) after named argument \
|
|
in call to macro {name:?}"
|
|
),
|
|
call.span(),
|
|
));
|
|
}
|
|
arg_expr
|
|
}
|
|
Some(arg_expr) if used_named_args[index] => {
|
|
let Expr::NamedArgument(name, _) = **arg_expr else { unreachable!() };
|
|
return Err(ctx.generate_error(
|
|
format_args!("`{name}` is passed more than once"),
|
|
call.span(),
|
|
));
|
|
}
|
|
_ => {
|
|
if let Some(default_value) = default_value {
|
|
default_value
|
|
} else {
|
|
return Err(ctx.generate_error(format_args!("missing `{arg}` argument"), call.span()));
|
|
}
|
|
}
|
|
}
|
|
};
|
|
match &**expr {
|
|
// If `expr` is already a form of variable then
|
|
// don't reintroduce a new variable. This is
|
|
// to avoid moving non-copyable values.
|
|
Expr::Var(name) if *name != "self" => {
|
|
let var = this.locals.resolve_or_self(name);
|
|
this.locals
|
|
.insert(Cow::Borrowed(arg), LocalMeta::var_with_ref(var));
|
|
}
|
|
Expr::AssociatedItem(obj, associated_item) => {
|
|
let mut associated_item_buf = Buffer::new();
|
|
this.visit_associated_item(ctx, &mut associated_item_buf, obj, associated_item)?;
|
|
|
|
let associated_item = associated_item_buf.into_string();
|
|
let var = this.locals.resolve(&associated_item).unwrap_or(associated_item);
|
|
this.locals
|
|
.insert(Cow::Borrowed(arg), LocalMeta::var_with_ref(var));
|
|
}
|
|
// Everything else still needs to become variables,
|
|
// to avoid having the same logic be executed
|
|
// multiple times, e.g. in the case of macro
|
|
// parameters being used multiple times.
|
|
_ => {
|
|
value.clear();
|
|
let (before, after) = if !is_copyable(expr) {
|
|
("&(", ")")
|
|
} else {
|
|
("", "")
|
|
};
|
|
value.write(this.visit_expr_root(ctx, expr)?);
|
|
// We need to normalize the arg to write it, thus we need to add it to
|
|
// locals in the normalized manner
|
|
let normalized_arg = normalize_identifier(arg);
|
|
buf.write(format_args!("let {normalized_arg} = {before}{value}{after};"));
|
|
this.locals.insert_with_default(Cow::Borrowed(normalized_arg));
|
|
}
|
|
}
|
|
}
|
|
|
|
let mut size_hint = this.handle(own_ctx, &def.nodes, buf, AstLevel::Nested)?;
|
|
|
|
this.flush_ws(def.ws2);
|
|
size_hint += this.write_buf_writable(ctx, buf)?;
|
|
buf.write('}');
|
|
Ok(size_hint)
|
|
})?;
|
|
self.prepare_ws(ws2);
|
|
self.seen_callers.pop();
|
|
Ok(size_hint)
|
|
}
|
|
|
|
fn write_filter_block(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
buf: &mut Buffer,
|
|
filter: &'a WithSpan<'a, FilterBlock<'_>>,
|
|
) -> Result<usize, CompileError> {
|
|
self.write_buf_writable(ctx, buf)?;
|
|
self.flush_ws(filter.ws1);
|
|
self.is_in_filter_block += 1;
|
|
self.write_buf_writable(ctx, buf)?;
|
|
buf.write('{');
|
|
|
|
// build `FmtCell` that contains the inner block
|
|
buf.write(format_args!(
|
|
"let {FILTER_SOURCE} = askama::helpers::FmtCell::new(\
|
|
|__askama_writer: &mut askama::helpers::core::fmt::Formatter<'_>| -> askama::Result<()> {{"
|
|
));
|
|
let size_hint = self.push_locals(|this| {
|
|
this.prepare_ws(filter.ws1);
|
|
let size_hint = this.handle(ctx, &filter.nodes, buf, AstLevel::Nested)?;
|
|
this.flush_ws(filter.ws2);
|
|
this.write_buf_writable(ctx, buf)?;
|
|
Ok(size_hint)
|
|
})?;
|
|
buf.write(
|
|
"\
|
|
askama::Result::Ok(())\
|
|
});",
|
|
);
|
|
|
|
// display the `FmtCell`
|
|
let mut filter_buf = Buffer::new();
|
|
let display_wrap = self.visit_filter(
|
|
ctx,
|
|
&mut filter_buf,
|
|
&filter.filters.name,
|
|
&filter.filters.arguments,
|
|
&filter.filters.generics,
|
|
filter.span(),
|
|
)?;
|
|
let filter_buf = match display_wrap {
|
|
DisplayWrap::Wrapped => fmt_left!("{filter_buf}"),
|
|
DisplayWrap::Unwrapped => fmt_right!(
|
|
"(&&askama::filters::AutoEscaper::new(&({filter_buf}), {})).askama_auto_escape()?",
|
|
self.input.escaper,
|
|
),
|
|
};
|
|
buf.write(format_args!(
|
|
"if askama::helpers::core::write!(__askama_writer, \"{{}}\", {filter_buf}).is_err() {{\
|
|
return {FILTER_SOURCE}.take_err();\
|
|
}}"
|
|
));
|
|
|
|
buf.write('}');
|
|
self.is_in_filter_block -= 1;
|
|
self.prepare_ws(filter.ws2);
|
|
Ok(size_hint)
|
|
}
|
|
|
|
fn handle_include(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
buf: &mut Buffer,
|
|
i: &'a WithSpan<'a, Include<'_>>,
|
|
) -> Result<usize, CompileError> {
|
|
self.flush_ws(i.ws);
|
|
self.write_buf_writable(ctx, buf)?;
|
|
let file_info = ctx
|
|
.path
|
|
.map(|path| FileInfo::of(i.span(), path, ctx.parsed));
|
|
let path = self
|
|
.input
|
|
.config
|
|
.find_template(i.path, Some(&self.input.path), file_info)?;
|
|
|
|
// We clone the context of the child in order to preserve their macros and imports.
|
|
// But also add all the imports and macros from this template that don't override the
|
|
// child's ones to preserve this template's context.
|
|
let child_ctx = &mut self.contexts[&path].clone();
|
|
for (name, mac) in &ctx.macros {
|
|
child_ctx.macros.entry(name).or_insert(mac);
|
|
}
|
|
for (name, import) in &ctx.imports {
|
|
child_ctx
|
|
.imports
|
|
.entry(name)
|
|
.or_insert_with(|| import.clone());
|
|
}
|
|
|
|
// Create a new generator for the child, and call it like in `impl_template` as if it were
|
|
// a full template, while preserving the context.
|
|
let heritage = if !child_ctx.blocks.is_empty() || child_ctx.extends.is_some() {
|
|
Some(Heritage::new(child_ctx, self.contexts))
|
|
} else {
|
|
None
|
|
};
|
|
|
|
let handle_ctx = match &heritage {
|
|
Some(heritage) => heritage.root,
|
|
None => child_ctx,
|
|
};
|
|
|
|
let size_hint = self.with_child(heritage.as_ref(), |child| {
|
|
let mut size_hint = 0;
|
|
size_hint += child.handle(handle_ctx, handle_ctx.nodes, buf, AstLevel::Top)?;
|
|
size_hint += child.write_buf_writable(handle_ctx, buf)?;
|
|
Ok(size_hint)
|
|
})?;
|
|
|
|
self.prepare_ws(i.ws);
|
|
|
|
Ok(size_hint)
|
|
}
|
|
|
|
fn is_shadowing_variable(
|
|
&self,
|
|
ctx: &Context<'_>,
|
|
var: &Target<'a>,
|
|
l: Span<'_>,
|
|
) -> Result<bool, CompileError> {
|
|
match var {
|
|
Target::Name(name) => {
|
|
let name = normalize_identifier(name);
|
|
match self.locals.get(name) {
|
|
// declares a new variable
|
|
None => Ok(false),
|
|
// an initialized variable gets shadowed
|
|
Some(meta) if meta.initialized => Ok(true),
|
|
// initializes a variable that was introduced in a LetDecl before
|
|
_ => Ok(false),
|
|
}
|
|
}
|
|
Target::Placeholder(_) => Ok(false),
|
|
Target::Rest(var_name) => {
|
|
if let Some(var_name) = **var_name {
|
|
match self.is_shadowing_variable(ctx, &Target::Name(var_name), l) {
|
|
Ok(false) => {}
|
|
outcome => return outcome,
|
|
}
|
|
}
|
|
Ok(false)
|
|
}
|
|
Target::Tuple(_, targets) => {
|
|
for target in targets {
|
|
match self.is_shadowing_variable(ctx, target, l) {
|
|
Ok(false) => continue,
|
|
outcome => return outcome,
|
|
}
|
|
}
|
|
Ok(false)
|
|
}
|
|
Target::Struct(_, named_targets) => {
|
|
for (_, target) in named_targets {
|
|
match self.is_shadowing_variable(ctx, target, l) {
|
|
Ok(false) => continue,
|
|
outcome => return outcome,
|
|
}
|
|
}
|
|
Ok(false)
|
|
}
|
|
_ => Err(ctx.generate_error(
|
|
"literals are not allowed on the left-hand side of an assignment",
|
|
l,
|
|
)),
|
|
}
|
|
}
|
|
|
|
fn write_let(
|
|
&mut self,
|
|
ctx: &Context<'_>,
|
|
buf: &mut Buffer,
|
|
l: &'a WithSpan<'a, Let<'_>>,
|
|
) -> Result<(), CompileError> {
|
|
self.handle_ws(l.ws);
|
|
|
|
let Some(val) = &l.val else {
|
|
self.write_buf_writable(ctx, buf)?;
|
|
buf.write("let ");
|
|
if l.is_mutable {
|
|
buf.write("mut ");
|
|
}
|
|
self.visit_target(buf, false, true, &l.var);
|
|
buf.write(';');
|
|
return Ok(());
|
|
};
|
|
|
|
// Handle when this statement creates a new alias of a caller variable (or of another alias),
|
|
if let Target::Name(dstvar) = l.var
|
|
&& let Expr::Var(srcvar) = **val
|
|
&& let Some(caller_alias) = self.locals.get_caller(srcvar)
|
|
{
|
|
self.locals
|
|
.insert(dstvar.into(), LocalMeta::CallerAlias(caller_alias.clone()));
|
|
return Ok(());
|
|
}
|
|
|
|
let mut expr_buf = Buffer::new();
|
|
self.visit_expr(ctx, &mut expr_buf, val)?;
|
|
|
|
let shadowed = self.is_shadowing_variable(ctx, &l.var, l.span())?;
|
|
if shadowed {
|
|
// Need to flush the buffer if the variable is being shadowed,
|
|
// to ensure the old variable is used.
|
|
self.write_buf_writable(ctx, buf)?;
|
|
}
|
|
if shadowed
|
|
|| !matches!(l.var, Target::Name(_))
|
|
|| matches!(&l.var, Target::Name(name) if self.locals.get(name).is_none())
|
|
{
|
|
buf.write("let ");
|
|
if l.is_mutable {
|
|
buf.write("mut ");
|
|
}
|
|
}
|
|
|
|
self.visit_target(buf, true, true, &l.var);
|
|
// If it's not taking the ownership of a local variable or copyable, then we need to add
|
|
// a reference.
|
|
let (before, after) = if !matches!(**val, Expr::Try(..))
|
|
&& !matches!(**val, Expr::Var(name) if self.locals.get(name).is_some())
|
|
&& !is_copyable(val)
|
|
{
|
|
("&(", ")")
|
|
} else {
|
|
("", "")
|
|
};
|
|
buf.write(format_args!(" = {before}{expr_buf}{after};"));
|
|
Ok(())
|
|
}
|
|
|
|
// If `name` is `Some`, this is a call to a block definition, and we have to find
|
|
// the first block for that name from the ancestry chain. If name is `None`, this
|
|
// is from a `super()` call, and we can get the name from `self.super_block`.
|
|
fn write_block(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
buf: &mut Buffer,
|
|
name: Option<&'a str>,
|
|
outer: Ws,
|
|
node: Span<'_>,
|
|
) -> Result<usize, CompileError> {
|
|
if self.is_in_filter_block > 0 {
|
|
return Err(ctx.generate_error("cannot have a block inside a filter block", node));
|
|
}
|
|
// Flush preceding whitespace according to the outer WS spec
|
|
self.flush_ws(outer);
|
|
|
|
let cur = match (name, self.super_block) {
|
|
// The top-level context contains a block definition
|
|
(Some(cur_name), None) => (cur_name, 0),
|
|
// A block definition contains a block definition of the same name
|
|
(Some(cur_name), Some((prev_name, _))) if cur_name == prev_name => {
|
|
return Err(ctx.generate_error(
|
|
format_args!("cannot define recursive blocks ({cur_name})"),
|
|
node,
|
|
));
|
|
}
|
|
// A block definition contains a definition of another block
|
|
(Some(cur_name), Some((_, _))) => (cur_name, 0),
|
|
// `super()` was called inside a block
|
|
(None, Some((prev_name, r#gen))) => (prev_name, r#gen + 1),
|
|
// `super()` is called from outside a block
|
|
(None, None) => {
|
|
return Err(ctx.generate_error("cannot call 'super()' outside block", node));
|
|
}
|
|
};
|
|
|
|
self.write_buf_writable(ctx, buf)?;
|
|
|
|
let block_fragment_write =
|
|
self.input.block.map(|(block, _)| block) == name && self.buf_writable.discard;
|
|
// Allow writing to the buffer if we're in the block fragment
|
|
if block_fragment_write {
|
|
self.buf_writable.discard = false;
|
|
}
|
|
let prev_buf_discard = buf.is_discard();
|
|
buf.set_discard(self.buf_writable.discard);
|
|
|
|
// Get the block definition from the heritage chain
|
|
let heritage = self
|
|
.heritage
|
|
.ok_or_else(|| ctx.generate_error("no block ancestors available", node))?;
|
|
let (child_ctx, def) = *heritage.blocks[cur.0].get(cur.1).ok_or_else(|| {
|
|
ctx.generate_error(
|
|
match name {
|
|
None => fmt_left!("no super() block found for block '{}'", cur.0),
|
|
Some(name) => fmt_right!(move "no block found for name '{name}'"),
|
|
},
|
|
node,
|
|
)
|
|
})?;
|
|
|
|
// We clone the context of the child in order to preserve their macros and imports.
|
|
// But also add all the imports and macros from this template that don't override the
|
|
// child's ones to preserve this template's context.
|
|
let mut child_ctx = child_ctx.clone();
|
|
for (name, mac) in &ctx.macros {
|
|
child_ctx.macros.entry(name).or_insert(mac);
|
|
}
|
|
for (name, import) in &ctx.imports {
|
|
child_ctx
|
|
.imports
|
|
.entry(name)
|
|
.or_insert_with(|| import.clone());
|
|
}
|
|
|
|
let size_hint = self.with_child(Some(heritage), |child| {
|
|
// Handle inner whitespace suppression spec and process block nodes
|
|
child.prepare_ws(def.ws1);
|
|
|
|
child.super_block = Some(cur);
|
|
let size_hint = child.handle(&child_ctx, &def.nodes, buf, AstLevel::Block)?;
|
|
|
|
if !child.locals.is_current_empty() {
|
|
// Need to flush the buffer before popping the variable stack
|
|
child.write_buf_writable(ctx, buf)?;
|
|
}
|
|
|
|
child.flush_ws(def.ws2);
|
|
Ok(size_hint)
|
|
})?;
|
|
|
|
// Restore original block context and set whitespace suppression for
|
|
// succeeding whitespace according to the outer WS spec
|
|
self.prepare_ws(outer);
|
|
|
|
// If we are rendering a specific block and the discard changed, it means that we're done
|
|
// with the block we want to render and that from this point, everything will be discarded.
|
|
//
|
|
// To get this block content rendered as well, we need to write to the buffer before then.
|
|
if buf.is_discard() != prev_buf_discard {
|
|
self.write_buf_writable(ctx, buf)?;
|
|
}
|
|
// Restore the original buffer discarding state
|
|
if block_fragment_write {
|
|
self.buf_writable.discard = true;
|
|
}
|
|
buf.set_discard(prev_buf_discard);
|
|
|
|
Ok(size_hint)
|
|
}
|
|
|
|
fn write_expr(
|
|
&mut self,
|
|
ctx: &Context<'a>,
|
|
buf: &mut Buffer,
|
|
ws: Ws,
|
|
s: &'a WithSpan<'a, Expr<'a>>,
|
|
) -> Result<usize, CompileError> {
|
|
if let Expr::Call(v) = &**s {
|
|
fn check_num_args<'a>(
|
|
s: &'a WithSpan<'a, Expr<'a>>,
|
|
ctx: &Context<'a>,
|
|
expected: usize,
|
|
found: usize,
|
|
name: &str,
|
|
) -> Result<(), CompileError> {
|
|
if expected != found {
|
|
Err(ctx.generate_error(
|
|
format!(
|
|
"expected {expected} argument{} in `{name}`, found {found}",
|
|
if expected != 1 { "s" } else { "" }
|
|
),
|
|
s.span(),
|
|
))
|
|
} else {
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
let var_name = match *v.path {
|
|
Expr::Var(var_name) => Some(var_name),
|
|
_ => None,
|
|
};
|
|
let caller_alias = var_name.and_then(|vn| self.locals.get_caller(vn));
|
|
|
|
if let Some("super") = var_name {
|
|
check_num_args(s, ctx, 0, v.args.len(), "super")?;
|
|
return self.write_block(ctx, buf, None, ws, s.span());
|
|
} else if let Some("caller") = var_name
|
|
&& caller_alias.is_none()
|
|
{
|
|
// attempted to use keyword `caller` - but no caller is currently in scope
|
|
return Err(ctx.generate_error("block is not defined for `caller`", s.span()));
|
|
} else if let Some(LocalCallerMeta { call_ctx, def }) = caller_alias.cloned() {
|
|
self.handle_ws(ws);
|
|
let size_hint = self.push_locals(|this| {
|
|
// Block-out the special caller() variable from this scope onward until it is defined by a
|
|
// new call-block again. This prohibits a caller from calling itself.
|
|
this.locals.insert("caller".into(), LocalMeta::Negative);
|
|
|
|
this.write_buf_writable(&call_ctx, buf)?;
|
|
buf.write('{');
|
|
this.prepare_ws(def.ws1);
|
|
let mut value = Buffer::new();
|
|
check_num_args(s, &call_ctx, def.caller_args.len(), v.args.len(), "caller")?;
|
|
for (index, arg) in def.caller_args.iter().enumerate() {
|
|
match v.args.get(index) {
|
|
Some(expr) => {
|
|
value.clear();
|
|
match &**expr {
|
|
// If `expr` is already a form of variable then
|
|
// don't reintroduce a new variable. This is
|
|
// to avoid moving non-copyable values.
|
|
&Expr::Var(name) if name != "self" => {
|
|
let var = this.locals.resolve_or_self(name);
|
|
this.locals.insert(
|
|
Cow::Borrowed(arg),
|
|
LocalMeta::var_with_ref(var),
|
|
);
|
|
}
|
|
Expr::AssociatedItem(obj, associated_item) => {
|
|
let mut associated_item_buf = Buffer::new();
|
|
this.visit_associated_item(
|
|
&call_ctx,
|
|
&mut associated_item_buf,
|
|
obj,
|
|
associated_item,
|
|
)?;
|
|
|
|
let associated_item = associated_item_buf.into_string();
|
|
let var = this
|
|
.locals
|
|
.resolve(&associated_item)
|
|
.unwrap_or(associated_item);
|
|
this.locals.insert(
|
|
Cow::Borrowed(arg),
|
|
LocalMeta::var_with_ref(var),
|
|
);
|
|
}
|
|
// Everything else still needs to become variables,
|
|
// to avoid having the same logic be executed
|
|
// multiple times, e.g. in the case of macro
|
|
// parameters being used multiple times.
|
|
_ => {
|
|
let (before, after) = if !is_copyable(expr) {
|
|
("&(", ")")
|
|
} else {
|
|
("", "")
|
|
};
|
|
value.write(this.visit_expr_root(&call_ctx, expr)?);
|
|
// We need to normalize the arg to write it, thus we need to add it to
|
|
// locals in the normalized manner
|
|
let normalized_arg = normalize_identifier(arg);
|
|
buf.write(format_args!(
|
|
"let {normalized_arg} = {before}{value}{after};"
|
|
));
|
|
this.locals
|
|
.insert_with_default(Cow::Borrowed(normalized_arg));
|
|
}
|
|
}
|
|
}
|
|
None => {
|
|
return Err(call_ctx.generate_error(
|
|
format_args!("missing `{arg}` argument in `caller`"),
|
|
s.span(),
|
|
));
|
|
}
|
|
}
|
|
}
|
|
let mut size_hint =
|
|
this.handle(&call_ctx, &def.nodes, buf, AstLevel::Nested)?;
|
|
|
|
this.flush_ws(def.ws2);
|
|
size_hint += this.write_buf_writable(&call_ctx, buf)?;
|
|
buf.write('}');
|
|
Ok(size_hint)
|
|
})?;
|
|
return Ok(size_hint);
|
|
}
|
|
}
|
|
|
|
self.handle_ws(ws);
|
|
let items = if let Expr::Concat(exprs) = &**s {
|
|
exprs
|
|
} else {
|
|
std::slice::from_ref(s)
|
|
};
|
|
|
|
for s in items {
|
|
self.buf_writable
|
|
.push(compile_time_escape(s, self.input.escaper).unwrap_or(Writable::Expr(s)));
|
|
}
|
|
Ok(0)
|
|
}
|
|
|
|
// Write expression buffer and empty
|
|
fn write_buf_writable(
|
|
&mut self,
|
|
ctx: &Context<'_>,
|
|
buf: &mut Buffer,
|
|
) -> Result<usize, CompileError> {
|
|
let mut size_hint = 0;
|
|
let items = mem::take(&mut self.buf_writable.buf);
|
|
let mut it = items.iter().enumerate().peekable();
|
|
|
|
while let Some((_, Writable::Lit(s))) = it.peek() {
|
|
size_hint += buf.write_writer(s);
|
|
it.next();
|
|
}
|
|
if it.peek().is_none() {
|
|
return Ok(size_hint);
|
|
}
|
|
|
|
let mut targets = Buffer::new();
|
|
let mut lines = Buffer::new();
|
|
let mut expr_cache = HashMap::with_capacity(self.buf_writable.len());
|
|
// the `last_line` contains any sequence of trailing simple `writer.write_str()` calls
|
|
let mut trailing_simple_lines = Vec::new();
|
|
|
|
buf.write("match (");
|
|
while let Some((idx, s)) = it.next() {
|
|
match s {
|
|
Writable::Lit(s) => {
|
|
let mut items = vec![s];
|
|
while let Some((_, Writable::Lit(s))) = it.peek() {
|
|
items.push(s);
|
|
it.next();
|
|
}
|
|
if it.peek().is_some() {
|
|
for s in items {
|
|
size_hint += lines.write_writer(s);
|
|
}
|
|
} else {
|
|
trailing_simple_lines = items;
|
|
break;
|
|
}
|
|
}
|
|
Writable::Expr(s) => {
|
|
size_hint += 3;
|
|
|
|
let mut expr_buf = Buffer::new();
|
|
let expr = match self.visit_expr(ctx, &mut expr_buf, s)? {
|
|
DisplayWrap::Wrapped => expr_buf.into_string(),
|
|
DisplayWrap::Unwrapped => format!(
|
|
"(&&askama::filters::AutoEscaper::new(&({expr_buf}), {})).\
|
|
askama_auto_escape()?",
|
|
self.input.escaper,
|
|
),
|
|
};
|
|
let idx = if is_cacheable(s) {
|
|
match expr_cache.entry(expr) {
|
|
Entry::Occupied(e) => *e.get(),
|
|
Entry::Vacant(e) => {
|
|
buf.write(format_args!("&({}),", e.key()));
|
|
targets.write(format_args!("expr{idx},"));
|
|
e.insert(idx);
|
|
idx
|
|
}
|
|
}
|
|
} else {
|
|
buf.write(format_args!("&({expr}),"));
|
|
targets.write(format_args!("expr{idx}, "));
|
|
idx
|
|
};
|
|
lines.write(format_args!(
|
|
"(&&&askama::filters::Writable(expr{idx})).\
|
|
askama_write(__askama_writer, __askama_values)?;",
|
|
));
|
|
}
|
|
}
|
|
}
|
|
buf.write(format_args!(
|
|
") {{\
|
|
({targets}) => {{\
|
|
{lines}\
|
|
}}\
|
|
}}"
|
|
));
|
|
|
|
for s in trailing_simple_lines {
|
|
size_hint += buf.write_writer(s);
|
|
}
|
|
|
|
Ok(size_hint)
|
|
}
|
|
|
|
fn write_comment(&mut self, comment: &'a WithSpan<'a, Comment<'_>>) {
|
|
self.handle_ws(comment.ws);
|
|
}
|
|
|
|
fn write_lit(&mut self, lit: &'a Lit<'_>) {
|
|
assert!(self.next_ws.is_none());
|
|
let Lit { lws, val, rws } = *lit;
|
|
if !lws.is_empty() {
|
|
match self.skip_ws {
|
|
Whitespace::Suppress => {}
|
|
_ if val.is_empty() => {
|
|
assert!(rws.is_empty());
|
|
self.next_ws = Some(lws);
|
|
}
|
|
Whitespace::Preserve => {
|
|
self.buf_writable.push(Writable::Lit(Cow::Borrowed(lws)));
|
|
}
|
|
Whitespace::Minimize => {
|
|
self.buf_writable.push(Writable::Lit(Cow::Borrowed(
|
|
match lws.contains('\n') {
|
|
true => "\n",
|
|
false => " ",
|
|
},
|
|
)));
|
|
}
|
|
}
|
|
}
|
|
|
|
if !val.is_empty() {
|
|
self.skip_ws = Whitespace::Preserve;
|
|
self.buf_writable.push(Writable::Lit(Cow::Borrowed(val)));
|
|
}
|
|
|
|
if !rws.is_empty() {
|
|
self.next_ws = Some(rws);
|
|
}
|
|
}
|
|
|
|
// Helper methods for dealing with whitespace nodes
|
|
|
|
// Combines `flush_ws()` and `prepare_ws()` to handle both trailing whitespace from the
|
|
// preceding literal and leading whitespace from the succeeding literal.
|
|
fn handle_ws(&mut self, ws: Ws) {
|
|
self.flush_ws(ws);
|
|
self.prepare_ws(ws);
|
|
}
|
|
|
|
fn should_trim_ws(&self, ws: Option<Whitespace>) -> Whitespace {
|
|
ws.unwrap_or(self.input.config.whitespace)
|
|
}
|
|
|
|
// If the previous literal left some trailing whitespace in `next_ws` and the
|
|
// prefix whitespace suppressor from the given argument, flush that whitespace.
|
|
// In either case, `next_ws` is reset to `None` (no trailing whitespace).
|
|
fn flush_ws(&mut self, ws: Ws) {
|
|
if self.next_ws.is_none() {
|
|
return;
|
|
}
|
|
|
|
// If `whitespace` is set to `suppress`, we keep the whitespace characters only if there is
|
|
// a `+` character.
|
|
match self.should_trim_ws(ws.0) {
|
|
Whitespace::Preserve => {
|
|
let val = self.next_ws.unwrap();
|
|
if !val.is_empty() {
|
|
self.buf_writable.push(Writable::Lit(Cow::Borrowed(val)));
|
|
}
|
|
}
|
|
Whitespace::Minimize => {
|
|
let val = self.next_ws.unwrap();
|
|
if !val.is_empty() {
|
|
self.buf_writable.push(Writable::Lit(Cow::Borrowed(
|
|
match val.contains('\n') {
|
|
true => "\n",
|
|
false => " ",
|
|
},
|
|
)));
|
|
}
|
|
}
|
|
Whitespace::Suppress => {}
|
|
}
|
|
self.next_ws = None;
|
|
}
|
|
|
|
// Sets `skip_ws` to match the suffix whitespace suppressor from the given
|
|
// argument, to determine whether to suppress leading whitespace from the
|
|
// next literal.
|
|
fn prepare_ws(&mut self, ws: Ws) {
|
|
self.skip_ws = self.should_trim_ws(ws.1);
|
|
}
|
|
}
|
|
|
|
fn bin_op<'a>(
|
|
span: impl Into<Span<'a>>,
|
|
op: &'a str,
|
|
lhs: WithSpan<'a, Expr<'a>>,
|
|
rhs: WithSpan<'a, Expr<'a>>,
|
|
) -> WithSpan<'a, Expr<'a>> {
|
|
WithSpan::new(Expr::BinOp(Box::new(BinOp { op, lhs, rhs })), span)
|
|
}
|
|
|
|
struct CondInfo<'a> {
|
|
cond: &'a WithSpan<'a, Cond<'a>>,
|
|
cond_expr: Option<WithSpan<'a, Expr<'a>>>,
|
|
generate_condition: bool,
|
|
generate_content: bool,
|
|
}
|
|
|
|
struct Conds<'a> {
|
|
conds: Vec<CondInfo<'a>>,
|
|
ws_before: Option<Ws>,
|
|
ws_after: Option<Ws>,
|
|
nb_conds: usize,
|
|
}
|
|
|
|
#[derive(Clone, Copy, PartialEq, Debug)]
|
|
enum EvaluatedResult {
|
|
AlwaysTrue,
|
|
AlwaysFalse,
|
|
Unknown,
|
|
}
|
|
|
|
impl<'a> Conds<'a> {
|
|
fn compute_branches(generator: &Generator<'a, '_>, i: &'a If<'a>) -> Self {
|
|
let mut conds = Vec::with_capacity(i.branches.len());
|
|
let mut ws_before = None;
|
|
let mut ws_after = None;
|
|
let mut nb_conds = 0;
|
|
let mut stop_loop = false;
|
|
|
|
for cond in &i.branches {
|
|
if stop_loop {
|
|
ws_after = Some(cond.ws);
|
|
break;
|
|
}
|
|
if let Some(CondTest {
|
|
expr,
|
|
contains_bool_lit_or_is_defined,
|
|
..
|
|
}) = &cond.cond
|
|
{
|
|
let mut only_contains_is_defined = true;
|
|
|
|
let (evaluated_result, cond_expr) = if *contains_bool_lit_or_is_defined {
|
|
let (evaluated_result, expr) =
|
|
generator.evaluate_condition(expr.clone(), &mut only_contains_is_defined);
|
|
(evaluated_result, Some(expr))
|
|
} else {
|
|
(EvaluatedResult::Unknown, None)
|
|
};
|
|
|
|
match evaluated_result {
|
|
// We generate the condition in case some calls are changing a variable, but
|
|
// no need to generate the condition body since it will never be called.
|
|
//
|
|
// However, if the condition only contains "is (not) defined" checks, then we
|
|
// can completely skip it.
|
|
EvaluatedResult::AlwaysFalse => {
|
|
if only_contains_is_defined {
|
|
if conds.is_empty() && ws_before.is_none() {
|
|
// If this is the first `if` and it's skipped, we definitely don't
|
|
// want its whitespace control to be lost.
|
|
ws_before = Some(cond.ws);
|
|
}
|
|
continue;
|
|
}
|
|
nb_conds += 1;
|
|
conds.push(CondInfo {
|
|
cond,
|
|
cond_expr,
|
|
generate_condition: true,
|
|
generate_content: false,
|
|
});
|
|
}
|
|
// This case is more interesting: it means that we will always enter this
|
|
// condition, meaning that any following should not be generated. Another
|
|
// thing to take into account: if there are no if branches before this one,
|
|
// no need to generate an `else`.
|
|
EvaluatedResult::AlwaysTrue => {
|
|
let generate_condition = !only_contains_is_defined;
|
|
if generate_condition {
|
|
nb_conds += 1;
|
|
}
|
|
conds.push(CondInfo {
|
|
cond,
|
|
cond_expr,
|
|
generate_condition,
|
|
generate_content: true,
|
|
});
|
|
// Since it's always true, we can stop here.
|
|
stop_loop = true;
|
|
}
|
|
EvaluatedResult::Unknown => {
|
|
nb_conds += 1;
|
|
conds.push(CondInfo {
|
|
cond,
|
|
cond_expr,
|
|
generate_condition: true,
|
|
generate_content: true,
|
|
});
|
|
}
|
|
}
|
|
} else {
|
|
let generate_condition = !conds.is_empty();
|
|
if generate_condition {
|
|
nb_conds += 1;
|
|
}
|
|
conds.push(CondInfo {
|
|
cond,
|
|
cond_expr: None,
|
|
generate_condition,
|
|
generate_content: true,
|
|
});
|
|
}
|
|
}
|
|
Self {
|
|
conds,
|
|
ws_before,
|
|
ws_after,
|
|
nb_conds,
|
|
}
|
|
}
|
|
}
|
|
|
|
fn median(sizes: &mut [usize]) -> usize {
|
|
if sizes.is_empty() {
|
|
return 0;
|
|
}
|
|
sizes.sort_unstable();
|
|
if sizes.len() % 2 == 1 {
|
|
sizes[sizes.len() / 2]
|
|
} else {
|
|
(sizes[sizes.len() / 2 - 1] + sizes[sizes.len() / 2]) / 2
|
|
}
|
|
}
|
|
|
|
fn macro_call_ensure_arg_count(
|
|
call: &WithSpan<'_, Call<'_>>,
|
|
def: &Macro<'_>,
|
|
ctx: &Context<'_>,
|
|
) -> Result<(), CompileError> {
|
|
if call.args.len() > def.args.len() {
|
|
return Err(ctx.generate_error(
|
|
format_args!(
|
|
"macro `{}` expected {} argument{}, found {}",
|
|
def.name,
|
|
def.args.len(),
|
|
if def.args.len() > 1 { "s" } else { "" },
|
|
call.args.len(),
|
|
),
|
|
call.span(),
|
|
));
|
|
}
|
|
|
|
// First we list of arguments position, then we remove every argument with a value.
|
|
let mut args: Vec<_> = def.args.iter().map(|&(name, _)| Some(name)).collect();
|
|
for (pos, arg) in call.args.iter().enumerate() {
|
|
let pos = match **arg {
|
|
Expr::NamedArgument(name, ..) => {
|
|
def.args.iter().position(|(arg_name, _)| *arg_name == name)
|
|
}
|
|
_ => Some(pos),
|
|
};
|
|
if let Some(pos) = pos
|
|
&& mem::take(&mut args[pos]).is_none()
|
|
{
|
|
// This argument was already passed, so error.
|
|
return Err(ctx.generate_error(
|
|
format_args!(
|
|
"argument `{}` was passed more than once when calling macro `{}`",
|
|
def.args[pos].0, def.name,
|
|
),
|
|
call.span(),
|
|
));
|
|
}
|
|
}
|
|
|
|
// Now we can check off arguments with a default value, too.
|
|
for (pos, (_, dflt)) in def.args.iter().enumerate() {
|
|
if dflt.is_some() {
|
|
args[pos] = None;
|
|
}
|
|
}
|
|
|
|
// Now that we have a needed information, we can print an error message (if needed).
|
|
struct FmtMissing<'a, I> {
|
|
count: usize,
|
|
missing: I,
|
|
name: &'a str,
|
|
}
|
|
|
|
impl<'a, I: Iterator<Item = &'a str> + Clone> fmt::Display for FmtMissing<'a, I> {
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
if self.count == 1 {
|
|
let a = self.missing.clone().next().unwrap();
|
|
write!(
|
|
f,
|
|
"missing argument when calling macro `{}`: `{a}`",
|
|
self.name
|
|
)
|
|
} else {
|
|
write!(f, "missing arguments when calling macro `{}`: ", self.name)?;
|
|
for (idx, a) in self.missing.clone().enumerate() {
|
|
if idx == self.count - 1 {
|
|
write!(f, " and ")?;
|
|
} else if idx > 0 {
|
|
write!(f, ", ")?;
|
|
}
|
|
write!(f, "`{a}`")?;
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
}
|
|
|
|
let missing = args.iter().filter_map(Option::as_deref);
|
|
let fmt_missing = FmtMissing {
|
|
count: missing.clone().count(),
|
|
missing,
|
|
name: def.name,
|
|
};
|
|
if fmt_missing.count == 0 {
|
|
Ok(())
|
|
} else {
|
|
Err(ctx.generate_error(fmt_missing, call.span()))
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy, PartialEq)]
|
|
enum AstLevel {
|
|
Top,
|
|
Block,
|
|
Nested,
|
|
}
|
|
|
|
/// Returns `true` if the outcome of this expression may be used multiple times in the same
|
|
/// `write!()` call, without evaluating the expression again, i.e. the expression should be
|
|
/// side-effect free.
|
|
fn is_cacheable(expr: &WithSpan<'_, Expr<'_>>) -> bool {
|
|
match &**expr {
|
|
// Literals are the definition of pure:
|
|
Expr::BoolLit(_) => true,
|
|
Expr::NumLit(_, _) => true,
|
|
Expr::StrLit(_) => true,
|
|
Expr::CharLit(_) => true,
|
|
// fmt::Display should have no effects:
|
|
Expr::Var(_) => true,
|
|
Expr::Path(_) => true,
|
|
// Check recursively:
|
|
Expr::Array(args) => args.iter().all(is_cacheable),
|
|
Expr::AssociatedItem(lhs, _) => is_cacheable(lhs),
|
|
Expr::Index(lhs, rhs) => is_cacheable(lhs) && is_cacheable(rhs),
|
|
Expr::Filter(v) => v.arguments.iter().all(is_cacheable),
|
|
Expr::Unary(_, arg) => is_cacheable(arg),
|
|
Expr::BinOp(v) => is_cacheable(&v.lhs) && is_cacheable(&v.rhs),
|
|
Expr::IsDefined(_) | Expr::IsNotDefined(_) => true,
|
|
Expr::Range(v) => {
|
|
v.lhs.as_ref().is_none_or(is_cacheable) && v.rhs.as_ref().is_none_or(is_cacheable)
|
|
}
|
|
Expr::Group(arg) => is_cacheable(arg),
|
|
Expr::Tuple(args) => args.iter().all(is_cacheable),
|
|
Expr::NamedArgument(_, expr) => is_cacheable(expr),
|
|
Expr::As(expr, _) => is_cacheable(expr),
|
|
Expr::Try(expr) => is_cacheable(expr),
|
|
Expr::Concat(args) => args.iter().all(is_cacheable),
|
|
// Doesn't make sense in this context.
|
|
Expr::LetCond(_) => false,
|
|
// We have too little information to tell if the expression is pure:
|
|
Expr::Call { .. } => false,
|
|
Expr::RustMacro(_, _) => false,
|
|
// Should never be encountered:
|
|
Expr::FilterSource => unreachable!("FilterSource in expression?"),
|
|
Expr::ArgumentPlaceholder => unreachable!("ExpressionPlaceholder in expression?"),
|
|
}
|
|
}
|