aboutsummaryrefslogtreecommitdiffstats
path: root/spec/engines/sql/unit/predicates/in_spec.rb
blob: 933e33fe998d2b368f582f02fd1c1c1141081193 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
require 'spec_helper'

module Arel
  module Predicates
    describe In do
      before do
        @relation = Arel::Table.new(:users)
        @attribute = @relation[:id]
      end

      describe '#to_sql' do
        describe 'when relating to an array' do
          describe 'when the array\'s elements are the same type as the attribute' do
            before do
              @array = [1, 2, 3]
            end

            it 'manufactures sql with a comma separated list' do
              sql = In.new(@attribute, @array).to_sql

              adapter_is :mysql do
                sql.should be_like(%Q{`users`.`id` IN (1, 2, 3)})
              end

              adapter_is :oracle do
                sql.should be_like(%Q{"USERS"."ID" IN (1, 2, 3)})
              end

              adapter_is_not :mysql, :oracle do
                sql.should be_like(%Q{"users"."id" IN (1, 2, 3)})
              end
            end
          end

          describe 'when the array\'s elements are not same type as the attribute' do
            before do
              @array = ['1-asdf', 2, 3]
            end

            it 'formats values in the array as the type of the attribute' do
              sql = In.new(@attribute, @array).to_sql

              adapter_is :mysql do
                sql.should be_like(%Q{`users`.`id` IN (1, 2, 3)})
              end

              adapter_is :oracle do
                sql.should be_like(%Q{"USERS"."ID" IN (1, 2, 3)})
              end

              adapter_is_not :mysql, :oracle do
                sql.should be_like(%Q{"users"."id" IN (1, 2, 3)})
              end
            end
          end

          describe 'when the array is empty' do
            before do
              @array = []
            end

            it 'manufactures sql with a comma separated list' do
              sql = In.new(@attribute, @array).to_sql

              adapter_is :mysql do
                sql.should be_like(%Q{`users`.`id` IN (NULL)})
              end

              adapter_is :oracle do
                sql.should be_like(%Q{"USERS"."ID" IN (NULL)})
              end

              adapter_is_not :mysql, :oracle do
                sql.should be_like(%Q{"users"."id" IN (NULL)})
              end
            end
          end

        end

        describe 'when relating to a range' do
          before do
            @range = 1..2
          end

          it 'manufactures sql with a between' do
            sql = In.new(@attribute, @range).to_sql

            adapter_is :mysql do
              sql.should be_like(%Q{`users`.`id` BETWEEN 1 AND 2})
            end

            adapter_is :oracle do
              sql.should be_like(%Q{"USERS"."ID" BETWEEN 1 AND 2})
            end

            adapter_is_not :mysql, :oracle do
              sql.should be_like(%Q{"users"."id" BETWEEN 1 AND 2})
            end
          end
        end

        describe 'when relating to a range with an excluded end' do
          before do
            @range = 1...3
          end

          it 'manufactures sql with a >= and <' do
            sql = In.new(@attribute, @range).to_sql

            adapter_is :mysql do
              sql.should be_like(%Q{(`users`.`id` >= 1 AND `users`.`id` < 3)})
            end

            adapter_is :oracle do
              sql.should be_like(%Q{("USERS"."ID" >= 1 AND "USERS"."ID" < 3)})
            end

            adapter_is_not :mysql, :oracle do
              sql.should be_like(%Q{("users"."id" >= 1 AND "users"."id" < 3)})
            end
          end
        end

        describe 'when relating to a time range' do
          before do
            @relation = Arel::Table.new(:developers)
            @attribute = @relation[:created_at]
            @range = Time.mktime(2010, 01, 01)..Time.mktime(2010, 02, 01)
          end

          it 'manufactures sql with a between' do
            sql = In.new(@attribute, @range).to_sql

            adapter_is :mysql do
              sql.should be_like(%Q{`developers`.`created_at` BETWEEN '2010-01-01 00:00:00' AND '2010-02-01 00:00:00'})
            end

            adapter_is :sqlite3 do
              sql.should match(/"developers"."created_at" BETWEEN '2010-01-01 00:00:00(?:\.\d+)' AND '2010-02-01 00:00:00(?:\.\d+)'/)
            end

            adapter_is :postgresql do
              sql.should be_like(%Q{"developers"."created_at" BETWEEN '2010-01-01 00:00:00.000000' AND '2010-02-01 00:00:00.000000'})
            end

            adapter_is :oracle do
              sql.should be_like(%Q{"DEVELOPERS"."CREATED_AT" BETWEEN TO_TIMESTAMP('2010-01-01 00:00:00:000000','YYYY-MM-DD HH24:MI:SS:FF6') AND TO_TIMESTAMP('2010-02-01 00:00:00:000000','YYYY-MM-DD HH24:MI:SS:FF6')})
            end
          end
        end

        describe 'when relating to a relation' do
          it 'manufactures sql with a subselect' do
            sql = In.new(@attribute, @relation).to_sql

            adapter_is :mysql do
              sql.should be_like(%Q{
                `users`.`id` IN (SELECT `users`.`id`, `users`.`name` FROM `users`)
              })
            end

            adapter_is :oracle do
              sql.should be_like(%Q{
                "USERS"."ID" IN (SELECT "USERS"."ID", "USERS"."NAME" FROM "USERS")
              })
            end

            adapter_is_not :mysql, :oracle do
              sql.should be_like(%Q{
                "users"."id" IN (SELECT "users"."id", "users"."name" FROM "users")
              })
            end
          end
        end
      end
    end
  end
end