2021-04-22 19:24:08 +02:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
MINUTES_PER_HOUR = 60.0
|
|
|
|
SECONDS_PER_MINUTE = 60.0
|
|
|
|
|
2021-12-20 17:08:14 +01:00
|
|
|
GET_SLOT_PRICE_DEFAULT_OPTS = { has_credits: false, elements: nil, is_division: true, prepaid: { minutes: 0 }, custom_duration: nil }.freeze
|
2021-04-22 19:24:08 +02:00
|
|
|
|
|
|
|
# A generic reservation added to the shopping cart
|
|
|
|
class CartItem::Reservation < CartItem::BaseItem
|
|
|
|
def initialize(customer, operator, reservable, slots)
|
|
|
|
@customer = customer
|
|
|
|
@operator = operator
|
|
|
|
@reservable = reservable
|
|
|
|
@slots = slots
|
2021-05-31 15:39:56 +02:00
|
|
|
super
|
2021-04-22 19:24:08 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
def price
|
2021-12-20 17:08:14 +01:00
|
|
|
base_amount = get_hourly_rate
|
2021-06-29 16:56:40 +02:00
|
|
|
is_privileged = @operator.privileged? && @operator.id != @customer.id
|
2021-06-30 10:53:05 +02:00
|
|
|
prepaid = { minutes: PrepaidPackService.minutes_available(@customer, @reservable) }
|
2021-04-22 19:24:08 +02:00
|
|
|
|
|
|
|
elements = { slots: [] }
|
|
|
|
amount = 0
|
|
|
|
|
|
|
|
hours_available = credits
|
|
|
|
@slots.each_with_index do |slot, index|
|
2021-06-29 16:56:40 +02:00
|
|
|
amount += get_slot_price(base_amount, slot, is_privileged,
|
|
|
|
elements: elements,
|
|
|
|
has_credits: (index < hours_available),
|
2021-06-30 10:53:05 +02:00
|
|
|
prepaid: prepaid)
|
2021-04-22 19:24:08 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
{ elements: elements, amount: amount }
|
|
|
|
end
|
|
|
|
|
2021-04-23 12:52:06 +02:00
|
|
|
def name
|
|
|
|
@reservable.name
|
|
|
|
end
|
|
|
|
|
2021-05-31 15:39:56 +02:00
|
|
|
def valid?(all_items)
|
|
|
|
pending_subscription = all_items.find { |i| i.is_a?(CartItem::Subscription) }
|
|
|
|
@slots.each do |slot|
|
|
|
|
availability = Availability.find(slot[:availability_id])
|
|
|
|
next if availability.plan_ids.empty?
|
|
|
|
next if (@customer.subscribed_plan && availability.plan_ids.include?(@customer.subscribed_plan.id)) ||
|
|
|
|
(pending_subscription && availability.plan_ids.include?(pending_subscription.plan.id)) ||
|
|
|
|
(@operator.manager? && @customer.id != @operator.id) ||
|
|
|
|
@operator.admin?
|
|
|
|
|
|
|
|
@errors[:slot] = 'slot is restricted for subscribers'
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2021-04-22 19:24:08 +02:00
|
|
|
protected
|
|
|
|
|
|
|
|
def credits
|
|
|
|
0
|
|
|
|
end
|
|
|
|
|
|
|
|
##
|
|
|
|
# Compute the price of a single slot, according to the base price and the ability for an admin
|
|
|
|
# to offer the slot.
|
|
|
|
# @param hourly_rate {Number} base price of a slot
|
|
|
|
# @param slot {Hash} Slot object
|
|
|
|
# @param is_privileged {Boolean} true if the current user has a privileged role (admin or manager)
|
|
|
|
# @param [options] {Hash} optional parameters, allowing the following options:
|
|
|
|
# - elements {Array} if provided the resulting price will be append into elements.slots
|
|
|
|
# - has_credits {Boolean} true if the user still has credits for the given slot, false if not provided
|
|
|
|
# - is_division {boolean} false if the slot covers a full availability, true if it is a subdivision (default)
|
2021-06-29 16:56:40 +02:00
|
|
|
# - prepaid_minutes {Number} number of remaining prepaid minutes for the customer
|
2021-04-22 19:24:08 +02:00
|
|
|
# @return {Number} price of the slot
|
|
|
|
##
|
|
|
|
def get_slot_price(hourly_rate, slot, is_privileged, options = {})
|
|
|
|
options = GET_SLOT_PRICE_DEFAULT_OPTS.merge(options)
|
|
|
|
|
|
|
|
slot_rate = options[:has_credits] || (slot[:offered] && is_privileged) ? 0 : hourly_rate
|
2021-06-30 10:53:05 +02:00
|
|
|
slot_minutes = (slot[:end_at].to_time - slot[:start_at].to_time) / SECONDS_PER_MINUTE
|
|
|
|
# apply the base price to the real slot duration
|
2021-04-22 19:24:08 +02:00
|
|
|
real_price = if options[:is_division]
|
2021-06-30 10:53:05 +02:00
|
|
|
(slot_rate / MINUTES_PER_HOUR) * slot_minutes
|
2021-04-22 19:24:08 +02:00
|
|
|
else
|
|
|
|
slot_rate
|
|
|
|
end
|
2021-06-30 10:53:05 +02:00
|
|
|
# subtract free minutes from prepaid packs
|
2021-07-02 15:45:52 +02:00
|
|
|
if real_price.positive? && options[:prepaid][:minutes]&.positive?
|
2021-06-30 10:53:05 +02:00
|
|
|
consumed = slot_minutes
|
|
|
|
consumed = options[:prepaid][:minutes] if slot_minutes > options[:prepaid][:minutes]
|
|
|
|
real_price = (slot_minutes - consumed) * (slot_rate / MINUTES_PER_HOUR)
|
|
|
|
options[:prepaid][:minutes] -= consumed
|
2021-06-29 16:56:40 +02:00
|
|
|
end
|
2021-04-22 19:24:08 +02:00
|
|
|
|
|
|
|
unless options[:elements].nil?
|
|
|
|
options[:elements][:slots].push(
|
|
|
|
start_at: slot[:start_at],
|
|
|
|
price: real_price,
|
|
|
|
promo: (slot_rate != hourly_rate)
|
|
|
|
)
|
|
|
|
end
|
|
|
|
real_price
|
|
|
|
end
|
|
|
|
|
2021-12-20 17:08:14 +01:00
|
|
|
# We compute the hourly rate according to the prices of the current reservation
|
|
|
|
# If there are prices for durations longer than 1 hour, but shorter than the total duration,
|
|
|
|
# we use these prices before using the hourly rate.
|
|
|
|
# Eg. If the reservation is for 12 hours, and there are prices for 3 hours, 7 hours,
|
|
|
|
# and the base price (1 hours), we use the 7 hours price, then 3 hours price, and finally the base price.
|
|
|
|
# Then we divide the total price by the total duration to get the hourly rate.
|
|
|
|
def get_hourly_rate
|
|
|
|
total_duration = @slots.map { |slot| (slot[:end_at].to_time - slot[:start_at].to_time) / SECONDS_PER_MINUTE }.reduce(:+)
|
|
|
|
price = 0
|
|
|
|
|
|
|
|
remaining_duration = total_duration
|
|
|
|
while remaining_duration > 60
|
|
|
|
max_duration = @reservable.prices.where(group_id: @customer.group_id, plan_id: @plan.try(:id))
|
|
|
|
.where(Price.arel_table[:duration].lteq(remaining_duration))
|
|
|
|
.maximum(:duration)
|
|
|
|
max_duration_amount = @reservable.prices.find_by(group_id: @customer.group_id, plan_id: @plan.try(:id), duration: max_duration)
|
|
|
|
.amount
|
|
|
|
|
|
|
|
price += max_duration_amount
|
|
|
|
remaining_duration -= max_duration
|
|
|
|
end
|
|
|
|
|
|
|
|
# base price for the last hour or less
|
|
|
|
base_amount = @reservable.prices.find_by(group_id: @customer.group_id, plan_id: @plan.try(:id), duration: 60).amount
|
|
|
|
price += (base_amount / MINUTES_PER_HOUR) * remaining_duration
|
|
|
|
|
|
|
|
price / (total_duration / MINUTES_PER_HOUR)
|
|
|
|
end
|
|
|
|
|
2021-04-22 19:24:08 +02:00
|
|
|
##
|
|
|
|
# Compute the number of remaining hours in the users current credits (for machine or space)
|
|
|
|
##
|
|
|
|
def credits_hours(credits, new_plan_being_bought = false)
|
2021-04-23 17:54:59 +02:00
|
|
|
return 0 unless credits
|
|
|
|
|
2021-04-22 19:24:08 +02:00
|
|
|
hours_available = credits.hours
|
|
|
|
unless new_plan_being_bought
|
|
|
|
user_credit = @customer.users_credits.find_by(credit_id: credits.id)
|
|
|
|
hours_available = credits.hours - user_credit.hours_used if user_credit
|
|
|
|
end
|
|
|
|
hours_available
|
|
|
|
end
|
2021-05-19 18:12:52 +02:00
|
|
|
|
|
|
|
def slots_params
|
|
|
|
@slots.map { |slot| slot.permit(:id, :start_at, :end_at, :availability_id, :offered) }
|
|
|
|
end
|
2021-04-22 19:24:08 +02:00
|
|
|
end
|