Commit 95aad49c authored by Dallas Reedy's avatar Dallas Reedy

Validate that there is exactly one subject present

- Update the database constraint
- Adjust the existing customer validator on ExperimentSubject :base
- Add private non_nil_subjects helper method to model
parent 35cfaf5e
...@@ -10,15 +10,19 @@ class ExperimentSubject < ApplicationRecord ...@@ -10,15 +10,19 @@ class ExperimentSubject < ApplicationRecord
validates :experiment, presence: true validates :experiment, presence: true
validates :variant, presence: true validates :variant, presence: true
validate :must_have_at_least_one_subject validate :must_have_one_subject_present
enum variant: { GROUP_CONTROL => 0, GROUP_EXPERIMENTAL => 1 } enum variant: { GROUP_CONTROL => 0, GROUP_EXPERIMENTAL => 1 }
private private
def must_have_at_least_one_subject def must_have_one_subject_present
if [user, group, project].all?(&:blank?) if non_nil_subjects.length != 1
errors.add(:base, s_("ExperimentSubject|Must have at least one of User, Group, or Project.")) errors.add(:base, s_("ExperimentSubject|Must have exactly one of User, Group, or Project."))
end end
end end
def non_nil_subjects
@non_nil_subjects ||= [user, group, project].reject(&:blank?)
end
end end
...@@ -13,9 +13,9 @@ class CreateExperimentSubjects < ActiveRecord::Migration[6.0] ...@@ -13,9 +13,9 @@ class CreateExperimentSubjects < ActiveRecord::Migration[6.0]
t.timestamps_with_timezone null: false t.timestamps_with_timezone null: false
end end
# Require at least one of user_id, group_id, or project_id to be NOT NULL # Require exactly one of user_id, group_id, or project_id to be NOT NULL
execute <<-SQL execute <<-SQL
ALTER TABLE experiment_subjects ADD CONSTRAINT chk_at_least_one_subject CHECK (NOT ROW(user_id, group_id, project_id) IS NULL); ALTER TABLE experiment_subjects ADD CONSTRAINT chk_has_one_subject CHECK (num_nonnulls(user_id, group_id, project_id) = 1);
SQL SQL
end end
......
...@@ -12147,7 +12147,7 @@ CREATE TABLE experiment_subjects ( ...@@ -12147,7 +12147,7 @@ CREATE TABLE experiment_subjects (
variant smallint DEFAULT 0 NOT NULL, variant smallint DEFAULT 0 NOT NULL,
created_at timestamp with time zone NOT NULL, created_at timestamp with time zone NOT NULL,
updated_at timestamp with time zone NOT NULL, updated_at timestamp with time zone NOT NULL,
CONSTRAINT chk_at_least_one_subject CHECK ((NOT (ROW(user_id, group_id, project_id) IS NULL))) CONSTRAINT chk_has_one_subject CHECK ((num_nonnulls(user_id, group_id, project_id) = 1))
); );
CREATE SEQUENCE experiment_subjects_id_seq CREATE SEQUENCE experiment_subjects_id_seq
......
...@@ -11307,7 +11307,7 @@ msgstr "" ...@@ -11307,7 +11307,7 @@ msgstr ""
msgid "Experienced" msgid "Experienced"
msgstr "" msgstr ""
msgid "ExperimentSubject|Must have at least one of User, Group, or Project." msgid "ExperimentSubject|Must have exactly one of User, Group, or Project."
msgstr "" msgstr ""
msgid "Expiration" msgid "Expiration"
......
...@@ -12,36 +12,43 @@ RSpec.describe ExperimentSubject, type: :model do ...@@ -12,36 +12,43 @@ RSpec.describe ExperimentSubject, type: :model do
describe 'validations' do describe 'validations' do
it { is_expected.to validate_presence_of(:experiment) } it { is_expected.to validate_presence_of(:experiment) }
end
describe 'validate must_have_at_least_one_subject' do
let(:experiment_subject) { build(:experiment_subject, user: nil, group: nil, project: nil) }
it 'fails if user, group, & project are blank' do
expect(experiment_subject).not_to be_valid
expect(experiment_subject.errors[:base]).to include("Must have at least one of User, Group, or Project.")
end
it 'passes when user is present' do
experiment_subject.user = build(:user)
expect(experiment_subject).to be_valid
end
it 'passes when group is present' do
experiment_subject.group = build(:group)
expect(experiment_subject).to be_valid
end
it 'passes when project is present' do
experiment_subject.project = build(:project)
expect(experiment_subject).to be_valid
end
it 'passes when multiple subjects are present' do describe 'must_have_one_subject_present' do
experiment_subject.user = build(:user) let(:experiment_subject) { build(:experiment_subject, user: nil, group: nil, project: nil) }
experiment_subject.group = build(:group) let(:error_message) { 'Must have exactly one of User, Group, or Project.' }
experiment_subject.project = build(:project)
expect(experiment_subject).to be_valid it 'fails when no subject is present' do
expect(experiment_subject).not_to be_valid
expect(experiment_subject.errors[:base]).to include(error_message)
end
it 'passes when user subject is present' do
experiment_subject.user = build(:user)
expect(experiment_subject).to be_valid
end
it 'passes when group subject is present' do
experiment_subject.group = build(:group)
expect(experiment_subject).to be_valid
end
it 'passes when project subject is present' do
experiment_subject.project = build(:project)
expect(experiment_subject).to be_valid
end
it 'fails when more than one subject is present', :aggregate_failures do
# two subjects
experiment_subject.user = build(:user)
experiment_subject.group = build(:group)
expect(experiment_subject).not_to be_valid
expect(experiment_subject.errors[:base]).to include(error_message)
# three subjects
experiment_subject.project = build(:project)
expect(experiment_subject).not_to be_valid
expect(experiment_subject.errors[:base]).to include(error_message)
end
end end
end end
end end
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment