reflection_test.rb 17.4 KB
Newer Older
1
require "cases/helper"
J
Jeremy Kemper 已提交
2 3 4 5
require 'models/topic'
require 'models/customer'
require 'models/company'
require 'models/company_in_module'
6
require 'models/ship'
7
require 'models/pirate'
8
require 'models/price_estimate'
9 10 11 12 13 14 15 16 17 18
require 'models/essay'
require 'models/author'
require 'models/organization'
require 'models/post'
require 'models/tagging'
require 'models/category'
require 'models/book'
require 'models/subscriber'
require 'models/subscription'
require 'models/tag'
19
require 'models/sponsor'
20
require 'models/edge'
21 22 23 24 25
require 'models/hotel'
require 'models/chef'
require 'models/department'
require 'models/cake_designer'
require 'models/drink_designer'
D
Initial  
David Heinemeier Hansson 已提交
26

27
class ReflectionTest < ActiveRecord::TestCase
28 29
  include ActiveRecord::Reflection

30
  fixtures :topics, :customers, :companies, :subscribers, :price_estimates
31

D
Initial  
David Heinemeier Hansson 已提交
32 33 34 35
  def setup
    @first = Topic.find(1)
  end

36
  def test_human_name
37 38
    assert_equal "Price estimate", PriceEstimate.model_name.human
    assert_equal "Subscriber", Subscriber.model_name.human
39 40
  end

D
Initial  
David Heinemeier Hansson 已提交
41 42
  def test_read_attribute_names
    assert_equal(
43
      %w( id title author_name author_email_address bonus_time written_on last_read content important group approved replies_count unique_replies_count parent_id parent_title type created_at updated_at ).sort,
44
      @first.attribute_names.sort
D
Initial  
David Heinemeier Hansson 已提交
45 46
    )
  end
47

D
Initial  
David Heinemeier Hansson 已提交
48
  def test_columns
49
    assert_equal 18, Topic.columns.length
D
Initial  
David Heinemeier Hansson 已提交
50 51
  end

52 53
  def test_columns_are_returned_in_the_order_they_were_declared
    column_names = Topic.columns.map { |column| column.name }
54
    assert_equal %w(id title author_name author_email_address written_on bonus_time last_read content important approved replies_count unique_replies_count parent_id parent_title type group created_at updated_at), column_names
55 56
  end

D
Initial  
David Heinemeier Hansson 已提交
57
  def test_content_columns
58 59
    content_columns        = Topic.content_columns
    content_column_names   = content_columns.map {|column| column.name}
A
Alvaro Bautista 已提交
60 61
    assert_equal 13, content_columns.length
    assert_equal %w(title author_name author_email_address written_on bonus_time last_read content important group approved parent_title created_at updated_at).sort, content_column_names.sort
D
Initial  
David Heinemeier Hansson 已提交
62
  end
63

D
Initial  
David Heinemeier Hansson 已提交
64 65
  def test_column_string_type_and_limit
    assert_equal :string, @first.column_for_attribute("title").type
66
    assert_equal 250, @first.column_for_attribute("title").limit
D
Initial  
David Heinemeier Hansson 已提交
67
  end
J
Jeremy Kemper 已提交
68

69
  def test_column_null_not_null
70
    subscriber = Subscriber.first
71 72 73
    assert subscriber.column_for_attribute("name").null
    assert !subscriber.column_for_attribute("nick").null
  end
D
Initial  
David Heinemeier Hansson 已提交
74 75 76 77

  def test_human_name_for_column
    assert_equal "Author name", @first.column_for_attribute("author_name").human_name
  end
78

D
Initial  
David Heinemeier Hansson 已提交
79 80
  def test_integer_columns
    assert_equal :integer, @first.column_for_attribute("id").type
81
  end
D
Initial  
David Heinemeier Hansson 已提交
82

83
  def test_reflection_klass_for_nested_class_name
84
    reflection = MacroReflection.new(:company, nil, nil, { :class_name => 'MyApplication::Business::Company' }, ActiveRecord::Base)
85 86 87 88 89
    assert_nothing_raised do
      assert_equal MyApplication::Business::Company, reflection.klass
    end
  end

