We were getting `undefined method `lock!' for nil:NilClass` when the question allowed multiple answers.
145 lines
5.2 KiB
Ruby
145 lines
5.2 KiB
Ruby
require "rails_helper"
|
|
|
|
describe Poll::Answer do
|
|
describe "validations" do
|
|
let(:answer) { build(:poll_answer) }
|
|
|
|
it "is valid" do
|
|
expect(answer).to be_valid
|
|
end
|
|
|
|
it "is not valid wihout a question" do
|
|
answer.question = nil
|
|
expect(answer).not_to be_valid
|
|
end
|
|
|
|
it "is not valid without an author" do
|
|
answer.author = nil
|
|
expect(answer).not_to be_valid
|
|
end
|
|
|
|
it "is not valid without an author when multiple answers are allowed" do
|
|
answer.author = nil
|
|
answer.question = create(:poll_question_multiple)
|
|
|
|
expect(answer).not_to be_valid
|
|
end
|
|
|
|
it "is not valid without an answer" do
|
|
answer.answer = nil
|
|
expect(answer).not_to be_valid
|
|
end
|
|
|
|
it "is not valid when user already reached multiple answers question max votes" do
|
|
author = create(:user)
|
|
question = create(:poll_question_multiple, :abc, max_votes: 2)
|
|
create(:poll_answer, author: author, question: question, answer: "Answer A")
|
|
create(:poll_answer, author: author, question: question, answer: "Answer B")
|
|
answer = build(:poll_answer, author: author, question: question, answer: "Answer C")
|
|
|
|
expect(answer).not_to be_valid
|
|
end
|
|
|
|
it "validates max votes when creating answers at the same time", :race_condition do
|
|
author = create(:user, :level_two)
|
|
question = create(:poll_question_multiple, :abc, max_votes: 2)
|
|
create(:poll_answer, question: question, answer: "Answer A", author: author)
|
|
answer = build(:poll_answer, question: question, answer: "Answer B", author: author)
|
|
other_answer = build(:poll_answer, question: question, answer: "Answer C", author: author)
|
|
|
|
[answer, other_answer].map do |a|
|
|
Thread.new { a.save }
|
|
end.each(&:join)
|
|
|
|
expect(Poll::Answer.count).to be 2
|
|
end
|
|
|
|
it "is valid for answers included in the Poll::Question's question_options list" do
|
|
question = create(:poll_question)
|
|
create(:poll_question_option, title: "One", question: question)
|
|
create(:poll_question_option, title: "Two", question: question)
|
|
create(:poll_question_option, title: "Three", question: question)
|
|
|
|
expect(build(:poll_answer, question: question, answer: "One")).to be_valid
|
|
expect(build(:poll_answer, question: question, answer: "Two")).to be_valid
|
|
expect(build(:poll_answer, question: question, answer: "Three")).to be_valid
|
|
|
|
expect(build(:poll_answer, question: question, answer: "Four")).not_to be_valid
|
|
end
|
|
end
|
|
|
|
describe "#save_and_record_voter_participation" do
|
|
let(:author) { create(:user, :level_two) }
|
|
let(:poll) { create(:poll) }
|
|
let(:question) { create(:poll_question, :yes_no, poll: poll) }
|
|
|
|
it "creates a poll_voter with user and poll data" do
|
|
answer = create(:poll_answer, question: question, author: author, answer: "Yes")
|
|
expect(answer.poll.voters).to be_blank
|
|
|
|
answer.save_and_record_voter_participation
|
|
expect(poll.reload.voters.size).to eq(1)
|
|
voter = poll.voters.first
|
|
|
|
expect(voter.document_number).to eq(answer.author.document_number)
|
|
expect(voter.poll_id).to eq(answer.poll.id)
|
|
expect(voter.officer_id).to be nil
|
|
end
|
|
|
|
it "updates a poll_voter with user and poll data" do
|
|
answer = create(:poll_answer, question: question, author: author, answer: "Yes")
|
|
answer.save_and_record_voter_participation
|
|
|
|
expect(poll.reload.voters.size).to eq(1)
|
|
|
|
answer = create(:poll_answer, question: question, author: author, answer: "No")
|
|
answer.save_and_record_voter_participation
|
|
|
|
expect(poll.reload.voters.size).to eq(1)
|
|
|
|
voter = poll.voters.first
|
|
expect(voter.document_number).to eq(answer.author.document_number)
|
|
expect(voter.poll_id).to eq(answer.poll.id)
|
|
end
|
|
|
|
it "does not save the answer if the voter is invalid" do
|
|
allow_any_instance_of(Poll::Voter).to receive(:valid?).and_return(false)
|
|
answer = build(:poll_answer)
|
|
|
|
expect do
|
|
answer.save_and_record_voter_participation
|
|
end.to raise_error(ActiveRecord::RecordInvalid)
|
|
|
|
expect(answer).not_to be_persisted
|
|
end
|
|
end
|
|
|
|
describe "#destroy_and_remove_voter_participation" do
|
|
let(:poll) { create(:poll) }
|
|
let(:question) { create(:poll_question, :yes_no, poll: poll) }
|
|
|
|
it "destroys voter record and answer when it was the only user's answer" do
|
|
answer = build(:poll_answer, question: question)
|
|
answer.save_and_record_voter_participation
|
|
|
|
expect { answer.destroy_and_remove_voter_participation }
|
|
.to change { Poll::Answer.count }.by(-1)
|
|
.and change { Poll::Voter.count }.by(-1)
|
|
end
|
|
|
|
it "destroys the answer but does not destroy the voter record when the user
|
|
has answered other poll questions" do
|
|
answer = build(:poll_answer, question: question)
|
|
answer.save_and_record_voter_participation
|
|
other_question = create(:poll_question, :yes_no, poll: poll)
|
|
other_answer = build(:poll_answer, question: other_question, author: answer.author)
|
|
other_answer.save_and_record_voter_participation
|
|
|
|
expect(other_answer).to be_persisted
|
|
expect { answer.destroy_and_remove_voter_participation }
|
|
.to change { Poll::Answer.count }.by(-1)
|
|
.and change { Poll::Voter.count }.by(0)
|
|
end
|
|
end
|
|
end
|