1
0
mirror of https://github.com/LaCasemate/fab-manager.git synced 2025-01-24 13:52:21 +01:00
fab-manager/lib/integrity/archive_helper.rb
2021-05-24 11:34:35 +02:00

93 lines
3.0 KiB
Ruby

# frozen_string_literal: true
# Accounting integrity verifications
module Integrity; end
# Provides various helpers methods for interacting with accounting archives
class Integrity::ArchiveHelper
class << self
## Checks the validity of all closed periods and raise an error otherwise
def check_footprints
if AccountingPeriod.count.positive?
last_period = AccountingPeriod.order(start_at: :desc).first
puts "Checking invoices footprints from #{last_period.end_at}. This may take a while..."
Invoice.where('created_at > ?', last_period.end_at).order(:id).each do |i|
next if i.check_footprint
i.debug_footprint
raise "Invalid footprint for invoice #{i.id}"
end
else
puts 'Checking all invoices footprints. This may take a while...'
Invoice.order(:id).all.each do |i|
next if i.check_footprint
i.debug_footprint
raise "Invalid footprint for invoice #{i.id}"
end
end
end
# will return an array of hash containing the removed periods data
def backup_and_remove_periods(range_start: nil, range_end: nil)
range_periods = get_periods(range_start: range_start, range_end: range_end)
return [] unless range_periods.count.positive?
puts 'Removing accounting archives...'
# 1. remove protection for AccountingPeriods
execute("DROP RULE IF EXISTS accounting_periods_del_protect ON #{AccountingPeriod.arel_table.name};")
# 2. backup AccountingPeriods in memory
periods = []
range_periods.each do |p|
periods.push(
start_at: p.start_at,
end_at: p.end_at,
closed_at: p.closed_at,
closed_by: p.closed_by
)
end
# 3. Delete periods from database
range_periods.each do |ap|
execute("DELETE FROM accounting_periods WHERE ID=#{ap.id};")
end
periods
end
def restore_periods(periods)
return unless periods.size.positive?
# 1. recreate AccountingPeriods
puts 'Recreating accounting archives. This may take a while...'
periods.each do |p|
AccountingPeriod.create!(
start_at: p[:start_at],
end_at: p[:end_at],
closed_at: p[:closed_at],
closed_by: p[:closed_by]
)
end
# 2. reset protection for AccountingPeriods
execute("CREATE RULE accounting_periods_del_protect AS ON DELETE TO #{AccountingPeriod.arel_table.name} DO INSTEAD NOTHING;")
end
private
def get_periods(range_start: nil, range_end: nil)
if range_start && range_end
AccountingPeriod.where('created_at > ? AND created_at < ?', range_start, range_end)
elsif range_start
AccountingPeriod.where('created_at > ?', range_start)
elsif range_end
AccountingPeriod.where('created_at < ?', range_end)
else
AccountingPeriod.all
end
end
def execute(query)
ActiveRecord::Base.connection.execute(query)
end
end
end