90 91 92 93 94 95 96 97
  def test_irregular_reflection_class_name
    ActiveSupport::Inflector.inflections do |inflect|
      inflect.irregular 'plural_irregular', 'plurales_irregulares'
    end
    reflection = AssociationReflection.new(:has_many, 'plurales_irregulares', nil, {}, ActiveRecord::Base)
    assert_equal 'PluralIrregular', reflection.class_name
  end

98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121
  def test_aggregation_reflection
    reflection_for_address = AggregateReflection.new(
      :composed_of, :address, nil, { :mapping => [ %w(address_street street), %w(address_city city), %w(address_country country) ] }, Customer
    )

    reflection_for_balance = AggregateReflection.new(
      :composed_of, :balance, nil, { :class_name => "Money", :mapping => %w(balance amount) }, Customer
    )

    reflection_for_gps_location = AggregateReflection.new(
      :composed_of, :gps_location, nil, { }, Customer
    )

    assert Customer.reflect_on_all_aggregations.include?(reflection_for_gps_location)
    assert Customer.reflect_on_all_aggregations.include?(reflection_for_balance)
    assert Customer.reflect_on_all_aggregations.include?(reflection_for_address)

    assert_equal reflection_for_address, Customer.reflect_on_aggregation(:address)

    assert_equal Address, Customer.reflect_on_aggregation(:address).klass

    assert_equal Money, Customer.reflect_on_aggregation(:balance).klass
  end

122 123 124 125 126 127 128 129 130
  def test_reflect_on_all_autosave_associations
    expected = Pirate.reflect_on_all_associations.select { |r| r.options[:autosave] }
    received = Pirate.reflect_on_all_autosave_associations

    assert !received.empty?
    assert_not_equal Pirate.reflect_on_all_associations.length, received.length
    assert_equal expected, received
  end

131
  def test_has_many_reflection
132
    reflection_for_clients = AssociationReflection.new(:has_many, :clients, nil, { :order => "id", :dependent => :destroy }, Firm)
D
Initial  
David Heinemeier Hansson 已提交
133 134 135 136

    assert_equal reflection_for_clients, Firm.reflect_on_association(:clients)

    assert_equal Client, Firm.reflect_on_association(:clients).klass
137 138
    assert_equal 'companies', Firm.reflect_on_association(:clients).table_name

D
Initial  
David Heinemeier Hansson 已提交
139
    assert_equal Client, Firm.reflect_on_association(:clients_of_firm).klass
140 141 142 143
    assert_equal 'companies', Firm.reflect_on_association(:clients_of_firm).table_name
  end

  def test_has_one_reflection
144
    reflection_for_account = AssociationReflection.new(:has_one, :account, nil, { :foreign_key => "firm_id", :dependent => :destroy }, Firm)
145 146 147 148
    assert_equal reflection_for_account, Firm.reflect_on_association(:account)

    assert_equal Account, Firm.reflect_on_association(:account).klass
    assert_equal 'accounts', Firm.reflect_on_association(:account).table_name
D
Initial  
David Heinemeier Hansson 已提交
149
  end
150

151 152
  def test_belongs_to_inferred_foreign_key_from_assoc_name
    Company.belongs_to :foo
153
    assert_equal "foo_id", Company.reflect_on_association(:foo).foreign_key
154
    Company.belongs_to :bar, :class_name => "Xyzzy"
155
    assert_equal "bar_id", Company.reflect_on_association(:bar).foreign_key
156
    Company.belongs_to :baz, :class_name => "Xyzzy", :foreign_key => "xyzzy_id"
157
    assert_equal "xyzzy_id", Company.reflect_on_association(:baz).foreign_key
158 159
  end

D
Initial  
David Heinemeier Hansson 已提交
160
  def test_association_reflection_in_modules
161
    ActiveRecord::Base.store_full_sti_class = false
162

