relation_test.rb 12.1 KB
Newer Older
1
require "cases/helper"
2 3 4 5
require "models/post"
require "models/comment"
require "models/author"
require "models/rating"
6 7 8

module ActiveRecord
  class RelationTest < ActiveRecord::TestCase
9
    fixtures :posts, :comments, :authors, :author_addresses, :ratings
10

11
    FakeKlass = Struct.new(:table_name, :name) do
12 13 14 15
      extend ActiveRecord::Delegation::DelegateCache

      inherited self

16 17 18
      def self.connection
        Post.connection
      end
19 20

      def self.table_name
21
        "fake_table"
22
      end
23 24 25 26

      def self.sanitize_sql_for_order(sql)
        sql
      end
A
Aaron Patterson 已提交
27 28
    end

29
    def test_construction
30
      relation = Relation.new(FakeKlass, :b, nil)
31
      assert_equal FakeKlass, relation.klass
32
      assert_equal :b, relation.table
33
      assert !relation.loaded, "relation is not loaded"
34 35
    end

36
    def test_responds_to_model_and_returns_klass
37
      relation = Relation.new(FakeKlass, :b, nil)
38
      assert_equal FakeKlass, relation.model
39 40
    end

41
    def test_initialize_single_values
42
      relation = Relation.new(FakeKlass, :b, nil)
43
      (Relation::SINGLE_VALUE_METHODS - [:create_with, :readonly]).each do |method|
44 45
        assert_nil relation.send("#{method}_value"), method.to_s
      end
46
      assert_equal false, relation.readonly_value
47 48 49
      value = relation.create_with_value
      assert_equal({}, value)
      assert_predicate value, :frozen?
50 51 52
    end

    def test_multi_value_initialize
53
      relation = Relation.new(FakeKlass, :b, nil)
54
      Relation::MULTI_VALUE_METHODS.each do |method|
55 56 57
        values = relation.send("#{method}_values")
        assert_equal [], values, method.to_s
        assert_predicate values, :frozen?, method.to_s
58 59 60 61
      end
    end

    def test_extensions
62
      relation = Relation.new(FakeKlass, :b, nil)
63 64
      assert_equal [], relation.extensions
    end
A
Aaron Patterson 已提交
65

66
    def test_empty_where_values_hash
67
      relation = Relation.new(FakeKlass, :b, nil)
A
Aaron Patterson 已提交
68 69 70
      assert_equal({}, relation.where_values_hash)
    end

71
    def test_has_values
72
      relation = Relation.new(Post, Post.arel_table, Post.predicate_builder)
73
      relation.where! relation.table[:id].eq(10)
74
      assert_equal({ id: 10 }, relation.where_values_hash)
75 76 77
    end

    def test_values_wrong_table
78
      relation = Relation.new(Post, Post.arel_table, Post.predicate_builder)
79
      relation.where! Comment.arel_table[:id].eq(10)
80 81 82
      assert_equal({}, relation.where_values_hash)
    end

83
    def test_tree_is_not_traversed
84
      relation = Relation.new(Post, Post.arel_table, Post.predicate_builder)
85 86
      left     = relation.table[:id].eq(10)
      right    = relation.table[:id].eq(10)
87
      combine  = left.and right
88
      relation.where! combine
89 90 91
      assert_equal({}, relation.where_values_hash)
    end

A
Aaron Patterson 已提交
92
    def test_table_name_delegates_to_klass
93 94
      relation = Relation.new(FakeKlass.new("posts"), :b, Post.predicate_builder)
      assert_equal "posts", relation.table_name
A
Aaron Patterson 已提交
95 96 97
    end

    def test_scope_for_create
98
      relation = Relation.new(FakeKlass, :b, nil)
A
Aaron Patterson 已提交
99 100
      assert_equal({}, relation.scope_for_create)
    end
101 102

    def test_create_with_value
103
      relation = Relation.new(Post, Post.arel_table, Post.predicate_builder)
104
      hash = { hello: "world" }
105 106 107 108 109
      relation.create_with_value = hash
      assert_equal hash, relation.scope_for_create
    end

    def test_create_with_value_with_wheres
110
      relation = Relation.new(Post, Post.arel_table, Post.predicate_builder)
111
      relation.where! relation.table[:id].eq(10)
112 113
      relation.create_with_value = { hello: "world" }
      assert_equal({ hello: "world", id: 10 }, relation.scope_for_create)
114
    end
115 116 117

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

121
      relation.where! relation.table[:id].eq(10)
122 123
      assert_equal({}, relation.scope_for_create)

124
      relation.create_with_value = { hello: "world" }
125 126
      assert_equal({}, relation.scope_for_create)
    end
A
Aaron Patterson 已提交
127

