1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00

Refactor predication methods to be available to SqlLiterals as well.

This commit is contained in:
Ernie Miller 2010-10-27 21:43:20 +08:00 committed by Aaron Patterson
parent f515283f51
commit 01e7ceef45
5 changed files with 204 additions and 174 deletions

View file

@ -1,6 +1,7 @@
require 'arel/crud'
require 'arel/expressions'
require 'arel/predications'
require 'arel/table'
require 'arel/attributes'
require 'arel/compatibility/wheres'

View file

@ -2,180 +2,7 @@ module Arel
module Attributes
class Attribute < Struct.new :relation, :name, :column
include Arel::Expressions
def not_eq other
Nodes::NotEqual.new self, other
end
def not_eq_any others
grouping_any :not_eq, others
end
def not_eq_all others
grouping_all :not_eq, others
end
def eq other
Nodes::Equality.new self, other
end
def eq_any others
grouping_any :eq, others
end
def eq_all others
grouping_all :eq, others
end
def in other
case other
when Arel::SelectManager
Nodes::In.new self, other.to_a.map { |x| x.id }
when Range
if other.exclude_end?
left = Nodes::GreaterThanOrEqual.new(self, other.min)
right = Nodes::LessThan.new(self, other.max + 1)
Nodes::And.new left, right
else
Nodes::Between.new(self, Nodes::And.new(other.min, other.max))
end
else
Nodes::In.new self, other
end
end
def in_any others
grouping_any :in, others
end
def in_all others
grouping_all :in, others
end
def not_in other
case other
when Arel::SelectManager
Nodes::NotIn.new self, other.to_a.map { |x| x.id }
when Range
if other.exclude_end?
left = Nodes::LessThan.new(self, other.min)
right = Nodes::GreaterThanOrEqual.new(self, other.max)
Nodes::Or.new left, right
else
left = Nodes::LessThan.new(self, other.min)
right = Nodes::GreaterThan.new(self, other.max)
Nodes::Or.new left, right
end
else
Nodes::NotIn.new self, other
end
end
def not_in_any others
grouping_any :not_in, others
end
def not_in_all others
grouping_all :not_in, others
end
def matches other
Nodes::Matches.new self, other
end
def matches_any others
grouping_any :matches, others
end
def matches_all others
grouping_all :matches, others
end
def does_not_match other
Nodes::DoesNotMatch.new self, other
end
def does_not_match_any others
grouping_any :does_not_match, others
end
def does_not_match_all others
grouping_all :does_not_match, others
end
def gteq right
Nodes::GreaterThanOrEqual.new self, right
end
def gteq_any others
grouping_any :gteq, others
end
def gteq_all others
grouping_all :gteq, others
end
def gt right
Nodes::GreaterThan.new self, right
end
def gt_any others
grouping_any :gt, others
end
def gt_all others
grouping_all :gt, others
end
def lt right
Nodes::LessThan.new self, right
end
def lt_any others
grouping_any :lt, others
end
def lt_all others
grouping_all :lt, others
end
def lteq right
Nodes::LessThanOrEqual.new self, right
end
def lteq_any others
grouping_any :lteq, others
end
def lteq_all others
grouping_all :lteq, others
end
def asc
Nodes::Ordering.new self, :asc
end
def desc
Nodes::Ordering.new self, :desc
end
private
def grouping_any method_id, others
first = send method_id, others.shift
Nodes::Grouping.new others.inject(first) { |memo,expr|
Nodes::Or.new(memo, send(method_id, expr))
}
end
def grouping_all method_id, others
first = send method_id, others.shift
Nodes::Grouping.new others.inject(first) { |memo,expr|
Nodes::And.new(memo, send(method_id, expr))
}
end
include Arel::Predications
end
class String < Attribute; end

View file

@ -2,6 +2,7 @@ module Arel
module Nodes
class SqlLiteral < String
include Arel::Expressions
include Arel::Predications
end
end
end

177
lib/arel/predications.rb Normal file
View file

