我正在努力使用动态方法元编程的最佳方法,我将根据条件限制结果...例如:
class Timeslip < ActiveRecord::Base
def self.by_car_trans(car, trans)
joins(:car)
.where("cars.trans IN (?) and cars.year IN (?) and cars.model ILIKE ?", trans, 1993..2002, car)
.order('et1320')
end
end
让我们说代替传入我的参数,我传入一个条件数组,其中key是fieldname,value是字段值。所以,例如,我会做这样的事情:
我传入[[“field”,“value”,“operator”],[“field”,“value”,“operator”]]
def self.using_conditions(conditions)
joins(:car)
conditions.each do |key, value|
where("cars.#{key} #{operator} ?", value)
end
end
然而,这不起作用,并且它不是很灵活......我希望能够检测到值是否为数组,并使用IN()而不是=,并且可能能够使用ILIKE对于不区分大小写的条件......
任何建议表示赞赏。我的主要目标是拥有一个“列表”模型,用户可以在其中动态构建条件,然后保存该列表以备将来使用。此列表将根据关联的汽车表过滤时间点模型......也许有更简单的方法可以解决这个问题?
答案 0 :(得分:2)
首先,您可能会对Squeel gem感兴趣。
除此之外,使用arel_table
表示IN或LIKE谓词:
joins( :car ).where( Car.arel_table[key].in values )
joins( :car ).where( Car.arel_table[key].matches value )
您可以检测值的类型以选择适当的谓词(不是很好的OO,但仍然):
column = Car.arel_table[key]
predicate = value.respond_to?( :to_str ) ? :in : :matches # or any logic you want
joins( :car ).where( column.send predicate, value )
你可以根据需要连接多个链接:
conditions.each do |(key, value, predicate)|
scope = scope.where( Car.arel_table[key].send predicate, value )
end
return scope
答案 1 :(得分:0)
因此,您希望最终用户可以在运行时指定动态查询(并且可以存储和检索以供以后使用)?
我认为你走在正确的轨道上。唯一的细节是您如何建模和存储您的标准。我不明白为什么以下内容不起作用:
def self.using_conditions(conditions)
joins(:car)
crit = conditions.each_with_object({}) {|(field, op, value), m|
m["#{field} #{op} ?"] = value
}
where crit.keys.join(' AND '), *crit.values
end
CAVEAT 以上代码原样不安全且易于SQL注入。
此外,没有简单的方法来指定AND
与OR
条件。最后,简单的"#{field} #{op} ?", value
大部分仅适用于数字字段和二元运算符。
但这说明这种方法可以起作用,只是有很大的改进空间。