1
0
mirror of https://github.com/LaCasemate/fab-manager.git synced 2025-01-10 00:46:15 +01:00
fab-manager/app/services/coupon_service.rb

106 lines
3.7 KiB
Ruby
Raw Normal View History

2019-09-17 17:39:05 +02:00
# frozen_string_literal: true
# This class provides helper methods to deal with coupons
class CouponService
##
# Apply the provided coupon, if active, to the given price. Usability tests will be run depending on the
# provided parameters.
# If no coupon/coupon code or if the code does not match, return origin price without change
#
# @param total {Number} invoice total, before any coupon is applied
# @param coupon {String|Coupon} Coupon's code OR Coupon object
# @param user_id {Number} user's id against the coupon will be tested for usability
# @return {Number}
##
def apply(total, coupon = nil, user_id = nil)
price = total
coupon_object = if coupon.instance_of? Coupon
coupon
elsif coupon.instance_of? String
Coupon.find_by(code: coupon)
else
nil
end
return price if coupon_object.nil?
if coupon_object.status(user_id, total) == 'active'
case coupon_object.type
when 'percent_off'
price -= (Rational(price * coupon_object.percent_off) / Rational(100.0)).to_f.ceil
when 'amount_off'
# do not apply cash coupon unless it has a lower amount that the total price
price -= coupon_object.amount_off if coupon_object.amount_off <= price
else
raise InvalidCouponError("unsupported coupon type #{coupon_object.type}")
end
end
price
end
# Apply the provided coupon to the given amount, considering that this applies to a refund invoice (Avoir),
# potentially partial
def self.apply_on_refund(amount, coupon, paid_items = 1, refund_items = 1)
return amount if coupon.nil?
case coupon.type
when 'percent_off'
amount - (Rational(amount * coupon.percent_off) / Rational(100.0)).to_f.ceil
when 'amount_off'
amount - (Rational(coupon.amount_off / paid_items) * Rational(refund_items)).to_f.ceil
else
raise InvalidCouponError
end
end
##
# Find the coupon associated with the given code and check it is valid for the given user
# @param code {String} the literal code of the coupon
# @param user_id {Number} identifier of the user who is applying the coupon
# @return {Coupon}
##
def validate(code, user_id)
return nil unless code && user_id
coupon = Coupon.find_by(code: code)
raise InvalidCouponError if coupon.nil? || coupon.status(user_id) != 'active'
coupon
end
##
# Ventilate the discount of the provided coupon over the given amount proportionately to the invoice's total
# @param total {Number} total amount of the invoice expressed in centimes
# @param amount {Number} price of the invoice's sub-item expressed in centimes
# @param coupon {Coupon} coupon applied to the invoice, amount_off expressed in centimes if applicable
##
def ventilate(total, amount, coupon)
price = amount
if !coupon.nil? && total != 0
case coupon.type
when 'percent_off'
price = amount - (Rational(amount * coupon.percent_off) / Rational(100.00)).to_f.round
when 'amount_off'
ratio = Rational(amount) / Rational(total)
discount = (coupon.amount_off * ratio.abs)
price = (amount - discount).to_f.round
else
raise InvalidCouponError("unsupported coupon type #{coupon.type}")
end
end
price
end
2019-09-17 17:39:05 +02:00
##
# Compute the total amount of the given invoice, without the applied coupon
# Invoice.total stores the amount payed by the customer, coupon deducted
# @param invoice {Invoice} invoice object, its total before discount will be computed
# @return {Number} total in centimes
2019-09-17 17:39:05 +02:00
##
def invoice_total_no_coupon(invoice)
invoice.invoice_items.map(&:amount).map(&:to_i).reduce(:+) or 0
2019-09-17 17:39:05 +02:00
end
2019-09-10 11:46:14 +02:00
end