named_scoping_test.rb 16.7 KB
Newer Older
1
require "cases/helper"
2 3 4
require 'models/post'
require 'models/topic'
require 'models/comment'
5
require 'models/reply'
6
require 'models/author'
7
require 'models/developer'
A
Arun Agrawal 已提交
8
require 'models/computer'
9

10
class NamedScopingTest < ActiveRecord::TestCase
11
  fixtures :posts, :authors, :topics, :comments, :author_addresses
12

13
  def test_implements_enumerable
J
Jon Leighton 已提交
14
    assert !Topic.all.empty?
15

J
Jon Leighton 已提交
16 17 18 19
    assert_equal Topic.all.to_a, Topic.base
    assert_equal Topic.all.to_a, Topic.base.to_a
    assert_equal Topic.first,    Topic.base.first
    assert_equal Topic.all.to_a, Topic.base.map { |i| i }
20
  end
21

22
  def test_found_items_are_cached
23
    all_posts = Topic.base
24

25 26 27 28 29
    assert_queries(1) do
      all_posts.collect
      all_posts.collect
    end
  end
30

31
  def test_reload_expires_cache_of_found_items
32
    all_posts = Topic.base
33
    all_posts.to_a
34

35 36 37 38
    new_post = Topic.create!
    assert !all_posts.include?(new_post)
    assert all_posts.reload.include?(new_post)
  end
39

40
  def test_delegates_finds_and_calculations_to_the_base_class
J
Jon Leighton 已提交
41
    assert !Topic.all.empty?
42

J
Jon Leighton 已提交
43 44 45
    assert_equal Topic.all.to_a,                Topic.base.to_a
    assert_equal Topic.first,                   Topic.base.first
    assert_equal Topic.count,                   Topic.base.count
46
    assert_equal Topic.average(:replies_count), Topic.base.average(:replies_count)
47
  end
48

49 50 51 52 53 54
  def test_method_missing_priority_when_delegating
    klazz = Class.new(ActiveRecord::Base) do
      self.table_name = "topics"
      scope :since, Proc.new { where('written_on >= ?', Time.now - 1.day) }
      scope :to,    Proc.new { where('written_on <= ?', Time.now) }
    end
55
    assert_equal klazz.to.since.to_a, klazz.since.to.to_a
56 57
  end

58
  def test_scope_should_respond_to_own_methods_and_methods_of_the_proxy
59
    assert Topic.approved.respond_to?(:limit)
60 61 62 63
    assert Topic.approved.respond_to?(:count)
    assert Topic.approved.respond_to?(:length)
  end

64
  def test_scopes_with_options_limit_finds_to_those_matching_the_criteria_specified
65
    assert !Topic.all.merge!(:where => {:approved => true}).to_a.empty?
66

67
    assert_equal Topic.all.merge!(:where => {:approved => true}).to_a, Topic.approved
J
Jon Leighton 已提交
68
    assert_equal Topic.where(:approved => true).count, Topic.approved.count
69
  end
70

71 72 73 74 75 76
  def test_scopes_with_string_name_can_be_composed
    # NOTE that scopes defined with a string as a name worked on their own
    # but when called on another scope the other scope was completely replaced
    assert_equal Topic.replied.approved, Topic.replied.approved_as_string
  end

77
  def test_scopes_are_composable
78 79
    assert_equal((approved = Topic.all.merge!(:where => {:approved => true}).to_a), Topic.approved)
    assert_equal((replied = Topic.all.merge!(:where => 'replies_count > 0').to_a), Topic.replied)
80
    assert !(approved == replied)
81
    assert !(approved & replied).empty?
82

83 84 85 86
    assert_equal approved & replied, Topic.approved.replied
  end

  def test_procedural_scopes
87 88
    topics_written_before_the_third = Topic.where('written_on < ?', topics(:third).written_on)
    topics_written_before_the_second = Topic.where('written_on < ?', topics(:second).written_on)
89
    assert_not_equal topics_written_before_the_second, topics_written_before_the_third
90

91 92 93
    assert_equal topics_written_before_the_third, Topic.written_before(topics(:third).written_on)
    assert_equal topics_written_before_the_second, Topic.written_before(topics(:second).written_on)
  end
