DEV: Refactor flag related services a bit
Extracted from https://github.com/discourse/discourse/pull/29129. This patch makes the code more compliant with the upcoming service docs best practices.
This commit is contained in:
parent
f8360f9665
commit
7f607699b8
|
@ -3,6 +3,7 @@
|
||||||
class Flags::CreateFlag
|
class Flags::CreateFlag
|
||||||
include Service::Base
|
include Service::Base
|
||||||
|
|
||||||
|
policy :invalid_access
|
||||||
contract do
|
contract do
|
||||||
attribute :name, :string
|
attribute :name, :string
|
||||||
attribute :description, :string
|
attribute :description, :string
|
||||||
|
@ -16,7 +17,6 @@ class Flags::CreateFlag
|
||||||
validates :description, length: { maximum: Flag::MAX_DESCRIPTION_LENGTH }
|
validates :description, length: { maximum: Flag::MAX_DESCRIPTION_LENGTH }
|
||||||
validates :applies_to, inclusion: { in: -> { Flag.valid_applies_to_types } }, allow_nil: false
|
validates :applies_to, inclusion: { in: -> { Flag.valid_applies_to_types } }, allow_nil: false
|
||||||
end
|
end
|
||||||
policy :invalid_access
|
|
||||||
policy :unique_name
|
policy :unique_name
|
||||||
model :flag, :instantiate_flag
|
model :flag, :instantiate_flag
|
||||||
transaction do
|
transaction do
|
||||||
|
@ -26,25 +26,18 @@ class Flags::CreateFlag
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def unique_name(name:)
|
|
||||||
!Flag.custom.where(name: name).exists?
|
|
||||||
end
|
|
||||||
|
|
||||||
def instantiate_flag(name:, description:, applies_to:, require_message:, enabled:)
|
|
||||||
Flag.new(
|
|
||||||
name: name,
|
|
||||||
description: description,
|
|
||||||
applies_to: applies_to,
|
|
||||||
require_message: require_message,
|
|
||||||
enabled: enabled,
|
|
||||||
notify_type: true,
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
def invalid_access(guardian:)
|
def invalid_access(guardian:)
|
||||||
guardian.can_create_flag?
|
guardian.can_create_flag?
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def unique_name(contract:)
|
||||||
|
!Flag.custom.where(name: contract.name).exists?
|
||||||
|
end
|
||||||
|
|
||||||
|
def instantiate_flag(contract:)
|
||||||
|
Flag.new(contract.attributes.merge(notify_type: true))
|
||||||
|
end
|
||||||
|
|
||||||
def create(flag:)
|
def create(flag:)
|
||||||
flag.save!
|
flag.save!
|
||||||
end
|
end
|
||||||
|
|
|
@ -7,7 +7,6 @@ class Flags::DestroyFlag
|
||||||
policy :not_system
|
policy :not_system
|
||||||
policy :not_used
|
policy :not_used
|
||||||
policy :invalid_access
|
policy :invalid_access
|
||||||
|
|
||||||
transaction do
|
transaction do
|
||||||
step :destroy
|
step :destroy
|
||||||
step :log
|
step :log
|
||||||
|
@ -16,7 +15,7 @@ class Flags::DestroyFlag
|
||||||
private
|
private
|
||||||
|
|
||||||
def fetch_flag(id:)
|
def fetch_flag(id:)
|
||||||
Flag.find(id)
|
Flag.find_by(id: id)
|
||||||
end
|
end
|
||||||
|
|
||||||
def not_system(flag:)
|
def not_system(flag:)
|
||||||
|
@ -38,12 +37,7 @@ class Flags::DestroyFlag
|
||||||
def log(guardian:, flag:)
|
def log(guardian:, flag:)
|
||||||
StaffActionLogger.new(guardian.user).log_custom(
|
StaffActionLogger.new(guardian.user).log_custom(
|
||||||
"delete_flag",
|
"delete_flag",
|
||||||
{
|
flag.slice(:name, :description, :applies_to, :enabled),
|
||||||
name: flag.name,
|
|
||||||
description: flag.description,
|
|
||||||
applies_to: flag.applies_to,
|
|
||||||
enabled: flag.enabled,
|
|
||||||
},
|
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,7 +1,5 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
VALID_DIRECTIONS = %w[up down]
|
|
||||||
|
|
||||||
class Flags::ReorderFlag
|
class Flags::ReorderFlag
|
||||||
include Service::Base
|
include Service::Base
|
||||||
|
|
||||||
|
@ -10,10 +8,11 @@ class Flags::ReorderFlag
|
||||||
attribute :direction, :string
|
attribute :direction, :string
|
||||||
|
|
||||||
validates :flag_id, presence: true
|
validates :flag_id, presence: true
|
||||||
validates :direction, inclusion: { in: VALID_DIRECTIONS }
|
validates :direction, inclusion: { in: %w[up down] }
|
||||||
end
|
end
|
||||||
model :flag
|
model :flag
|
||||||
policy :invalid_access
|
policy :invalid_access
|
||||||
|
model :all_flags
|
||||||
policy :invalid_move
|
policy :invalid_move
|
||||||
transaction do
|
transaction do
|
||||||
step :move
|
step :move
|
||||||
|
@ -22,37 +21,37 @@ class Flags::ReorderFlag
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def fetch_flag(flag_id:)
|
def fetch_flag(contract:)
|
||||||
Flag.find(flag_id)
|
Flag.find_by(id: contract.flag_id)
|
||||||
end
|
end
|
||||||
|
|
||||||
def invalid_access(guardian:, flag:)
|
def invalid_access(guardian:, flag:)
|
||||||
guardian.can_reorder_flag?(flag)
|
guardian.can_reorder_flag?(flag)
|
||||||
end
|
end
|
||||||
|
|
||||||
def all_flags
|
def fetch_all_flags
|
||||||
@all_flags ||= Flag.where.not(name_key: "notify_user").order(:position)
|
Flag.where.not(name_key: "notify_user").order(:position).to_a
|
||||||
end
|
end
|
||||||
|
|
||||||
def invalid_move(flag:, direction:)
|
def invalid_move(flag:, contract:, all_flags:)
|
||||||
return false if all_flags.first == flag && direction == "up"
|
return false if all_flags.first == flag && contract.direction == "up"
|
||||||
return false if all_flags.last == flag && direction == "down"
|
return false if all_flags.last == flag && contract.direction == "down"
|
||||||
true
|
true
|
||||||
end
|
end
|
||||||
|
|
||||||
def move(flag:, direction:)
|
def move(flag:, contract:, all_flags:)
|
||||||
old_position = flag.position
|
old_position = flag.position
|
||||||
index = all_flags.index(flag)
|
index = all_flags.index(flag)
|
||||||
target_flag = all_flags[direction == "up" ? index - 1 : index + 1]
|
target_flag = all_flags[contract.direction == "up" ? index - 1 : index + 1]
|
||||||
|
|
||||||
flag.update!(position: target_flag.position)
|
flag.update!(position: target_flag.position)
|
||||||
target_flag.update!(position: old_position)
|
target_flag.update!(position: old_position)
|
||||||
end
|
end
|
||||||
|
|
||||||
def log(guardian:, flag:, direction:)
|
def log(guardian:, flag:, contract:)
|
||||||
StaffActionLogger.new(guardian.user).log_custom(
|
StaffActionLogger.new(guardian.user).log_custom(
|
||||||
"move_flag",
|
"move_flag",
|
||||||
{ flag: flag.name, direction: direction },
|
{ flag: flag.name, direction: contract.direction },
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -3,13 +3,13 @@
|
||||||
class Flags::ToggleFlag
|
class Flags::ToggleFlag
|
||||||
include Service::Base
|
include Service::Base
|
||||||
|
|
||||||
|
policy :invalid_access
|
||||||
contract do
|
contract do
|
||||||
attribute :flag_id, :integer
|
attribute :flag_id, :integer
|
||||||
|
|
||||||
validates :flag_id, presence: true
|
validates :flag_id, presence: true
|
||||||
end
|
end
|
||||||
model :flag
|
model :flag
|
||||||
policy :invalid_access
|
|
||||||
transaction do
|
transaction do
|
||||||
step :toggle
|
step :toggle
|
||||||
step :log
|
step :log
|
||||||
|
@ -17,14 +17,14 @@ class Flags::ToggleFlag
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def fetch_flag(flag_id:)
|
|
||||||
Flag.find(flag_id)
|
|
||||||
end
|
|
||||||
|
|
||||||
def invalid_access(guardian:)
|
def invalid_access(guardian:)
|
||||||
guardian.can_toggle_flag?
|
guardian.can_toggle_flag?
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def fetch_flag(contract:)
|
||||||
|
Flag.find_by(id: contract.flag_id)
|
||||||
|
end
|
||||||
|
|
||||||
def toggle(flag:)
|
def toggle(flag:)
|
||||||
flag.update!(enabled: !flag.enabled)
|
flag.update!(enabled: !flag.enabled)
|
||||||
end
|
end
|
||||||
|
|
|
@ -4,12 +4,14 @@ class Flags::UpdateFlag
|
||||||
include Service::Base
|
include Service::Base
|
||||||
|
|
||||||
contract do
|
contract do
|
||||||
|
attribute :id, :integer
|
||||||
attribute :name, :string
|
attribute :name, :string
|
||||||
attribute :description, :string
|
attribute :description, :string
|
||||||
attribute :require_message, :boolean
|
attribute :require_message, :boolean
|
||||||
attribute :enabled, :boolean
|
attribute :enabled, :boolean
|
||||||
attribute :applies_to
|
attribute :applies_to
|
||||||
|
|
||||||
|
validates :id, presence: true
|
||||||
validates :name, presence: true
|
validates :name, presence: true
|
||||||
validates :description, presence: true
|
validates :description, presence: true
|
||||||
validates :name, length: { maximum: Flag::MAX_NAME_LENGTH }
|
validates :name, length: { maximum: Flag::MAX_NAME_LENGTH }
|
||||||
|
@ -28,12 +30,8 @@ class Flags::UpdateFlag
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def unique_name(id:, name:)
|
def fetch_flag(contract:)
|
||||||
!Flag.custom.where(name: name).where.not(id: id).exists?
|
Flag.find_by(id: contract.id)
|
||||||
end
|
|
||||||
|
|
||||||
def fetch_flag(id:)
|
|
||||||
Flag.find(id)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def not_system(flag:)
|
def not_system(flag:)
|
||||||
|
@ -48,26 +46,18 @@ class Flags::UpdateFlag
|
||||||
guardian.can_edit_flag?(flag)
|
guardian.can_edit_flag?(flag)
|
||||||
end
|
end
|
||||||
|
|
||||||
def update(flag:, name:, description:, applies_to:, require_message:, enabled:)
|
def unique_name(contract:)
|
||||||
flag.update!(
|
!Flag.custom.where(name: contract.name).where.not(id: contract.id).exists?
|
||||||
name: name,
|
end
|
||||||
description: description,
|
|
||||||
applies_to: applies_to,
|
def update(flag:, contract:)
|
||||||
require_message: require_message,
|
flag.update!(contract.attributes)
|
||||||
enabled: enabled,
|
|
||||||
)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def log(guardian:, flag:)
|
def log(guardian:, flag:)
|
||||||
StaffActionLogger.new(guardian.user).log_custom(
|
StaffActionLogger.new(guardian.user).log_custom(
|
||||||
"update_flag",
|
"update_flag",
|
||||||
{
|
flag.slice(:name, :description, :applies_to, :require_message, :enabled),
|
||||||
name: flag.name,
|
|
||||||
description: flag.description,
|
|
||||||
applies_to: flag.applies_to,
|
|
||||||
require_message: flag.require_message,
|
|
||||||
enabled: flag.enabled,
|
|
||||||
},
|
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,105 +1,78 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
RSpec.describe(Flags::CreateFlag) do
|
RSpec.describe(Flags::CreateFlag) do
|
||||||
subject(:result) do
|
describe described_class::Contract, type: :model do
|
||||||
described_class.call(
|
it { is_expected.to validate_presence_of(:name) }
|
||||||
guardian: current_user.guardian,
|
it { is_expected.to validate_presence_of(:description) }
|
||||||
name: name,
|
it { is_expected.to validate_length_of(:name).is_at_most(Flag::MAX_NAME_LENGTH) }
|
||||||
description: description,
|
it { is_expected.to validate_length_of(:description).is_at_most(Flag::MAX_DESCRIPTION_LENGTH) }
|
||||||
applies_to: applies_to,
|
it { is_expected.to validate_inclusion_of(:applies_to).in_array(Flag.valid_applies_to_types) }
|
||||||
require_message: require_message,
|
|
||||||
enabled: enabled,
|
|
||||||
)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
let(:name) { "custom flag name" }
|
describe ".call" do
|
||||||
let(:description) { "custom flag description" }
|
subject(:result) { described_class.call(**params, **dependencies) }
|
||||||
let(:applies_to) { ["Topic"] }
|
|
||||||
let(:enabled) { true }
|
|
||||||
let(:require_message) { true }
|
|
||||||
|
|
||||||
context "when user is not allowed to perform the action" do
|
|
||||||
fab!(:current_user) { Fabricate(:user) }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_policy(:invalid_access) }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when title is not unique" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
fab!(:current_user) { Fabricate(:admin) }
|
||||||
fab!(:flag) { Fabricate(:flag, name: "custom flag name") }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_policy(:unique_name) }
|
let(:params) { { name:, description:, applies_to:, require_message:, enabled: } }
|
||||||
|
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 }
|
||||||
|
|
||||||
after { Flag.destroy_by(name: "custom flag name") }
|
context "when user is not allowed to perform the action" do
|
||||||
end
|
fab!(:current_user) { Fabricate(:user) }
|
||||||
|
|
||||||
context "when applies to is invalid" do
|
it { is_expected.to fail_a_policy(:invalid_access) }
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:applies_to) { ["User"] }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when title is empty" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:name) { nil }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when title is too long" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:name) { "a" * 201 }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when description is empty" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:description) { nil }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when description is too long" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:description) { "a" * 1001 }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when user is allowed to perform the action" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:applies_to) { ["Topic::Custom"] }
|
|
||||||
|
|
||||||
before do
|
|
||||||
DiscoursePluginRegistry.register_flag_applies_to_type(
|
|
||||||
"Topic::Custom",
|
|
||||||
OpenStruct.new(enabled?: true),
|
|
||||||
)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
after { Flag.destroy_by(name: "custom flag name") }
|
context "when contract is invalid" do
|
||||||
|
let(:name) { nil }
|
||||||
|
|
||||||
it { is_expected.to run_successfully }
|
it { is_expected.to fail_a_contract }
|
||||||
|
|
||||||
it "creates the flag" do
|
|
||||||
result
|
|
||||||
flag = Flag.last
|
|
||||||
expect(flag.name).to eq("custom flag name")
|
|
||||||
expect(flag.description).to eq("custom flag description")
|
|
||||||
expect(flag.applies_to).to eq(["Topic::Custom"])
|
|
||||||
expect(flag.require_message).to be true
|
|
||||||
expect(flag.enabled).to be true
|
|
||||||
end
|
end
|
||||||
|
|
||||||
it "logs the action" do
|
context "when name is not unique" do
|
||||||
expect { result }.to change { UserHistory.count }.by(1)
|
let!(:flag) { Fabricate(:flag, name:) }
|
||||||
expect(UserHistory.last).to have_attributes(
|
|
||||||
custom_type: "create_flag",
|
it { is_expected.to fail_a_policy(:unique_name) }
|
||||||
details:
|
end
|
||||||
"name: custom flag name\ndescription: custom flag description\napplies_to: [\"Topic::Custom\"]\nrequire_message: true\nenabled: true",
|
|
||||||
)
|
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
|
||||||
|
|
||||||
|
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,
|
||||||
|
)
|
||||||
|
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
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,11 +1,32 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
RSpec.describe(Flags::DestroyFlag) do
|
RSpec.describe(Flags::DestroyFlag) do
|
||||||
subject(:result) { described_class.call(id: flag.id, guardian: current_user.guardian) }
|
subject(:result) { described_class.call(**params, **dependencies) }
|
||||||
|
|
||||||
fab!(:flag)
|
fab!(:flag)
|
||||||
|
fab!(:current_user) { Fabricate(:admin) }
|
||||||
|
|
||||||
after { flag.destroy }
|
let(:params) { { id: flag_id } }
|
||||||
|
let(:dependencies) { { guardian: current_user.guardian } }
|
||||||
|
let(:flag_id) { flag.id }
|
||||||
|
|
||||||
|
context "when model is not found" do
|
||||||
|
let(:flag_id) { 0 }
|
||||||
|
|
||||||
|
it { is_expected.to fail_to_find_a_model(:flag) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when the flag is a system one" do
|
||||||
|
let(:flag) { Flag.first }
|
||||||
|
|
||||||
|
it { is_expected.to fail_a_policy(:not_system) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when the flag has been used" do
|
||||||
|
let!(:post_action) { Fabricate(:post_action, post_action_type_id: flag.id) }
|
||||||
|
|
||||||
|
it { is_expected.to fail_a_policy(:not_used) }
|
||||||
|
end
|
||||||
|
|
||||||
context "when user is not allowed to perform the action" do
|
context "when user is not allowed to perform the action" do
|
||||||
fab!(:current_user) { Fabricate(:user) }
|
fab!(:current_user) { Fabricate(:user) }
|
||||||
|
@ -13,18 +34,11 @@ RSpec.describe(Flags::DestroyFlag) do
|
||||||
it { is_expected.to fail_a_policy(:invalid_access) }
|
it { is_expected.to fail_a_policy(:invalid_access) }
|
||||||
end
|
end
|
||||||
|
|
||||||
context "when user is allowed to perform the action" do
|
context "when everything's ok" do
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
|
|
||||||
it { is_expected.to run_successfully }
|
it { is_expected.to run_successfully }
|
||||||
|
|
||||||
it "sets the service result as successful" do
|
|
||||||
expect(result).to be_a_success
|
|
||||||
end
|
|
||||||
|
|
||||||
it "destroys the flag" do
|
it "destroys the flag" do
|
||||||
result
|
expect { result }.to change { Flag.where(id: flag).count }.by(-1)
|
||||||
expect { flag.reload }.to raise_error(ActiveRecord::RecordNotFound)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
it "logs the action" do
|
it "logs the action" do
|
||||||
|
|
|
@ -1,58 +1,62 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
RSpec.describe(Flags::ReorderFlag) do
|
RSpec.describe(Flags::ReorderFlag) do
|
||||||
subject(:result) do
|
describe described_class::Contract, type: :model do
|
||||||
described_class.call(flag_id: flag.id, guardian: current_user.guardian, direction: direction)
|
it { is_expected.to validate_presence_of(:flag_id) }
|
||||||
|
it { is_expected.to validate_inclusion_of(:direction).in_array(%w[up down]) }
|
||||||
end
|
end
|
||||||
|
|
||||||
let(:flag) { Flag.order(:position).last }
|
describe ".call" do
|
||||||
let(:direction) { "up" }
|
subject(:result) { described_class.call(**params, **dependencies) }
|
||||||
|
|
||||||
context "when user is not allowed to perform the action" do
|
|
||||||
fab!(:current_user) { Fabricate(:user) }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_policy(:invalid_access) }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when direction is invalid" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:direction) { "side" }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when move is invalid" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:direction) { "down" }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_policy(:invalid_move) }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when user is allowed to perform the action" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
fab!(:current_user) { Fabricate(:admin) }
|
||||||
|
|
||||||
after do
|
let(:params) { { flag_id: flag_id, direction: } }
|
||||||
described_class.call(flag_id: flag.id, guardian: current_user.guardian, direction: "down")
|
let(:dependencies) { { guardian: current_user.guardian } }
|
||||||
|
let(:flag_id) { flag.id }
|
||||||
|
let(:flag) { Flag.order(:position).last }
|
||||||
|
let(:direction) { "up" }
|
||||||
|
|
||||||
|
context "when contract is invalid" do
|
||||||
|
let(:direction) { "left" }
|
||||||
|
|
||||||
|
it { is_expected.to fail_a_contract }
|
||||||
end
|
end
|
||||||
|
|
||||||
it { is_expected.to run_successfully }
|
context "when model is not found" do
|
||||||
|
let(:flag_id) { 0 }
|
||||||
|
|
||||||
it "moves the flag" do
|
it { is_expected.to fail_to_find_a_model(:flag) }
|
||||||
expect(Flag.order(:position).map(&:name)).to eq(
|
|
||||||
%w[notify_user off_topic inappropriate spam illegal notify_moderators],
|
|
||||||
)
|
|
||||||
result
|
|
||||||
expect(Flag.order(:position).map(&:name)).to eq(
|
|
||||||
%w[notify_user off_topic inappropriate spam notify_moderators illegal],
|
|
||||||
)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
it "logs the action" do
|
context "when user is not allowed to perform the action" do
|
||||||
expect { result }.to change { UserHistory.count }.by(1)
|
fab!(:current_user) { Fabricate(:user) }
|
||||||
expect(UserHistory.last).to have_attributes(
|
|
||||||
custom_type: "move_flag",
|
it { is_expected.to fail_a_policy(:invalid_access) }
|
||||||
details: "flag: #{result[:flag].name}\ndirection: up",
|
end
|
||||||
)
|
|
||||||
|
context "when move is invalid" do
|
||||||
|
let(:direction) { "down" }
|
||||||
|
|
||||||
|
it { is_expected.to fail_a_policy(:invalid_move) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when everything's ok" do
|
||||||
|
it { is_expected.to run_successfully }
|
||||||
|
|
||||||
|
it "moves the flag" do
|
||||||
|
expect { result }.to change { Flag.order(:position).map(&:name) }.from(
|
||||||
|
%w[notify_user off_topic inappropriate spam illegal notify_moderators],
|
||||||
|
).to(%w[notify_user off_topic inappropriate spam notify_moderators illegal])
|
||||||
|
end
|
||||||
|
|
||||||
|
it "logs the action" do
|
||||||
|
expect { result }.to change { UserHistory.count }.by(1)
|
||||||
|
expect(UserHistory.last).to have_attributes(
|
||||||
|
custom_type: "move_flag",
|
||||||
|
details: "flag: #{result[:flag].name}\ndirection: up",
|
||||||
|
)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,33 +1,52 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
RSpec.describe(Flags::ToggleFlag) do
|
RSpec.describe(Flags::ToggleFlag) do
|
||||||
subject(:result) { described_class.call(flag_id: flag.id, guardian: current_user.guardian) }
|
describe described_class::Contract, type: :model do
|
||||||
|
it { is_expected.to validate_presence_of(:flag_id) }
|
||||||
let(:flag) { Flag.system.last }
|
|
||||||
|
|
||||||
context "when user is not allowed to perform the action" do
|
|
||||||
fab!(:current_user) { Fabricate(:user) }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_policy(:invalid_access) }
|
|
||||||
end
|
end
|
||||||
|
|
||||||
context "when user is allowed to perform the action" do
|
describe ".call" do
|
||||||
|
subject(:result) { described_class.call(**params, **dependencies) }
|
||||||
|
|
||||||
|
fab!(:flag)
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
fab!(:current_user) { Fabricate(:admin) }
|
||||||
|
|
||||||
after { flag.reload.update!(enabled: true) }
|
let(:flag_id) { flag.id }
|
||||||
|
let(:params) { { flag_id: flag_id } }
|
||||||
|
let(:dependencies) { { guardian: current_user.guardian } }
|
||||||
|
|
||||||
it { is_expected.to run_successfully }
|
context "when user is not allowed to perform the action" do
|
||||||
|
fab!(:current_user) { Fabricate(:user) }
|
||||||
|
|
||||||
it "toggles the flag" do
|
it { is_expected.to fail_a_policy(:invalid_access) }
|
||||||
expect(result[:flag].enabled).to be false
|
|
||||||
end
|
end
|
||||||
|
|
||||||
it "logs the action" do
|
context "when contract is invalid" do
|
||||||
expect { result }.to change { UserHistory.count }.by(1)
|
let(:flag_id) { nil }
|
||||||
expect(UserHistory.last).to have_attributes(
|
|
||||||
custom_type: "toggle_flag",
|
it { is_expected.to fail_a_contract }
|
||||||
details: "flag: #{result[:flag].name}\nenabled: #{result[:flag].enabled}",
|
end
|
||||||
)
|
|
||||||
|
context "when model is not found" do
|
||||||
|
let(:flag_id) { 0 }
|
||||||
|
|
||||||
|
it { is_expected.to fail_to_find_a_model(:flag) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when everything's ok" do
|
||||||
|
it { is_expected.to run_successfully }
|
||||||
|
|
||||||
|
it "toggles the flag" do
|
||||||
|
expect(result[:flag].enabled).to be false
|
||||||
|
end
|
||||||
|
|
||||||
|
it "logs the action" do
|
||||||
|
expect { result }.to change { UserHistory.count }.by(1)
|
||||||
|
expect(UserHistory.last).to have_attributes(
|
||||||
|
custom_type: "toggle_flag",
|
||||||
|
details: "flag: #{result[:flag].name}\nenabled: #{result[:flag].enabled}",
|
||||||
|
)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,99 +1,88 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
RSpec.describe(Flags::UpdateFlag) do
|
RSpec.describe(Flags::UpdateFlag) do
|
||||||
subject(:result) do
|
describe described_class::Contract, type: :model do
|
||||||
described_class.call(
|
it { is_expected.to validate_presence_of(:id) }
|
||||||
id: flag.id,
|
it { is_expected.to validate_presence_of(:name) }
|
||||||
guardian: current_user.guardian,
|
it { is_expected.to validate_presence_of(:description) }
|
||||||
name: name,
|
it { is_expected.to validate_length_of(:name).is_at_most(Flag::MAX_NAME_LENGTH) }
|
||||||
description: description,
|
it { is_expected.to validate_length_of(:description).is_at_most(Flag::MAX_DESCRIPTION_LENGTH) }
|
||||||
applies_to: applies_to,
|
it { is_expected.to validate_inclusion_of(:applies_to).in_array(Flag.valid_applies_to_types) }
|
||||||
require_message: require_message,
|
|
||||||
enabled: enabled,
|
|
||||||
)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
fab!(:flag)
|
describe ".call" do
|
||||||
|
subject(:result) { described_class.call(**params, **dependencies) }
|
||||||
|
|
||||||
after { flag.destroy }
|
fab!(:flag)
|
||||||
|
|
||||||
let(:name) { "edited custom flag name" }
|
|
||||||
let(:description) { "edited custom flag description" }
|
|
||||||
let(:applies_to) { ["Topic"] }
|
|
||||||
let(:require_message) { true }
|
|
||||||
let(:enabled) { false }
|
|
||||||
|
|
||||||
context "when user is not allowed to perform the action" do
|
|
||||||
fab!(:current_user) { Fabricate(:user) }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_policy(:invalid_access) }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when applies to is invalid" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:applies_to) { ["User"] }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when title is empty" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:name) { nil }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when title is not unique" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
fab!(:flag_2) { Fabricate(:flag, name: "edited custom flag name") }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_policy(:unique_name) }
|
|
||||||
|
|
||||||
after { Flag.destroy_by(name: "edited custom flag name") }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when title is too long" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:name) { "a" * 201 }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when description is empty" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:description) { nil }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when description is too long" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
|
||||||
let(:description) { "a" * 1001 }
|
|
||||||
|
|
||||||
it { is_expected.to fail_a_contract }
|
|
||||||
end
|
|
||||||
|
|
||||||
context "when user is allowed to perform the action" do
|
|
||||||
fab!(:current_user) { Fabricate(:admin) }
|
fab!(:current_user) { Fabricate(:admin) }
|
||||||
|
|
||||||
it { is_expected.to run_successfully }
|
let(:params) { { id: flag_id, name:, description:, applies_to:, require_message:, enabled: } }
|
||||||
|
let(:dependencies) { { guardian: current_user.guardian } }
|
||||||
|
let(:flag_id) { flag.id }
|
||||||
|
let(:name) { "edited custom flag name" }
|
||||||
|
let(:description) { "edited custom flag description" }
|
||||||
|
let(:applies_to) { ["Topic"] }
|
||||||
|
let(:require_message) { true }
|
||||||
|
let(:enabled) { false }
|
||||||
|
|
||||||
it "updates the flag" do
|
context "when contract is invalid" do
|
||||||
result
|
let(:name) { nil }
|
||||||
expect(flag.reload.name).to eq("edited custom flag name")
|
|
||||||
expect(flag.description).to eq("edited custom flag description")
|
it { is_expected.to fail_a_contract }
|
||||||
expect(flag.applies_to).to eq(["Topic"])
|
|
||||||
expect(flag.require_message).to be true
|
|
||||||
expect(flag.enabled).to be false
|
|
||||||
end
|
end
|
||||||
|
|
||||||
it "logs the action" do
|
context "when model is not found" do
|
||||||
expect { result }.to change { UserHistory.count }.by(1)
|
let(:flag_id) { 0 }
|
||||||
expect(UserHistory.last).to have_attributes(
|
|
||||||
custom_type: "update_flag",
|
it { is_expected.to fail_to_find_a_model(:flag) }
|
||||||
details:
|
end
|
||||||
"name: edited custom flag name\ndescription: edited custom flag description\napplies_to: [\"Topic\"]\nrequire_message: true\nenabled: false",
|
|
||||||
)
|
context "when the flag is a system one" do
|
||||||
|
let(:flag) { Flag.first }
|
||||||
|
|
||||||
|
it { is_expected.to fail_a_policy(:not_system) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when the flag has been used" do
|
||||||
|
let!(:post_action) { Fabricate(:post_action, post_action_type_id: flag.id) }
|
||||||
|
|
||||||
|
it { is_expected.to fail_a_policy(:not_used) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when user is not allowed to perform the action" do
|
||||||
|
fab!(:current_user) { Fabricate(:user) }
|
||||||
|
|
||||||
|
it { is_expected.to fail_a_policy(:invalid_access) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when title is not unique" do
|
||||||
|
let!(:flag_2) { Fabricate(:flag, name:) }
|
||||||
|
|
||||||
|
it { is_expected.to fail_a_policy(:unique_name) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when everything's ok" do
|
||||||
|
it { is_expected.to run_successfully }
|
||||||
|
|
||||||
|
it "updates the flag" do
|
||||||
|
result
|
||||||
|
expect(flag.reload).to have_attributes(
|
||||||
|
name: "edited custom flag name",
|
||||||
|
description: "edited custom flag description",
|
||||||
|
applies_to: ["Topic"],
|
||||||
|
require_message: true,
|
||||||
|
enabled: false,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "logs the action" do
|
||||||
|
expect { result }.to change { UserHistory.count }.by(1)
|
||||||
|
expect(UserHistory.last).to have_attributes(
|
||||||
|
custom_type: "update_flag",
|
||||||
|
details:
|
||||||
|
"name: edited custom flag name\ndescription: edited custom flag description\napplies_to: [\"Topic\"]\nrequire_message: true\nenabled: false",
|
||||||
|
)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in New Issue