@ -0,0 +1,177 @@
module Arel
module Predications
def not_eq other
Nodes::NotEqual.new self, other
end
def not_eq_any others
grouping_any :not_eq, others
end
def not_eq_all others
grouping_all :not_eq, others
end
def eq other
Nodes::Equality.new self, other
end
def eq_any others
grouping_any :eq, others
end
def eq_all others
grouping_all :eq, others
end
def in other
case other
when Arel::SelectManager
Nodes::In.new self, other.to_a.map { |x| x.id }
when Range
if other.exclude_end?
left = Nodes::GreaterThanOrEqual.new(self, other.min)
right = Nodes::LessThan.new(self, other.max + 1)
Nodes::And.new left, right
else
Nodes::Between.new(self, Nodes::And.new(other.min, other.max))
end
else
Nodes::In.new self, other
end
end
def in_any others
grouping_any :in, others
end
def in_all others
grouping_all :in, others
end
def not_in other
case other
when Arel::SelectManager
Nodes::NotIn.new self, other.to_a.map { |x| x.id }
when Range
if other.exclude_end?
left = Nodes::LessThan.new(self, other.min)
right = Nodes::GreaterThanOrEqual.new(self, other.max)
Nodes::Or.new left, right
else
left = Nodes::LessThan.new(self, other.min)
right = Nodes::GreaterThan.new(self, other.max)
Nodes::Or.new left, right
end
else
Nodes::NotIn.new self, other
end
end
def not_in_any others
grouping_any :not_in, others
end
def not_in_all others
grouping_all :not_in, others
end
def matches other
Nodes::Matches.new self, other
end
def matches_any others
grouping_any :matches, others
end
def matches_all others
grouping_all :matches, others
end
def does_not_match other
Nodes::DoesNotMatch.new self, other
end
def does_not_match_any others
grouping_any :does_not_match, others
end
def does_not_match_all others
grouping_all :does_not_match, others
end
def gteq right
Nodes::GreaterThanOrEqual.new self, right
end
def gteq_any others
grouping_any :gteq, others
end
def gteq_all others
grouping_all :gteq, others
end
def gt right
Nodes::GreaterThan.new self, right
end
def gt_any others
grouping_any :gt, others
end
def gt_all others
grouping_all :gt, others
end
def lt right
Nodes::LessThan.new self, right
end
def lt_any others
grouping_any :lt, others
end
def lt_all others
grouping_all :lt, others
end
def lteq right
Nodes::LessThanOrEqual.new self, right
end
def lteq_any others
grouping_any :lteq, others
end
def lteq_all others
grouping_all :lteq, others
end
def asc
Nodes::Ordering.new self, :asc
end
def desc
Nodes::Ordering.new self, :desc
end
private
def grouping_any method_id, others
first = send method_id, others.shift
Nodes::Grouping.new others.inject(first) { |memo,expr|
Nodes::Or.new(memo, send(method_id, expr))
}
end
def grouping_all method_id, others
first = send method_id, others.shift
Nodes::Grouping.new others.inject(first) { |memo,expr|
Nodes::And.new(memo, send(method_id, expr))
}
end
end
end

View file

@ -23,6 +23,30 @@ module Arel
viz.accept(node).must_be_like %{ COUNT(DISTINCT *) }
end
end
describe 'equality' do
it 'makes an equality node' do
node = SqlLiteral.new('foo').eq(1)
viz = Visitors::ToSql.new Table.engine
viz.accept(node).must_be_like %{ foo = 1 }
end
end
describe 'grouped "or" equality' do
it 'makes a grouping node with an or node' do
node = SqlLiteral.new('foo').eq_any([1,2])
viz = Visitors::ToSql.new Table.engine
viz.accept(node).must_be_like %{ (foo = 1 OR foo = 2) }
end
end
describe 'grouped "and" equality' do
it 'makes a grouping node with an or node' do
node = SqlLiteral.new('foo').eq_all([1,2])
viz = Visitors::ToSql.new Table.engine
viz.accept(node).must_be_like %{ (foo = 1 AND foo = 2) }
end
end
end
end
end