diff options
Diffstat (limited to 'activerecord/test')
36 files changed, 284 insertions, 257 deletions
| diff --git a/activerecord/test/cases/adapters/mysql/connection_test.rb b/activerecord/test/cases/adapters/mysql/connection_test.rb index c3f82bc63d..4bccd2cc59 100644 --- a/activerecord/test/cases/adapters/mysql/connection_test.rb +++ b/activerecord/test/cases/adapters/mysql/connection_test.rb @@ -128,11 +128,12 @@ class MysqlConnectionTest < ActiveRecord::TestCase      assert_equal [["STRICT_ALL_TABLES"]], result.rows    end -  def test_mysql_strict_mode_disabled +  def test_mysql_strict_mode_disabled_dont_override_global_sql_mode      run_without_connection do |orig_connection|        ActiveRecord::Model.establish_connection(orig_connection.merge({:strict => false})) -      result = ActiveRecord::Model.connection.exec_query "SELECT @@SESSION.sql_mode" -      assert_equal [['']], result.rows +      global_sql_mode = ActiveRecord::Model.connection.exec_query "SELECT @@GLOBAL.sql_mode" +      session_sql_mode = ActiveRecord::Model.connection.exec_query "SELECT @@SESSION.sql_mode" +      assert_equal global_sql_mode.rows, session_sql_mode.rows      end    end diff --git a/activerecord/test/cases/adapters/mysql2/connection_test.rb b/activerecord/test/cases/adapters/mysql2/connection_test.rb index 276c499276..c63e4fe5b6 100644 --- a/activerecord/test/cases/adapters/mysql2/connection_test.rb +++ b/activerecord/test/cases/adapters/mysql2/connection_test.rb @@ -44,11 +44,12 @@ class MysqlConnectionTest < ActiveRecord::TestCase      assert_equal [["STRICT_ALL_TABLES"]], result.rows    end -  def test_mysql_strict_mode_disabled +  def test_mysql_strict_mode_disabled_dont_override_global_sql_mode      run_without_connection do |orig_connection|        ActiveRecord::Model.establish_connection(orig_connection.merge({:strict => false})) -      result = ActiveRecord::Model.connection.exec_query "SELECT @@SESSION.sql_mode" -      assert_equal [['']], result.rows +      global_sql_mode = ActiveRecord::Model.connection.exec_query "SELECT @@GLOBAL.sql_mode" +      session_sql_mode = ActiveRecord::Model.connection.exec_query "SELECT @@SESSION.sql_mode" +      assert_equal global_sql_mode.rows, session_sql_mode.rows      end    end diff --git a/activerecord/test/cases/adapters/postgresql/connection_test.rb b/activerecord/test/cases/adapters/postgresql/connection_test.rb index 202f7e27c5..1ff307c735 100644 --- a/activerecord/test/cases/adapters/postgresql/connection_test.rb +++ b/activerecord/test/cases/adapters/postgresql/connection_test.rb @@ -85,16 +85,17 @@ module ActiveRecord        assert @connection.active?        original_connection_pid = @connection.query('select pg_backend_pid()') -      # Fail with bad connection after next query attempt. -      connection_class = class << @connection ; self ; end -      connection_class.class_eval <<-CODE +      # Fail with bad connection on next query attempt. +      raw_connection = @connection.raw_connection +      raw_connection_class = class << raw_connection ; self ; end +      raw_connection_class.class_eval <<-CODE, __FILE__, __LINE__ + 1          def query_fake(*args) -          if @called ||= false -            @connection.stubs(:status).returns(PCconn::CONNECTION_BAD) +          if !( @called ||= false ) +            self.stubs(:status).returns(PGconn::CONNECTION_BAD) +            @called = true              raise PGError            else -            @called = true -            @connection.unstub(:status) +            self.unstub(:status)              query_unfake(*args)            end          end @@ -107,13 +108,13 @@ module ActiveRecord          @connection.verify!          new_connection_pid = @connection.query('select pg_backend_pid()')        ensure -        connection_class.class_eval <<-CODE +        raw_connection_class.class_eval <<-CODE            alias query query_unfake            undef query_fake          CODE        end -      assert_equal original_connection_pid, new_connection_pid, "Should have a new underlying connection pid" +      assert_not_equal original_connection_pid, new_connection_pid, "Should have a new underlying connection pid"      end      # Must have with_manual_interventions set to true for this diff --git a/activerecord/test/cases/associations/belongs_to_associations_test.rb b/activerecord/test/cases/associations/belongs_to_associations_test.rb index ec7e4f5fb7..5f7825783b 100644 --- a/activerecord/test/cases/associations/belongs_to_associations_test.rb +++ b/activerecord/test/cases/associations/belongs_to_associations_test.rb @@ -524,13 +524,13 @@ class BelongsToAssociationsTest < ActiveRecord::TestCase    def test_invalid_belongs_to_dependent_option_nullify_raises_exception      assert_raise ArgumentError do -      Author.belongs_to :special_author_address, :dependent => :nullify +      Class.new(Author).belongs_to :special_author_address, :dependent => :nullify      end    end    def test_invalid_belongs_to_dependent_option_restrict_raises_exception      assert_raise ArgumentError do -      Author.belongs_to :special_author_address, :dependent => :restrict +      Class.new(Author).belongs_to :special_author_address, :dependent => :restrict      end    end diff --git a/activerecord/test/cases/associations/has_many_associations_test.rb b/activerecord/test/cases/associations/has_many_associations_test.rb index 43440c1146..04714f42e9 100644 --- a/activerecord/test/cases/associations/has_many_associations_test.rb +++ b/activerecord/test/cases/associations/has_many_associations_test.rb @@ -1091,9 +1091,6 @@ class HasManyAssociationsTest < ActiveRecord::TestCase    end    def test_restrict -    option_before = ActiveRecord::Base.dependent_restrict_raises -    ActiveRecord::Base.dependent_restrict_raises = true -      firm = RestrictedFirm.create!(:name => 'restrict')      firm.companies.create(:name => 'child') @@ -1101,15 +1098,25 @@ class HasManyAssociationsTest < ActiveRecord::TestCase      assert_raise(ActiveRecord::DeleteRestrictionError) { firm.destroy }      assert RestrictedFirm.exists?(:name => 'restrict')      assert firm.companies.exists?(:name => 'child') -  ensure -    ActiveRecord::Base.dependent_restrict_raises = option_before    end -  def test_restrict_when_dependent_restrict_raises_config_set_to_false -    option_before = ActiveRecord::Base.dependent_restrict_raises -    ActiveRecord::Base.dependent_restrict_raises = false +  def test_restrict_is_deprecated +    klass = Class.new(ActiveRecord::Base) +    assert_deprecated { klass.has_many :posts, dependent: :restrict } +  end -    firm = RestrictedFirm.create!(:name => 'restrict') +  def test_restrict_with_exception +    firm = RestrictedWithExceptionFirm.create!(:name => 'restrict') +    firm.companies.create(:name => 'child') + +    assert !firm.companies.empty? +    assert_raise(ActiveRecord::DeleteRestrictionError) { firm.destroy } +    assert RestrictedWithExceptionFirm.exists?(:name => 'restrict') +    assert firm.companies.exists?(:name => 'child') +  end + +  def test_restrict_with_error +    firm = RestrictedWithErrorFirm.create!(:name => 'restrict')      firm.companies.create(:name => 'child')      assert !firm.companies.empty? @@ -1119,10 +1126,8 @@ class HasManyAssociationsTest < ActiveRecord::TestCase      assert !firm.errors.empty?      assert_equal "Cannot delete record because dependent companies exist", firm.errors[:base].first -    assert RestrictedFirm.exists?(:name => 'restrict') +    assert RestrictedWithErrorFirm.exists?(:name => 'restrict')      assert firm.companies.exists?(:name => 'child') -  ensure -    ActiveRecord::Base.dependent_restrict_raises = option_before    end    def test_included_in_collection @@ -1602,18 +1607,6 @@ class HasManyAssociationsTest < ActiveRecord::TestCase      assert_equal [bulb1, bulb3], result    end -  def test_building_has_many_association_with_restrict_dependency -    option_before = ActiveRecord::Base.dependent_restrict_raises -    ActiveRecord::Base.dependent_restrict_raises = true - -    klass = Class.new(ActiveRecord::Base) - -    assert_deprecated     { klass.has_many :companies, :dependent => :restrict } -    assert_not_deprecated { klass.has_many :companies } -  ensure -    ActiveRecord::Base.dependent_restrict_raises = option_before -  end -    def test_collection_association_with_private_kernel_method      firm = companies(:first_firm)      assert_equal [accounts(:signals37)], firm.accounts.open diff --git a/activerecord/test/cases/associations/has_one_associations_test.rb b/activerecord/test/cases/associations/has_one_associations_test.rb index 112735839f..8bc633f2b5 100644 --- a/activerecord/test/cases/associations/has_one_associations_test.rb +++ b/activerecord/test/cases/associations/has_one_associations_test.rb @@ -156,10 +156,7 @@ class HasOneAssociationsTest < ActiveRecord::TestCase      assert_nothing_raised { firm.destroy }    end -  def test_dependence_with_restrict -    option_before = ActiveRecord::Base.dependent_restrict_raises -    ActiveRecord::Base.dependent_restrict_raises = true - +  def test_restrict      firm = RestrictedFirm.create!(:name => 'restrict')      firm.create_account(:credit_limit => 10) @@ -168,38 +165,26 @@ class HasOneAssociationsTest < ActiveRecord::TestCase      assert_raise(ActiveRecord::DeleteRestrictionError) { firm.destroy }      assert RestrictedFirm.exists?(:name => 'restrict')      assert firm.account.present? -  ensure -    ActiveRecord::Base.dependent_restrict_raises = option_before    end -  def test_dependence_with_restrict_with_dependent_restrict_raises_config_set_to_false -    option_before = ActiveRecord::Base.dependent_restrict_raises -    ActiveRecord::Base.dependent_restrict_raises = false +  def test_restrict_is_deprecated +    klass = Class.new(ActiveRecord::Base) +    assert_deprecated { klass.has_one :post, dependent: :restrict } +  end -    firm = RestrictedFirm.create!(:name => 'restrict') +  def test_restrict_with_exception +    firm = RestrictedWithExceptionFirm.create!(:name => 'restrict')      firm.create_account(:credit_limit => 10)      assert_not_nil firm.account -    firm.destroy - -    assert !firm.errors.empty? -    assert_equal "Cannot delete record because a dependent account exists", firm.errors[:base].first -    assert RestrictedFirm.exists?(:name => 'restrict') +    assert_raise(ActiveRecord::DeleteRestrictionError) { firm.destroy } +    assert RestrictedWithExceptionFirm.exists?(:name => 'restrict')      assert firm.account.present? -  ensure -    ActiveRecord::Base.dependent_restrict_raises = option_before    end -  def test_dependence_with_restrict_with_dependent_restrict_raises_config_set_to_false_and_attribute_name -    old_backend = I18n.backend -    I18n.backend = I18n::Backend::Simple.new -    I18n.backend.store_translations 'en', :activerecord => {:attributes => {:restricted_firm => {:account => "account model"}}} - -    option_before = ActiveRecord::Base.dependent_restrict_raises -    ActiveRecord::Base.dependent_restrict_raises = false - -    firm = RestrictedFirm.create!(:name => 'restrict') +  def test_restrict_with_error +    firm = RestrictedWithErrorFirm.create!(:name => 'restrict')      firm.create_account(:credit_limit => 10)      assert_not_nil firm.account @@ -207,12 +192,9 @@ class HasOneAssociationsTest < ActiveRecord::TestCase      firm.destroy      assert !firm.errors.empty? -    assert_equal "Cannot delete record because a dependent account model exists", firm.errors[:base].first -    assert RestrictedFirm.exists?(:name => 'restrict') +    assert_equal "Cannot delete record because a dependent account exists", firm.errors[:base].first +    assert RestrictedWithErrorFirm.exists?(:name => 'restrict')      assert firm.account.present? -  ensure -    ActiveRecord::Base.dependent_restrict_raises = option_before -    I18n.backend = old_backend    end    def test_successful_build_association @@ -524,15 +506,16 @@ class HasOneAssociationsTest < ActiveRecord::TestCase      assert_equal car.id, bulb.attributes_after_initialize['car_id']    end -  def test_building_has_one_association_with_dependent_restrict -    option_before = ActiveRecord::Base.dependent_restrict_raises -    ActiveRecord::Base.dependent_restrict_raises = true +  def test_has_one_transaction +    company = companies(:first_firm) +    account = Account.find(1) -    klass = Class.new(ActiveRecord::Base) +    company.account # force loading +    assert_no_queries { company.account = account } -    assert_deprecated     { klass.has_one :account, :dependent => :restrict } -    assert_not_deprecated { klass.has_one :account } -  ensure -    ActiveRecord::Base.dependent_restrict_raises = option_before +    company.account = nil +    assert_no_queries { company.account = nil } +    account = Account.find(2) +    assert_queries { company.account = account }    end  end diff --git a/activerecord/test/cases/associations/inverse_associations_test.rb b/activerecord/test/cases/associations/inverse_associations_test.rb index 8cb8a5a861..aad48e7ce9 100644 --- a/activerecord/test/cases/associations/inverse_associations_test.rb +++ b/activerecord/test/cases/associations/inverse_associations_test.rb @@ -259,6 +259,12 @@ class InverseHasManyTests < ActiveRecord::TestCase      assert_equal m.name, i.man.name, "Name of man should be the same after changes to replaced-child-owned instance"    end +  def test_parent_instance_should_be_shared_with_first_and_last_child +    man = Man.first +    assert man.interests.first.man.equal? man +    assert man.interests.last.man.equal? man +  end +    def test_trying_to_use_inverses_that_dont_exist_should_raise_an_error      assert_raise(ActiveRecord::InverseOfAssociationNotFoundError) { Man.first.secret_interests }    end diff --git a/activerecord/test/cases/associations/join_model_test.rb b/activerecord/test/cases/associations/join_model_test.rb index d4b7960047..86893ec4b3 100644 --- a/activerecord/test/cases/associations/join_model_test.rb +++ b/activerecord/test/cases/associations/join_model_test.rb @@ -385,7 +385,7 @@ class AssociationsJoinModelTest < ActiveRecord::TestCase    end    def test_has_many_through_polymorphic_has_one -    assert_equal Tagging.find(1,2).sort_by { |t| t.id }, authors(:david).tagging +    assert_equal Tagging.find(1,2).sort_by { |t| t.id }, authors(:david).taggings_2    end    def test_has_many_through_polymorphic_has_many @@ -453,7 +453,7 @@ class AssociationsJoinModelTest < ActiveRecord::TestCase      assert saved_post.tags.include?(new_tag)      assert new_tag.persisted? -    assert new_tag.in?(saved_post.reload.tags(true)) +    assert saved_post.reload.tags(true).include?(new_tag)      new_post = Post.new(:title => "Association replacmenet works!", :body => "You best believe it.") @@ -466,7 +466,7 @@ class AssociationsJoinModelTest < ActiveRecord::TestCase      new_post.save!      assert new_post.persisted? -    assert saved_tag.in?(new_post.reload.tags(true)) +    assert new_post.reload.tags(true).include?(saved_tag)      assert !posts(:thinking).tags.build.persisted?      assert !posts(:thinking).tags.new.persisted? diff --git a/activerecord/test/cases/attribute_methods/read_test.rb b/activerecord/test/cases/attribute_methods/read_test.rb index 55d45d9d93..da5d9d8c2a 100644 --- a/activerecord/test/cases/attribute_methods/read_test.rb +++ b/activerecord/test/cases/attribute_methods/read_test.rb @@ -47,13 +47,13 @@ module ActiveRecord          instance = @klass.new          @klass.column_names.each do |name| -          assert !name.in?(instance.methods.map(&:to_s)) +          assert !instance.methods.map(&:to_s).include?(name)          end          @klass.define_attribute_methods          @klass.column_names.each do |name| -          assert name.in?(instance.methods.map(&:to_s)), "#{name} is not defined" +          assert instance.methods.map(&:to_s).include?(name), "#{name} is not defined"          end        end diff --git a/activerecord/test/cases/attribute_methods_test.rb b/activerecord/test/cases/attribute_methods_test.rb index 807971d678..4bc68acd13 100644 --- a/activerecord/test/cases/attribute_methods_test.rb +++ b/activerecord/test/cases/attribute_methods_test.rb @@ -34,7 +34,6 @@ class AttributeMethodsTest < ActiveRecord::TestCase      assert t.attribute_present?("written_on")      assert !t.attribute_present?("content")      assert !t.attribute_present?("author_name") -        end    def test_attribute_present_with_booleans diff --git a/activerecord/test/cases/base_test.rb b/activerecord/test/cases/base_test.rb index 062f196a12..63981a68a9 100644 --- a/activerecord/test/cases/base_test.rb +++ b/activerecord/test/cases/base_test.rb @@ -231,6 +231,7 @@ class BasicsTest < ActiveRecord::TestCase        assert_equal 11, Topic.find(1).written_on.sec        assert_equal 223300, Topic.find(1).written_on.usec        assert_equal 9900, Topic.find(2).written_on.usec +      assert_equal 129346, Topic.find(3).written_on.usec      end    end @@ -603,6 +604,12 @@ class BasicsTest < ActiveRecord::TestCase      assert_equal "changed", post.body    end +  def test_attr_readonly_is_class_level_setting +    post = ReadonlyTitlePost.new +    assert_raise(NoMethodError) { post._attr_readonly = [:title] } +    assert_deprecated { post._attr_readonly } +  end +    def test_non_valid_identifier_column_name      weird = Weird.create('a$b' => 'value')      weird.reload diff --git a/activerecord/test/cases/counter_cache_test.rb b/activerecord/test/cases/counter_cache_test.rb index cd3d19e783..ee443741ca 100644 --- a/activerecord/test/cases/counter_cache_test.rb +++ b/activerecord/test/cases/counter_cache_test.rb @@ -8,9 +8,11 @@ require 'models/category'  require 'models/categorization'  require 'models/dog'  require 'models/dog_lover' +require 'models/person' +require 'models/friendship'  class CounterCacheTest < ActiveRecord::TestCase -  fixtures :topics, :categories, :categorizations, :cars, :dogs, :dog_lovers +  fixtures :topics, :categories, :categorizations, :cars, :dogs, :dog_lovers, :people, :friendships    class ::SpecialTopic < ::Topic      has_many :special_replies, :foreign_key => 'parent_id' @@ -109,4 +111,11 @@ class CounterCacheTest < ActiveRecord::TestCase        Topic.update_counters([t1.id, t2.id], :replies_count => 2)      end    end + +  test "reset the right counter if two have the same foreign key" do +    michael = people(:michael) +    assert_nothing_raised(ActiveRecord::StatementInvalid) do +      Person.reset_counters(michael.id, :followers) +    end +  end  end diff --git a/activerecord/test/cases/defaults_test.rb b/activerecord/test/cases/defaults_test.rb index 81c5850a47..deaf5252db 100644 --- a/activerecord/test/cases/defaults_test.rb +++ b/activerecord/test/cases/defaults_test.rb @@ -94,7 +94,7 @@ if current_adapter?(:MysqlAdapter) or current_adapter?(:Mysql2Adapter)        assert_equal 0, klass.columns_hash['zero'].default        assert !klass.columns_hash['zero'].null        # 0 in MySQL 4, nil in 5. -      assert klass.columns_hash['omit'].default.in?([0, nil]) +      assert [0, nil].include?(klass.columns_hash['omit'].default)        assert !klass.columns_hash['omit'].null        assert_raise(ActiveRecord::StatementInvalid) { klass.create! } diff --git a/activerecord/test/cases/deprecated_dynamic_methods_test.rb b/activerecord/test/cases/deprecated_dynamic_methods_test.rb index fe307bc49b..392f5f4cd5 100644 --- a/activerecord/test/cases/deprecated_dynamic_methods_test.rb +++ b/activerecord/test/cases/deprecated_dynamic_methods_test.rb @@ -1,4 +1,4 @@ -# This file should be deleted when active_record_deprecated_finders is removed as +# This file should be deleted when activerecord-deprecated_finders is removed as  # a dependency.  #  # It is kept for now as there is some fairly nuanced behaviour in the dynamic diff --git a/activerecord/test/cases/dirty_test.rb b/activerecord/test/cases/dirty_test.rb index 248f4efe3e..92677b9926 100644 --- a/activerecord/test/cases/dirty_test.rb +++ b/activerecord/test/cases/dirty_test.rb @@ -525,6 +525,21 @@ class DirtyTest < ActiveRecord::TestCase      end    end +  def test_setting_time_attributes_with_time_zone_field_to_same_time_should_not_be_marked_as_a_change +    in_time_zone 'Paris' do +      target = Class.new(ActiveRecord::Base) +      target.table_name = 'pirates' + +      created_on = Time.now + +      pirate = target.create(:created_on => created_on) +      pirate.reload # Here mysql truncate the usec value to 0 +       +      pirate.created_on = created_on +      assert !pirate.created_on_changed? +    end +  end +    private      def with_partial_updates(klass, on = true)        old = klass.partial_updates? diff --git a/activerecord/test/cases/helper.rb b/activerecord/test/cases/helper.rb index 018064233a..4c6d4666ed 100644 --- a/activerecord/test/cases/helper.rb +++ b/activerecord/test/cases/helper.rb @@ -19,9 +19,6 @@ require 'support/connection'  # Show backtraces for deprecated behavior for quicker cleanup.  ActiveSupport::Deprecation.debug = true -# Avoid deprecation warning setting dependent_restrict_raises to false. The default is true -ActiveRecord::Base.dependent_restrict_raises = false -  # Connect to the database  ARTest.connect diff --git a/activerecord/test/cases/locking_test.rb b/activerecord/test/cases/locking_test.rb index afb0bd6fd9..2392516395 100644 --- a/activerecord/test/cases/locking_test.rb +++ b/activerecord/test/cases/locking_test.rb @@ -3,6 +3,7 @@ require "cases/helper"  require 'models/person'  require 'models/job'  require 'models/reader' +require 'models/ship'  require 'models/legacy_thing'  require 'models/reference'  require 'models/string_key_object' @@ -18,8 +19,8 @@ class LockWithCustomColumnWithoutDefault < ActiveRecord::Base    self.locking_column = :custom_lock_version  end -class ReadonlyFirstNamePerson < Person -  attr_readonly :first_name +class ReadonlyNameShip < Ship +  attr_readonly :name  end  class OptimisticLockingTest < ActiveRecord::TestCase @@ -200,15 +201,15 @@ class OptimisticLockingTest < ActiveRecord::TestCase    end    def test_readonly_attributes -    assert_equal Set.new([ 'first_name' ]), ReadonlyFirstNamePerson.readonly_attributes +    assert_equal Set.new([ 'name' ]), ReadonlyNameShip.readonly_attributes -    p = ReadonlyFirstNamePerson.create(:first_name => "unchangeable name") -    p.reload -    assert_equal "unchangeable name", p.first_name +    s = ReadonlyNameShip.create(:name => "unchangeable name") +    s.reload +    assert_equal "unchangeable name", s.name -    p.update_attributes(:first_name => "changed name") -    p.reload -    assert_equal "unchangeable name", p.first_name +    s.update_attributes(:name => "changed name") +    s.reload +    assert_equal "unchangeable name", s.name    end    def test_quote_table_name diff --git a/activerecord/test/cases/mass_assignment_security_test.rb b/activerecord/test/cases/mass_assignment_security_test.rb index 73a01906b9..a36b2c2506 100644 --- a/activerecord/test/cases/mass_assignment_security_test.rb +++ b/activerecord/test/cases/mass_assignment_security_test.rb @@ -313,7 +313,7 @@ class MassAssignmentSecurityTest < ActiveRecord::TestCase  end -# This class should be deleted when we removed active_record_deprecated_finders as a +# This class should be deleted when we remove activerecord-deprecated_finders as a  # dependency.  class MassAssignmentSecurityDeprecatedFindersTest < ActiveRecord::TestCase    include MassAssignmentTestHelpers diff --git a/activerecord/test/cases/migration/column_attributes_test.rb b/activerecord/test/cases/migration/column_attributes_test.rb index 9584d5dd06..b88db384a0 100644 --- a/activerecord/test/cases/migration/column_attributes_test.rb +++ b/activerecord/test/cases/migration/column_attributes_test.rb @@ -7,6 +7,14 @@ module ActiveRecord        self.use_transactional_fixtures = false +      def test_add_column_newline_default +        string = "foo\nbar" +        add_column 'test_models', 'command', :string, :default => string +        TestModel.reset_column_information + +        assert_equal string, TestModel.new.command +      end +        def test_add_remove_single_field_using_string_arguments          refute TestModel.column_methods_hash.key?(:last_name) diff --git a/activerecord/test/cases/multiple_db_test.rb b/activerecord/test/cases/multiple_db_test.rb index 06d6596725..42461e8ecb 100644 --- a/activerecord/test/cases/multiple_db_test.rb +++ b/activerecord/test/cases/multiple_db_test.rb @@ -1,9 +1,7 @@  require "cases/helper"  require 'models/entrant'  require 'models/bird' - -# So we can test whether Course.connection survives a reload. -require_dependency 'models/course' +require 'models/course'  class MultipleDbTest < ActiveRecord::TestCase    self.use_transactional_fixtures = false diff --git a/activerecord/test/cases/relation_test.rb b/activerecord/test/cases/relation_test.rb index 5fb54b1ca1..6399111be6 100644 --- a/activerecord/test/cases/relation_test.rb +++ b/activerecord/test/cases/relation_test.rb @@ -19,6 +19,11 @@ module ActiveRecord        assert !relation.loaded, 'relation is not loaded'      end +    def test_responds_to_model_and_returns_klass +      relation = Relation.new :a, :b +      assert_equal :a, relation.model +    end +      def test_initialize_single_values        relation = Relation.new :a, :b        (Relation::SINGLE_VALUE_METHODS - [:create_with]).each do |method| diff --git a/activerecord/test/cases/relations_test.rb b/activerecord/test/cases/relations_test.rb index 0bd48913e1..684538940a 100644 --- a/activerecord/test/cases/relations_test.rb +++ b/activerecord/test/cases/relations_test.rb @@ -668,6 +668,25 @@ class RelationTest < ActiveRecord::TestCase      assert_equal [developers(:poor_jamis)], dev_with_count.to_a    end +  def test_relation_merging_with_arel_equalities_keeps_last_equality +    devs = Developer.where(Developer.arel_table[:salary].eq(80000)).merge( +      Developer.where(Developer.arel_table[:salary].eq(9000)) +    ) +    assert_equal [developers(:poor_jamis)], devs.to_a +  end + +  def test_relation_merging_with_arel_equalities_keeps_last_equality_with_non_attribute_left_hand +    salary_attr = Developer.arel_table[:salary] +    devs = Developer.where( +      Arel::Nodes::NamedFunction.new('abs', [salary_attr]).eq(80000) +    ).merge( +      Developer.where( +        Arel::Nodes::NamedFunction.new('abs', [salary_attr]).eq(9000) +      ) +    ) +    assert_equal [developers(:poor_jamis)], devs.to_a +  end +    def test_relation_merging_with_eager_load      relations = []      relations << Post.order('comments.id DESC').merge(Post.eager_load(:last_comment)).merge(Post.all) @@ -1122,6 +1141,10 @@ class RelationTest < ActiveRecord::TestCase      assert_equal authors(:david), Author.order('id DESC , name DESC').last    end +  def test_update_all_with_blank_argument +    assert_raises(ArgumentError) { Comment.update_all({}) } +  end +    def test_update_all_with_joins      comments = Comment.joins(:post).where('posts.id' => posts(:welcome).id)      count    = comments.count diff --git a/activerecord/test/cases/serialization_test.rb b/activerecord/test/cases/serialization_test.rb index ce167509c1..10d8ccc711 100644 --- a/activerecord/test/cases/serialization_test.rb +++ b/activerecord/test/cases/serialization_test.rb @@ -53,9 +53,8 @@ class SerializationTest < ActiveRecord::TestCase    end    def test_serialized_attributes_are_class_level_settings -    assert_raise NoMethodError do -      topic = Topic.new -      topic.serialized_attributes = [] -    end +    topic = Topic.new +    assert_raise(NoMethodError) { topic.serialized_attributes = [] } +    assert_deprecated { topic.serialized_attributes }    end  end diff --git a/activerecord/test/cases/store_test.rb b/activerecord/test/cases/store_test.rb index 3e60b62fd5..fb0d116c08 100644 --- a/activerecord/test/cases/store_test.rb +++ b/activerecord/test/cases/store_test.rb @@ -122,9 +122,8 @@ class StoreTest < ActiveRecord::TestCase    end    test "stores_attributes are class level settings" do -    assert_raise NoMethodError do -      @john.stored_attributes = {} -    end +    assert_raise(NoMethodError) { @john.stored_attributes = Hash.new } +    assert_raise(NoMethodError) { @john.stored_attributes }    end  end diff --git a/activerecord/test/cases/timestamp_test.rb b/activerecord/test/cases/timestamp_test.rb index 7444dc5de1..bb034848e1 100644 --- a/activerecord/test/cases/timestamp_test.rb +++ b/activerecord/test/cases/timestamp_test.rb @@ -114,9 +114,12 @@ class TimestampTest < ActiveRecord::TestCase    end    def test_saving_a_record_with_a_belongs_to_that_specifies_touching_a_specific_attribute_the_parent_should_update_that_attribute -    Pet.belongs_to :owner, :touch => :happy_at +    klass = Class.new(ActiveRecord::Base) do +      def self.name; 'Pet'; end +      belongs_to :owner, :touch => :happy_at +    end -    pet   = Pet.first +    pet   = klass.first      owner = pet.owner      previously_owner_happy_at = owner.happy_at @@ -124,14 +127,15 @@ class TimestampTest < ActiveRecord::TestCase      pet.save      assert_not_equal previously_owner_happy_at, pet.owner.happy_at -  ensure -    Pet.belongs_to :owner, :touch => true    end    def test_touching_a_record_with_a_belongs_to_that_uses_a_counter_cache_should_update_the_parent -    Pet.belongs_to :owner, :counter_cache => :use_count, :touch => true +    klass = Class.new(ActiveRecord::Base) do +      def self.name; 'Pet'; end +      belongs_to :owner, :counter_cache => :use_count, :touch => true +    end -    pet = Pet.first +    pet = klass.first      owner = pet.owner      owner.update_columns(happy_at: 3.days.ago)      previously_owner_updated_at = owner.updated_at @@ -140,15 +144,15 @@ class TimestampTest < ActiveRecord::TestCase      pet.save      assert_not_equal previously_owner_updated_at, pet.owner.updated_at -  ensure -    Pet.belongs_to :owner, :touch => true    end    def test_touching_a_record_touches_parent_record_and_grandparent_record -    Toy.belongs_to :pet, :touch => true -    Pet.belongs_to :owner, :touch => true +    klass = Class.new(ActiveRecord::Base) do +      def self.name; 'Toy'; end +      belongs_to :pet, :touch => true +    end -    toy = Toy.first +    toy = klass.first      pet = toy.pet      owner = pet.owner      time = 3.days.ago @@ -158,8 +162,6 @@ class TimestampTest < ActiveRecord::TestCase      owner.reload      assert_not_equal time, owner.updated_at -  ensure -    Toy.belongs_to :pet    end    def test_timestamp_attributes_for_create diff --git a/activerecord/test/cases/transactions_test.rb b/activerecord/test/cases/transactions_test.rb index a9ccd00fac..0d0de455b3 100644 --- a/activerecord/test/cases/transactions_test.rb +++ b/activerecord/test/cases/transactions_test.rb @@ -91,18 +91,14 @@ class TransactionTest < ActiveRecord::TestCase    end    def test_raising_exception_in_callback_rollbacks_in_save -    add_exception_raising_after_save_callback_to_topic - -    begin -      @first.approved = true -      @first.save -      flunk -    rescue => e -      assert_equal "Make the transaction rollback", e.message -      assert !Topic.find(1).approved? -    ensure -      remove_exception_raising_after_save_callback_to_topic +    def @first.after_save_for_transaction +      raise 'Make the transaction rollback'      end + +    @first.approved = true +    e = assert_raises(RuntimeError) { @first.save } +    assert_equal "Make the transaction rollback", e.message +    assert !Topic.find(1).approved?    end    def test_update_attributes_should_rollback_on_failure @@ -125,85 +121,85 @@ class TransactionTest < ActiveRecord::TestCase    end    def test_cancellation_from_before_destroy_rollbacks_in_destroy -    add_cancelling_before_destroy_with_db_side_effect_to_topic -    begin -      nbooks_before_destroy = Book.count -      status = @first.destroy -      assert !status -      assert_nothing_raised(ActiveRecord::RecordNotFound) { @first.reload } -      assert_equal nbooks_before_destroy, Book.count -    ensure -      remove_cancelling_before_destroy_with_db_side_effect_to_topic -    end +    add_cancelling_before_destroy_with_db_side_effect_to_topic @first +    nbooks_before_destroy = Book.count +    status = @first.destroy +    assert !status +    @first.reload +    assert_equal nbooks_before_destroy, Book.count    end -  def test_cancellation_from_before_filters_rollbacks_in_save -    %w(validation save).each do |filter| -      send("add_cancelling_before_#{filter}_with_db_side_effect_to_topic") -      begin -        nbooks_before_save = Book.count -        original_author_name = @first.author_name -        @first.author_name += '_this_should_not_end_up_in_the_db' -        status = @first.save -        assert !status -        assert_equal original_author_name, @first.reload.author_name -        assert_equal nbooks_before_save, Book.count -      ensure -        send("remove_cancelling_before_#{filter}_with_db_side_effect_to_topic") -      end +  %w(validation save).each do |filter| +    define_method("test_cancellation_from_before_filters_rollbacks_in_#{filter}") do +      send("add_cancelling_before_#{filter}_with_db_side_effect_to_topic", @first) +      nbooks_before_save = Book.count +      original_author_name = @first.author_name +      @first.author_name += '_this_should_not_end_up_in_the_db' +      status = @first.save +      assert !status +      assert_equal original_author_name, @first.reload.author_name +      assert_equal nbooks_before_save, Book.count      end -  end -  def test_cancellation_from_before_filters_rollbacks_in_save! -    %w(validation save).each do |filter| -      send("add_cancelling_before_#{filter}_with_db_side_effect_to_topic") +    define_method("test_cancellation_from_before_filters_rollbacks_in_#{filter}!") do +      send("add_cancelling_before_#{filter}_with_db_side_effect_to_topic", @first) +      nbooks_before_save = Book.count +      original_author_name = @first.author_name +      @first.author_name += '_this_should_not_end_up_in_the_db' +        begin -        nbooks_before_save = Book.count -        original_author_name = @first.author_name -        @first.author_name += '_this_should_not_end_up_in_the_db'          @first.save! -        flunk -      rescue -        assert_equal original_author_name, @first.reload.author_name -        assert_equal nbooks_before_save, Book.count -      ensure -        send("remove_cancelling_before_#{filter}_with_db_side_effect_to_topic") +      rescue ActiveRecord::RecordInvalid, ActiveRecord::RecordNotSaved        end + +      assert_equal original_author_name, @first.reload.author_name +      assert_equal nbooks_before_save, Book.count      end    end    def test_callback_rollback_in_create -    new_topic = Topic.new( -      :title => "A new topic", -      :author_name => "Ben", -      :author_email_address => "ben@example.com", -      :written_on => "2003-07-16t15:28:11.2233+01:00", -      :last_read => "2004-04-15", -      :bonus_time => "2005-01-30t15:28:00.00+01:00", -      :content => "Have a nice day", -      :approved => false) +    topic = Class.new(Topic) { +      def after_create_for_transaction +        raise 'Make the transaction rollback' +      end +    } + +    new_topic = topic.new(:title                => "A new topic", +                          :author_name          => "Ben", +                          :author_email_address => "ben@example.com", +                          :written_on           => "2003-07-16t15:28:11.2233+01:00", +                          :last_read            => "2004-04-15", +                          :bonus_time           => "2005-01-30t15:28:00.00+01:00", +                          :content              => "Have a nice day", +                          :approved             => false) +      new_record_snapshot = !new_topic.persisted?      id_present = new_topic.has_attribute?(Topic.primary_key)      id_snapshot = new_topic.id      # Make sure the second save gets the after_create callback called.      2.times do -      begin -        add_exception_raising_after_create_callback_to_topic -        new_topic.approved = true -        new_topic.save -        flunk -      rescue => e -        assert_equal "Make the transaction rollback", e.message -        assert_equal new_record_snapshot, !new_topic.persisted?, "The topic should have its old persisted value" -        assert_equal id_snapshot, new_topic.id, "The topic should have its old id" -        assert_equal id_present, new_topic.has_attribute?(Topic.primary_key) -      ensure -        remove_exception_raising_after_create_callback_to_topic -      end +      new_topic.approved = true +      e = assert_raises(RuntimeError) { new_topic.save } +      assert_equal "Make the transaction rollback", e.message +      assert_equal new_record_snapshot, !new_topic.persisted?, "The topic should have its old persisted value" +      assert_equal id_snapshot, new_topic.id, "The topic should have its old id" +      assert_equal id_present, new_topic.has_attribute?(Topic.primary_key)      end    end +  def test_callback_rollback_in_create_with_record_invalid_exception +    topic = Class.new(Topic) { +      def after_create_for_transaction +        raise ActiveRecord::RecordInvalid.new(Author.new) +      end +    } + +    new_topic = topic.create(:title => "A new topic") +    assert !new_topic.persisted?, "The topic should not be persisted" +    assert_nil new_topic.id, "The topic should not have an ID" +  end +    def test_nested_explicit_transactions      Topic.transaction do        Topic.transaction do @@ -461,62 +457,16 @@ class TransactionTest < ActiveRecord::TestCase    end    private -    def define_callback_method(callback_method) -      define_method(callback_method) do -        self.history << [callback_method, :method] -      end -    end -    def add_exception_raising_after_save_callback_to_topic -      Topic.class_eval <<-eoruby, __FILE__, __LINE__ + 1 -        remove_method(:after_save_for_transaction) -        def after_save_for_transaction -          raise 'Make the transaction rollback' -        end -      eoruby -    end - -    def remove_exception_raising_after_save_callback_to_topic -      Topic.class_eval <<-eoruby, __FILE__, __LINE__ + 1 -        remove_method :after_save_for_transaction -        def after_save_for_transaction; end -      eoruby -    end - -    def add_exception_raising_after_create_callback_to_topic -      Topic.class_eval <<-eoruby, __FILE__, __LINE__ + 1 -        remove_method(:after_create_for_transaction) -        def after_create_for_transaction -          raise 'Make the transaction rollback' -        end -      eoruby -    end - -    def remove_exception_raising_after_create_callback_to_topic -      Topic.class_eval <<-eoruby, __FILE__, __LINE__ + 1 -        remove_method :after_create_for_transaction -        def after_create_for_transaction; end -      eoruby -    end - -    %w(validation save destroy).each do |filter| -      define_method("add_cancelling_before_#{filter}_with_db_side_effect_to_topic") do -        Topic.class_eval <<-eoruby, __FILE__, __LINE__ + 1 -          remove_method :before_#{filter}_for_transaction -          def before_#{filter}_for_transaction -            Book.create -            false -          end -        eoruby -      end - -      define_method("remove_cancelling_before_#{filter}_with_db_side_effect_to_topic") do -        Topic.class_eval <<-eoruby, __FILE__, __LINE__ + 1 -          remove_method :before_#{filter}_for_transaction -          def before_#{filter}_for_transaction; end -        eoruby +  %w(validation save destroy).each do |filter| +    define_method("add_cancelling_before_#{filter}_with_db_side_effect_to_topic") do |topic| +      meta = class << topic; self; end +      meta.send("define_method", "before_#{filter}_for_transaction") do +        Book.create +        false        end      end +  end  end  class TransactionsWithTransactionalFixturesTest < ActiveRecord::TestCase diff --git a/activerecord/test/fixtures/friendships.yml b/activerecord/test/fixtures/friendships.yml new file mode 100644 index 0000000000..1ee09175bf --- /dev/null +++ b/activerecord/test/fixtures/friendships.yml @@ -0,0 +1,4 @@ +Connection 1: +  id: 1 +  person_id: 1 +  friend_id: 2
\ No newline at end of file diff --git a/activerecord/test/fixtures/people.yml b/activerecord/test/fixtures/people.yml index 123673a2af..e640a38f1f 100644 --- a/activerecord/test/fixtures/people.yml +++ b/activerecord/test/fixtures/people.yml @@ -4,15 +4,18 @@ michael:    primary_contact_id: 2    number1_fan_id: 3    gender: M +  followers_count: 1  david:    id: 2    first_name: David    primary_contact_id: 3    number1_fan_id: 1    gender: M +  followers_count: 1  susan:    id: 3    first_name: Susan    primary_contact_id: 2    number1_fan_id: 1    gender: F +  followers_count: 1 diff --git a/activerecord/test/fixtures/topics.yml b/activerecord/test/fixtures/topics.yml index 93f48aedc4..2b042bd135 100644 --- a/activerecord/test/fixtures/topics.yml +++ b/activerecord/test/fixtures/topics.yml @@ -25,7 +25,7 @@ third:    id: 3    title: The Third Topic of the day    author_name: Carl -  written_on: 2005-07-15t15:28:00.0099+01:00 +  written_on: 2012-08-12t20:24:22.129346+00:00    content: I'm a troll    approved: true    replies_count: 1 diff --git a/activerecord/test/models/author.rb b/activerecord/test/models/author.rb index 3157d8fe7f..77f4a2ec87 100644 --- a/activerecord/test/models/author.rb +++ b/activerecord/test/models/author.rb @@ -93,8 +93,8 @@ class Author < ActiveRecord::Base    has_many :author_favorites    has_many :favorite_authors, -> { order('name') }, :through => :author_favorites -  has_many :tagging,         :through => :posts    has_many :taggings,        :through => :posts +  has_many :taggings_2,      :through => :posts, :source => :tagging    has_many :tags,            :through => :posts    has_many :post_categories, :through => :posts, :source => :categories    has_many :tagging_tags,    :through => :taggings, :source => :tag diff --git a/activerecord/test/models/company.rb b/activerecord/test/models/company.rb index 5bfbb5e855..75f38d275c 100644 --- a/activerecord/test/models/company.rb +++ b/activerecord/test/models/company.rb @@ -115,8 +115,20 @@ class DependentFirm < Company  end  class RestrictedFirm < Company -  has_one :account, -> { order("id") }, :foreign_key => "firm_id", :dependent => :restrict -  has_many :companies, -> { order("id") }, :foreign_key => 'client_of', :dependent => :restrict +  ActiveSupport::Deprecation.silence do +    has_one :account, -> { order("id") }, :foreign_key => "firm_id", :dependent => :restrict +    has_many :companies, -> { order("id") }, :foreign_key => 'client_of', :dependent => :restrict +  end +end + +class RestrictedWithExceptionFirm < Company +  has_one :account, -> { order("id") }, :foreign_key => "firm_id", :dependent => :restrict_with_exception +  has_many :companies, -> { order("id") }, :foreign_key => 'client_of', :dependent => :restrict_with_exception +end + +class RestrictedWithErrorFirm < Company +  has_one :account, -> { order("id") }, :foreign_key => "firm_id", :dependent => :restrict_with_error +  has_many :companies, -> { order("id") }, :foreign_key => 'client_of', :dependent => :restrict_with_error  end  class Client < Company diff --git a/activerecord/test/models/friendship.rb b/activerecord/test/models/friendship.rb new file mode 100644 index 0000000000..6b4f7acc38 --- /dev/null +++ b/activerecord/test/models/friendship.rb @@ -0,0 +1,4 @@ +class Friendship < ActiveRecord::Base +  belongs_to :friend, class_name: 'Person' +  belongs_to :follower, foreign_key: 'friend_id', class_name: 'Person', counter_cache: :followers_count +end diff --git a/activerecord/test/models/member.rb b/activerecord/test/models/member.rb index 359b29fac3..1134b09d8b 100644 --- a/activerecord/test/models/member.rb +++ b/activerecord/test/models/member.rb @@ -24,11 +24,10 @@ class Member < ActiveRecord::Base    has_one :club_category, :through => :club, :source => :category -  has_many :current_memberships -  has_one :club_through_many, :through => :current_memberships, :source => :club -    has_many :current_memberships, -> { where :favourite => true }    has_many :clubs, :through => :current_memberships + +  has_one :club_through_many, :through => :current_memberships, :source => :club  end  class SelfMember < ActiveRecord::Base diff --git a/activerecord/test/models/person.rb b/activerecord/test/models/person.rb index e204508986..6e6ff29f77 100644 --- a/activerecord/test/models/person.rb +++ b/activerecord/test/models/person.rb @@ -8,6 +8,8 @@ class Person < ActiveRecord::Base    has_many :posts_with_no_comments, -> { includes(:comments).where('comments.id is null').references(:comments) },                                      :through => :readers, :source => :post +  has_many :followers, foreign_key: 'friend_id', class_name: 'Friendship' +    has_many :references    has_many :bad_references    has_many :fixed_bad_references, -> { where :favourite => true }, :class_name => 'BadReference' diff --git a/activerecord/test/schema/schema.rb b/activerecord/test/schema/schema.rb index 6c919a2b02..7c45ca27c0 100644 --- a/activerecord/test/schema/schema.rb +++ b/activerecord/test/schema/schema.rb @@ -270,6 +270,11 @@ ActiveRecord::Schema.define do      t.string :name    end +  create_table :friendships, :force => true do |t| +    t.integer :friend_id +    t.integer :person_id +  end +    create_table :goofy_string_id, :force => true, :id => false do |t|      t.string :id, :null => false      t.string :info @@ -476,6 +481,7 @@ ActiveRecord::Schema.define do      t.references :number1_fan      t.integer    :lock_version, :null => false, :default => 0      t.string     :comments +    t.integer    :followers_count, :default => 0      t.references :best_friend      t.references :best_friend_of      t.timestamps diff --git a/activerecord/test/support/connection.rb b/activerecord/test/support/connection.rb index c176316a05..92736e0ca9 100644 --- a/activerecord/test/support/connection.rb +++ b/activerecord/test/support/connection.rb @@ -1,6 +1,6 @@  require 'active_support/logger' -require_dependency 'models/college' -require_dependency 'models/course' +require 'models/college' +require 'models/course'  module ARTest    def self.connection_name | 
