# frozen_string_literal: true require "cases/helper" require "models/post" require "models/author" require "models/project" require "models/developer" require "models/computer" require "models/company" class AssociationCallbacksTest < ActiveRecord::TestCase fixtures :posts, :authors, :author_addresses, :projects, :developers def setup @david = authors(:david) @thinking = posts(:thinking) @authorless = posts(:authorless) assert_empty @david.post_log end def test_adding_macro_callbacks @david.posts_with_callbacks << @thinking assert_equal ["before_adding#{@thinking.id}", "after_adding#{@thinking.id}"], @david.post_log @david.posts_with_callbacks << @thinking assert_equal ["before_adding#{@thinking.id}", "after_adding#{@thinking.id}", "before_adding#{@thinking.id}", "after_adding#{@thinking.id}"], @david.post_log end def test_adding_with_proc_callbacks @david.posts_with_proc_callbacks << @thinking assert_equal ["before_adding#{@thinking.id}", "after_adding#{@thinking.id}"], @david.post_log @david.posts_with_proc_callbacks << @thinking assert_equal ["before_adding#{@thinking.id}", "after_adding#{@thinking.id}", "before_adding#{@thinking.id}", "after_adding#{@thinking.id}"], @david.post_log end def test_removing_with_macro_callbacks first_post, second_post = @david.posts_with_callbacks[0, 2] @david.posts_with_callbacks.delete(first_post) assert_equal ["before_removing#{first_post.id}", "after_removing#{first_post.id}"], @david.post_log @david.posts_with_callbacks.delete(second_post) assert_equal ["before_removing#{first_post.id}", "after_removing#{first_post.id}", "before_removing#{second_post.id}", "after_removing#{second_post.id}"], @david.post_log end def test_removing_with_proc_callbacks first_post, second_post = @david.posts_with_callbacks[0, 2] @david.posts_with_proc_callbacks.delete(first_post) assert_equal ["before_removing#{first_post.id}", "after_removing#{first_post.id}"], @david.post_log @david.posts_with_proc_callbacks.delete(second_post) assert_equal ["before_removing#{first_post.id}", "after_removing#{first_post.id}", "before_removing#{second_post.id}", "after_removing#{second_post.id}"], @david.post_log end def test_multiple_callbacks @david.posts_with_multiple_callbacks << @thinking assert_equal ["before_adding#{@thinking.id}", "before_adding_proc#{@thinking.id}", "after_adding#{@thinking.id}", "after_adding_proc#{@thinking.id}"], @david.post_log @david.posts_with_multiple_callbacks << @thinking assert_equal ["before_adding#{@thinking.id}", "before_adding_proc#{@thinking.id}", "after_adding#{@thinking.id}", "after_adding_proc#{@thinking.id}", "before_adding#{@thinking.id}", "before_adding_proc#{@thinking.id}", "after_adding#{@thinking.id}", "after_adding_proc#{@thinking.id}"], @david.post_log end def test_has_many_callbacks_halt_execution_when_abort_is_trown_when_adding_to_association author = Author.create!(name: "Roger") post = Post.create!(title: "hello", body: "abc") author.posts_with_thrown_callbacks << post assert_empty(author.posts_with_callbacks) end def test_has_many_callbacks_halt_execution_when_abort_is_trown_when_removing_from_association author = Author.create!(name: "Roger") post = Post.create!(title: "hello", body: "abc", author: author) assert_equal(1, author.posts_with_thrown_callbacks.size) author.posts_with_thrown_callbacks.destroy(post.id) assert_equal(1, author.posts_with_thrown_callbacks.size) end def test_has_many_callbacks_with_create morten = Author.create name: "Morten" post = morten.posts_with_proc_callbacks.create! title: "Hello", body: "How are you doing?" assert_equal ["before_adding", "after_adding#{post.id}"], morten.post_log end def test_has_many_callbacks_with_create! morten = Author.create! name: "Morten" post = morten.posts_with_proc_callbacks.create title: "Hello", body: "How are you doing?" assert_equal ["before_adding", "after_adding#{post.id}"], morten.post_log end def test_has_many_callbacks_for_save_on_parent jack = Author.new name: "Jack" jack.posts_with_callbacks.build title: "Call me back!", body: "Before you wake up and after you sleep" callback_log = ["before_adding", "after_adding#{jack.posts_with_callbacks.first.id}"] assert_equal callback_log, jack.post_log assert jack.save assert_equal 1, jack.posts_with_callbacks.count assert_equal callback_log, jack.post_log end def test_has_many_callbacks_for_destroy_on_parent firm = Firm.create! name: "Firm" client = firm.clients.create! name: "Client" firm.destroy assert_equal ["before_remove#{client.id}", "after_remove#{client.id}"], firm.log end def test_has_and_belongs_to_many_add_callback david = developers(:david) ar = projects(:active_record) assert_empty ar.developers_log ar.developers_with_callbacks << david assert_equal ["before_adding#{david.id}", "after_adding#{david.id}"], ar.developers_log ar.developers_with_callbacks << david assert_equal ["before_adding#{david.id}", "after_adding#{david.id}", "before_adding#{david.id}", "after_adding#{david.id}"], ar.developers_log end def test_has_and_belongs_to_many_before_add_called_before_save dev = nil new_dev = nil klass = Class.new(Project) do def self.name; Project.name; end has_and_belongs_to_many :developers_with_callbacks, class_name: "Developer", before_add: lambda { |o, r| dev = r new_dev = r.new_record? } end rec = klass.create! alice = Developer.new(name: "alice") rec.developers_with_callbacks << alice assert_equal alice, dev assert_not_nil new_dev assert new_dev, "record should not have been saved" assert_not_predicate alice, :new_record? end def test_has_and_belongs_to_many_after_add_called_after_save ar = projects(:active_record) assert_empty ar.developers_log alice = Developer.new(name: "alice") ar.developers_with_callbacks << alice assert_equal "after_adding#{alice.id}", ar.developers_log.last bob = ar.developers_with_callbacks.create(name: "bob") assert_equal "after_adding#{bob.id}", ar.developers_log.last ar.developers_with_callbacks.build(name: "charlie") assert_equal "after_adding", ar.developers_log.last end def test_has_and_belongs_to_many_remove_callback david = developers(:david) jamis = developers(:jamis) activerecord = projects(:active_record) assert_empty activerecord.developers_log activerecord.developers_with_callbacks.delete(david) assert_equal ["before_removing#{david.id}", "after_removing#{david.id}"], activerecord.developers_log activerecord.developers_with_callbacks.delete(jamis) assert_equal ["before_removing#{david.id}", "after_removing#{david.id}", "before_removing#{jamis.id}", "after_removing#{jamis.id}"], activerecord.developers_log end def test_has_and_belongs_to_many_does_not_fire_callbacks_on_clear activerecord = projects(:active_record) assert_empty activerecord.developers_log if activerecord.developers_with_callbacks.size == 0 activerecord.developers << developers(:david) activerecord.developers << developers(:jamis) activerecord.reload assert activerecord.developers_with_callbacks.size == 2 end activerecord.developers_with_callbacks.flat_map { |d| ["before_removing#{d.id}", "after_removing#{d.id}"] }.sort assert activerecord.developers_with_callbacks.clear assert_empty activerecord.developers_log end def test_has_and_belongs_to_many_callbacks_for_save_on_parent project = Project.new name: "Callbacks" project.developers_with_callbacks.build name: "Jack", salary: 95000 callback_log = ["before_adding", "after_adding"] assert_equal callback_log, project.developers_log assert project.save assert_equal 1, project.developers_with_callbacks.size assert_equal callback_log, project.developers_log end def test_dont_add_if_before_callback_raises_exception assert_not_includes @david.unchangeable_posts, @authorless begin @david.unchangeable_posts << @authorless rescue Exception end assert_empty @david.post_log assert_not_includes @david.unchangeable_posts, @authorless @david.reload assert_not_includes @david.unchangeable_posts, @authorless end end