elixir - 使用动态运算符创建 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...
解决方案
你可以,用一点元编程。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,因此它不执行检查。
推荐阅读
- arrays - 数组中值和索引之间的高效一对一映射
- codenameone - 根据虚拟键盘何时可见隐藏/显示选项卡
- bash - 从 Bash Unix shell 获取 Windows 用户名?
- android - 如何使用 Android 导航组件实现导航抽屉
- sql-server - 如何防止空值被插入到数据库表中?
- primefaces - 如何开发 primefaces 响应式工具栏
- python-sphinx - 使用德语变音符号时,Python-Sphinx 发出“警告:不可解码的源字符,替换为“?”:”
- c++ - 在成员函数中保留对对象本身的引用
- airflow - 气流清除的回填任务不会被调度程序选中
- reactjs - 在重新图表中使用自定义刻度时刻度计数出错?