使用动态运算符创建 Ecto 查询
Create Ecto query with dynamic operators
我知道如何在 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...
你可以,需要一些元编程。 ecto 支持自定义宏。
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>
这里的技巧是我们欺骗 ecto 就地注入 AST,因此它不执行检查。
我知道如何在 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...
你可以,需要一些元编程。 ecto 支持自定义宏。
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>
这里的技巧是我们欺骗 ecto 就地注入 AST,因此它不执行检查。