2020-02-11 13:22:20 +01:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2020-03-25 17:45:53 +01:00
|
|
|
# Reservation is a Slot or a Ticket booked by a member.
|
|
|
|
# Slots are for Machine, Space and Training reservations.
|
|
|
|
# Tickets are for Event reservations.
|
2020-03-25 10:16:47 +01:00
|
|
|
class Reservation < ApplicationRecord
|
2016-03-23 18:39:41 +01:00
|
|
|
include NotifyWith::NotificationAttachedObject
|
|
|
|
|
2019-06-04 16:50:23 +02:00
|
|
|
belongs_to :statistic_profile
|
2017-02-28 13:23:31 +01:00
|
|
|
|
|
|
|
has_many :slots_reservations, dependent: :destroy
|
|
|
|
has_many :slots, through: :slots_reservations
|
|
|
|
|
2016-03-23 18:39:41 +01:00
|
|
|
accepts_nested_attributes_for :slots, allow_destroy: true
|
|
|
|
belongs_to :reservable, polymorphic: true
|
|
|
|
|
2016-08-25 18:41:33 +02:00
|
|
|
has_many :tickets
|
|
|
|
accepts_nested_attributes_for :tickets, allow_destroy: false
|
|
|
|
|
2021-05-25 17:28:35 +02:00
|
|
|
has_many :invoice_items, as: :object, dependent: :destroy
|
2021-05-27 16:11:23 +02:00
|
|
|
has_one :payment_schedule_object, as: :object, dependent: :destroy
|
2016-03-23 18:39:41 +01:00
|
|
|
|
|
|
|
validates_presence_of :reservable_id, :reservable_type
|
2018-12-11 17:27:25 +01:00
|
|
|
validate :machine_not_already_reserved, if: -> { reservable.is_a?(Machine) }
|
|
|
|
validate :training_not_fully_reserved, if: -> { reservable.is_a?(Training) }
|
2021-05-21 18:25:18 +02:00
|
|
|
validate :slots_not_locked
|
2016-03-23 18:39:41 +01:00
|
|
|
|
|
|
|
after_commit :notify_member_create_reservation, on: :create
|
|
|
|
after_commit :notify_admin_member_create_reservation, on: :create
|
2021-05-21 18:25:18 +02:00
|
|
|
after_commit :extend_subscription, on: :create
|
2018-12-11 17:27:25 +01:00
|
|
|
after_save :update_event_nb_free_places, if: proc { |reservation| reservation.reservable_type == 'Event' }
|
2016-03-23 18:39:41 +01:00
|
|
|
|
|
|
|
|
2019-11-25 10:45:54 +01:00
|
|
|
# @param canceled if true, count the number of seats for this reservation, including canceled seats
|
|
|
|
def total_booked_seats(canceled: false)
|
2019-11-25 10:30:03 +01:00
|
|
|
# cases:
|
|
|
|
# - machine/training/space reservation => 1 slot = 1 seat (currently not covered by this function)
|
|
|
|
# - event reservation => seats = nb_reserve_place (normal price) + tickets.booked (other prices)
|
2019-11-25 10:45:54 +01:00
|
|
|
return 0 if slots.first.canceled_at && !canceled
|
|
|
|
|
2019-11-25 10:30:03 +01:00
|
|
|
total = nb_reserve_places
|
2019-01-10 16:50:54 +01:00
|
|
|
total += tickets.map(&:booked).map(&:to_i).reduce(:+) if tickets.count.positive?
|
|
|
|
|
2016-08-30 09:47:03 +02:00
|
|
|
total
|
|
|
|
end
|
|
|
|
|
2019-06-04 16:50:23 +02:00
|
|
|
def user
|
|
|
|
statistic_profile.user
|
|
|
|
end
|
|
|
|
|
2019-11-20 17:06:42 +01:00
|
|
|
def update_event_nb_free_places
|
|
|
|
return unless reservable_type == 'Event'
|
|
|
|
|
|
|
|
reservable.update_nb_free_places
|
|
|
|
reservable.save!
|
|
|
|
end
|
|
|
|
|
2021-05-28 17:34:20 +02:00
|
|
|
def original_payment_schedule
|
|
|
|
payment_schedule_object&.payment_schedule
|
|
|
|
end
|
|
|
|
|
|
|
|
def original_invoice
|
|
|
|
invoice_items.select(:invoice_id)
|
|
|
|
.group(:invoice_id)
|
|
|
|
.map(&:invoice_id)
|
|
|
|
.map { |id| Invoice.find_by(id: id, type: nil) }
|
|
|
|
.first
|
|
|
|
end
|
|
|
|
|
2016-08-10 16:33:26 +02:00
|
|
|
private
|
2019-01-10 16:50:54 +01:00
|
|
|
|
2016-03-23 18:39:41 +01:00
|
|
|
def machine_not_already_reserved
|
|
|
|
already_reserved = false
|
2019-01-10 16:50:54 +01:00
|
|
|
slots.each do |slot|
|
2017-02-28 13:23:31 +01:00
|
|
|
same_hour_slots = Slot.joins(:reservations).where(
|
2019-01-10 16:50:54 +01:00
|
|
|
reservations: { reservable_type: reservable_type, reservable_id: reservable_id },
|
|
|
|
start_at: slot.start_at,
|
|
|
|
end_at: slot.end_at,
|
|
|
|
availability_id: slot.availability_id,
|
|
|
|
canceled_at: nil
|
|
|
|
)
|
2016-03-23 18:39:41 +01:00
|
|
|
if same_hour_slots.any?
|
|
|
|
already_reserved = true
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
2019-01-10 16:50:54 +01:00
|
|
|
errors.add(:machine, 'already reserved') if already_reserved
|
2016-03-23 18:39:41 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
def training_not_fully_reserved
|
2019-01-10 16:50:54 +01:00
|
|
|
slot = slots.first
|
|
|
|
errors.add(:training, 'already fully reserved') if Availability.find(slot.availability_id).completed?
|
2016-03-23 18:39:41 +01:00
|
|
|
end
|
|
|
|
|
2021-05-21 18:25:18 +02:00
|
|
|
def slots_not_locked
|
|
|
|
# check that none of the reserved availabilities was locked
|
|
|
|
slots.each do |slot|
|
|
|
|
errors.add(:slots, 'locked') if slot.availability.lock
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def extend_subscription
|
|
|
|
SubscriptionExtensionAfterReservation.new(self).extend_subscription_if_eligible
|
|
|
|
end
|
|
|
|
|
2016-03-23 18:39:41 +01:00
|
|
|
def notify_member_create_reservation
|
|
|
|
NotificationCenter.call type: 'notify_member_create_reservation',
|
|
|
|
receiver: user,
|
|
|
|
attached_object: self
|
|
|
|
end
|
|
|
|
|
|
|
|
def notify_admin_member_create_reservation
|
|
|
|
NotificationCenter.call type: 'notify_admin_member_create_reservation',
|
2020-04-29 15:34:30 +02:00
|
|
|
receiver: User.admins_and_managers,
|
2016-03-23 18:39:41 +01:00
|
|
|
attached_object: self
|
|
|
|
end
|
|
|
|
end
|