Skip to content

Commit

Permalink
Removed implicit borrowing of literals (fixes #404)
Browse files Browse the repository at this point in the history
  • Loading branch information
vallentin committed Jan 2, 2021
1 parent c29ecd6 commit 1d60d57
Show file tree
Hide file tree
Showing 4 changed files with 89 additions and 44 deletions.
74 changes: 37 additions & 37 deletions askama_shared/src/filters/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -241,12 +241,12 @@ pub fn trim(s: &dyn fmt::Display) -> Result<String> {
}

/// Limit string length, appends '...' if truncated
pub fn truncate(s: &dyn fmt::Display, len: &usize) -> Result<String> {
pub fn truncate(s: &dyn fmt::Display, len: usize) -> Result<String> {
let mut s = s.to_string();
if s.len() <= *len {
if s.len() <= len {
Ok(s)
} else {
let mut real_len = *len;
let mut real_len = len;
while !s.is_char_boundary(real_len) {
real_len += 1;
}
Expand All @@ -257,7 +257,7 @@ pub fn truncate(s: &dyn fmt::Display, len: &usize) -> Result<String> {
}

/// Indent lines with `width` spaces
pub fn indent(s: &dyn fmt::Display, width: &usize) -> Result<String> {
pub fn indent(s: &dyn fmt::Display, width: usize) -> Result<String> {
let s = s.to_string();

let mut indented = String::new();
Expand All @@ -266,7 +266,7 @@ pub fn indent(s: &dyn fmt::Display, width: &usize) -> Result<String> {
indented.push(c);

if c == '\n' && i < s.len() - 1 {
for _ in 0..*width {
for _ in 0..width {
indented.push(' ');
}
}
Expand All @@ -277,7 +277,7 @@ pub fn indent(s: &dyn fmt::Display, width: &usize) -> Result<String> {

#[cfg(feature = "num-traits")]
/// Casts number to f64
pub fn into_f64<T>(number: &T) -> Result<f64>
pub fn into_f64<T>(number: T) -> Result<f64>
where
T: NumCast,
{
Expand All @@ -286,7 +286,7 @@ where

#[cfg(feature = "num-traits")]
/// Casts number to isize
pub fn into_isize<T>(number: &T) -> Result<isize>
pub fn into_isize<T>(number: T) -> Result<isize>
where
T: NumCast,
{
Expand Down Expand Up @@ -480,36 +480,36 @@ mod tests {

#[test]
fn test_truncate() {
assert_eq!(truncate(&"hello", &2).unwrap(), "he...");
assert_eq!(truncate(&"hello", 2).unwrap(), "he...");
let a = String::from("您好");
assert_eq!(a.len(), 6);
assert_eq!(String::from("您").len(), 3);
assert_eq!(truncate(&"您好", &1).unwrap(), "您...");
assert_eq!(truncate(&"您好", &2).unwrap(), "您...");
assert_eq!(truncate(&"您好", &3).unwrap(), "您...");
assert_eq!(truncate(&"您好", &4).unwrap(), "您好...");
assert_eq!(truncate(&"您好", &6).unwrap(), "您好");
assert_eq!(truncate(&"您好", &7).unwrap(), "您好");
assert_eq!(truncate(&"您好", 1).unwrap(), "您...");
assert_eq!(truncate(&"您好", 2).unwrap(), "您...");
assert_eq!(truncate(&"您好", 3).unwrap(), "您...");
assert_eq!(truncate(&"您好", 4).unwrap(), "您好...");
assert_eq!(truncate(&"您好", 6).unwrap(), "您好");
assert_eq!(truncate(&"您好", 7).unwrap(), "您好");
let s = String::from("🤚a🤚");
assert_eq!(s.len(), 9);
assert_eq!(String::from("🤚").len(), 4);
assert_eq!(truncate(&"🤚a🤚", &1).unwrap(), "🤚...");
assert_eq!(truncate(&"🤚a🤚", &2).unwrap(), "🤚...");
assert_eq!(truncate(&"🤚a🤚", &3).unwrap(), "🤚...");
assert_eq!(truncate(&"🤚a🤚", &4).unwrap(), "🤚...");
assert_eq!(truncate(&"🤚a🤚", &5).unwrap(), "🤚a...");
assert_eq!(truncate(&"🤚a🤚", &6).unwrap(), "🤚a🤚...");
assert_eq!(truncate(&"🤚a🤚", &9).unwrap(), "🤚a🤚");
assert_eq!(truncate(&"🤚a🤚", &10).unwrap(), "🤚a🤚");
assert_eq!(truncate(&"🤚a🤚", 1).unwrap(), "🤚...");
assert_eq!(truncate(&"🤚a🤚", 2).unwrap(), "🤚...");
assert_eq!(truncate(&"🤚a🤚", 3).unwrap(), "🤚...");
assert_eq!(truncate(&"🤚a🤚", 4).unwrap(), "🤚...");
assert_eq!(truncate(&"🤚a🤚", 5).unwrap(), "🤚a...");
assert_eq!(truncate(&"🤚a🤚", 6).unwrap(), "🤚a🤚...");
assert_eq!(truncate(&"🤚a🤚", 9).unwrap(), "🤚a🤚");
assert_eq!(truncate(&"🤚a🤚", 10).unwrap(), "🤚a🤚");
}

#[test]
fn test_indent() {
assert_eq!(indent(&"hello", &2).unwrap(), "hello");
assert_eq!(indent(&"hello\n", &2).unwrap(), "hello\n");
assert_eq!(indent(&"hello\nfoo", &2).unwrap(), "hello\n foo");
assert_eq!(indent(&"hello", 2).unwrap(), "hello");
assert_eq!(indent(&"hello\n", 2).unwrap(), "hello\n");
assert_eq!(indent(&"hello\nfoo", 2).unwrap(), "hello\n foo");
assert_eq!(
indent(&"hello\nfoo\n bar", &4).unwrap(),
indent(&"hello\nfoo\n bar", 4).unwrap(),
"hello\n foo\n bar"
);
}
Expand All @@ -518,22 +518,22 @@ mod tests {
#[test]
#[allow(clippy::float_cmp)]
fn test_into_f64() {
assert_eq!(into_f64(&1).unwrap(), 1.0_f64);
assert_eq!(into_f64(&1.9).unwrap(), 1.9_f64);
assert_eq!(into_f64(&-1.9).unwrap(), -1.9_f64);
assert_eq!(into_f64(&(INFINITY as f32)).unwrap(), INFINITY);
assert_eq!(into_f64(&(-INFINITY as f32)).unwrap(), -INFINITY);
assert_eq!(into_f64(1).unwrap(), 1.0_f64);
assert_eq!(into_f64(1.9).unwrap(), 1.9_f64);
assert_eq!(into_f64(-1.9).unwrap(), -1.9_f64);
assert_eq!(into_f64(INFINITY as f32).unwrap(), INFINITY);
assert_eq!(into_f64(-INFINITY as f32).unwrap(), -INFINITY);
}

#[cfg(feature = "num-traits")]
#[test]
fn test_into_isize() {
assert_eq!(into_isize(&1).unwrap(), 1_isize);
assert_eq!(into_isize(&1.9).unwrap(), 1_isize);
assert_eq!(into_isize(&-1.9).unwrap(), -1_isize);
assert_eq!(into_isize(&(1.5_f64)).unwrap(), 1_isize);
assert_eq!(into_isize(&(-1.5_f64)).unwrap(), -1_isize);
match into_isize(&INFINITY) {
assert_eq!(into_isize(1).unwrap(), 1_isize);
assert_eq!(into_isize(1.9).unwrap(), 1_isize);
assert_eq!(into_isize(-1.9).unwrap(), -1_isize);
assert_eq!(into_isize(1.5_f64).unwrap(), 1_isize);
assert_eq!(into_isize(-1.5_f64).unwrap(), -1_isize);
match into_isize(INFINITY) {
Err(Fmt(fmt::Error)) => {}
_ => panic!("Should return error of type Err(Fmt(fmt::Error))"),
};
Expand Down
13 changes: 10 additions & 3 deletions askama_shared/src/generator.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1275,9 +1275,12 @@ impl<'a, S: std::hash::BuildHasher> Generator<'a, S> {

for (i, arg) in args.iter().enumerate() {
if i > 0 {
buf.write(", &");
} else {
buf.write("&");
buf.write(", ");
}

let borrow = !arg.is_copyable();
if borrow {
buf.write("&(");
}

let scoped = matches!(arg,
Expand All @@ -1293,6 +1296,10 @@ impl<'a, S: std::hash::BuildHasher> Generator<'a, S> {
} else {
self.visit_expr(buf, arg)?;
}

if borrow {
buf.write(")");
}
}
Ok(())
}
Expand Down
38 changes: 38 additions & 0 deletions askama_shared/src/parser.rs
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,44 @@ pub enum Expr<'a> {
RustMacro(&'a str, &'a str),
}

impl Expr<'_> {
/// Returns `true` if enough assumptions can be made,
/// to determine that `self` is copyable.
pub fn is_copyable(&self) -> bool {
fn is_copyable(expr: &Expr, within_op: bool) -> bool {
use Expr::*;
match expr {
BoolLit(_) | NumLit(_) | StrLit(_) | CharLit(_) => true,
Unary(.., expr) => is_copyable(expr, true),
BinOp(_, lhs, rhs) => is_copyable(lhs, true) && is_copyable(rhs, true),
// FIXME: The parser should produce `NumLit(_)` instead of `Attr(NumLit(_), _)` for floats
Attr(obj, _) if matches!(obj.as_ref(), NumLit(_)) => true,
// The result of a call likely doesn't need to be borrowed,
// as in that case the call is more likely to return a
// reference in the first place then.
VarCall(..) | PathCall(..) | MethodCall(..) => true,
// If the `expr` is within a `Unary` or `BinOp` then
// an assumption can be made that the operand is copy.
// If not, then the value is moved and adding `.clone()`
// will solve that issue. However, if the operand is
// implicitly borrowed, then it's likely not even possible
// to get the template to compile.
_ => within_op && expr.is_attr_self(),
}
}
is_copyable(self, false)
}

/// Returns `true` if this is an `Attr` where the `obj` is `"self"`.
pub fn is_attr_self(&self) -> bool {
match self {
Expr::Attr(obj, _) if matches!(obj.as_ref(), Expr::Var("self")) => true,
Expr::Attr(obj, _) if matches!(obj.as_ref(), Expr::Attr(..)) => obj.is_attr_self(),
_ => false,
}
}
}

pub type When<'a> = (
WS,
Option<MatchVariant<'a>>,
Expand Down
8 changes: 4 additions & 4 deletions testing/tests/simple.rs
Original file line number Diff line number Diff line change
Expand Up @@ -265,7 +265,7 @@ fn test_slice_literal() {
#[derive(Template)]
#[template(source = "Hello, {{ world(\"123\", 4) }}!", ext = "txt")]
struct FunctionRefTemplate {
world: fn(s: &str, v: &u8) -> String,
world: fn(s: &str, v: u8) -> String,
}

#[test]
Expand All @@ -277,7 +277,7 @@ fn test_func_ref_call() {
}

#[allow(clippy::trivially_copy_pass_by_ref)]
fn world2(s: &str, v: &u8) -> String {
fn world2(s: &str, v: u8) -> String {
format!("world{}{}", v, s)
}

Expand All @@ -291,7 +291,7 @@ fn test_path_func_call() {
}

#[derive(Template)]
#[template(source = "{{ ::std::string::ToString::to_string(123) }}", ext = "txt")]
#[template(source = "{{ ::std::string::String::from(\"123\") }}", ext = "txt")]
struct RootPathFunctionTemplate;

#[test]
Expand All @@ -305,7 +305,7 @@ struct FunctionTemplate;

impl FunctionTemplate {
#[allow(clippy::trivially_copy_pass_by_ref)]
fn world3(&self, s: &str, v: &u8) -> String {
fn world3(&self, s: &str, v: u8) -> String {
format!("world{}{}", s, v)
}
}
Expand Down

0 comments on commit 1d60d57

Please sign in to comment.