From 6f93b217fe43b7635077bd5bf065469c56c972c9 Mon Sep 17 00:00:00 2001 From: Martin Meyerhoff Date: Thu, 5 Dec 2024 13:05:31 +0100 Subject: [PATCH 1/4] Add pending integration spec for orders adjustments page None of this works right now, but let's write a spec that details how things should work. --- .../backend/orders/adjustments_spec.rb | 87 +++++++++++++++++++ 1 file changed, 87 insertions(+) create mode 100644 promotions/spec/system/solidus_promotions/backend/orders/adjustments_spec.rb diff --git a/promotions/spec/system/solidus_promotions/backend/orders/adjustments_spec.rb b/promotions/spec/system/solidus_promotions/backend/orders/adjustments_spec.rb new file mode 100644 index 00000000000..806e971d882 --- /dev/null +++ b/promotions/spec/system/solidus_promotions/backend/orders/adjustments_spec.rb @@ -0,0 +1,87 @@ +# frozen_string_literal: true + +require "rails_helper" + +RSpec.describe "Adjustments", :pending, type: :feature do + stub_authorization! + + let!(:ship_address) { create(:address) } + let!(:tax_zone) { create(:global_zone) } # will include the above address + let!(:tax_rate) { create(:tax_rate, name: "Sales Tax", amount: 0.20, zone: tax_zone, tax_categories: [tax_category]) } + + let!(:line_item) { order.line_items[0] } + + let(:tax_category) { create(:tax_category) } + let(:variant) { create(:variant, tax_category:) } + + before(:each) do + order.recalculate + + visit spree.admin_path + click_link "Orders" + uncheck "Only show complete orders" + click_button "Filter Results" + within_row(1) { click_icon :edit } + click_link "Adjustments" + end + + let!(:order) { create(:order, line_items_attributes: [{ price: 10, variant: }]) } + + context "when the order is completed" do + let!(:order) do + create( + :completed_order_with_totals, + line_items_attributes: [{ price: 10, variant: }], + ship_address: + ) + end + + let!(:adjustment) { order.adjustments.create!(order:, label: "Rebate", amount: 10) } + + it "shows adjustments" do + expect(page).to have_content("Adjustments") + end + + context "when the promotion system is configured to allow applying promotions to completed orders" do + before do + expect(SolidusPromotions.config).to receive(:recalculate_complete_orders).and_return(true) + end + + it "shows input field for promotion code" do + expect(page).to have_content("Adjustments") + expect(page).to have_field("coupon_code") + end + end + + context "when the promotion system is configured to not allow applying promotions to completed orders" do + before do + expect(SolidusPromotions.config).to receive(:recalculate_complete_orders).and_return(false) + end + + it "does not show input field for promotion code" do + expect(page).to have_content("Adjustments") + expect(page).not_to have_field("coupon_code") + end + end + end + + it "shows the input field for applying a promotion" do + expect(page).to have_field("coupon_code") + end + + context "creating a manual adjustment" do + let!(:adjustment_reason) { create(:adjustment_reason, name: "Friendly customer") } + before do + click_link "New Adjustment" + end + + it "creates a new adjustment" do + fill_in "adjustment_amount", with: "5" + fill_in "adjustment_label", with: "Test Adjustment" + select "Friendly customer", from: "Reason" + click_button "Continue" + expect(page).to have_content("Adjustment has been successfully created!") + expect(page).to have_content("Test Adjustment") + end + end +end From d7bb89525fb21cf6801666631591f42a4c9a4b3c Mon Sep 17 00:00:00 2001 From: Martin Meyerhoff Date: Thu, 5 Dec 2024 13:11:34 +0100 Subject: [PATCH 2/4] Feat(promotions): Add PromotionHandler::Coupon#can_apply? The promotion class should decide whether it can apply. --- .../solidus_promotions/promotion_handler/coupon.rb | 4 ++++ .../promotion_handler/coupon_spec.rb | 11 +++++++++++ 2 files changed, 15 insertions(+) diff --git a/promotions/app/models/solidus_promotions/promotion_handler/coupon.rb b/promotions/app/models/solidus_promotions/promotion_handler/coupon.rb index 48db345a41e..99fe4918b66 100644 --- a/promotions/app/models/solidus_promotions/promotion_handler/coupon.rb +++ b/promotions/app/models/solidus_promotions/promotion_handler/coupon.rb @@ -26,6 +26,10 @@ def apply self end + def can_apply? + SolidusPromotions::Promotion.order_activatable?(order) + end + def remove if promotion.blank? set_error_code :coupon_code_not_found diff --git a/promotions/spec/models/solidus_promotions/promotion_handler/coupon_spec.rb b/promotions/spec/models/solidus_promotions/promotion_handler/coupon_spec.rb index 0e82d6c8bdc..bf84e0f120a 100644 --- a/promotions/spec/models/solidus_promotions/promotion_handler/coupon_spec.rb +++ b/promotions/spec/models/solidus_promotions/promotion_handler/coupon_spec.rb @@ -452,4 +452,15 @@ def expect_adjustment_creation(adjustable:, promotion:) ) end end + + describe "#can_apply?", :pending do + let(:order) { double("Order").as_null_object } + + subject { described_class.new(order).can_apply? } + + it "forwards to SolidusPromotions::Promotion.order_activatable?" do + expect(SolidusPromotions::Promotion).to receive(:order_activatable?).with(order) + subject + end + end end From a2e034ea9736bbed94512058a17b6eba23b5bf68 Mon Sep 17 00:00:00 2001 From: Martin Meyerhoff Date: Thu, 5 Dec 2024 14:21:57 +0100 Subject: [PATCH 3/4] Feat(Promotions): Implement SolidusPromotions#order_activatable? --- .../models/solidus_promotions/promotion.rb | 10 ++++ .../promotion_handler/coupon_spec.rb | 2 +- .../solidus_promotions/promotion_spec.rb | 52 +++++++++++++++++++ .../backend/orders/adjustments_spec.rb | 12 ++--- 4 files changed, 67 insertions(+), 9 deletions(-) diff --git a/promotions/app/models/solidus_promotions/promotion.rb b/promotions/app/models/solidus_promotions/promotion.rb index 7eacbbf61af..fced9b24d75 100644 --- a/promotions/app/models/solidus_promotions/promotion.rb +++ b/promotions/app/models/solidus_promotions/promotion.rb @@ -2,6 +2,8 @@ module SolidusPromotions class Promotion < Spree::Base + UNACTIVATABLE_ORDER_STATES = ["awaiting_return", "returned", "canceled"] + include Spree::SoftDeletable belongs_to :category, class_name: "SolidusPromotions::PromotionCategory", @@ -59,6 +61,14 @@ def self.ordered_lanes lanes.sort_by(&:last).to_h end + def self.order_activatable?(order) + return false if UNACTIVATABLE_ORDER_STATES.include?(order.state) + return false if order.shipped? + return false if order.complete? && !SolidusPromotions.config.recalculate_complete_orders + + true + end + self.allowed_ransackable_associations = ["codes"] self.allowed_ransackable_attributes = %w[name customer_label path promotion_category_id lane updated_at] self.allowed_ransackable_scopes = %i[active with_discarded] diff --git a/promotions/spec/models/solidus_promotions/promotion_handler/coupon_spec.rb b/promotions/spec/models/solidus_promotions/promotion_handler/coupon_spec.rb index bf84e0f120a..7b6df6178de 100644 --- a/promotions/spec/models/solidus_promotions/promotion_handler/coupon_spec.rb +++ b/promotions/spec/models/solidus_promotions/promotion_handler/coupon_spec.rb @@ -453,7 +453,7 @@ def expect_adjustment_creation(adjustable:, promotion:) end end - describe "#can_apply?", :pending do + describe "#can_apply?" do let(:order) { double("Order").as_null_object } subject { described_class.new(order).can_apply? } diff --git a/promotions/spec/models/solidus_promotions/promotion_spec.rb b/promotions/spec/models/solidus_promotions/promotion_spec.rb index e2fea9fad2a..6d941eabf6b 100644 --- a/promotions/spec/models/solidus_promotions/promotion_spec.rb +++ b/promotions/spec/models/solidus_promotions/promotion_spec.rb @@ -686,4 +686,56 @@ expect(subject).to be_nil end end + + describe ".order_activatable" do + let(:order) { create :order } + + subject { described_class.order_activatable?(order) } + + it "is true" do + expect(subject).to be true + end + + context "when the order is in the cart state" do + let(:order) { create :order, state: "cart" } + + it { is_expected.to be true } + end + + context "when the order is shipped" do + let(:order) { create :order, state: "complete", shipment_state: "shipped" } + + it { is_expected.to be false } + end + + context "when the order is completed but not shipped" do + let(:order) { create :order, state: "complete", shipment_state: "ready" } + + it { is_expected.to be true } + + context "when the promotion system is configured to prohibit applying promotions to completed orders" do + before { stub_spree_preferences(SolidusPromotions.configuration, recalculate_complete_orders: false) } + + it { is_expected.to be false } + end + end + + context "when the order is canceled" do + let(:order) { create :order, state: "canceled" } + + it { is_expected.to be false } + end + + context "when the order is awaiting return" do + let(:order) { create :order, state: "awaiting_return" } + + it { is_expected.to be false } + end + + context "when the order is returned" do + let(:order) { create :order, state: "returned" } + + it { is_expected.to be false } + end + end end diff --git a/promotions/spec/system/solidus_promotions/backend/orders/adjustments_spec.rb b/promotions/spec/system/solidus_promotions/backend/orders/adjustments_spec.rb index 806e971d882..d861d498fa2 100644 --- a/promotions/spec/system/solidus_promotions/backend/orders/adjustments_spec.rb +++ b/promotions/spec/system/solidus_promotions/backend/orders/adjustments_spec.rb @@ -2,7 +2,7 @@ require "rails_helper" -RSpec.describe "Adjustments", :pending, type: :feature do +RSpec.describe "Adjustments", type: :feature do stub_authorization! let!(:ship_address) { create(:address) } @@ -13,8 +13,10 @@ let(:tax_category) { create(:tax_category) } let(:variant) { create(:variant, tax_category:) } + let(:preferences) { {} } before(:each) do + stub_spree_preferences(SolidusPromotions.configuration, preferences) order.recalculate visit spree.admin_path @@ -43,10 +45,6 @@ end context "when the promotion system is configured to allow applying promotions to completed orders" do - before do - expect(SolidusPromotions.config).to receive(:recalculate_complete_orders).and_return(true) - end - it "shows input field for promotion code" do expect(page).to have_content("Adjustments") expect(page).to have_field("coupon_code") @@ -54,9 +52,7 @@ end context "when the promotion system is configured to not allow applying promotions to completed orders" do - before do - expect(SolidusPromotions.config).to receive(:recalculate_complete_orders).and_return(false) - end + let(:preferences) { { recalculate_complete_orders: false } } it "does not show input field for promotion code" do expect(page).to have_content("Adjustments") From 3588c152bc45c604c5fb5d15e66686f29d572610 Mon Sep 17 00:00:00 2001 From: Martin Meyerhoff Date: Thu, 5 Dec 2024 15:43:36 +0100 Subject: [PATCH 4/4] Feat(Promotions): Use order_activatable in OrderAdjuster This duplicated the logic from SolidusPromotions::Promotion.order_activatable, and this way it reads nicer. --- promotions/app/models/solidus_promotions/order_adjuster.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/promotions/app/models/solidus_promotions/order_adjuster.rb b/promotions/app/models/solidus_promotions/order_adjuster.rb index 8769670be57..ea16b895166 100644 --- a/promotions/app/models/solidus_promotions/order_adjuster.rb +++ b/promotions/app/models/solidus_promotions/order_adjuster.rb @@ -13,7 +13,7 @@ def initialize(order, dry_run_promotion: nil) def call order.reset_current_discounts - return order if (!SolidusPromotions.config.recalculate_complete_orders && order.complete?) || order.shipped? + return order unless SolidusPromotions::Promotion.order_activatable?(order) discounted_order = DiscountOrder.new(order, promotions, dry_run: dry_run).call