mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
7abd9b5f66
This saves checking the inehritance chain on each access.
661 lines
22 KiB
Ruby
661 lines
22 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require "cases/helper"
|
|
require "models/author"
|
|
require "models/company"
|
|
require "models/membership"
|
|
require "models/person"
|
|
require "models/post"
|
|
require "models/project"
|
|
require "models/subscriber"
|
|
require "models/vegetables"
|
|
require "models/shop"
|
|
require "models/sponsor"
|
|
|
|
module InheritanceTestHelper
|
|
def with_store_full_sti_class(&block)
|
|
assign_store_full_sti_class true, &block
|
|
end
|
|
|
|
def without_store_full_sti_class(&block)
|
|
assign_store_full_sti_class false, &block
|
|
end
|
|
|
|
def assign_store_full_sti_class(flag)
|
|
old_store_full_sti_class = ActiveRecord::Base.store_full_sti_class
|
|
ActiveRecord::Base.store_full_sti_class = flag
|
|
yield
|
|
ensure
|
|
ActiveRecord::Base.store_full_sti_class = old_store_full_sti_class
|
|
end
|
|
end
|
|
|
|
class InheritanceTest < ActiveRecord::TestCase
|
|
include InheritanceTestHelper
|
|
fixtures :companies, :projects, :subscribers, :accounts, :vegetables, :memberships
|
|
|
|
def test_class_with_store_full_sti_class_returns_full_name
|
|
with_store_full_sti_class do
|
|
assert_equal "Namespaced::Company", Namespaced::Company.sti_name
|
|
end
|
|
end
|
|
|
|
def test_class_with_blank_sti_name
|
|
company = Company.first
|
|
company = company.dup
|
|
company.update!(type: " ")
|
|
company = Company.find(company.id)
|
|
assert_equal " ", company.type
|
|
end
|
|
|
|
def test_class_without_store_full_sti_class_returns_demodulized_name
|
|
without_store_full_sti_class do
|
|
assert_equal "Company", Namespaced::Company.sti_name
|
|
end
|
|
end
|
|
|
|
def test_compute_type_success
|
|
assert_equal Author, Company.send(:compute_type, "Author")
|
|
end
|
|
|
|
def test_compute_type_nonexistent_constant
|
|
e = assert_raises NameError do
|
|
Company.send :compute_type, "NonexistentModel"
|
|
end
|
|
assert_equal "uninitialized constant Company::NonexistentModel", e.message
|
|
assert_equal "Company::NonexistentModel", e.name
|
|
end
|
|
|
|
def test_compute_type_no_method_error
|
|
ActiveSupport::Dependencies.stub(:safe_constantize, proc { raise NoMethodError }) do
|
|
assert_raises NoMethodError do
|
|
Company.send :compute_type, "InvalidModel"
|
|
end
|
|
end
|
|
end
|
|
|
|
def test_compute_type_on_undefined_method
|
|
error = nil
|
|
begin
|
|
Class.new(Author) do
|
|
alias_method :foo, :bar
|
|
end
|
|
rescue => e
|
|
error = e
|
|
end
|
|
|
|
ActiveSupport::Dependencies.stub(:safe_constantize, proc { raise e }) do
|
|
exception = assert_raises NameError do
|
|
Company.send :compute_type, "InvalidModel"
|
|
end
|
|
assert_equal error.message, exception.message
|
|
end
|
|
end
|
|
|
|
def test_compute_type_argument_error
|
|
ActiveSupport::Dependencies.stub(:safe_constantize, proc { raise ArgumentError }) do
|
|
assert_raises ArgumentError do
|
|
Company.send :compute_type, "InvalidModel"
|
|
end
|
|
end
|
|
end
|
|
|
|
def test_should_store_demodulized_class_name_with_store_full_sti_class_option_disabled
|
|
without_store_full_sti_class do
|
|
item = Namespaced::Company.new
|
|
assert_equal "Company", item[:type]
|
|
end
|
|
end
|
|
|
|
def test_should_store_full_class_name_with_store_full_sti_class_option_enabled
|
|
with_store_full_sti_class do
|
|
item = Namespaced::Company.new
|
|
assert_equal "Namespaced::Company", item[:type]
|
|
end
|
|
end
|
|
|
|
def test_different_namespace_subclass_should_load_correctly_with_store_full_sti_class_option
|
|
with_store_full_sti_class do
|
|
item = Namespaced::Company.create name: "Wolverine 2"
|
|
assert_not_nil Company.find(item.id)
|
|
assert_not_nil Namespaced::Company.find(item.id)
|
|
end
|
|
end
|
|
|
|
def test_descends_from_active_record
|
|
assert_not_predicate ActiveRecord::Base, :descends_from_active_record?
|
|
|
|
# Abstract subclass of AR::Base.
|
|
assert_predicate LoosePerson, :descends_from_active_record?
|
|
|
|
# Concrete subclass of an abstract class.
|
|
assert_predicate LooseDescendant, :descends_from_active_record?
|
|
|
|
# Concrete subclass of AR::Base.
|
|
assert_predicate TightPerson, :descends_from_active_record?
|
|
|
|
# Concrete subclass of a concrete class but has no type column.
|
|
assert_predicate TightDescendant, :descends_from_active_record?
|
|
|
|
# Concrete subclass of AR::Base.
|
|
assert_predicate Post, :descends_from_active_record?
|
|
|
|
# Concrete subclasses of a concrete class which has a type column.
|
|
assert_not_predicate StiPost, :descends_from_active_record?
|
|
assert_not_predicate SubStiPost, :descends_from_active_record?
|
|
|
|
# Abstract subclass of a concrete class which has a type column.
|
|
# This is pathological, as you'll never have Sub < Abstract < Concrete.
|
|
assert_not_predicate AbstractStiPost, :descends_from_active_record?
|
|
|
|
# Concrete subclass of an abstract class which has a type column.
|
|
assert_not_predicate SubAbstractStiPost, :descends_from_active_record?
|
|
end
|
|
|
|
def test_company_descends_from_active_record
|
|
assert_not_predicate ActiveRecord::Base, :descends_from_active_record?
|
|
assert AbstractCompany.descends_from_active_record?, "AbstractCompany should descend from ActiveRecord::Base"
|
|
assert Company.descends_from_active_record?, "Company should descend from ActiveRecord::Base"
|
|
assert_not Class.new(Company).descends_from_active_record?, "Company subclass should not descend from ActiveRecord::Base"
|
|
end
|
|
|
|
def test_abstract_class
|
|
assert_not_predicate ActiveRecord::Base, :abstract_class?
|
|
assert_predicate LoosePerson, :abstract_class?
|
|
assert_not_predicate LooseDescendant, :abstract_class?
|
|
end
|
|
|
|
def test_inheritance_base_class
|
|
assert_equal Post, Post.base_class
|
|
assert_predicate Post, :base_class?
|
|
assert_equal Post, SpecialPost.base_class
|
|
assert_not_predicate SpecialPost, :base_class?
|
|
assert_equal Post, StiPost.base_class
|
|
assert_not_predicate StiPost, :base_class?
|
|
assert_equal Post, SubStiPost.base_class
|
|
assert_not_predicate SubStiPost, :base_class?
|
|
assert_equal SubAbstractStiPost, SubAbstractStiPost.base_class
|
|
assert_predicate SubAbstractStiPost, :base_class?
|
|
end
|
|
|
|
def test_abstract_inheritance_base_class
|
|
assert_equal LoosePerson, LoosePerson.base_class
|
|
assert_predicate LoosePerson, :base_class?
|
|
assert_equal LooseDescendant, LooseDescendant.base_class
|
|
assert_predicate LooseDescendant, :base_class?
|
|
assert_equal TightPerson, TightPerson.base_class
|
|
assert_predicate TightPerson, :base_class?
|
|
assert_equal TightPerson, TightDescendant.base_class
|
|
assert_not_predicate TightDescendant, :base_class?
|
|
end
|
|
|
|
def test_base_class_activerecord_error
|
|
assert_raise(ActiveRecord::ActiveRecordError) do
|
|
Class.new { include ActiveRecord::Inheritance }
|
|
end
|
|
end
|
|
|
|
def test_a_bad_type_column
|
|
Company.connection.insert "INSERT INTO companies (id, #{QUOTED_TYPE}, name) VALUES(100, 'bad_class!', 'Not happening')"
|
|
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Company.find(100) }
|
|
end
|
|
|
|
def test_inheritance_find
|
|
assert_kind_of Firm, Company.find(1), "37signals should be a firm"
|
|
assert_kind_of Firm, Firm.find(1), "37signals should be a firm"
|
|
assert_kind_of Client, Company.find(2), "Summit should be a client"
|
|
assert_kind_of Client, Client.find(2), "Summit should be a client"
|
|
end
|
|
|
|
def test_alt_inheritance_find
|
|
assert_kind_of Cucumber, Vegetable.find(1)
|
|
assert_kind_of Cucumber, Cucumber.find(1)
|
|
assert_kind_of Cabbage, Vegetable.find(2)
|
|
assert_kind_of Cabbage, Cabbage.find(2)
|
|
end
|
|
|
|
def test_alt_becomes_works_with_sti
|
|
vegetable = Vegetable.find(1)
|
|
assert_kind_of Vegetable, vegetable
|
|
cabbage = vegetable.becomes(Cabbage)
|
|
assert_kind_of Cabbage, cabbage
|
|
end
|
|
|
|
def test_becomes_sets_variables_before_initialization_callbacks
|
|
vegetable = Vegetable.create!(name: "yelling carrot")
|
|
assert_kind_of Vegetable, vegetable
|
|
assert_equal "yelling carrot", vegetable.name
|
|
|
|
yelling_veggie = vegetable.becomes(YellingVegetable)
|
|
assert_equal "YELLING CARROT", yelling_veggie.name, "YellingVegetable name should be YELLING CARROT"
|
|
assert_equal "YELLING CARROT", vegetable.name, "Vegetable name should be YELLING CARROT after becoming a YellingVegetable"
|
|
end
|
|
|
|
def test_becomes_and_change_tracking_for_inheritance_columns
|
|
cucumber = Vegetable.find(1)
|
|
cabbage = cucumber.becomes!(Cabbage)
|
|
assert_equal ["Cucumber", "Cabbage"], cabbage.custom_type_change
|
|
end
|
|
|
|
def test_alt_becomes_bang_resets_inheritance_type_column
|
|
vegetable = Vegetable.create!(name: "Red Pepper")
|
|
assert_nil vegetable.custom_type
|
|
|
|
cabbage = vegetable.becomes!(Cabbage)
|
|
assert_equal "Cabbage", cabbage.custom_type
|
|
|
|
cabbage.becomes!(Vegetable)
|
|
assert_nil cabbage.custom_type
|
|
end
|
|
|
|
def test_inheritance_find_all
|
|
companies = Company.all.merge!(order: "id").to_a
|
|
assert_kind_of Firm, companies[0], "37signals should be a firm"
|
|
assert_kind_of Client, companies[1], "Summit should be a client"
|
|
end
|
|
|
|
def test_alt_inheritance_find_all
|
|
companies = Vegetable.all.merge!(order: "id").to_a
|
|
assert_kind_of Cucumber, companies[0]
|
|
assert_kind_of Cabbage, companies[1]
|
|
end
|
|
|
|
def test_inheritance_save
|
|
firm = Firm.new
|
|
firm.name = "Next Angle"
|
|
firm.save
|
|
|
|
next_angle = Company.find(firm.id)
|
|
assert_kind_of Firm, next_angle, "Next Angle should be a firm"
|
|
end
|
|
|
|
def test_alt_inheritance_save
|
|
cabbage = Cabbage.new(name: "Savoy")
|
|
cabbage.save!
|
|
|
|
savoy = Vegetable.find(cabbage.id)
|
|
assert_kind_of Cabbage, savoy
|
|
end
|
|
|
|
def test_inheritance_new_with_default_class
|
|
company = Company.new
|
|
assert_equal Company, company.class
|
|
end
|
|
|
|
def test_inheritance_new_with_base_class
|
|
company = Company.new(type: "Company")
|
|
assert_equal Company, company.class
|
|
end
|
|
|
|
def test_inheritance_new_with_subclass
|
|
firm = Company.new(type: "Firm")
|
|
assert_equal Firm, firm.class
|
|
end
|
|
|
|
def test_where_new_with_subclass
|
|
firm = Company.where(type: "Firm").new
|
|
assert_equal Firm, firm.class
|
|
end
|
|
|
|
def test_where_create_with_subclass
|
|
firm = Company.where(type: "Firm").create(name: "Basecamp")
|
|
assert_equal Firm, firm.class
|
|
end
|
|
|
|
def test_where_create_bang_with_subclass
|
|
firm = Company.where(type: "Firm").create!(name: "Basecamp")
|
|
assert_equal Firm, firm.class
|
|
end
|
|
|
|
def test_new_with_abstract_class
|
|
e = assert_raises(NotImplementedError) do
|
|
AbstractCompany.new
|
|
end
|
|
assert_equal("AbstractCompany is an abstract class and cannot be instantiated.", e.message)
|
|
end
|
|
|
|
def test_new_with_ar_base
|
|
e = assert_raises(NotImplementedError) do
|
|
ActiveRecord::Base.new
|
|
end
|
|
assert_equal("ActiveRecord::Base is an abstract class and cannot be instantiated.", e.message)
|
|
end
|
|
|
|
def test_new_with_invalid_type
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Company.new(type: "InvalidType") }
|
|
end
|
|
|
|
def test_new_with_unrelated_type
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Company.new(type: "Account") }
|
|
end
|
|
|
|
def test_where_new_with_invalid_type
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Company.where(type: "InvalidType").new }
|
|
end
|
|
|
|
def test_where_new_with_unrelated_type
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Company.where(type: "Account").new }
|
|
end
|
|
|
|
def test_where_create_with_invalid_type
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Company.where(type: "InvalidType").create }
|
|
end
|
|
|
|
def test_where_create_with_unrelated_type
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Company.where(type: "Account").create }
|
|
end
|
|
|
|
def test_where_create_bang_with_invalid_type
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Company.where(type: "InvalidType").create! }
|
|
end
|
|
|
|
def test_where_create_bang_with_unrelated_type
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Company.where(type: "Account").create! }
|
|
end
|
|
|
|
def test_new_with_unrelated_namespaced_type
|
|
without_store_full_sti_class do
|
|
e = assert_raises ActiveRecord::SubclassNotFound do
|
|
Namespaced::Company.new(type: "Firm")
|
|
end
|
|
|
|
assert_equal "Invalid single-table inheritance type: Namespaced::Firm is not a subclass of Namespaced::Company", e.message
|
|
end
|
|
end
|
|
|
|
def test_new_with_complex_inheritance
|
|
assert_nothing_raised { Client.new(type: "VerySpecialClient") }
|
|
end
|
|
|
|
def test_new_without_storing_full_sti_class
|
|
without_store_full_sti_class do
|
|
item = Company.new(type: "SpecialCo")
|
|
assert_instance_of Company::SpecialCo, item
|
|
end
|
|
end
|
|
|
|
def test_new_with_autoload_paths
|
|
path = File.expand_path("../models/autoloadable", __dir__)
|
|
ActiveSupport::Dependencies.autoload_paths << path
|
|
|
|
firm = Company.new(type: "ExtraFirm")
|
|
assert_equal ExtraFirm, firm.class
|
|
ensure
|
|
ActiveSupport::Dependencies.autoload_paths.reject! { |p| p == path }
|
|
ActiveSupport::Dependencies.clear
|
|
end
|
|
|
|
def test_inheritance_condition
|
|
assert_equal 12, Company.count
|
|
assert_equal 3, Firm.count
|
|
assert_equal 5, Client.count
|
|
end
|
|
|
|
def test_alt_inheritance_condition
|
|
assert_equal 4, Vegetable.count
|
|
assert_equal 1, Cucumber.count
|
|
assert_equal 3, Cabbage.count
|
|
end
|
|
|
|
def test_finding_incorrect_type_data
|
|
assert_raise(ActiveRecord::RecordNotFound) { Firm.find(2) }
|
|
assert_nothing_raised { Firm.find(1) }
|
|
end
|
|
|
|
def test_alt_finding_incorrect_type_data
|
|
assert_raise(ActiveRecord::RecordNotFound) { Cucumber.find(2) }
|
|
assert_nothing_raised { Cucumber.find(1) }
|
|
end
|
|
|
|
def test_update_all_within_inheritance
|
|
Client.update_all "name = 'I am a client'"
|
|
assert_equal "I am a client", Client.first.name
|
|
# Order by added as otherwise Oracle tests were failing because of different order of results
|
|
assert_equal "37signals", Firm.all.merge!(order: "id").to_a.first.name
|
|
end
|
|
|
|
def test_alt_update_all_within_inheritance
|
|
Cabbage.update_all "name = 'the cabbage'"
|
|
assert_equal "the cabbage", Cabbage.first.name
|
|
assert_equal ["my cucumber"], Cucumber.all.map(&:name).uniq
|
|
end
|
|
|
|
def test_destroy_all_within_inheritance
|
|
Client.destroy_all
|
|
assert_equal 0, Client.count
|
|
assert_equal 3, Firm.count
|
|
end
|
|
|
|
def test_alt_destroy_all_within_inheritance
|
|
Cabbage.destroy_all
|
|
assert_equal 0, Cabbage.count
|
|
assert_equal 1, Cucumber.count
|
|
end
|
|
|
|
def test_find_first_within_inheritance
|
|
assert_kind_of Firm, Company.all.merge!(where: "name = '37signals'").first
|
|
assert_kind_of Firm, Firm.all.merge!(where: "name = '37signals'").first
|
|
assert_nil Client.all.merge!(where: "name = '37signals'").first
|
|
end
|
|
|
|
def test_alt_find_first_within_inheritance
|
|
assert_kind_of Cabbage, Vegetable.all.merge!(where: "name = 'his cabbage'").first
|
|
assert_kind_of Cabbage, Cabbage.all.merge!(where: "name = 'his cabbage'").first
|
|
assert_nil Cucumber.all.merge!(where: "name = 'his cabbage'").first
|
|
end
|
|
|
|
def test_complex_inheritance
|
|
very_special_client = VerySpecialClient.create("name" => "veryspecial")
|
|
assert_equal very_special_client, VerySpecialClient.where("name = 'veryspecial'").first
|
|
assert_equal very_special_client, SpecialClient.all.merge!(where: "name = 'veryspecial'").first
|
|
assert_equal very_special_client, Company.all.merge!(where: "name = 'veryspecial'").first
|
|
assert_equal very_special_client, Client.all.merge!(where: "name = 'veryspecial'").first
|
|
assert_equal 1, Client.all.merge!(where: "name = 'Summit'").to_a.size
|
|
assert_equal very_special_client, Client.find(very_special_client.id)
|
|
end
|
|
|
|
def test_alt_complex_inheritance
|
|
king_cole = KingCole.create("name" => "uniform heads")
|
|
assert_equal king_cole, KingCole.where("name = 'uniform heads'").first
|
|
assert_equal king_cole, GreenCabbage.all.merge!(where: "name = 'uniform heads'").first
|
|
assert_equal king_cole, Cabbage.all.merge!(where: "name = 'uniform heads'").first
|
|
assert_equal king_cole, Vegetable.all.merge!(where: "name = 'uniform heads'").first
|
|
assert_equal 1, Cabbage.all.merge!(where: "name = 'his cabbage'").to_a.size
|
|
assert_equal king_cole, Cabbage.find(king_cole.id)
|
|
end
|
|
|
|
def test_eager_load_belongs_to_something_inherited
|
|
account = Account.all.merge!(includes: :firm).find(1)
|
|
assert account.association(:firm).loaded?, "association was not eager loaded"
|
|
end
|
|
|
|
def test_alt_eager_loading
|
|
cabbage = RedCabbage.all.merge!(includes: :seller).find(4)
|
|
assert cabbage.association(:seller).loaded?, "association was not eager loaded"
|
|
end
|
|
|
|
def test_eager_load_belongs_to_primary_key_quoting
|
|
c = Account.connection
|
|
bind_param = Arel::Nodes::BindParam.new(nil)
|
|
assert_sql(/#{Regexp.escape(c.quote_table_name("companies.id"))} = (?:#{Regexp.escape(bind_param.to_sql)}|1)/i) do
|
|
Account.all.merge!(includes: :firm).find(1)
|
|
end
|
|
end
|
|
|
|
def test_inherits_custom_primary_key
|
|
assert_equal Subscriber.primary_key, SpecialSubscriber.primary_key
|
|
end
|
|
|
|
def test_inheritance_without_mapping
|
|
assert_kind_of SpecialSubscriber, SpecialSubscriber.find("webster132")
|
|
assert_nothing_raised { s = SpecialSubscriber.new("name" => "And breaaaaathe!"); s.id = "roger"; s.save }
|
|
end
|
|
|
|
def test_scope_inherited_properly
|
|
assert_nothing_raised { Company.of_first_firm.to_a }
|
|
assert_nothing_raised { Client.of_first_firm.to_a }
|
|
end
|
|
|
|
def test_inheritance_with_default_scope
|
|
assert_equal 1, SelectedMembership.count(:all)
|
|
end
|
|
end
|
|
|
|
class InheritanceComputeTypeTest < ActiveRecord::TestCase
|
|
include InheritanceTestHelper
|
|
fixtures :companies
|
|
|
|
def test_instantiation_doesnt_try_to_require_corresponding_file
|
|
without_store_full_sti_class do
|
|
foo = Firm.first.clone
|
|
foo.type = "FirmOnTheFly"
|
|
foo.save!
|
|
|
|
# Should fail without FirmOnTheFly in the type condition.
|
|
assert_raise(ActiveRecord::RecordNotFound) { Firm.find(foo.id) }
|
|
assert_raise(ActiveRecord::RecordNotFound) { Firm.find_by!(id: foo.id) }
|
|
|
|
# Nest FirmOnTheFly in the test case where Dependencies won't see it.
|
|
self.class.const_set :FirmOnTheFly, Class.new(Firm)
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Firm.find(foo.id) }
|
|
assert_raise(ActiveRecord::SubclassNotFound) { Firm.find_by!(id: foo.id) }
|
|
|
|
# Nest FirmOnTheFly in Firm where Dependencies will see it.
|
|
# This is analogous to nesting models in a migration.
|
|
Firm.const_set :FirmOnTheFly, Class.new(Firm)
|
|
|
|
# And instantiate will find the existing constant rather than trying
|
|
# to require firm_on_the_fly.
|
|
assert_nothing_raised { assert_kind_of Firm::FirmOnTheFly, Firm.find(foo.id) }
|
|
assert_nothing_raised { assert_kind_of Firm::FirmOnTheFly, Firm.find_by!(id: foo.id) }
|
|
end
|
|
ensure
|
|
self.class.send(:remove_const, :FirmOnTheFly) rescue nil
|
|
Firm.send(:remove_const, :FirmOnTheFly) rescue nil
|
|
end
|
|
|
|
def test_sti_type_from_attributes_disabled_in_non_sti_class
|
|
phone = Shop::Product::Type.new(name: "Phone")
|
|
product = Shop::Product.new(type: phone)
|
|
assert product.save
|
|
end
|
|
|
|
def test_inheritance_new_with_subclass_as_default
|
|
original_type = Company.columns_hash["type"].default
|
|
ActiveRecord::Base.connection.change_column_default :companies, :type, "Firm"
|
|
Company.reset_column_information
|
|
|
|
firm = Company.new # without arguments
|
|
assert_equal "Firm", firm.type
|
|
assert_instance_of Firm, firm
|
|
|
|
firm = Company.new(firm_name: "Shri Hans Plastic") # with arguments
|
|
assert_equal "Firm", firm.type
|
|
assert_instance_of Firm, firm
|
|
|
|
client = Client.new
|
|
assert_equal "Client", client.type
|
|
assert_instance_of Client, client
|
|
|
|
firm = Company.new(type: "Client") # overwrite the default type
|
|
assert_equal "Client", firm.type
|
|
assert_instance_of Client, firm
|
|
ensure
|
|
ActiveRecord::Base.connection.change_column_default :companies, :type, original_type
|
|
Company.reset_column_information
|
|
end
|
|
end
|
|
|
|
class InheritanceAttributeTest < ActiveRecord::TestCase
|
|
class Company < ActiveRecord::Base
|
|
self.table_name = "companies"
|
|
attribute :type, :string, default: "InheritanceAttributeTest::Startup"
|
|
end
|
|
|
|
class Startup < Company
|
|
end
|
|
|
|
class Empire < Company
|
|
end
|
|
|
|
def test_inheritance_new_with_subclass_as_default
|
|
startup = Company.new # without arguments
|
|
assert_equal "InheritanceAttributeTest::Startup", startup.type
|
|
assert_instance_of Startup, startup
|
|
|
|
empire = Company.new(type: "InheritanceAttributeTest::Empire") # without arguments
|
|
assert_equal "InheritanceAttributeTest::Empire", empire.type
|
|
assert_instance_of Empire, empire
|
|
end
|
|
end
|
|
|
|
class InheritanceAttributeMappingTest < ActiveRecord::TestCase
|
|
setup do
|
|
Company.delete_all
|
|
Sponsor.delete_all
|
|
end
|
|
|
|
class OmgStiType < ActiveRecord::Type::String
|
|
def cast_value(value)
|
|
if value =~ /\Aomg_(.+)\z/
|
|
$1.classify
|
|
else
|
|
value
|
|
end
|
|
end
|
|
|
|
def serialize(value)
|
|
if value
|
|
"omg_%s" % value.underscore
|
|
end
|
|
end
|
|
end
|
|
|
|
ActiveRecord::Type.register :omg_sti, OmgStiType
|
|
|
|
class Company < ActiveRecord::Base
|
|
self.table_name = "companies"
|
|
attribute :type, :omg_sti
|
|
end
|
|
|
|
class Startup < Company; end
|
|
class Empire < Company; end
|
|
|
|
class Sponsor < ActiveRecord::Base
|
|
self.table_name = "sponsors"
|
|
attribute :sponsorable_type, :omg_sti
|
|
|
|
belongs_to :sponsorable, polymorphic: true
|
|
end
|
|
|
|
def test_sti_with_custom_type
|
|
Startup.create! name: "a Startup"
|
|
Empire.create! name: "an Empire"
|
|
|
|
assert_equal [["a Startup", "omg_inheritance_attribute_mapping_test/startup"],
|
|
["an Empire", "omg_inheritance_attribute_mapping_test/empire"]], ActiveRecord::Base.connection.select_rows("SELECT name, type FROM companies").sort
|
|
assert_equal [["a Startup", "InheritanceAttributeMappingTest::Startup"],
|
|
["an Empire", "InheritanceAttributeMappingTest::Empire"]], Company.all.map { |a| [a.name, a.type] }.sort
|
|
|
|
startup = Startup.first
|
|
startup.becomes! Empire
|
|
startup.save!
|
|
|
|
assert_equal [["a Startup", "omg_inheritance_attribute_mapping_test/empire"],
|
|
["an Empire", "omg_inheritance_attribute_mapping_test/empire"]], ActiveRecord::Base.connection.select_rows("SELECT name, type FROM companies").sort
|
|
|
|
assert_equal [["a Startup", "InheritanceAttributeMappingTest::Empire"],
|
|
["an Empire", "InheritanceAttributeMappingTest::Empire"]], Company.all.map { |a| [a.name, a.type] }.sort
|
|
end
|
|
|
|
def test_polymorphic_associations_custom_type
|
|
startup = Startup.create! name: "a Startup"
|
|
sponsor = Sponsor.create! sponsorable: startup
|
|
|
|
assert_equal ["omg_inheritance_attribute_mapping_test/company"], ActiveRecord::Base.connection.select_values("SELECT sponsorable_type FROM sponsors")
|
|
|
|
sponsor = Sponsor.find(sponsor.id)
|
|
assert_equal startup, sponsor.sponsorable
|
|
end
|
|
end
|