1
0
mirror of https://github.com/LaCasemate/fab-manager.git synced 2025-01-25 14:52:20 +01:00
fab-manager/lib/tasks/fablab/fix_availabilities.rake

119 lines
4.1 KiB
Ruby

# frozen_string_literal: true
# This take will ensure data integrity for availbilities.
# For an unknown reason, some slots are associated with unexisting availabilities. This script will try to re-create them
namespace :fablab do
desc 'regenerate missing availabilities'
task fix_availabilities: :environment do |_task, _args|
ActiveRecord::Base.transaction do
Slot.find_each do |slot|
next unless slot.availability.nil?
other_slots = Slot.where(availability_id: slot.availability_id)
reservations = SlotsReservation.where(slot_id: other_slots.map(&:id))
type = available_type(reservations)
a = Availability.new(
id: slot.availability_id,
start_at: other_slots.group('id').select('min(start_at) as min').first[:min],
end_at: other_slots.group('id').select('max(end_at) as max').first[:max],
available_type: type,
machine_ids: machines_ids(reservations, slot.availability_id),
space_ids: space_ids(reservations, slot.availability_id),
training_ids: training_ids(reservations, slot.availability_id)
)
create_mock_event(reservations, slot.availability_id) if type == 'event' && a.event.nil?
raise StandardError, "unable to save availability for slot #{slot.id}: #{a.errors.full_messages}" unless a.save(validate: false)
end
end
end
private
# @param reservations [ActiveRecord::Relation<SlotsReservation>]
def available_type(reservations)
return 'unknown' if reservations.count.zero?
type = reservations.first&.reservation&.reservable_type
case type
when 'Training', 'Space', 'Event'
type&.downcase
else
'machines'
end
end
# @param reservations [ActiveRecord::Relation<SlotsReservation>]
# @param availability_id [Number]
def machines_ids(reservations, availability_id)
type = reservations.first&.reservation&.reservable_type
return [] unless type == 'Machine'
ma = MachinesAvailability.where(availability_id: availability_id).map(&:machine_id)
return ma unless ma.empty?
rv = reservations.map(&:reservation).map(&:reservable_id)
return rv unless rv.empty?
[]
end
# @param reservations [ActiveRecord::Relation<SlotsReservation>]
# @param availability_id [Number]
def space_ids(reservations, availability_id)
type = reservations.first&.reservation&.reservable_type
return [] unless type == 'Space'
sa = SpacesAvailability.where(availability_id: availability_id).map(&:machine_id)
return sa unless sa.empty?
rv = reservations.map(&:reservation).map(&:reservable_id)
return rv unless rv.empty?
[]
end
# @param reservations [ActiveRecord::Relation<SlotsReservation>]
# @param availability_id [Number]
def training_ids(reservations, availability_id)
type = reservations.first&.reservation&.reservable_type
return [] unless type == 'Training'
ta = TrainingsAvailability.where(availability_id: availability_id).map(&:machine_id)
return ta unless ta.empty?
rv = reservations.map(&:reservation).map(&:reservable_id)
return rv unless rv.empty?
[]
end
# @param reservations [ActiveRecord::Relation<SlotsReservation>]
# @param availability_id [Number]
def create_mock_event(reservations, availability_id)
model = find_similar_event(reservations)
invoice_item = reservations.first&.reservation&.invoice_items&.find_by(main: true)
Event.create!(
title: model&.title || invoice_item&.description,
description: model&.description || invoice_item&.description,
category: model&.category || Category.first,
availability_id: availability_id
)
end
# @param reservations [ActiveRecord::Relation<SlotsReservation>]
# @return [Event,NilClass]
def find_similar_event(reservations)
reservations.each do |reservation|
reservation.reservation.invoice_items.each do |invoice_item|
words = invoice_item.description.split
(0..words.count).each do |w|
try_title = words[0..words.count - w].join(' ')
event = Event.find_by("title LIKE '#{try_title}%'")
return event unless event.nil?
end
end
end
end
end