首页 > 解决方案 > 使用动态运算符创建 Ecto 查询

问题描述

我知道如何在 Ecto 查询中使用动态字段和值,例如:

field = :age
value = 20
Ecto.Query.where(App.User, [x], field(x, ^field) < ^value)

但是,是否可以<动态定义运算符(在本例中)?如何?

我已经尝试过fragment并插入字符串:

operator = ">"
Ecto.Query.where(App.User, [x], fragment("? #{operator} ?", field(x, ^field), ^value))

但它引发了异常:Ecto.Query.CompileError) to prevent SQL injection attacks, fragment(...) does not allow strings...

标签: elixirphoenix-frameworkectophoenix

解决方案


你可以,用一点元编程。支持自定义宏。

defmacrop custom_where(t, f, v, sign) do
  {sign, [context: Elixir, import: Kernel],
   [
     {:field, [], [t, {:^, [], [f]}]},
     {:^, [], [v]}
   ]}
end

并像使用它一样

field = :age
value = 20

Ecto.Query.where(User, [x],
  custom_where(x, field, value, :<))
#⇒ #Ecto.Query<from u0 in User, where: u0.age < ^20>

这里的诀窍是我们欺骗了就地注入 AST,因此它不执行检查。


推荐阅读