163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197
    assert_reflection MyApplication::Business::Firm,
      :clients_of_firm,
      :klass      => MyApplication::Business::Client,
      :class_name => 'Client',
      :table_name => 'companies'

    assert_reflection MyApplication::Billing::Account,
      :firm,
      :klass      => MyApplication::Business::Firm,
      :class_name => 'MyApplication::Business::Firm',
      :table_name => 'companies'

    assert_reflection MyApplication::Billing::Account,
      :qualified_billing_firm,
      :klass      => MyApplication::Billing::Firm,
      :class_name => 'MyApplication::Billing::Firm',
      :table_name => 'companies'

    assert_reflection MyApplication::Billing::Account,
      :unqualified_billing_firm,
      :klass      => MyApplication::Billing::Firm,
      :class_name => 'Firm',
      :table_name => 'companies'

    assert_reflection MyApplication::Billing::Account,
      :nested_qualified_billing_firm,
      :klass      => MyApplication::Billing::Nested::Firm,
      :class_name => 'MyApplication::Billing::Nested::Firm',
      :table_name => 'companies'

    assert_reflection MyApplication::Billing::Account,
      :nested_unqualified_billing_firm,
      :klass      => MyApplication::Billing::Nested::Firm,
      :class_name => 'Nested::Firm',
      :table_name => 'companies'
198 199
  ensure
    ActiveRecord::Base.store_full_sti_class = true
D
Initial  
David Heinemeier Hansson 已提交
200
  end
J
Jeremy Kemper 已提交
201

202
  def test_reflection_should_not_raise_error_when_compared_to_other_object
203 204 205 206 207 208
    assert_not_equal Object.new, Firm._reflections['clients']
  end

  def test_has_and_belongs_to_many_reflection
    assert_equal :has_and_belongs_to_many, Category.reflections['posts'].macro
    assert_equal :posts, Category.reflect_on_all_associations(:has_and_belongs_to_many).first.name
209 210
  end

211
  def test_has_many_through_reflection
212
    assert_kind_of ThroughReflection, Subscriber.reflect_on_association(:books)
213
  end
J
Jon Leighton 已提交
214

215
  def test_chain
216
    expected = [
217
      Organization.reflect_on_association(:author_essay_categories),
218 219 220
      Author.reflect_on_association(:essays),
      Organization.reflect_on_association(:authors)
    ]
221
    actual = Organization.reflect_on_association(:author_essay_categories).chain
J
Jon Leighton 已提交
222

223 224
    assert_equal expected, actual
  end
J
Jon Leighton 已提交
225

226
  def test_scope_chain
227
    expected = [
228 229 230
      [Tagging.reflect_on_association(:tag).scope, Post.reflect_on_association(:first_blue_tags).scope],
      [Post.reflect_on_association(:first_taggings).scope],
      [Author.reflect_on_association(:misc_posts).scope]
231
    ]
232
    actual = Author.reflect_on_association(:misc_post_first_blue_tags).scope_chain
233
    assert_equal expected, actual
J
Jon Leighton 已提交
234

235
    expected = [
236 237 238 239 240
      [
        Tagging.reflect_on_association(:blue_tag).scope,
        Post.reflect_on_association(:first_blue_tags_2).scope,
        Author.reflect_on_association(:misc_post_first_blue_tags_2).scope
      ],
241
      [],
242 243
      []
    ]
244
    actual = Author.reflect_on_association(:misc_post_first_blue_tags_2).scope_chain
245 246
    assert_equal expected, actual
  end
J
Jon Leighton 已提交
247

248 249 250 251 252 253 254 255 256 257 258
  def test_scope_chain_does_not_interfere_with_hmt_with_polymorphic_case
    @hotel = Hotel.create!
    @department = @hotel.departments.create!
    @department.chefs.create!(employable: CakeDesigner.create!)
    @department.chefs.create!(employable: DrinkDesigner.create!)

    assert_equal 1, @hotel.cake_designers.size
    assert_equal 1, @hotel.drink_designers.size
    assert_equal 2, @hotel.chefs.size
  end

259 260 261
  def test_nested?
    assert !Author.reflect_on_association(:comments).nested?
    assert Author.reflect_on_association(:tags).nested?
J
Jon Leighton 已提交
262

263 264 265 266
    # Only goes :through once, but the through_reflection is a has_and_belongs_to_many, so this is
    # a nested through association
    assert Category.reflect_on_association(:post_comments).nested?
  end
J
Jon Leighton 已提交
267

268 269 270 271
  def test_association_primary_key
    # Normal association
    assert_equal "id",   Author.reflect_on_association(:posts).association_primary_key.to_s
    assert_equal "name", Author.reflect_on_association(:essay).association_primary_key.to_s