128 129 130 131 132 133
    def test_bad_constants_raise_errors
      assert_raises(NameError) do
        ActiveRecord::Relation::HelloWorld
      end
    end

A
Aaron Patterson 已提交
134
    def test_empty_eager_loading?
135
      relation = Relation.new(FakeKlass, :b, nil)
A
Aaron Patterson 已提交
136 137 138 139
      assert !relation.eager_loading?
    end

    def test_eager_load_values
140
      relation = Relation.new(FakeKlass, :b, nil)
141
      relation.eager_load! :b
A
Aaron Patterson 已提交
142 143
      assert relation.eager_loading?
    end
144 145

    def test_references_values
146
      relation = Relation.new(FakeKlass, :b, nil)
147 148
      assert_equal [], relation.references_values
      relation = relation.references(:foo).references(:omg, :lol)
149
      assert_equal ["foo", "omg", "lol"], relation.references_values
150 151 152
    end

    def test_references_values_dont_duplicate
153
      relation = Relation.new(FakeKlass, :b, nil)
154
      relation = relation.references(:foo).references(:foo)
155
      assert_equal ["foo"], relation.references_values
156
    end
157

158
    test "merging a hash into a relation" do
159
      relation = Relation.new(Post, Post.arel_table, Post.predicate_builder)
160
      relation = relation.merge where: { name: :lol }, readonly: true
161

162
      assert_equal({ "name" => :lol }, relation.where_clause.to_h)
163 164 165
      assert_equal true, relation.readonly_value
    end

166
    test "merging an empty hash into a relation" do
167
      assert_equal Relation::WhereClause.empty, Relation.new(FakeKlass, :b, nil).merge({}).where_clause
168
    end
J
Jon Leighton 已提交
169

170 171
    test "merging a hash with unknown keys raises" do
      assert_raises(ArgumentError) { Relation::HashMerger.new(nil, omg: "lol") }
J
Jon Leighton 已提交
172
    end
173

174
    test "merging nil or false raises" do
175 176 177 178 179 180
      relation = Relation.new(FakeKlass, :b, nil)

      e = assert_raises(ArgumentError) do
        relation = relation.merge nil
      end

181
      assert_equal "invalid argument: nil.", e.message
182 183 184 185 186

      e = assert_raises(ArgumentError) do
        relation = relation.merge false
      end

187
      assert_equal "invalid argument: false.", e.message
188 189
    end

190
    test "#values returns a dup of the values" do
191
      relation = Relation.new(Post, Post.arel_table, Post.predicate_builder).where!(name: :foo)
192 193 194
      values   = relation.values

      values[:where] = nil
195
      assert_not_nil relation.where_clause
196 197
    end

198
    test "relations can be created with a values hash" do
S
Sean Griffin 已提交
199 200
      relation = Relation.new(FakeKlass, :b, nil, select: [:foo])
      assert_equal [:foo], relation.select_values
201
    end
202

203
    test "merging a hash interpolates conditions" do
204 205
      klass = Class.new(FakeKlass) do
        def self.sanitize_sql(args)
206 207
          raise unless args == ["foo = ?", "bar"]
          "foo = bar"
208 209
        end
      end
210

211
      relation = Relation.new(klass, :b, nil)
212 213
      relation.merge!(where: ["foo = ?", "bar"])
      assert_equal Relation::WhereClause.new(["foo = bar"], []), relation.where_clause
214
    end
215

216
    def test_merging_readonly_false
217
      relation = Relation.new(FakeKlass, :b, nil)
218 219 220 221 222 223
      readonly_false_relation = relation.readonly(false)
      # test merging in both directions
      assert_equal false, relation.merge(readonly_false_relation).readonly_value
      assert_equal false, readonly_false_relation.merge(relation).readonly_value
    end

224
    def test_relation_merging_with_merged_joins_as_symbols
225 226
      special_comments_with_ratings = SpecialComment.joins(:ratings)
      posts_with_special_comments_with_ratings = Post.group("posts.id").joins(:special_comments).merge(special_comments_with_ratings)
227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246
      assert_equal({ 4 => 2 }, authors(:david).posts.merge(posts_with_special_comments_with_ratings).count)
    end

    def test_relation_merging_with_merged_symbol_joins_keeps_inner_joins
      queries = capture_sql { authors_with_commented_posts = Author.joins(:posts).merge(Post.joins(:comments)).to_a }

      nb_inner_join = queries.sum { |sql| sql.scan(/INNER\s+JOIN/i).size }
      assert_equal 2, nb_inner_join, "Wrong amount of INNER JOIN in query"
      assert queries.none? { |sql| /LEFT\s+(OUTER)?\s+JOIN/i.match?(sql) }, "Shouldn't have any LEFT JOIN in query"
    end

    def test_relation_merging_with_merged_symbol_joins_has_correct_size_and_count
      # Has one entry per comment
      merged_authors_with_commented_posts_relation = Author.joins(:posts).merge(Post.joins(:comments))

      post_ids_with_author = Post.joins(:author).pluck(:id)
      manual_comments_on_post_that_have_author = Comment.where(post_id: post_ids_with_author).pluck(:id)

      assert_equal manual_comments_on_post_that_have_author.size, merged_authors_with_commented_posts_relation.count
      assert_equal manual_comments_on_post_that_have_author.size, merged_authors_with_commented_posts_relation.to_a.size