94

95
  def test_procedural_scopes_returning_nil
J
Jon Leighton 已提交
96
    all_topics = Topic.all
97 98 99

    assert_equal all_topics, Topic.written_before(nil)
  end
100

101 102 103 104 105 106
  def test_scope_with_object
    objects = Topic.with_object
    assert_operator objects.length, :>, 0
    assert objects.all?(&:approved?), 'all objects should be approved'
  end

107
  def test_has_many_associations_have_access_to_scopes
108 109
    assert_not_equal Post.containing_the_letter_a, authors(:david).posts
    assert !Post.containing_the_letter_a.empty?
110

111 112
    assert_equal authors(:david).posts & Post.containing_the_letter_a, authors(:david).posts.containing_the_letter_a
  end
113

114
  def test_scope_with_STI
115 116 117 118
    assert_equal 3,Post.containing_the_letter_a.count
    assert_equal 1,SpecialPost.containing_the_letter_a.count
  end

119
  def test_has_many_through_associations_have_access_to_scopes
120
    assert_not_equal Comment.containing_the_letter_e, authors(:david).comments
121
    assert !Comment.containing_the_letter_e.empty?
122

123 124
    assert_equal authors(:david).comments & Comment.containing_the_letter_e, authors(:david).comments.containing_the_letter_e
  end
125

126
  def test_scopes_honor_current_scopes_from_when_defined
127 128 129
    assert !Post.ranked_by_comments.limit_by(5).empty?
    assert !authors(:david).posts.ranked_by_comments.limit_by(5).empty?
    assert_not_equal Post.ranked_by_comments.limit_by(5), authors(:david).posts.ranked_by_comments.limit_by(5)
130
    assert_not_equal Post.top(5), authors(:david).posts.top(5)
131
    # Oracle sometimes sorts differently if WHERE condition is changed
132
    assert_equal authors(:david).posts.ranked_by_comments.limit_by(5).to_a.sort_by(&:id), authors(:david).posts.top(5).to_a.sort_by(&:id)
133
    assert_equal Post.ranked_by_comments.limit_by(5), Post.top(5)
134 135
  end

136
  def test_scopes_body_is_a_callable
137 138
    e = assert_raises ArgumentError do
      Class.new(Post).class_eval { scope :containing_the_letter_z, where("body LIKE '%z%'") }
139
    end
140
    assert_equal "The scope body needs to be callable.", e.message
141 142
  end

143
  def test_active_records_have_scope_named__all__
J
Jon Leighton 已提交
144
    assert !Topic.all.empty?
145

J
Jon Leighton 已提交
146
    assert_equal Topic.all.to_a, Topic.base
147
  end
148

149
  def test_active_records_have_scope_named__scoped__
J
Jon Leighton 已提交
150 151
    scope = Topic.where("content LIKE '%Have%'")
    assert !scope.empty?
152

153
    assert_equal scope, Topic.all.merge!(where: "content LIKE '%Have%'")
154
  end
155

156
  def test_first_and_last_should_allow_integers_for_limit
157
    assert_equal Topic.base.first(2), Topic.base.to_a.first(2)
158
    assert_equal Topic.base.last(2), Topic.base.order("id").to_a.last(2)
159 160 161 162 163 164 165 166 167 168 169
  end

  def test_first_and_last_should_not_use_query_when_results_are_loaded
    topics = Topic.base
    topics.reload # force load
    assert_no_queries do
      topics.first
      topics.last
    end
  end

170 171 172 173 174 175 176 177
  def test_empty_should_not_load_results
    topics = Topic.base
    assert_queries(2) do
      topics.empty?  # use count query
      topics.collect # force load
      topics.empty?  # use loaded (no query)
    end
  end
178

179 180
  def test_any_should_not_load_results
    topics = Topic.base
181 182 183 184
    assert_queries(2) do
      topics.any?    # use count query
      topics.collect # force load
      topics.any?    # use loaded (no query)
185 186 187 188 189 190
    end
  end

  def test_any_should_call_proxy_found_if_using_a_block
    topics = Topic.base
    assert_queries(1) do
191 192 193
      assert_not_called(topics, :empty?) do
        topics.any? { true }
      end
194 195 196
    end
  end

197
  def test_any_should_not_fire_query_if_scope_loaded
198 199 200 201 202
    topics = Topic.base
    topics.collect # force load
    assert_no_queries { assert topics.any? }
  end

203 204 205 206 207 208
  def test_model_class_should_respond_to_any
    assert Topic.any?
    Topic.delete_all
    assert !Topic.any?
  end

209 210 211 212 213 214 215 216 217 218 219 220
  def test_many_should_not_load_results
    topics = Topic.base
    assert_queries(2) do
      topics.many?   # use count query
      topics.collect # force load
      topics.many?   # use loaded (no query)
    end
  end

  def test_many_should_call_proxy_found_if_using_a_block
    topics = Topic.base
    assert_queries(1) do
221 222 223
      assert_not_called(topics, :size) do
        topics.many? { true }
      end
224 225 226
    end
  end

227
  def test_many_should_not_fire_query_if_scope_loaded
228 229 230 231 232 233
    topics = Topic.base
    topics.collect # force load
    assert_no_queries { assert topics.many? }
  end

  def test_many_should_return_false_if_none_or_one
J
Jon Leighton 已提交
234
    topics = Topic.base.where(:id => 0)
235
    assert !topics.many?
J
Jon Leighton 已提交
236
    topics = Topic.base.where(:id => 1)
237 238 239 240 241 242 243
    assert !topics.many?
  end

  def test_many_should_return_true_if_more_than_one
    assert Topic.base.many?
  end

244 245 246 247 248 249 250 251 252
  def test_model_class_should_respond_to_many
    Topic.delete_all
    assert !Topic.many?
    Topic.create!
    assert !Topic.many?
    Topic.create!
    assert Topic.many?
  end

253
  def test_should_build_on_top_of_scope
254 255 256 257
    topic = Topic.approved.build({})
    assert topic.approved
  end

258
  def test_should_build_new_on_top_of_scope
259 260 261 262
    topic = Topic.approved.new
    assert topic.approved
  end

263
  def test_should_create_on_top_of_scope
264 265 266 267
    topic = Topic.approved.create({})
    assert topic.approved
  end

268
  def test_should_create_with_bang_on_top_of_scope
269 270 271
    topic = Topic.approved.create!({})
    assert topic.approved
  end
272

273
  def test_should_build_on_top_of_chained_scopes
274 275 276 277
    topic = Topic.approved.by_lifo.build({})
    assert topic.approved
    assert_equal 'lifo', topic.author_name
  end
278

279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303
  def test_reserved_scope_names
    klass = Class.new(ActiveRecord::Base) do
      self.table_name = "topics"

      scope :approved, -> { where(approved: true) }

      class << self
        public
          def pub; end

        private
          def pri; end

        protected
          def pro; end
      end
    end

    subklass = Class.new(klass)

    conflicts = [
      :create,        # public class method on AR::Base
      :relation,      # private class method on AR::Base
      :new,           # redefined class method on AR::Base
      :all,           # a default scope
304
      :public,        # some imporant methods on Module and Class
305
      :protected,
306 307 308 309
      :private,
      :name,
      :parent,
      :superclass
310 311 312 313 314 315 316 317 318 319 320 321
    ]

    non_conflicts = [
      :find_by_title, # dynamic finder method
      :approved,      # existing scope
      :pub,           # existing public class method
      :pri,           # existing private class method
      :pro,           # existing protected class method
      :open,          # a ::Kernel method
    ]

    conflicts.each do |name|
F
Franky W 已提交
322
      e = assert_raises(ArgumentError, "scope `#{name}` should not be allowed") do
323 324
        klass.class_eval { scope name, ->{ where(approved: true) } }
      end
