mirror of
https://github.com/LaCasemate/fab-manager.git
synced 2025-01-25 14:52:20 +01:00
119 lines
4.1 KiB
Ruby
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
|