predications.rb 5.4 KB
Newer Older
1
# frozen_string_literal: true
2 3 4
module Arel
  module Predications
    def not_eq other
5
      Nodes::NotEqual.new self, quoted_node(other)
6 7 8 9 10 11 12 13 14 15 16
    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
17
      Nodes::Equality.new self, quoted_node(other)
18 19 20 21 22 23 24
    end

    def eq_any others
      grouping_any :eq, others
    end

    def eq_all others
25
      grouping_all :eq, quoted_array(others)
26 27
    end

S
Sean Griffin 已提交
28
    def between other
29 30
      if equals_quoted?(other.begin, -Float::INFINITY)
        if equals_quoted?(other.end, Float::INFINITY)
31 32 33 34 35 36
          not_in([])
        elsif other.exclude_end?
          lt(other.end)
        else
          lteq(other.end)
        end
37
      elsif equals_quoted?(other.end, Float::INFINITY)
38 39 40 41
        gteq(other.begin)
      elsif other.exclude_end?
        gteq(other.begin).and(lt(other.end))
      else
42 43
        left = quoted_node(other.begin)
        right = quoted_node(other.end)
44 45
        Nodes::Between.new(self, left.and(right))
      end
S
Sean Griffin 已提交
46 47
    end

48 49 50
    def in other
      case other
      when Arel::SelectManager
51
        Arel::Nodes::In.new(self, other.ast)
52
      when Range
53 54 55 56
        if $VERBOSE
          warn <<-eowarn
Passing a range to `#in` is deprecated. Call `#between`, instead.
          eowarn
57
        end
58
        between(other)
59
      when Enumerable
60
        Nodes::In.new self, quoted_array(other)
61
      else
62
        Nodes::In.new self, quoted_node(other)
63 64 65 66 67 68 69 70 71 72 73
      end
    end

    def in_any others
      grouping_any :in, others
    end

    def in_all others
      grouping_all :in, others
    end

74
    def not_between other
75 76
      if equals_quoted?(other.begin, -Float::INFINITY)
        if equals_quoted?(other.end, Float::INFINITY)
77 78 79 80 81 82
          self.in([])
        elsif other.exclude_end?
          gteq(other.end)
        else
          gt(other.end)
        end
83
      elsif equals_quoted?(other.end, Float::INFINITY)
84 85 86 87 88 89 90 91 92 93 94 95
        lt(other.begin)
      else
        left = lt(other.begin)
        right = if other.exclude_end?
          gteq(other.end)
        else
          gt(other.end)
        end
        left.or(right)
      end
    end

96 97 98
    def not_in other
      case other
      when Arel::SelectManager
99
        Arel::Nodes::NotIn.new(self, other.ast)
100
      when Range
101 102 103 104
        if $VERBOSE
          warn <<-eowarn
Passing a range to `#not_in` is deprecated. Call `#not_between`, instead.
          eowarn
105
        end
106
        not_between(other)
107
      when Enumerable
108
        Nodes::NotIn.new self, quoted_array(other)
109
      else
110
        Nodes::NotIn.new self, quoted_node(other)
111 112 113 114 115 116 117 118 119 120 121
      end
    end

    def not_in_any others
      grouping_any :not_in, others
    end

    def not_in_all others
      grouping_all :not_in, others
    end

K
Keenan Brock 已提交
122 123
    def matches other, escape = nil, case_sensitive = false
      Nodes::Matches.new self, quoted_node(other), escape, case_sensitive
124 125
    end

126 127 128 129
    def matches_regexp other, case_sensitive = true
      Nodes::Regexp.new self, quoted_node(other), case_sensitive
    end

K
Keenan Brock 已提交
130 131
    def matches_any others, escape = nil, case_sensitive = false
      grouping_any :matches, others, escape, case_sensitive
132 133
    end

K
Keenan Brock 已提交
134 135
    def matches_all others, escape = nil, case_sensitive = false
      grouping_all :matches, others, escape, case_sensitive
136 137
    end

K
Keenan Brock 已提交
138 139
    def does_not_match other, escape = nil, case_sensitive = false
      Nodes::DoesNotMatch.new self, quoted_node(other), escape, case_sensitive
140 141
    end

142 143 144 145
    def does_not_match_regexp other, case_sensitive = true
      Nodes::NotRegexp.new self, quoted_node(other), case_sensitive
    end

146 147
    def does_not_match_any others, escape = nil
      grouping_any :does_not_match, others, escape
148 149
    end

150 151
    def does_not_match_all others, escape = nil
      grouping_all :does_not_match, others, escape
152 153 154
    end

    def gteq right
155
      Nodes::GreaterThanOrEqual.new self, quoted_node(right)
156 157 158 159 160 161 162 163 164 165 166
    end

    def gteq_any others
      grouping_any :gteq, others
    end

    def gteq_all others
      grouping_all :gteq, others
    end

    def gt right
167
      Nodes::GreaterThan.new self, quoted_node(right)
168 169 170 171 172 173 174 175 176 177 178
    end

    def gt_any others
      grouping_any :gt, others
    end

    def gt_all others
      grouping_all :gt, others
    end

    def lt right
179
      Nodes::LessThan.new self, quoted_node(right)
180 181 182 183 184 185 186 187 188 189 190
    end

    def lt_any others
      grouping_any :lt, others
    end

    def lt_all others
      grouping_all :lt, others
    end

    def lteq right
191
      Nodes::LessThanOrEqual.new self, quoted_node(right)
192 193 194 195 196 197 198 199 200 201
    end

    def lteq_any others
      grouping_any :lteq, others
    end

    def lteq_all others
      grouping_all :lteq, others
    end

202 203 204 205
    def when right
      Nodes::Case.new(self).when quoted_node(right)
    end

206 207 208 209
    def concat other
      Nodes::Concat.new self, other
    end

210 211
    private

212 213
    def grouping_any method_id, others, *extras
      nodes = others.map {|expr| send(method_id, expr, *extras)}
214 215
      Nodes::Grouping.new nodes.inject { |memo,node|
        Nodes::Or.new(memo, node)
216 217 218
      }
    end

219 220 221
    def grouping_all method_id, others, *extras
      nodes = others.map {|expr| send(method_id, expr, *extras)}
      Nodes::Grouping.new Nodes::And.new(nodes)
222
    end
223 224 225 226 227 228 229 230

    def quoted_node(other)
      Nodes.build_quoted(other, self)
    end

    def quoted_array(others)
      others.map { |v| quoted_node(v) }
    end
231 232 233 234 235 236 237 238

    def equals_quoted?(maybe_quoted, value)
      if maybe_quoted.is_a?(Nodes::Quoted)
        maybe_quoted.val == value
      else
        maybe_quoted == value
      end
    end
239
  end
A
Aaron Patterson 已提交
240
end