-
-
Notifications
You must be signed in to change notification settings - Fork 1.4k
In-memory order updater #5872
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: main
Are you sure you want to change the base?
In-memory order updater #5872
Changes from all commits
8343681
1664546
d455aab
a859efc
db760b0
caf4754
04cf743
53a32ee
849333d
784d3b5
59beeec
d6da25d
f0e0cc3
13d1c15
07f5599
65b5435
fde33aa
e43df3f
d9cad88
bafd816
ebdd0ad
29b60c6
8aac5e6
a8cd034
f500efd
5f11b5d
89c97da
6c75072
ea9e25f
0485e88
070b81c
6e2a452
1a0a387
fbc1f96
79e04c6
e2e6ed2
c736cd8
ff11e8d
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,262 @@ | ||
| # frozen_string_literal: true | ||
|
|
||
| require 'spree/manipulative_query_monitor' | ||
|
|
||
| module Spree | ||
| class InMemoryOrderUpdater | ||
| attr_reader :order | ||
|
|
||
| # logs a warning when a manipulative query is made when the persist flag is set to false | ||
| class_attribute :log_manipulative_queries | ||
| self.log_manipulative_queries = true | ||
|
|
||
| delegate :payments, :line_items, :adjustments, :all_adjustments, :shipments, :quantity, to: :order | ||
|
|
||
| def initialize(order) | ||
| @order = order | ||
| end | ||
|
|
||
| # This is a multi-purpose method for processing logic related to changes in the Order. | ||
| # It is meant to be called from various observers so that the Order is aware of changes | ||
| # that affect totals and other values stored in the Order. | ||
| # | ||
| # This method should never do anything to the Order that results in a save call on the | ||
| # object with callbacks (otherwise you will end up in an infinite recursion as the | ||
| # associations try to save and then in turn try to call +update!+ again.) | ||
| def recalculate(persist: true) | ||
| monitor = | ||
| if log_manipulative_queries | ||
| Spree::ManipulativeQueryMonitor | ||
| else | ||
| proc { |&block| block.call } | ||
| end | ||
|
|
||
| order.transaction do | ||
| monitor.call do | ||
| recalculate_item_count | ||
| assign_shipment_amounts | ||
| end | ||
|
|
||
| if persist | ||
| update_totals(persist:) | ||
| else | ||
| monitor.call do | ||
| update_totals(persist:) | ||
| end | ||
| end | ||
|
|
||
| monitor.call do | ||
| if order.completed? | ||
| recalculate_payment_state | ||
| recalculate_shipment_state | ||
| end | ||
| end | ||
|
|
||
| Spree::Bus.publish(:order_recalculated, order:) | ||
|
|
||
| persist_totals if persist | ||
| end | ||
| end | ||
| alias_method :update, :recalculate | ||
| deprecate update: :recalculate, deprecator: Spree.deprecator | ||
|
|
||
| # Recalculates the state on all of them shipments, then recalculates the | ||
| # +shipment_state+ attribute according to the following logic: | ||
| # | ||
| # shipped when all Shipments are in the "shipped" state | ||
| # partial when at least one Shipment has a state of "shipped" and there is another Shipment with a state other than "shipped" | ||
| # or there are InventoryUnits associated with the order that have a state of "sold" but are not associated with a Shipment. | ||
| # ready when all Shipments are in the "ready" state | ||
| # backorder when there is backordered inventory associated with an order | ||
| # pending when all Shipments are in the "pending" state | ||
| # | ||
| # The +shipment_state+ value helps with reporting, etc. since it provides a quick and easy way to locate Orders needing attention. | ||
| def recalculate_shipment_state | ||
| shipments.each(&:recalculate_state) | ||
|
|
||
| order.shipment_state = determine_shipment_state | ||
| order.shipment_state | ||
| end | ||
| alias_method :update_shipment_state, :recalculate_shipment_state | ||
| deprecate update_shipment_state: :recalculate_shipment_state, deprecator: Spree.deprecator | ||
|
|
||
| # Recalculates the +payment_state+ attribute according to the following logic: | ||
| # | ||
| # paid when +payment_total+ is equal to +total+ | ||
| # balance_due when +payment_total+ is less than +total+ | ||
| # credit_owed when +payment_total+ is greater than +total+ | ||
| # failed when most recent payment is in the failed state | ||
| # void when the order has been canceled and the payment total is 0 | ||
| # | ||
| # The +payment_state+ value helps with reporting, etc. since it provides a quick and easy way to locate Orders needing attention. | ||
| def recalculate_payment_state | ||
| order.payment_state = determine_payment_state | ||
| order.payment_state | ||
| end | ||
| alias_method :update_payment_state, :recalculate_payment_state | ||
| deprecate update_payment_state: :recalculate_payment_state, deprecator: Spree.deprecator | ||
|
Member
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. could also be extracted into PR |
||
|
|
||
| private | ||
|
|
||
| def determine_payment_state | ||
| if payments.present? && payments.valid.empty? && order.outstanding_balance != 0 | ||
| 'failed' | ||
| elsif order.state == 'canceled' && order.payment_total.zero? | ||
| 'void' | ||
| elsif order.outstanding_balance > 0 | ||
| 'balance_due' | ||
| elsif order.outstanding_balance < 0 | ||
| 'credit_owed' | ||
| else | ||
| # outstanding_balance == 0 | ||
| 'paid' | ||
| end | ||
| end | ||
|
|
||
| def determine_shipment_state | ||
| if order.backordered? | ||
| 'backorder' | ||
| else | ||
| # get all the shipment states for this order | ||
| shipment_states = shipments.states | ||
| if shipment_states.size > 1 | ||
| # multiple shiment states means it's most likely partially shipped | ||
| 'partial' | ||
| else | ||
| # will return nil if no shipments are found | ||
| shipment_states.first | ||
| end | ||
| end | ||
| end | ||
|
|
||
| # This will update and select the best promotion adjustment, update tax | ||
| # adjustments, update cancellation adjustments, and then update the total | ||
| # fields (promo_total, included_tax_total, additional_tax_total, and | ||
| # adjustment_total) on the item. | ||
| # @return [void] | ||
| def update_adjustments(persist:) | ||
| # Promotion adjustments must be applied first, then tax adjustments. | ||
| # This fits the criteria for VAT tax as outlined here: | ||
| # http://www.hmrc.gov.uk/vat/managing/charging/discounts-etc.htm#1 | ||
| # It also fits the criteria for sales tax as outlined here: | ||
| # http://www.boe.ca.gov/formspubs/pub113/ | ||
| update_promotions(persist:) | ||
| update_tax_adjustments | ||
| assign_item_totals | ||
| end | ||
|
|
||
| # Updates the following Order total values: | ||
| # | ||
| # +payment_total+ The total value of all finalized Payments (NOTE: non-finalized Payments are excluded) | ||
| # +item_total+ The total value of all LineItems | ||
| # +adjustment_total+ The total value of all adjustments (promotions, credits, etc.) | ||
| # +promo_total+ The total value of all promotion adjustments | ||
| # +total+ The so-called "order total." This is equivalent to +item_total+ plus +adjustment_total+. | ||
| def update_totals(persist:) | ||
| recalculate_payment_total | ||
| recalculate_item_total | ||
| recalculate_shipment_total | ||
|
Member
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. these should also be deprecated because even though they are private we all have evidence that this are overwritten in subclasses. can be extracted into a PR |
||
| update_adjustment_total(persist:) | ||
| end | ||
|
|
||
| def assign_shipment_amounts | ||
| shipments.each(&:assign_amounts) | ||
| end | ||
|
|
||
| def update_adjustment_total(persist:) | ||
| update_adjustments(persist:) | ||
|
|
||
| all_items = line_items + shipments | ||
| # Ignore any adjustments that have been marked for destruction in our | ||
| # calculations. They'll get removed when/if we persist the order. | ||
| valid_adjustments = adjustments.reject(&:marked_for_destruction?) | ||
| order_tax_adjustments = valid_adjustments.select(&:tax?) | ||
|
|
||
| order.adjustment_total = all_items.sum(&:adjustment_total) + valid_adjustments.sum(&:amount) | ||
| order.included_tax_total = all_items.sum(&:included_tax_total) + order_tax_adjustments.select(&:included?).sum(&:amount) | ||
| order.additional_tax_total = all_items.sum(&:additional_tax_total) + order_tax_adjustments.reject(&:included?).sum(&:amount) | ||
|
|
||
| recalculate_order_total | ||
| end | ||
|
|
||
| def update_promotions(persist:) | ||
| Spree::Config.promotions.order_adjuster_class.new(order).call(persist:) | ||
| end | ||
|
|
||
| def update_tax_adjustments | ||
| Spree::Config.tax_adjuster_class.new(order).adjust! | ||
| end | ||
|
|
||
| def update_cancellations | ||
| end | ||
| deprecate :update_cancellations, deprecator: Spree.deprecator | ||
|
|
||
| def assign_item_totals | ||
| [*line_items, *shipments].each do |item| | ||
| Spree::Config.item_total_class.new(item).recalculate! | ||
| end | ||
| end | ||
|
|
||
| def persist_item_totals | ||
| [*line_items, *shipments].each do |item| | ||
| next unless item.changed? | ||
|
|
||
| item.save! unless item.persisted? | ||
|
|
||
| item.update_columns( | ||
| promo_total: item.promo_total, | ||
| included_tax_total: item.included_tax_total, | ||
| additional_tax_total: item.additional_tax_total, | ||
| adjustment_total: item.adjustment_total, | ||
| updated_at: Time.current, | ||
| ) | ||
| end | ||
| end | ||
|
|
||
| def recalculate_payment_total | ||
| order.payment_total = payments.completed.includes(:refunds).sum { |payment| payment.amount - payment.refunds.sum(:amount) } | ||
| end | ||
|
|
||
| def recalculate_shipment_total | ||
| order.shipment_total = shipments.to_a.sum(&:cost) | ||
| recalculate_order_total | ||
| end | ||
|
|
||
| def recalculate_order_total | ||
| order.total = order.item_total + order.shipment_total + order.adjustment_total | ||
| end | ||
|
|
||
| def recalculate_item_count | ||
| order.item_count = line_items.to_a.sum(&:quantity) | ||
| end | ||
|
|
||
| def recalculate_item_total | ||
| order.item_total = line_items.to_a.sum(&:amount) | ||
| recalculate_order_total | ||
| end | ||
|
|
||
| def persist_totals | ||
| shipments.each(&:persist_amounts) | ||
| persist_item_totals | ||
| log_state_change("payment") | ||
| log_state_change("shipment") | ||
| order.save! | ||
| end | ||
|
|
||
| def log_state_change(name) | ||
| state = "#{name}_state" | ||
| previous_state, current_state = order.changes[state] | ||
|
|
||
| if previous_state != current_state | ||
| # Enqueue the job to track this state change | ||
| StateChangeTrackingJob.perform_later( | ||
| order, | ||
| previous_state, | ||
| current_state, | ||
| Time.current, | ||
| name | ||
| ) | ||
| end | ||
| end | ||
| end | ||
| end | ||
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
could be extracted into a PR