PageRenderTime 26ms CodeModel.GetById 19ms RepoModel.GetById 0ms app.codeStats 0ms

/activerecord/test/cases/relation_test.rb

https://github.com/Tho85/rails
Ruby | 235 lines | 189 code | 44 blank | 2 comment | 1 complexity | 284e68149eba0e89e4fab99b30cec6b1 MD5 | raw file
  1. require "cases/helper"
  2. require 'models/post'
  3. require 'models/comment'
  4. require 'models/author'
  5. require 'models/rating'
  6. module ActiveRecord
  7. class RelationTest < ActiveRecord::TestCase
  8. fixtures :posts, :comments, :authors
  9. class FakeKlass < Struct.new(:table_name, :name)
  10. extend ActiveRecord::Delegation::DelegateCache
  11. inherited self
  12. def self.connection
  13. Post.connection
  14. end
  15. end
  16. def test_construction
  17. relation = Relation.new FakeKlass, :b
  18. assert_equal FakeKlass, relation.klass
  19. assert_equal :b, relation.table
  20. assert !relation.loaded, 'relation is not loaded'
  21. end
  22. def test_responds_to_model_and_returns_klass
  23. relation = Relation.new FakeKlass, :b
  24. assert_equal FakeKlass, relation.model
  25. end
  26. def test_initialize_single_values
  27. relation = Relation.new FakeKlass, :b
  28. (Relation::SINGLE_VALUE_METHODS - [:create_with]).each do |method|
  29. assert_nil relation.send("#{method}_value"), method.to_s
  30. end
  31. assert_equal({}, relation.create_with_value)
  32. end
  33. def test_multi_value_initialize
  34. relation = Relation.new FakeKlass, :b
  35. Relation::MULTI_VALUE_METHODS.each do |method|
  36. assert_equal [], relation.send("#{method}_values"), method.to_s
  37. end
  38. end
  39. def test_extensions
  40. relation = Relation.new FakeKlass, :b
  41. assert_equal [], relation.extensions
  42. end
  43. def test_empty_where_values_hash
  44. relation = Relation.new FakeKlass, :b
  45. assert_equal({}, relation.where_values_hash)
  46. relation.where! :hello
  47. assert_equal({}, relation.where_values_hash)
  48. end
  49. def test_has_values
  50. relation = Relation.new Post, Post.arel_table
  51. relation.where! relation.table[:id].eq(10)
  52. assert_equal({:id => 10}, relation.where_values_hash)
  53. end
  54. def test_values_wrong_table
  55. relation = Relation.new Post, Post.arel_table
  56. relation.where! Comment.arel_table[:id].eq(10)
  57. assert_equal({}, relation.where_values_hash)
  58. end
  59. def test_tree_is_not_traversed
  60. relation = Relation.new Post, Post.arel_table
  61. left = relation.table[:id].eq(10)
  62. right = relation.table[:id].eq(10)
  63. combine = left.and right
  64. relation.where! combine
  65. assert_equal({}, relation.where_values_hash)
  66. end
  67. def test_table_name_delegates_to_klass
  68. relation = Relation.new FakeKlass.new('posts'), :b
  69. assert_equal 'posts', relation.table_name
  70. end
  71. def test_scope_for_create
  72. relation = Relation.new FakeKlass, :b
  73. assert_equal({}, relation.scope_for_create)
  74. end
  75. def test_create_with_value
  76. relation = Relation.new Post, Post.arel_table
  77. hash = { :hello => 'world' }
  78. relation.create_with_value = hash
  79. assert_equal hash, relation.scope_for_create
  80. end
  81. def test_create_with_value_with_wheres
  82. relation = Relation.new Post, Post.arel_table
  83. relation.where! relation.table[:id].eq(10)
  84. relation.create_with_value = {:hello => 'world'}
  85. assert_equal({:hello => 'world', :id => 10}, relation.scope_for_create)
  86. end
  87. # FIXME: is this really wanted or expected behavior?
  88. def test_scope_for_create_is_cached
  89. relation = Relation.new Post, Post.arel_table
  90. assert_equal({}, relation.scope_for_create)
  91. relation.where! relation.table[:id].eq(10)
  92. assert_equal({}, relation.scope_for_create)
  93. relation.create_with_value = {:hello => 'world'}
  94. assert_equal({}, relation.scope_for_create)
  95. end
  96. def test_bad_constants_raise_errors
  97. assert_raises(NameError) do
  98. ActiveRecord::Relation::HelloWorld
  99. end
  100. end
  101. def test_empty_eager_loading?
  102. relation = Relation.new FakeKlass, :b
  103. assert !relation.eager_loading?
  104. end
  105. def test_eager_load_values
  106. relation = Relation.new FakeKlass, :b
  107. relation.eager_load! :b
  108. assert relation.eager_loading?
  109. end
  110. def test_references_values
  111. relation = Relation.new FakeKlass, :b
  112. assert_equal [], relation.references_values
  113. relation = relation.references(:foo).references(:omg, :lol)
  114. assert_equal ['foo', 'omg', 'lol'], relation.references_values
  115. end
  116. def test_references_values_dont_duplicate
  117. relation = Relation.new FakeKlass, :b
  118. relation = relation.references(:foo).references(:foo)
  119. assert_equal ['foo'], relation.references_values
  120. end
  121. test 'merging a hash into a relation' do
  122. relation = Relation.new FakeKlass, :b
  123. relation = relation.merge where: :lol, readonly: true
  124. assert_equal [:lol], relation.where_values
  125. assert_equal true, relation.readonly_value
  126. end
  127. test 'merging an empty hash into a relation' do
  128. assert_equal [], Relation.new(FakeKlass, :b).merge({}).where_values
  129. end
  130. test 'merging a hash with unknown keys raises' do
  131. assert_raises(ArgumentError) { Relation::HashMerger.new(nil, omg: 'lol') }
  132. end
  133. test '#values returns a dup of the values' do
  134. relation = Relation.new(FakeKlass, :b).where! :foo
  135. values = relation.values
  136. values[:where] = nil
  137. assert_not_nil relation.where_values
  138. end
  139. test 'relations can be created with a values hash' do
  140. relation = Relation.new(FakeKlass, :b, where: [:foo])
  141. assert_equal [:foo], relation.where_values
  142. end
  143. test 'merging a single where value' do
  144. relation = Relation.new(FakeKlass, :b)
  145. relation.merge!(where: :foo)
  146. assert_equal [:foo], relation.where_values
  147. end
  148. test 'merging a hash interpolates conditions' do
  149. klass = Class.new(FakeKlass) do
  150. def self.sanitize_sql(args)
  151. raise unless args == ['foo = ?', 'bar']
  152. 'foo = bar'
  153. end
  154. end
  155. relation = Relation.new(klass, :b)
  156. relation.merge!(where: ['foo = ?', 'bar'])
  157. assert_equal ['foo = bar'], relation.where_values
  158. end
  159. def test_merging_readonly_false
  160. relation = Relation.new FakeKlass, :b
  161. readonly_false_relation = relation.readonly(false)
  162. # test merging in both directions
  163. assert_equal false, relation.merge(readonly_false_relation).readonly_value
  164. assert_equal false, readonly_false_relation.merge(relation).readonly_value
  165. end
  166. def test_relation_merging_with_merged_joins_as_symbols
  167. special_comments_with_ratings = SpecialComment.joins(:ratings)
  168. posts_with_special_comments_with_ratings = Post.group("posts.id").joins(:special_comments).merge(special_comments_with_ratings)
  169. assert_equal 3, authors(:david).posts.merge(posts_with_special_comments_with_ratings).count.length
  170. end
  171. def test_relation_merging_with_joins_as_join_dependency_pick_proper_parent
  172. post = Post.create!(title: "haha", body: "huhu")
  173. comment = post.comments.create!(body: "hu")
  174. 3.times { comment.ratings.create! }
  175. relation = Post.joins(:comments).merge Comment.joins(:ratings)
  176. assert_equal 3, relation.where(id: post.id).pluck(:id).size
  177. end
  178. def test_respond_to_for_non_selected_element
  179. post = Post.select(:title).first
  180. assert_equal false, post.respond_to?(:body), "post should not respond_to?(:body) since invoking it raises exception"
  181. silence_warnings { post = Post.select("'title' as post_title").first }
  182. assert_equal false, post.respond_to?(:title), "post should not respond_to?(:body) since invoking it raises exception"
  183. end
  184. def test_relation_merging_with_merged_joins_as_strings
  185. join_string = "LEFT OUTER JOIN #{Rating.quoted_table_name} ON #{SpecialComment.quoted_table_name}.id = #{Rating.quoted_table_name}.comment_id"
  186. special_comments_with_ratings = SpecialComment.joins join_string
  187. posts_with_special_comments_with_ratings = Post.group("posts.id").joins(:special_comments).merge(special_comments_with_ratings)
  188. assert_equal 3, authors(:david).posts.merge(posts_with_special_comments_with_ratings).count.length
  189. end
  190. end
  191. end