272
    assert_equal "name", Essay.reflect_on_association(:writer).association_primary_key.to_s
J
Jon Leighton 已提交
273

274 275 276 277 278
    # Through association (uses the :primary_key option from the source reflection)
    assert_equal "nick", Author.reflect_on_association(:subscribers).association_primary_key.to_s
    assert_equal "name", Author.reflect_on_association(:essay_category).association_primary_key.to_s
    assert_equal "custom_primary_key", Author.reflect_on_association(:tags_with_primary_key).association_primary_key.to_s # nested
  end
J
Jon Leighton 已提交
279

280
  def test_association_primary_key_raises_when_missing_primary_key
281
    reflection = ActiveRecord::Reflection::AssociationReflection.new(:fuu, :edge, nil, {}, Author)
282 283
    assert_raises(ActiveRecord::UnknownPrimaryKey) { reflection.association_primary_key }

284 285 286
    through = Class.new(ActiveRecord::Reflection::ThroughReflection) {
      define_method(:source_reflection) { reflection }
    }.new(:fuu, :edge, nil, {}, Author)
287 288 289
    assert_raises(ActiveRecord::UnknownPrimaryKey) { through.association_primary_key }
  end

290 291 292 293
  def test_active_record_primary_key
    assert_equal "nick", Subscriber.reflect_on_association(:subscriptions).active_record_primary_key.to_s
    assert_equal "name", Author.reflect_on_association(:essay).active_record_primary_key.to_s
  end
294

295
  def test_active_record_primary_key_raises_when_missing_primary_key
296
    reflection = ActiveRecord::Reflection::AssociationReflection.new(:fuu, :author, nil, {}, Edge)
297 298 299
    assert_raises(ActiveRecord::UnknownPrimaryKey) { reflection.active_record_primary_key }
  end

300 301 302 303 304
  def test_foreign_type
    assert_equal "sponsorable_type", Sponsor.reflect_on_association(:sponsorable).foreign_type.to_s
    assert_equal "sponsorable_type", Sponsor.reflect_on_association(:thing).foreign_type.to_s
  end

305
  def test_collection_association
306 307
    assert Pirate.reflect_on_association(:birds).collection?
    assert Pirate.reflect_on_association(:parrots).collection?
308

309 310
    assert !Pirate.reflect_on_association(:ship).collection?
    assert !Ship.reflect_on_association(:pirate).collection?
311 312
  end

313
  def test_default_association_validation
314
    assert AssociationReflection.new(:has_many, :clients, nil, {}, Firm).validate?
315

316 317 318
    assert !AssociationReflection.new(:has_one, :client, nil, {}, Firm).validate?
    assert !AssociationReflection.new(:belongs_to, :client, nil, {}, Firm).validate?
    assert !AssociationReflection.new(:has_and_belongs_to_many, :clients, nil, {}, Firm).validate?
319 320 321
  end

  def test_always_validate_association_if_explicit
322 323 324 325
    assert AssociationReflection.new(:has_one, :client, nil, { :validate => true }, Firm).validate?
    assert AssociationReflection.new(:belongs_to, :client, nil, { :validate => true }, Firm).validate?
    assert AssociationReflection.new(:has_many, :clients, nil, { :validate => true }, Firm).validate?
    assert AssociationReflection.new(:has_and_belongs_to_many, :clients, nil, { :validate => true }, Firm).validate?
326 327 328
  end

  def test_validate_association_if_autosave
329 330 331 332
    assert AssociationReflection.new(:has_one, :client, nil, { :autosave => true }, Firm).validate?
    assert AssociationReflection.new(:belongs_to, :client, nil, { :autosave => true }, Firm).validate?
    assert AssociationReflection.new(:has_many, :clients, nil, { :autosave => true }, Firm).validate?
    assert AssociationReflection.new(:has_and_belongs_to_many, :clients, nil, { :autosave => true }, Firm).validate?
333 334 335
  end

  def test_never_validate_association_if_explicit
