Module: Ardm::Ar::PredicateBuilder::Rails3::ClassMethods

Defined in:
lib/ardm/ar/predicate_builder/rails3.rb

Instance Method Summary collapse

Instance Method Details

#build_from_hash(klass, attributes, default_table) ⇒ Object



41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
# File 'lib/ardm/ar/predicate_builder/rails3.rb', line 41

def build_from_hash(klass, attributes, default_table)
  queries = []
  # HAX (this method is added to the attributes hash by expand_hash_conditions_for_aggregates
  # Rails 3 calls build_form_hash with the first arg that is not the klass.
  klass = attributes.klass

  attributes.each do |column, value|
    table = default_table

    if value.is_a?(Hash)
      if value.empty?
        queries << '1=0'
      else
        table       = Arel::Table.new(column, default_table.engine)
        association = klass.reflect_on_association(column.to_sym)

        value.each do |k, v|
          queries.concat expand(association && association.klass, table, k, v)
        end
      end
    else
      if Ardm::Query::Operator === column
        original = column
        operator = column.operator
        column   = column.target.to_s
      else
        column = column.to_s
      end

      if column.include?('.')
        table_name, column = column.split('.', 2)
        table = Arel::Table.new(table_name, default_table.engine)
      end

      query = expand(klass, table, column, value)
      # TODO make nicer
      if operator == :not
        # Logical not factorization !(a && b) == (!a || !b)
        query.map! &:not
        query = [query.inject { |composite, predicate| composite.or(predicate) }]
      end
      queries.concat query
    end
  end

  queries
end

#expand(klass, table, column, value) ⇒ Object



89
90
91
92
93
94
95
# File 'lib/ardm/ar/predicate_builder/rails3.rb', line 89

def expand(klass, table, column, value)
  if klass && association = klass.reflect_on_association(column.to_sym)
    expand_association(association, table, column, value)
  else
    [build(table[column], value)]
  end
end

#expand_association(association, table, column, value) ⇒ Object

Find the foreign key when using queries such as: Post.where(author: author)

For polymorphic relationships, find the foreign key and type: PriceEstimate.where(estimate_of: treasure)

Attempt to build a query that makes sense for an association name in the query, but if we can’t generate a propery query, fallback to using the original key we received.



106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
# File 'lib/ardm/ar/predicate_builder/rails3.rb', line 106

def expand_association(association, table, column, value)
  queries = []
  case association.macro
  when :belongs_to
    if association.polymorphic? && base_class = polymorphic_base_class_from_value(value)
      queries << build(table[association.foreign_type], base_class)
    end
    queries << build(table[association.foreign_key], value)
  when :has_many, :has_one
    table = Arel::Table.new(association.klass.table_name, table.engine)
    queries << build(table[association.klass.primary_key], value)
  else
    queries << build(table[column], value)
  end
  queries
end

#polymorphic_base_class_from_value(value) ⇒ Object



123
124
125
126
127
128
129
130
131
132
133
# File 'lib/ardm/ar/predicate_builder/rails3.rb', line 123

def polymorphic_base_class_from_value(value)
  case value
  when Relation
    value.klass.base_class
  when Array
    val = value.compact.first
    val.class.base_class if val.is_a?(Base)
  when Base
    value.class.base_class
  end
end

#references(attributes) ⇒ Object



135
136
137
138
139
140
141
142
143
144
# File 'lib/ardm/ar/predicate_builder/rails3.rb', line 135

def references(attributes)
  attributes.map do |key, value|
    if value.is_a?(Hash)
      key
    else
      key = key.to_s
      key.split('.').first if key.include?('.')
    end
  end.compact
end

#register_handler(klass, handler) ⇒ Object

Define how a class is converted to Arel nodes when passed to where. The handler can be any object that responds to call, and will be used for any value that === the class given. For example:

MyCustomDateRange = Struct.new(:start, :end)
handler = proc do |column, range|
  Arel::Nodes::Between.new(column,
    Arel::Nodes::And.new([range.start, range.end])
  )
end
ActiveRecord::PredicateBuilder.register_handler(MyCustomDateRange, handler)


157
158
159
# File 'lib/ardm/ar/predicate_builder/rails3.rb', line 157

def register_handler(klass, handler)
  handlers.unshift([klass, handler])
end

#resolve_column_aliases(klass, hash) ⇒ Object



31
32
33
34
35
36
37
38
39
# File 'lib/ardm/ar/predicate_builder/rails3.rb', line 31

def resolve_column_aliases(klass, hash)
  hash = hash.dup
  hash.keys.grep(Symbol) do |key|
    if klass.attribute_alias? key
      hash[klass.attribute_alias(key)] = hash.delete key
    end
  end
  hash
end