247 248
    end

249 250 251 252 253
    def test_relation_merging_with_joins_as_join_dependency_pick_proper_parent
      post = Post.create!(title: "haha", body: "huhu")
      comment = post.comments.create!(body: "hu")
      3.times { comment.ratings.create! }

254
      relation = Post.joins(:comments).merge Comment.joins(:ratings)
255

256
      assert_equal 3, relation.where(id: post.id).pluck(:id).size
257 258
    end

259 260 261 262 263 264 265
    def test_merge_raises_with_invalid_argument
      assert_raises ArgumentError do
        relation = Relation.new(FakeKlass, :b, nil)
        relation.merge(true)
      end
    end

266
    def test_respond_to_for_non_selected_element
267 268 269
      post = Post.select(:title).first
      assert_equal false, post.respond_to?(:body), "post should not respond_to?(:body) since invoking it raises exception"

270
      silence_warnings { post = Post.select("'title' as post_title").first }
271
      assert_equal false, post.respond_to?(:title), "post should not respond_to?(:body) since invoking it raises exception"
272 273
    end

274
    def test_select_quotes_when_using_from_clause
275
      skip_if_sqlite3_version_includes_quoting_bug
276 277 278 279 280
      quoted_join = ActiveRecord::Base.connection.quote_table_name("join")
      selected = Post.select(:join).from(Post.select("id as #{quoted_join}")).map(&:join)
      assert_equal Post.pluck(:id), selected
    end

281 282 283 284 285 286 287 288 289 290 291
    def test_selecting_aliased_attribute_quotes_column_name_when_from_is_used
      skip_if_sqlite3_version_includes_quoting_bug
      klass = Class.new(ActiveRecord::Base) do
        self.table_name = :test_with_keyword_column_name
        alias_attribute :description, :desc
      end
      klass.create!(description: "foo")

      assert_equal ["foo"], klass.select(:description).from(klass.all).map(&:desc)
    end

292 293 294 295
    def test_relation_merging_with_merged_joins_as_strings
      join_string = "LEFT OUTER JOIN #{Rating.quoted_table_name} ON #{SpecialComment.quoted_table_name}.id = #{Rating.quoted_table_name}.comment_id"
      special_comments_with_ratings = SpecialComment.joins join_string
      posts_with_special_comments_with_ratings = Post.group("posts.id").joins(:special_comments).merge(special_comments_with_ratings)
296
      assert_equal({ 2 => 1, 4 => 3, 5 => 1 }, authors(:david).posts.merge(posts_with_special_comments_with_ratings).count)
297
    end
298 299 300 301 302 303

    class EnsureRoundTripTypeCasting < ActiveRecord::Type::Value
      def type
        :string
      end

304
      def deserialize(value)
305 306 307 308
        raise value unless value == "type cast for database"
        "type cast from database"
      end

309
      def serialize(value)
310 311 312 313 314 315
        raise value unless value == "value from user"
        "type cast for database"
      end
    end

    class UpdateAllTestModel < ActiveRecord::Base
316
      self.table_name = "posts"
317 318 319 320 321 322 323 324 325

      attribute :body, EnsureRoundTripTypeCasting.new
    end

    def test_update_all_goes_through_normal_type_casting
      UpdateAllTestModel.update_all(body: "value from user", type: nil) # No STI

      assert_equal "type cast from database", UpdateAllTestModel.first.body
    end
326 327 328

    private

329 330 331
      def skip_if_sqlite3_version_includes_quoting_bug
        if sqlite3_version_includes_quoting_bug?
          skip <<-ERROR.squish
332 333 334 335
            You are using an outdated version of SQLite3 which has a bug in
            quoted column names. Please update SQLite3 and rebuild the sqlite3
            ruby gem
          ERROR
336
        end
337 338
      end

339 340 341 342 343 344 345
      def sqlite3_version_includes_quoting_bug?
        if current_adapter?(:SQLite3Adapter)
          selected_quoted_column_names = ActiveRecord::Base.connection.exec_query(
            'SELECT "join" FROM (SELECT id AS "join" FROM posts) subquery'
          ).columns
          ["join"] != selected_quoted_column_names
        end
346
      end
347 348
  end
end