336 337 338 339
    assert !AssociationReflection.new(:has_one, :client, nil, { :autosave => true, :validate => false }, Firm).validate?
    assert !AssociationReflection.new(:belongs_to, :client, nil, { :autosave => true, :validate => false }, Firm).validate?
    assert !AssociationReflection.new(:has_many, :clients, nil, { :autosave => true, :validate => false }, Firm).validate?
    assert !AssociationReflection.new(:has_and_belongs_to_many, :clients, nil, { :autosave => true, :validate => false }, Firm).validate?
340 341
  end

342 343 344 345 346
  def test_foreign_key
    assert_equal "author_id", Author.reflect_on_association(:posts).foreign_key.to_s
    assert_equal "category_id", Post.reflect_on_association(:categorizations).foreign_key.to_s
  end

347 348 349 350
  def test_through_reflection_scope_chain_does_not_modify_other_reflections
    orig_conds = Post.reflect_on_association(:first_blue_tags_2).scope_chain.inspect
    Author.reflect_on_association(:misc_post_first_blue_tags_2).scope_chain
    assert_equal orig_conds, Post.reflect_on_association(:first_blue_tags_2).scope_chain.inspect
J
Jon Leighton 已提交
351 352
  end

353 354 355 356
  def test_symbol_for_class_name
    assert_equal Client, Firm.reflect_on_association(:unsorted_clients_with_symbol).klass
  end

357 358 359 360
  def test_join_table
    category = Struct.new(:table_name, :pluralize_table_names).new('categories', true)
    product = Struct.new(:table_name, :pluralize_table_names).new('products', true)

361
    reflection = AssociationReflection.new(:has_and_belongs_to_many, :categories, nil, {}, product)
362 363 364
    reflection.stubs(:klass).returns(category)
    assert_equal 'categories_products', reflection.join_table

365
    reflection = AssociationReflection.new(:has_and_belongs_to_many, :products, nil, {}, category)
366 367 368 369 370 371 372 373
    reflection.stubs(:klass).returns(product)
    assert_equal 'categories_products', reflection.join_table
  end

  def test_join_table_with_common_prefix
    category = Struct.new(:table_name, :pluralize_table_names).new('catalog_categories', true)
    product = Struct.new(:table_name, :pluralize_table_names).new('catalog_products', true)

374
    reflection = AssociationReflection.new(:has_and_belongs_to_many, :categories, nil, {}, product)
375 376 377
    reflection.stubs(:klass).returns(category)
    assert_equal 'catalog_categories_products', reflection.join_table

378
    reflection = AssociationReflection.new(:has_and_belongs_to_many, :products, nil, {}, category)
379 380 381 382 383 384 385 386
    reflection.stubs(:klass).returns(product)
    assert_equal 'catalog_categories_products', reflection.join_table
  end

  def test_join_table_with_different_prefix
    category = Struct.new(:table_name, :pluralize_table_names).new('catalog_categories', true)
    page = Struct.new(:table_name, :pluralize_table_names).new('content_pages', true)

387
    reflection = AssociationReflection.new(:has_and_belongs_to_many, :categories, nil, {}, page)
388 389 390
    reflection.stubs(:klass).returns(category)
    assert_equal 'catalog_categories_content_pages', reflection.join_table

391
    reflection = AssociationReflection.new(:has_and_belongs_to_many, :pages, nil, {}, category)
392 393 394 395 396 397 398 399
    reflection.stubs(:klass).returns(page)
    assert_equal 'catalog_categories_content_pages', reflection.join_table
  end

  def test_join_table_can_be_overridden
    category = Struct.new(:table_name, :pluralize_table_names).new('categories', true)
    product = Struct.new(:table_name, :pluralize_table_names).new('products', true)

400
    reflection = AssociationReflection.new(:has_and_belongs_to_many, :categories, nil, { :join_table => 'product_categories' }, product)
401 402 403
    reflection.stubs(:klass).returns(category)
    assert_equal 'product_categories', reflection.join_table

404
    reflection = AssociationReflection.new(:has_and_belongs_to_many, :products, nil, { :join_table => 'product_categories' }, category)
405 406 407 408
    reflection.stubs(:klass).returns(product)
    assert_equal 'product_categories', reflection.join_table
  end

409 410 411 412 413 414 415
  private
    def assert_reflection(klass, association, options)
      assert reflection = klass.reflect_on_association(association)
      options.each do |method, value|
        assert_equal(value, reflection.send(method))
      end
    end
416
end