2024-07-02 18:45:37 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
RSpec.describe(Flags::CreateFlag) do
|
2024-10-17 11:15:35 -04:00
|
|
|
describe described_class::Contract, type: :model do
|
|
|
|
it { is_expected.to validate_presence_of(:name) }
|
|
|
|
it { is_expected.to validate_presence_of(:description) }
|
|
|
|
it { is_expected.to validate_length_of(:name).is_at_most(Flag::MAX_NAME_LENGTH) }
|
|
|
|
it { is_expected.to validate_length_of(:description).is_at_most(Flag::MAX_DESCRIPTION_LENGTH) }
|
|
|
|
it { is_expected.to validate_inclusion_of(:applies_to).in_array(Flag.valid_applies_to_types) }
|
2024-07-02 18:45:37 -04:00
|
|
|
end
|
|
|
|
|
2024-10-17 11:15:35 -04:00
|
|
|
describe ".call" do
|
|
|
|
subject(:result) { described_class.call(**params, **dependencies) }
|
2024-07-02 18:45:37 -04:00
|
|
|
|
|
|
|
fab!(:current_user) { Fabricate(:admin) }
|
|
|
|
|
2024-10-21 19:56:31 -04:00
|
|
|
let(:params) do
|
|
|
|
{ name:, description:, applies_to:, require_message:, enabled:, auto_action_type: }
|
|
|
|
end
|
2024-10-17 11:15:35 -04:00
|
|
|
let(:dependencies) { { guardian: current_user.guardian } }
|
|
|
|
let(:name) { "custom flag name" }
|
|
|
|
let(:description) { "custom flag description" }
|
|
|
|
let(:applies_to) { ["Topic"] }
|
|
|
|
let(:enabled) { true }
|
|
|
|
let(:require_message) { true }
|
2024-10-21 19:56:31 -04:00
|
|
|
let(:auto_action_type) { true }
|
2024-07-02 18:45:37 -04:00
|
|
|
|
2024-10-17 11:15:35 -04:00
|
|
|
context "when user is not allowed to perform the action" do
|
|
|
|
fab!(:current_user) { Fabricate(:user) }
|
2024-07-02 18:45:37 -04:00
|
|
|
|
2024-10-17 11:15:35 -04:00
|
|
|
it { is_expected.to fail_a_policy(:invalid_access) }
|
|
|
|
end
|
2024-07-02 18:45:37 -04:00
|
|
|
|
2024-10-17 11:15:35 -04:00
|
|
|
context "when contract is invalid" do
|
|
|
|
let(:name) { nil }
|
2024-07-02 18:45:37 -04:00
|
|
|
|
2024-10-17 11:15:35 -04:00
|
|
|
it { is_expected.to fail_a_contract }
|
2024-07-22 21:47:50 -04:00
|
|
|
end
|
2024-08-28 08:54:10 -04:00
|
|
|
|
2024-10-17 11:15:35 -04:00
|
|
|
context "when name is not unique" do
|
|
|
|
let!(:flag) { Fabricate(:flag, name:) }
|
2024-07-02 18:45:37 -04:00
|
|
|
|
2024-10-22 04:07:24 -04:00
|
|
|
# DO NOT REMOVE: flags have side effects and their state will leak to
|
|
|
|
# other examples otherwise.
|
2024-10-21 22:18:57 -04:00
|
|
|
after { flag.destroy! }
|
2024-10-22 04:07:24 -04:00
|
|
|
|
|
|
|
it { is_expected.to fail_a_policy(:unique_name) }
|
2024-07-02 18:45:37 -04:00
|
|
|
end
|
|
|
|
|
2024-10-17 11:15:35 -04:00
|
|
|
context "when everything's ok" do
|
|
|
|
let(:applies_to) { ["Topic::Custom"] }
|
|
|
|
let(:flag) { Flag.last }
|
|
|
|
|
|
|
|
before do
|
|
|
|
DiscoursePluginRegistry.register_flag_applies_to_type(
|
|
|
|
"Topic::Custom",
|
|
|
|
OpenStruct.new(enabled?: true),
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2024-10-22 04:07:24 -04:00
|
|
|
# DO NOT REMOVE: flags have side effects and their state will leak to
|
|
|
|
# other examples otherwise.
|
2024-10-21 22:18:57 -04:00
|
|
|
after { flag.destroy! }
|
|
|
|
|
2024-10-17 11:15:35 -04:00
|
|
|
it { is_expected.to run_successfully }
|
|
|
|
|
|
|
|
it "creates the flag" do
|
|
|
|
expect { result }.to change { Flag.count }.by(1)
|
|
|
|
expect(flag).to have_attributes(
|
|
|
|
name: "custom flag name",
|
|
|
|
description: "custom flag description",
|
|
|
|
applies_to: ["Topic::Custom"],
|
|
|
|
require_message: true,
|
|
|
|
enabled: true,
|
|
|
|
notify_type: true,
|
2024-10-21 19:56:31 -04:00
|
|
|
auto_action_type: true,
|
2024-10-17 11:15:35 -04:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "logs the action" do
|
|
|
|
expect { result }.to change { UserHistory.count }.by(1)
|
|
|
|
expect(UserHistory.last).to have_attributes(
|
|
|
|
custom_type: "create_flag",
|
|
|
|
details:
|
|
|
|
"name: custom flag name\ndescription: custom flag description\napplies_to: [\"Topic::Custom\"]\nrequire_message: true\nenabled: true",
|
|
|
|
)
|
|
|
|
end
|
2024-07-02 18:45:37 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|