relation_test.rb 7.8 KB
Newer Older
1
require "cases/helper"
2 3
require 'models/post'
require 'models/comment'
4 5 6

module ActiveRecord
  class RelationTest < ActiveRecord::TestCase
7 8 9
    fixtures :posts, :comments

    class FakeKlass < Struct.new(:table_name)
A
Aaron Patterson 已提交
10 11
    end

12 13 14 15 16 17 18 19 20 21
    def test_construction
      relation = nil
      assert_nothing_raised do
        relation = Relation.new :a, :b
      end
      assert_equal :a, relation.klass
      assert_equal :b, relation.table
      assert !relation.loaded, 'relation is not loaded'
    end

22 23 24 25 26
    def test_responds_to_model_and_returns_klass
      relation = Relation.new :a, :b
      assert_equal :a, relation.model
    end

27 28
    def test_initialize_single_values
      relation = Relation.new :a, :b
29
      (Relation::SINGLE_VALUE_METHODS - [:create_with]).each do |method|
30 31
        assert_nil relation.send("#{method}_value"), method.to_s
      end
32
      assert_equal({}, relation.create_with_value)
33 34 35 36 37 38 39 40 41 42 43 44 45
    end

    def test_multi_value_initialize
      relation = Relation.new :a, :b
      Relation::MULTI_VALUE_METHODS.each do |method|
        assert_equal [], relation.send("#{method}_values"), method.to_s
      end
    end

    def test_extensions
      relation = Relation.new :a, :b
      assert_equal [], relation.extensions
    end
A
Aaron Patterson 已提交
46

47
    def test_empty_where_values_hash
A
Aaron Patterson 已提交
48 49 50
      relation = Relation.new :a, :b
      assert_equal({}, relation.where_values_hash)

51
      relation.where! :hello
A
Aaron Patterson 已提交
52 53 54
      assert_equal({}, relation.where_values_hash)
    end

55 56
    def test_has_values
      relation = Relation.new Post, Post.arel_table
57
      relation.where! relation.table[:id].eq(10)
58 59 60 61 62
      assert_equal({:id => 10}, relation.where_values_hash)
    end

    def test_values_wrong_table
      relation = Relation.new Post, Post.arel_table
63
      relation.where! Comment.arel_table[:id].eq(10)
64 65 66
      assert_equal({}, relation.where_values_hash)
    end

67 68 69 70 71
    def test_tree_is_not_traversed
      relation = Relation.new Post, Post.arel_table
      left     = relation.table[:id].eq(10)
      right    = relation.table[:id].eq(10)
      combine  = left.and right
72
      relation.where! combine
73 74 75
      assert_equal({}, relation.where_values_hash)
    end

A
Aaron Patterson 已提交
76
    def test_table_name_delegates_to_klass
77
      relation = Relation.new FakeKlass.new('foo'), :b
A
Aaron Patterson 已提交
78 79 80 81 82 83 84
      assert_equal 'foo', relation.table_name
    end

    def test_scope_for_create
      relation = Relation.new :a, :b
      assert_equal({}, relation.scope_for_create)
    end
85 86 87 88 89 90 91 92 93 94

    def test_create_with_value
      relation = Relation.new Post, Post.arel_table
      hash = { :hello => 'world' }
      relation.create_with_value = hash
      assert_equal hash, relation.scope_for_create
    end

    def test_create_with_value_with_wheres
      relation = Relation.new Post, Post.arel_table
95
      relation.where! relation.table[:id].eq(10)
96 97 98
      relation.create_with_value = {:hello => 'world'}
      assert_equal({:hello => 'world', :id => 10}, relation.scope_for_create)
    end
99 100 101 102 103 104

    # FIXME: is this really wanted or expected behavior?
    def test_scope_for_create_is_cached
      relation = Relation.new Post, Post.arel_table
      assert_equal({}, relation.scope_for_create)

105
      relation.where! relation.table[:id].eq(10)
106 107 108 109 110
      assert_equal({}, relation.scope_for_create)

      relation.create_with_value = {:hello => 'world'}
      assert_equal({}, relation.scope_for_create)
    end
A
Aaron Patterson 已提交
111 112 113 114 115 116 117 118

    def test_empty_eager_loading?
      relation = Relation.new :a, :b
      assert !relation.eager_loading?
    end

    def test_eager_load_values
      relation = Relation.new :a, :b
119
      relation.eager_load! :b
A
Aaron Patterson 已提交
120 121
      assert relation.eager_loading?
    end
122 123 124 125 126

    def test_references_values
      relation = Relation.new :a, :b
      assert_equal [], relation.references_values
      relation = relation.references(:foo).references(:omg, :lol)
J
Jon Leighton 已提交
127
      assert_equal ['foo', 'omg', 'lol'], relation.references_values
128 129 130 131 132
    end

    def test_references_values_dont_duplicate
      relation = Relation.new :a, :b
      relation = relation.references(:foo).references(:foo)
J
Jon Leighton 已提交
133
      assert_equal ['foo'], relation.references_values
134
    end
135

136 137
    test 'merging a hash into a relation' do
      relation = Relation.new :a, :b
138
      relation = relation.merge where: :lol, readonly: true
139

140
      assert_equal [:lol], relation.where_values
141 142 143 144 145 146
      assert_equal true, relation.readonly_value
    end

    test 'merging an empty hash into a relation' do
      assert_equal [], Relation.new(:a, :b).merge({}).where_values
    end
J
Jon Leighton 已提交
147 148 149 150

    test 'merging a hash with unknown keys raises' do
      assert_raises(ArgumentError) { Relation::HashMerger.new(nil, omg: 'lol') }
    end
151 152 153 154 155 156 157 158 159 160 161 162

    test '#values returns a dup of the values' do
      relation = Relation.new(:a, :b).where! :foo
      values   = relation.values

      values[:where] = nil
      assert_not_nil relation.where_values
    end

    test 'relations can be created with a values hash' do
      relation = Relation.new(:a, :b, where: [:foo])
      assert_equal [:foo], relation.where_values
163 164 165 166 167 168
    end

    test 'merging a single where value' do
      relation = Relation.new(:a, :b)
      relation.merge!(where: :foo)
      assert_equal [:foo], relation.where_values
169
    end
170 171 172 173 174 175 176 177 178

    test 'merging a hash interpolates conditions' do
      klass = stub
      klass.stubs(:sanitize_sql).with(['foo = ?', 'bar']).returns('foo = bar')

      relation = Relation.new(klass, :b)
      relation.merge!(where: ['foo = ?', 'bar'])
      assert_equal ['foo = bar'], relation.where_values
    end
179
  end
180 181 182 183 184 185

  class RelationMutationTest < ActiveSupport::TestCase
    def relation
      @relation ||= Relation.new :a, :b
    end

186
    (Relation::MULTI_VALUE_METHODS - [:references, :extending]).each do |method|
187 188 189 190 191 192 193 194 195 196 197
      test "##{method}!" do
        assert relation.public_send("#{method}!", :foo).equal?(relation)
        assert_equal [:foo], relation.public_send("#{method}_values")
      end
    end

    test '#references!' do
      assert relation.references!(:foo).equal?(relation)
      assert relation.references_values.include?('foo')
    end

198
    test 'extending!' do
J
Jon Leighton 已提交
199
      mod, mod2 = Module.new, Module.new
200 201

      assert relation.extending!(mod).equal?(relation)
J
Jon Leighton 已提交
202
      assert_equal [mod], relation.extending_values
203
      assert relation.is_a?(mod)
J
Jon Leighton 已提交
204 205 206

      relation.extending!(mod2)
      assert_equal [mod, mod2], relation.extending_values
207 208
    end

209 210 211 212 213
    test 'extending! with empty args' do
      relation.extending!
      assert_equal [], relation.extending_values
    end

214
    (Relation::SINGLE_VALUE_METHODS - [:from, :lock, :reordering, :reverse_order, :create_with]).each do |method|
215 216 217 218 219 220
      test "##{method}!" do
        assert relation.public_send("#{method}!", :foo).equal?(relation)
        assert_equal :foo, relation.public_send("#{method}_value")
      end
    end

221 222 223 224 225
    test '#from!' do
      assert relation.from!('foo').equal?(relation)
      assert_equal ['foo', nil], relation.from_value
    end

226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245
    test '#lock!' do
      assert relation.lock!('foo').equal?(relation)
      assert_equal 'foo', relation.lock_value
    end

    test '#reorder!' do
      relation = self.relation.order('foo')

      assert relation.reorder!('bar').equal?(relation)
      assert_equal ['bar'], relation.order_values
      assert relation.reordering_value
    end

    test 'reverse_order!' do
      assert relation.reverse_order!.equal?(relation)
      assert relation.reverse_order_value
      relation.reverse_order!
      assert !relation.reverse_order_value
    end

246 247 248 249
    test 'create_with!' do
      assert relation.create_with!(foo: 'bar').equal?(relation)
      assert_equal({foo: 'bar'}, relation.create_with_value)
    end
J
Jon Leighton 已提交
250 251

    test 'merge!' do
252 253
      assert relation.merge!(where: :foo).equal?(relation)
      assert_equal [:foo], relation.where_values
J
Jon Leighton 已提交
254
    end
255 256 257 258

    test 'merge with a proc' do
      assert_equal [:foo], relation.merge(-> { where(:foo) }).where_values
    end
259 260 261 262 263 264

    test 'none!' do
      assert relation.none!.equal?(relation)
      assert_equal [NullRelation], relation.extending_values
      assert relation.is_a?(NullRelation)
    end
265
  end
266
end