Files
openfoodnetwork/spec/models/proxy_order_spec.rb
Maikel Linke dcb6f4676d Remove all unnecessary spec_helper require statements
The `.rspec` file is doing this for us.
2026-01-21 12:35:34 +11:00

218 lines
7.5 KiB
Ruby

# frozen_string_literal: true
RSpec.describe ProxyOrder do
describe "cancel" do
let(:order_cycle) { create(:simple_order_cycle) }
let(:subscription) { create(:subscription) }
# We are testing if database columns have been set to "now".
before { freeze_time }
context "when the order cycle is not yet closed" do
let(:proxy_order) {
create(:proxy_order, subscription:, order:, order_cycle:)
}
before { order_cycle.update(orders_open_at: 1.day.ago, orders_close_at: 3.days.from_now) }
context "and an order has not been initialised" do
let(:order) { nil }
it "returns true and sets canceled_at to the current time" do
expect(proxy_order.cancel).to be true
expect_cancelled_now proxy_order
expect(proxy_order.state).to eq 'canceled'
end
end
context "and the order has already been completed" do
let(:order) { create(:completed_order_with_totals) }
it "returns true and sets canceled_at to the current time, and cancels the order" do
expect(Spree::OrderMailer).to receive(:cancel_email) {
double(:email, deliver_later: true)
}
expect(proxy_order.cancel).to be true
expect_cancelled_now proxy_order
expect(order.reload.state).to eq 'canceled'
expect(proxy_order.state).to eq 'canceled'
end
end
context "and the order has not already been completed" do
let(:order) { create(:order) }
it "returns true and sets canceled_at to the current time" do
expect(proxy_order.cancel).to be true
expect_cancelled_now proxy_order
expect(order.reload.state).to eq 'cart'
expect(proxy_order.state).to eq 'canceled'
end
end
end
context "when the order cycle is already closed" do
let(:proxy_order) {
create(:proxy_order, subscription:, order:, order_cycle:)
}
before { order_cycle.update(orders_open_at: 3.days.ago, orders_close_at: 1.minute.ago) }
context "and an order has not been initialised" do
let(:order) { nil }
it "returns false and does nothing" do
expect(proxy_order.cancel).to be false
expect(proxy_order.reload.canceled_at).to be nil
expect(proxy_order.state).to eq 'pending'
end
end
context "and an order has been initialised" do
let(:order) { create(:order) }
it "returns false and does nothing" do
expect(proxy_order.cancel).to be false
expect(proxy_order.reload.canceled_at).to be nil
expect(order.reload.state).to eq 'cart'
expect(proxy_order.state).to eq 'cart'
end
end
end
end
describe "resume" do
let!(:shipment) { create(:shipment) }
let(:order) {
create(:order_with_totals, ship_address: create(:address),
shipments: [shipment],
payments: [create(:payment)],
distributor: shipment.shipping_method.distributors.first)
}
let(:proxy_order) { create(:proxy_order, order:, canceled_at: Time.zone.now) }
let(:order_cycle) { proxy_order.order_cycle }
before { freeze_time }
context "when the order cycle is not yet closed" do
before { order_cycle.update(orders_open_at: 1.day.ago, orders_close_at: 3.days.from_now) }
context "and the order has not been initialised" do
let(:order) { nil }
it "returns true and clears canceled_at" do
expect(proxy_order.resume).to be true
expect(proxy_order.reload.canceled_at).to be nil
expect(proxy_order.state).to eq 'pending'
end
end
context "and the order has already been cancelled" do
before do
allow(Spree::OrderMailer).to receive(:cancel_email) {
double(:email, deliver_later: true)
}
Orders::WorkflowService.new(order).complete!
order.cancel
order.reload
end
it "returns true, clears canceled_at and resumes the order" do
expect(proxy_order.resume).to be true
expect(proxy_order.reload.canceled_at).to be nil
expect(order.reload.state).to eq 'resumed'
expect(proxy_order.state).to eq 'resumed'
end
end
context "and the order has not been cancelled" do
before { Orders::WorkflowService.new(order).complete! }
it "returns true and clears canceled_at" do
expect(proxy_order.resume).to be true
expect(proxy_order.reload.canceled_at).to be nil
expect(order.reload.state).to eq 'complete'
expect(proxy_order.state).to eq 'cart'
end
end
end
context "when the order cycle is already closed" do
before { order_cycle.update(orders_open_at: 3.days.ago, orders_close_at: 1.minute.ago) }
context "and the order has not been initialised" do
let(:order) { nil }
it "returns false and does nothing" do
expect(proxy_order.resume).to eq false
expect_cancelled_now proxy_order
expect(proxy_order.state).to eq 'canceled'
end
end
context "and the order has been cancelled" do
before do
allow(Spree::OrderMailer).to receive(:cancel_email) {
double(:email, deliver_later: true)
}
Orders::WorkflowService.new(order).complete!
order.cancel
end
it "returns false and does nothing" do
expect(proxy_order.resume).to eq false
expect_cancelled_now proxy_order
expect(order.reload.state).to eq 'canceled'
expect(proxy_order.state).to eq 'canceled'
end
end
context "and the order has not been cancelled" do
before { Orders::WorkflowService.new(order).complete! }
it "returns false and does nothing" do
expect(proxy_order.resume).to eq false
expect_cancelled_now proxy_order
expect(order.reload.state).to eq 'complete'
expect(proxy_order.state).to eq 'canceled'
end
end
end
end
describe "initialise_order!" do
let(:order) { create(:order) }
let(:factory) { instance_double(Orders::FactoryService) }
let!(:proxy_order) { create(:proxy_order) }
context "when the order has not already been initialised" do
it "creates a new order using the Orders::FactoryService, and returns it" do
expect(Orders::FactoryService).to receive(:new) { factory }
expect(factory).to receive(:create) { order }
expect(proxy_order.initialise_order!).to eq order
end
end
context "when the order has already been initialised" do
let(:existing_order) { create(:order) }
before do
proxy_order.update(order: existing_order)
end
it "returns the existing order" do
expect(Orders::FactoryService).not_to receive(:new)
expect(proxy_order).not_to receive(:save!)
expect(proxy_order.initialise_order!).to eq existing_order
end
end
end
private
def expect_cancelled_now(subject)
# We still need to use be_within, because the Database timestamp is not as
# accurate as the Rails timestamp. If we use `eq`, we have differing nano
# seconds.
expect(subject.reload.canceled_at).to be_within(2.seconds).of Time.zone.now
end
end