diff --git a/lib/acts_as_taggable_on/tag.rb b/lib/acts_as_taggable_on/tag.rb index 05f2a6773..870ee4cb0 100644 --- a/lib/acts_as_taggable_on/tag.rb +++ b/lib/acts_as_taggable_on/tag.rb @@ -78,7 +78,7 @@ def self.find_or_create_all_with_like_by_name(*list) existing_tag = existing_tags.find { |tag| comparable_name(tag.name) == comparable_tag_name } existing_tag || create(name: tag_name) rescue ActiveRecord::RecordNotUnique - if (tries -= 1).positive? + unless (tries -= 1).negative? ActiveRecord::Base.connection.execute 'ROLLBACK' retry end diff --git a/spec/acts_as_taggable_on/tag_spec.rb b/spec/acts_as_taggable_on/tag_spec.rb index 392e0f929..469e6bf79 100644 --- a/spec/acts_as_taggable_on/tag_spec.rb +++ b/spec/acts_as_taggable_on/tag_spec.rb @@ -167,6 +167,17 @@ it 'should return an empty array if no tags are specified' do expect(ActsAsTaggableOn::Tag.find_or_create_all_with_like_by_name([])).to be_empty end + + context 'retry 3 times on not unique exception' do + it 'performs 3 tries before raising the exception' do + allow(ActsAsTaggableOn::Tag).to receive(:named_any).and_raise(ActiveRecord::RecordNotUnique.new('error')) # trigger error inside block + expect(ActiveRecord::Base.connection).to receive(:execute).with('ROLLBACK').exactly(3).times + + expect { + ActsAsTaggableOn::Tag.find_or_create_all_with_like_by_name('AWESOME', 'awesome') + }.to raise_error ActsAsTaggableOn::DuplicateTagError + end + end end it 'should require a name' do