# frozen_string_literal: true require 'spree/order/checkout' require 'spree/order/number_generator' module Spree # The customers cart until completed, then acts as permanent record of the transaction. # # `Spree::Order` is the heart of the Solidus system, as it acts as the customer's # cart as they shop. Once an order is complete, it serves as the # permanent record of their purchase. It has many responsibilities: # # * Records and validates attributes like `total` and relationships like # `Spree::LineItem` as an ActiveRecord model. # # * Implements a customizable state machine to manage the lifecycle of an order. # # * Implements business logic to provide a single interface for quesitons like # `checkout_allowed?` or `payment_required?`. # # * Implements an interface for mutating the order with methods like # `empty!` and `fulfill!`. # class Order < Spree::Base ORDER_NUMBER_LENGTH = 9 ORDER_NUMBER_LETTERS = false ORDER_NUMBER_PREFIX = 'R' include Spree::Order::Checkout include Spree::Order::Payments class InsufficientStock < StandardError; end class CannotRebuildShipments < StandardError; end extend Spree::DisplayMoney money_methods :outstanding_balance, :item_total, :adjustment_total, :included_tax_total, :additional_tax_total, :tax_total, :shipment_total, :total, :order_total_after_store_credit, :total_available_store_credit alias :display_ship_total :display_shipment_total checkout_flow do go_to_state :address go_to_state :delivery go_to_state :payment, if: ->(order) { order.payment_required? } go_to_state :confirm end self.whitelisted_ransackable_associations = %w[shipments user order_promotions promotions bill_address ship_address line_items] self.whitelisted_ransackable_attributes = %w[completed_at created_at email number state payment_state shipment_state total store_id] attr_reader :coupon_code attr_accessor :temporary_address attr_accessor :temporary_payment_source alias_method :temporary_credit_card, :temporary_payment_source alias_method :temporary_credit_card=, :temporary_payment_source= deprecate temporary_credit_card: :temporary_payment_source, deprecator: Spree::Deprecation deprecate :temporary_credit_card= => :temporary_payment_source=, deprecator: Spree::Deprecation # Customer info belongs_to :user, class_name: Spree::UserClassHandle.new belongs_to :bill_address, foreign_key: :bill_address_id, class_name: 'Spree::Address' alias_attribute :billing_address, :bill_address belongs_to :ship_address, foreign_key: :ship_address_id, class_name: 'Spree::Address' alias_attribute :shipping_address, :ship_address alias_attribute :ship_total, :shipment_total belongs_to :store, class_name: 'Spree::Store' # Items has_many :line_items, -> { order(:created_at, :id) }, dependent: :destroy, inverse_of: :order has_many :variants, through: :line_items has_many :products, through: :variants # Shipping has_many :shipments, dependent: :destroy, inverse_of: :order do def states pluck(:state).uniq end end has_many :inventory_units, through: :shipments has_many :cartons, -> { distinct }, through: :inventory_units # Adjustments and promotions has_many :adjustments, -> { order(:created_at) }, as: :adjustable, inverse_of: :adjustable, dependent: :destroy has_many :line_item_adjustments, through: :line_items, source: :adjustments has_many :shipment_adjustments, through: :shipments, source: :adjustments has_many :all_adjustments, class_name: 'Spree::Adjustment', foreign_key: :order_id, dependent: :destroy, inverse_of: :order has_many :order_promotions, class_name: 'Spree::OrderPromotion' has_many :promotions, through: :order_promotions # Payments has_many :payments, dependent: :destroy, inverse_of: :order # Returns has_many :return_authorizations, dependent: :destroy, inverse_of: :order has_many :reimbursements, inverse_of: :order has_many :refunds, through: :payments # Logging has_many :state_changes, as: :stateful belongs_to :created_by, class_name: Spree::UserClassHandle.new belongs_to :approver, class_name: Spree::UserClassHandle.new belongs_to :canceler, class_name: Spree::UserClassHandle.new accepts_nested_attributes_for :line_items accepts_nested_attributes_for :bill_address accepts_nested_attributes_for :ship_address accepts_nested_attributes_for :payments accepts_nested_attributes_for :shipments # Needs to happen before save_permalink is called before_validation :associate_store before_validation :set_currency before_validation :generate_order_number, on: :create before_validation :assign_billing_to_shipping_address, if: :use_billing? attr_accessor :use_billing before_create :create_token before_create :link_by_email validates :email, presence: true, if: :require_email validates :email, 'spree/email' => true, allow_blank: true validates :guest_token, presence: { allow_nil: true } validates :number, presence: true, uniqueness: { allow_blank: true } validates :store_id, presence: true def self.find_by_param(value) find_by number: value end def self.find_by_param!(value) find_by! number: value end delegate :update_totals, :persist_totals, to: :updater delegate :firstname, :lastname, to: :bill_address, prefix: true, allow_nil: true alias_method :billing_firstname, :bill_address_firstname alias_method :billing_lastname, :bill_address_lastname class_attribute :update_hooks self.update_hooks = Set.new class_attribute :line_item_comparison_hooks self.line_item_comparison_hooks = Set.new scope :created_between, ->(start_date, end_date) { where(created_at: start_date..end_date) } scope :completed_between, ->(start_date, end_date) { where(completed_at: start_date..end_date) } scope :by_store, ->(store) { where(store_id: store.id) } # shows completed orders first, by their completed_at date, then uncompleted orders by their created_at scope :reverse_chronological, -> { order(Arel.sql('spree_orders.completed_at IS NULL'), completed_at: :desc, created_at: :desc) } def self.by_customer(customer) joins(:user).where("#{Spree.user_class.table_name}.email" => customer) end def self.by_state(state) where(state: state) end def self.complete where.not(completed_at: nil) end def self.incomplete where(completed_at: nil) end def self.canceled where(state: 'canceled') end def self.not_canceled where.not(state: 'canceled') end # Use this method in other gems that wish to register their own custom logic # that should be called after Order#update def self.register_update_hook(hook) update_hooks.add(hook) end # Use this method in other gems that wish to register their own custom logic # that should be called when determining if two line items are equal. def self.register_line_item_comparison_hook(hook) line_item_comparison_hooks.add(hook) end # For compatiblity with Calculator::PriceSack def amount line_items.map(&:amount).sum end # Sum of all line item amounts after promotions, before added tax def discounted_item_amount line_items.to_a.sum(&:discounted_amount) end deprecate discounted_item_amount: :item_total_before_tax, deprecator: Spree::Deprecation def item_total_before_tax line_items.to_a.sum(&:total_before_tax) end # Sum of all line item amounts pre-tax def item_total_excluding_vat line_items.to_a.sum(&:total_excluding_vat) end alias pre_tax_item_amount item_total_excluding_vat deprecate pre_tax_item_amount: :item_total_excluding_vat, deprecator: Spree::Deprecation def currency self[:currency] || Spree::Config[:currency] end def shipping_discount shipment_adjustments.eligible.sum(:amount) * - 1 end def to_param number end def completed? completed_at.present? end # Indicates whether or not the user is allowed to proceed to checkout. # Currently this is implemented as a check for whether or not there is at # least one LineItem in the Order. Feel free to override this logic in your # own application if you require additional steps before allowing a checkout. def checkout_allowed? line_items.count > 0 end # Is this a free order in which case the payment step should be skipped def payment_required? total > 0 end def confirmation_required? true end deprecate :confirmation_required?, deprecator: Spree::Deprecation def backordered? shipments.any?(&:backordered?) end # Returns the address for taxation based on configuration def tax_address if Spree::Config[:tax_using_ship_address] ship_address else bill_address end || store.default_cart_tax_location end def updater @updater ||= Spree::OrderUpdater.new(self) end def recalculate updater.update end def update!(*args) if args.empty? Spree::Deprecation.warn "Calling order.update! with no arguments as a way to invoke the OrderUpdater is deprecated, since it conflicts with AR::Base#update! Please use order.recalculate instead" recalculate else super end end def assign_billing_to_shipping_address self.ship_address = bill_address if bill_address true end def allow_cancel? return false unless completed? && state != 'canceled' shipment_state.nil? || %w{ready backorder pending}.include?(shipment_state) end def all_inventory_units_returned? inventory_units.all?(&:returned?) end def contents @contents ||= Spree::OrderContents.new(self) end def shipping @shipping ||= Spree::OrderShipping.new(self) end def cancellations @cancellations ||= Spree::OrderCancellations.new(self) end # Associates the specified user with the order. def associate_user!(user, override_email = true) self.user = user attrs_to_set = { user_id: user.try(:id) } attrs_to_set[:email] = user.try(:email) if override_email attrs_to_set[:created_by_id] = user.try(:id) if created_by.blank? if persisted? # immediately persist the changes we just made, but don't use save since we might have an invalid address associated self.class.unscoped.where(id: id).update_all(attrs_to_set) end assign_attributes(attrs_to_set) end def generate_order_number(options = nil) if options Spree::Deprecation.warn \ "Passing options to Order#generate_order_number is deprecated. " \ "Please add your own instance of the order number generator " \ "with your options (#{options.inspect}) and store it as " \ "Spree::Config.order_number_generator in your stores config." end self.number ||= Spree::Config.order_number_generator.generate end def shipped_shipments shipments.shipped end def contains?(variant, options = {}) find_line_item_by_variant(variant, options).present? end def quantity_of(variant, options = {}) line_item = find_line_item_by_variant(variant, options) line_item ? line_item.quantity : 0 end def find_line_item_by_variant(variant, options = {}) line_items.detect { |line_item| line_item.variant_id == variant.id && line_item_options_match(line_item, options) } end # This method enables extensions to participate in the # "Are these line items equal" decision. # # When adding to cart, an extension would send something like: # params[:product_customizations]=... # # and would provide: # # def product_customizations_match def line_item_options_match(line_item, options) return true unless options line_item_comparison_hooks.all? { |hook| send(hook, line_item, options) } end # Creates new tax charges if there are any applicable rates. If prices already # include taxes then price adjustments are created instead. # @deprecated This now happens during #update! def create_tax_charge! Spree::Config.tax_adjuster_class.new(self).adjust! end deprecate create_tax_charge!: :update!, deprecator: Spree::Deprecation def reimbursement_total reimbursements.sum(:total) end def outstanding_balance # If reimbursement has happened add it back to total to prevent balance_due payment state # See: https://github.com/spree/spree/issues/6229 if state == 'canceled' -1 * payment_total else total - reimbursement_total - payment_total end end def outstanding_balance? outstanding_balance != 0 end def refund_total payments.flat_map(&:refunds).sum(&:amount) end def name if (address = bill_address || ship_address) "#{address.firstname} #{address.lastname}" end end def can_ship? complete? || resumed? || awaiting_return? || returned? end def credit_cards credit_card_ids = payments.from_credit_card.pluck(:source_id).uniq Spree::CreditCard.where(id: credit_card_ids) end def valid_credit_cards credit_card_ids = payments.from_credit_card.valid.pluck(:source_id).uniq Spree::CreditCard.where(id: credit_card_ids) end # Finalizes an in progress order after checkout is complete. # Called after transition to complete state when payments will have been processed def finalize! # lock all adjustments (coupon promotions, etc.) all_adjustments.each(&:finalize!) # update payment and shipment(s) states, and save updater.update_payment_state shipments.each do |shipment| shipment.update_state shipment.finalize! end updater.update_shipment_state save! updater.run_hooks touch :completed_at deliver_order_confirmation_email unless confirmation_delivered? end def fulfill! shipments.each { |shipment| shipment.update_state if shipment.persisted? } updater.update_shipment_state save! end def deliver_order_confirmation_email Spree::Config.order_mailer_class.confirm_email(self).deliver_later update_column(:confirmation_delivered, true) end # Helper methods for checkout steps def paid? %w(paid credit_owed).include?(payment_state) end def available_payment_methods @available_payment_methods ||= Spree::PaymentMethod .active .available_to_store(store) .available_to_users .order(:position) end def insufficient_stock_lines line_items.select(&:insufficient_stock?) end ## # Check to see if any line item variants are soft, deleted. # If so add error and restart checkout. def ensure_line_item_variants_are_not_deleted if line_items.any? { |li| li.variant.discarded? } errors.add(:base, I18n.t('spree.deleted_variants_present')) restart_checkout_flow false else true end end def merge!(*args) Spree::Config.order_merger_class.new(self).merge!(*args) end def empty! line_items.destroy_all adjustments.destroy_all shipments.destroy_all order_promotions.destroy_all recalculate end alias_method :has_step?, :has_checkout_step? deprecate has_step?: :has_checkout_step?, deprecator: Spree::Deprecation def state_changed(name) state = "#{name}_state" if persisted? old_state = send("#{state}_was") new_state = send(state) unless old_state == new_state state_changes.create( previous_state: old_state, next_state: new_state, name: name, user_id: user_id ) end end end deprecate :state_changed, deprecator: Spree::Deprecation def coupon_code=(code) @coupon_code = begin code.strip.downcase rescue StandardError nil end end def can_add_coupon? Spree::Promotion.order_activatable?(self) end def shipped? %w(partial shipped).include?(shipment_state) end def ensure_shipping_address unless ship_address && ship_address.valid? errors.add(:base, I18n.t('spree.ship_address_required')) && (return false) end end def create_proposed_shipments if completed? raise CannotRebuildShipments.new(I18n.t('spree.cannot_rebuild_shipments_order_completed')) elsif shipments.any? { |s| !s.pending? } raise CannotRebuildShipments.new(I18n.t('spree.cannot_rebuild_shipments_shipments_not_pending')) else shipments.destroy_all self.shipments = Spree::Config.stock.coordinator_class.new(self).shipments end end def apply_shipping_promotions Spree::PromotionHandler::Shipping.new(self).activate recalculate end alias_method :apply_free_shipping_promotions, :apply_shipping_promotions deprecate apply_free_shipping_promotions: :apply_shipping_promotions, deprecator: Spree::Deprecation # Clean shipments and make order back to address state # # At some point the might need to force the order to transition from address # to delivery again so that proper updated shipments are created. # e.g. customer goes back from payment step and changes order items def ensure_updated_shipments if !completed? && shipments.all?(&:pending?) shipments.destroy_all update_column(:shipment_total, 0) restart_checkout_flow end end def restart_checkout_flow return if state == 'cart' update_columns( state: 'cart', updated_at: Time.current ) next! if !line_items.empty? end def refresh_shipment_rates shipments.map(&:refresh_rates) end def shipping_eq_billing_address? bill_address == ship_address end def set_shipments_cost shipments.each(&:update_amounts) recalculate end deprecate set_shipments_cost: :update!, deprecator: Spree::Deprecation def is_risky? payments.risky.count > 0 end def canceled_by(user) transaction do cancel! update_columns( canceler_id: user.id, canceled_at: Time.current ) end end def approved? !!approved_at end def can_approve? !approved? end def quantity line_items.sum(:quantity) end def has_non_reimbursement_related_refunds? refunds.non_reimbursement.exists? || payments.offset_payment.exists? # how old versions of spree stored refunds end def token Spree::Deprecation.warn("Spree::Order#token is DEPRECATED, please use #guest_token instead.", caller) guest_token end def tax_total additional_tax_total + included_tax_total end def add_store_credit_payments return if user.nil? return if payments.store_credits.checkout.empty? && user.available_store_credit_total(currency: currency).zero? payments.store_credits.checkout.each(&:invalidate!) # this can happen when multiple payments are present, auto_capture is # turned off, and one of the payments fails when the user tries to # complete the order, which sends the order back to the 'payment' state. authorized_total = payments.pending.sum(:amount) remaining_total = outstanding_balance - authorized_total matching_store_credits = user.store_credits.where(currency: currency) if matching_store_credits.any? payment_method = Spree::PaymentMethod::StoreCredit.first matching_store_credits.order_by_priority.each do |credit| break if remaining_total.zero? next if credit.amount_remaining.zero? amount_to_take = [credit.amount_remaining, remaining_total].min payments.create!(source: credit, payment_method: payment_method, amount: amount_to_take, state: 'checkout', response_code: credit.generate_authorization_code) remaining_total -= amount_to_take end end other_payments = payments.checkout.not_store_credits if remaining_total.zero? other_payments.each(&:invalidate!) elsif other_payments.size == 1 other_payments.first.update_attributes!(amount: remaining_total) end payments.reset if payments.where(state: %w(checkout pending completed)).sum(:amount) != total errors.add(:base, I18n.t('spree.store_credit.errors.unable_to_fund')) && (return false) end end def covered_by_store_credit? return false unless user user.available_store_credit_total(currency: currency) >= total end alias_method :covered_by_store_credit, :covered_by_store_credit? def total_available_store_credit return 0.0 unless user user.available_store_credit_total(currency: currency) end def order_total_after_store_credit total - total_applicable_store_credit end def total_applicable_store_credit if can_complete? || complete? payments.store_credits.valid.sum(:amount) else [total, (user.try(:available_store_credit_total, currency: currency) || 0.0)].min end end def display_total_applicable_store_credit Spree::Money.new(-total_applicable_store_credit, { currency: currency }) end def display_store_credit_remaining_after_capture Spree::Money.new(total_available_store_credit - total_applicable_store_credit, { currency: currency }) end def bill_address_attributes=(attributes) self.bill_address = Spree::Address.immutable_merge(bill_address, attributes) end def ship_address_attributes=(attributes) self.ship_address = Spree::Address.immutable_merge(ship_address, attributes) end # Assigns a default bill_address and ship_address to the order based on the # associated user's bill_address and ship_address. # @note This doesn't persist the change bill_address or ship_address def assign_default_user_addresses if user bill_address = (user.bill_address || user.default_address) ship_address = (user.ship_address || user.default_address) # this is one of 2 places still using User#bill_address self.bill_address ||= bill_address if bill_address.try!(:valid?) # Skip setting ship address if order doesn't have a delivery checkout step # to avoid triggering validations on shipping address self.ship_address ||= ship_address if ship_address.try!(:valid?) && checkout_steps.include?("delivery") end end alias_method :assign_default_user_addresses!, :assign_default_user_addresses deprecate assign_default_user_addresses!: :assign_default_user_addresses, deprecator: Spree::Deprecation alias_method :assign_default_addresses!, :assign_default_user_addresses deprecate assign_default_addresses!: :assign_default_user_addresses, deprecator: Spree::Deprecation def persist_user_address! if !temporary_address && user && user.respond_to?(:persist_order_address) && bill_address_id user.persist_order_address(self) end end def add_payment_sources_to_wallet Spree::Config. add_payment_sources_to_wallet_class.new(self). add_to_wallet end alias_method :persist_user_credit_card, :add_payment_sources_to_wallet deprecate persist_user_credit_card: :add_payment_sources_to_wallet, deprecator: Spree::Deprecation def add_default_payment_from_wallet builder = Spree::Config.default_payment_builder_class.new(self) if payment = builder.build payments << payment if bill_address.nil? # this is one of 2 places still using User#bill_address self.bill_address = payment.source.try(:address) || user.bill_address end end end alias_method :assign_default_credit_card, :add_default_payment_from_wallet deprecate assign_default_credit_card: :add_default_payment_from_wallet, deprecator: Spree::Deprecation def record_ip_address(ip_address) if last_ip_address != ip_address update_attributes!(last_ip_address: ip_address) end end def payments_attributes=(attributes) validate_payments_attributes(attributes) super(attributes) end def validate_payments_attributes(attributes) attributes = Array(attributes) attributes.each do |payment_attributes| payment_method_id = payment_attributes[:payment_method_id] # raise RecordNotFound unless it is an allowed payment method available_payment_methods.find(payment_method_id) if payment_method_id end end private def process_payments_before_complete return if !payment_required? if payments.valid.empty? errors.add(:base, I18n.t('spree.no_payment_found')) return false end if process_payments! true else saved_errors = errors[:base] payment_failed! saved_errors.each { |error| errors.add(:base, error) } false end end # In case a existing credit card is provided it needs to build the payment # attributes from scratch so we can set the amount. example payload: # # { # "order": { # "existing_card": "2" # } # } # def update_params_payment_source Spree::Deprecation.warn('update_params_payment_source is deprecated. Please use set_payment_parameters_amount instead.', caller) if @updating_params[:order] && (@updating_params[:order][:payments_attributes] || @updating_params[:order][:existing_card]) @updating_params[:order][:payments_attributes] ||= [{}] @updating_params[:order][:payments_attributes].first[:amount] = total end end def associate_store self.store ||= Spree::Store.default end def link_by_email self.email = user.email if user end # Determine if email is required (we don't want validation errors before we hit the checkout) def require_email true unless new_record? || ['cart', 'address'].include?(state) end def ensure_inventory_units if has_checkout_step?("delivery") inventory_validator = Spree::Stock::InventoryValidator.new errors = line_items.map { |line_item| inventory_validator.validate(line_item) }.compact raise InsufficientStock if errors.any? end end def ensure_promotions_eligible adjustment_changed = all_adjustments.eligible.promotion.any? do |adjustment| !adjustment.calculate_eligibility end if adjustment_changed restart_checkout_flow recalculate errors.add(:base, I18n.t('spree.promotion_total_changed_before_complete')) end errors.empty? end def validate_line_item_availability availability_validator = Spree::Stock::AvailabilityValidator.new raise InsufficientStock unless line_items.all? { |line_item| availability_validator.validate(line_item) } end def ensure_line_items_present unless line_items.present? errors.add(:base, I18n.t('spree.there_are_no_items_for_this_order')) && (return false) end end def ensure_available_shipping_rates if shipments.empty? || shipments.any? { |shipment| shipment.shipping_rates.blank? } # After this point, order redirects back to 'address' state and asks user to pick a proper address # Therefore, shipments are not necessary at this point. shipments.destroy_all errors.add(:base, I18n.t('spree.items_cannot_be_shipped')) && (return false) end end def after_cancel shipments.each(&:cancel!) payments.completed.each { |payment| payment.cancel! unless payment.fully_refunded? } payments.store_credits.pending.each(&:void_transaction!) send_cancel_email recalculate end def send_cancel_email Spree::Config.order_mailer_class.cancel_email(self).deliver_later end def after_resume shipments.each(&:resume!) end def use_billing? use_billing.in?([true, 'true', '1']) end def set_currency self.currency = Spree::Config[:currency] if self[:currency].nil? end def create_token self.guest_token ||= loop do random_token = SecureRandom.urlsafe_base64(nil, false) break random_token unless self.class.exists?(guest_token: random_token) end end end end