F
Franky W 已提交
325
      assert_match(/You tried to define a scope named \"#{name}\" on the model/, e.message)
326

F
Franky W 已提交
327
      e = assert_raises(ArgumentError, "scope `#{name}` should not be allowed") do
328 329
        subklass.class_eval { scope name, ->{ where(approved: true) } }
      end
F
Franky W 已提交
330
      assert_match(/You tried to define a scope named \"#{name}\" on the model/, e.message)
331 332 333 334
    end

    non_conflicts.each do |name|
      assert_nothing_raised do
335 336 337
        silence_warnings do
          klass.class_eval { scope name, ->{ where(approved: true) } }
        end
338 339 340 341 342 343 344 345
      end

      assert_nothing_raised do
        subklass.class_eval { scope name, ->{ where(approved: true) } }
      end
    end
  end

346 347 348 349 350 351 352 353 354 355 356 357 358
  # Method delegation for scope names which look like /\A[a-zA-Z_]\w*[!?]?\z/
  # has been done by evaluating a string with a plain def statement. For scope
  # names which contain spaces this approach doesn't work.
  def test_spaces_in_scope_names
    klass = Class.new(ActiveRecord::Base) do
      self.table_name = "topics"
      scope :"title containing space", -> { where("title LIKE '% %'") }
      scope :approved, -> { where(:approved => true) }
    end
    assert_equal klass.send(:"title containing space"), klass.where("title LIKE '% %'")
    assert_equal klass.approved.send(:"title containing space"), klass.approved.where("title LIKE '% %'")
  end

359
  def test_find_all_should_behave_like_select
360
    assert_equal Topic.base.to_a.select(&:approved), Topic.base.to_a.find_all(&:approved)
361
  end
362

363
  def test_rand_should_select_a_random_object_from_proxy
364
    assert_kind_of Topic, Topic.approved.sample
365 366
  end

367
  def test_should_use_where_in_query_for_scope
J
Jon Leighton 已提交
368
    assert_equal Developer.where(name: 'Jamis').to_set, Developer.where(id: Developer.jamises).to_set
369
  end
370 371 372 373 374 375 376 377 378 379 380 381 382 383 384

  def test_size_should_use_count_when_results_are_not_loaded
    topics = Topic.base
    assert_queries(1) do
      assert_sql(/COUNT/i) { topics.size }
    end
  end

  def test_size_should_use_length_when_results_are_loaded
    topics = Topic.base
    topics.reload # force load
    assert_no_queries do
      topics.size # use loaded (no query)
    end
  end
385

386
  def test_should_not_duplicates_where_values
387 388
    relation = Topic.where("1=1")
    assert_equal relation.where_clause, relation.scope_with_lambda.where_clause
389 390
  end

391 392 393
  def test_chaining_with_duplicate_joins
    join = "INNER JOIN comments ON comments.post_id = posts.id"
    post = Post.find(1)
J
Jon Leighton 已提交
394
    assert_equal post.comments.size, Post.joins(join).joins(join).where("posts.id = #{post.id}").size
395
  end
396

397
  def test_chaining_applies_last_conditions_when_creating
398 399 400 401 402
    post = Topic.rejected.new
    assert !post.approved?

    post = Topic.rejected.approved.new
    assert post.approved?
403

404 405 406 407 408
    post = Topic.approved.rejected.new
    assert !post.approved?

    post = Topic.approved.rejected.approved.new
    assert post.approved?
409 410
  end

411
  def test_chaining_combines_conditions_when_searching
412
    # Normal hash conditions
413 414
    assert_equal Topic.where(approved: false).where(approved: true).to_a, Topic.rejected.approved.to_a
    assert_equal Topic.where(approved: true).where(approved: false).to_a, Topic.approved.rejected.to_a
415 416

    # Nested hash conditions with same keys
417
    assert_equal [], Post.with_special_comments.with_very_special_comments.to_a
418 419

    # Nested hash conditions with different keys
420
    assert_equal [posts(:sti_comments)], Post.with_special_comments.with_post(4).to_a.uniq
421
  end
422

423
  def test_scopes_batch_finders
424
    assert_equal 4, Topic.approved.count
425

426
    assert_queries(5) do
427 428 429
      Topic.approved.find_each(:batch_size => 1) {|t| assert t.approved? }
    end

430
    assert_queries(3) do
431 432 433 434 435
      Topic.approved.find_in_batches(:batch_size => 2) do |group|
        group.each {|t| assert t.approved? }
      end
    end
  end
436 437 438

  def test_table_names_for_chaining_scopes_with_and_without_table_name_included
    assert_nothing_raised do
439
      Comment.for_first_post.for_first_author.to_a
440 441
    end
  end
442

443 444 445 446 447 448 449 450 451 452 453 454 455 456 457
  def test_scopes_with_reserved_names
    class << Topic
      def public_method; end
      public :public_method

      def protected_method; end
      protected :protected_method

      def private_method; end
      private :private_method
    end

    [:public_method, :protected_method, :private_method].each do |reserved_method|
      assert Topic.respond_to?(reserved_method, true)
      ActiveRecord::Base.logger.expects(:warn)
A
Akira Matsuda 已提交
458
      Topic.scope(reserved_method, -> { })
459 460 461
    end
  end

462
  def test_scopes_on_relations
463
    # Topic.replied
464
    approved_topics = Topic.all.approved.order('id DESC')
465
    assert_equal topics(:fifth), approved_topics.first
466 467 468 469 470

    replied_approved_topics = approved_topics.replied
    assert_equal topics(:third), replied_approved_topics.first
  end

471
  def test_index_on_scope
472 473 474 475
    approved = Topic.approved.order('id ASC')
    assert_equal topics(:second), approved[0]
    assert approved.loaded?
  end
476

477
  def test_nested_scopes_queries_size
478
    assert_queries(1) do
479
      Topic.approved.by_lifo.replied.written_before(Time.now).to_a
480 481
    end
  end
482

483 484 485
  # Note: these next two are kinda odd because they are essentially just testing that the
  # query cache works as it should, but they are here for legacy reasons as they was previously
  # a separate cache on association proxies, and these show that that is not necessary.
486
  def test_scopes_are_cached_on_associations
487 488
    post = posts(:welcome)

489
    Post.cache do
490 491
      assert_queries(1) { post.comments.containing_the_letter_e.to_a }
      assert_no_queries { post.comments.containing_the_letter_e.to_a }
492
    end
493
  end
494

495
  def test_scopes_with_arguments_are_cached_on_associations
496 497
    post = posts(:welcome)

498
    Post.cache do
499
      one = assert_queries(1) { post.comments.limit_by(1).to_a }
500
      assert_equal 1, one.size
501

502
      two = assert_queries(1) { post.comments.limit_by(2).to_a }
503
      assert_equal 2, two.size
504

505 506
      assert_no_queries { post.comments.limit_by(1).to_a }
      assert_no_queries { post.comments.limit_by(2).to_a }
507
    end
508 509
  end

A
Arun Agrawal 已提交
510 511 512 513 514 515 516 517
  def test_scopes_to_get_newest
    post = posts(:welcome)
    old_last_comment = post.comments.newest
    new_comment = post.comments.create(:body => "My new comment")
    assert_equal new_comment, post.comments.newest
    assert_not_equal old_last_comment, post.comments.newest
  end

518
  def test_scopes_are_reset_on_association_reload
519 520 521 522
    post = posts(:welcome)

    [:destroy_all, :reset, :delete_all].each do |method|
      before = post.comments.containing_the_letter_e
523
      post.association(:comments).send(method)
524
      assert before.object_id != post.comments.containing_the_letter_e.object_id, "CollectionAssociation##{method} should reset the named scopes cache"
525 526
    end
  end
527

528
  def test_scoped_are_lazy_loaded_if_table_still_does_not_exist
529 530 531 532
    assert_nothing_raised do
      require "models/without_table"
    end
  end
J
Jon Leighton 已提交
533

534
  def test_eager_default_scope_relations_are_remove
J
Jon Leighton 已提交
535 536 537
    klass = Class.new(ActiveRecord::Base)
    klass.table_name = 'posts'

538
    assert_raises(ArgumentError) do
J
Jon Leighton 已提交
539 540 541
      klass.send(:default_scope, klass.where(:id => posts(:welcome).id))
    end
  end
N
Neeraj Singh 已提交
542 543

  def test_subclass_merges_scopes_properly
J
Jon Leighton 已提交
544
    assert_equal 1, SpecialComment.where(body: 'go crazy').created.count
N
Neeraj Singh 已提交
545 546
  end

547
end