使用ActiveRecord的动态方法,传入条件的哈希值

时间:2013-01-30 20:12:46

标签: ruby-on-rails ruby ruby-on-rails-3 postgresql rails-activerecord

我正在努力使用动态方法元编程的最佳方法,我将根据条件限制结果...例如:

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对于不区分大小写的条件......

任何建议表示赞赏。我的主要目标是拥有一个“列表”模型,用户可以在其中动态构建条件,然后保存该列表以备将来使用。此列表将根据关联的汽车表过滤时间点模型......也许有更简单的方法可以解决这个问题?

2 个答案:

答案 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注入。

此外,没有简单的方法来指定ANDOR条件。最后,简单的"#{field} #{op} ?", value大部分仅适用于数字字段和二元运算符。

但这说明这种方法可以起作用,只是有很大的改进空间。