1
0
mirror of https://github.com/LaCasemate/fab-manager.git synced 2025-01-25 14:52:20 +01:00
fab-manager/app/services/trainings/auto_cancel_service.rb
2023-03-20 10:13:42 +01:00

87 lines
3.8 KiB
Ruby

# frozen_string_literal: true
# Business logic around trainings
module Trainings; end
# Automatically cancel trainings without enough reservation
class Trainings::AutoCancelService
class << self
# @param training [Training]
def auto_cancel_reservations(training)
return unless training.auto_cancel
training.availabilities
.includes(slots: :slots_reservations)
.where(availabilities: { lock: false })
.where('availabilities.start_at >= ? AND availabilities.start_at <= ?',
Time.current,
Time.current + training.auto_cancel_deadline.hours)
.find_each do |availability|
next if availability.reservations.count >= training.auto_cancel_threshold
auto_refund = Setting.get('wallet_module')
NotificationCenter.call type: 'notify_admin_training_auto_cancelled',
receiver: User.admins_and_managers,
attached_object: availability,
meta_data: { auto_refund: auto_refund }
availability.update(lock: true)
availability.slots_reservations.find_each do |sr|
NotificationCenter.call type: 'notify_member_training_auto_cancelled',
receiver: sr.reservation.user,
attached_object: sr,
meta_data: { auto_refund: auto_refund }
sr.update(canceled_at: Time.current)
refund_after_cancel(sr.reservation) if auto_refund
end
end
end
# update the given training, depending on the provided settings
# @param training [Training]
# @param auto_cancel [Setting,NilClass]
# @param threshold [Setting,NilClass]
# @param deadline [Setting,NilClass]
def update_auto_cancel(training, auto_cancel, threshold, deadline)
previous_auto_cancel = auto_cancel.nil? ? Setting.find_by(name: 'trainings_auto_cancel').value : auto_cancel.previous_value
previous_threshold = threshold.nil? ? Setting.find_by(name: 'trainings_auto_cancel_threshold').value : threshold.previous_value
previous_deadline = deadline.nil? ? Setting.find_by(name: 'trainings_auto_cancel_deadline').value : deadline.previous_value
is_default = training.auto_cancel.to_s == previous_auto_cancel.to_s &&
training.auto_cancel_threshold.to_s == previous_threshold.to_s &&
training.auto_cancel_deadline.to_s == previous_deadline.to_s
return unless is_default
# update parameters if the given training is default
params = {}
params[:auto_cancel] = auto_cancel.value unless auto_cancel.nil?
params[:auto_cancel_threshold] = threshold.value unless threshold.nil?
params[:auto_cancel_deadline] = deadline.value unless deadline.nil?
training.update(params)
end
# @param training [Training]
# @return [Boolean]
def override_settings?(training)
training.auto_cancel.to_s != Setting.find_by(name: 'trainings_auto_cancel')&.value.to_s ||
training.auto_cancel_threshold.to_s != Setting.find_by(name: 'trainings_auto_cancel_threshold')&.value.to_s ||
training.auto_cancel_deadline.to_s != Setting.find_by(name: 'trainings_auto_cancel_deadline')&.value.to_s
end
private
# @param reservation [Reservation]
def refund_after_cancel(reservation)
invoice_item = reservation.invoice_items.joins(:invoice).where(invoices: { type: nil }).first
amount = (invoice_item&.amount_after_coupon || 0) / 100.00
return if amount.zero?
service = WalletService.new(user: reservation.user, wallet: reservation.user.wallet)
transaction = service.credit(amount)
service.create_avoir(transaction, I18n.t('trainings.refund_for_auto_cancel')) if transaction
end
end
end