1
0
mirror of https://github.com/LaCasemate/fab-manager.git synced 2025-01-10 00:46:15 +01:00
fab-manager/app/models/accounting_period.rb

105 lines
3.5 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
2021-04-16 10:34:02 +02:00
require 'integrity/checksum'
require 'version'
require 'zip'
# AccountingPeriod is a period of N days (N > 0) which as been closed by an admin
# to prevent writing new accounting lines (invoices & refunds) during this period of time.
2020-03-25 10:16:47 +01:00
class AccountingPeriod < ApplicationRecord
2019-01-07 12:29:52 +01:00
before_destroy { false }
before_update { false }
before_create :compute_totals
2019-04-04 11:37:23 +02:00
after_commit :archive_closed_data, on: [:create]
2019-01-07 12:29:52 +01:00
validates :start_at, :end_at, :closed_at, :closed_by, presence: true
validates_with DateRangeValidator
validates_with DurationValidator
validates_with PastPeriodValidator
validates_with PeriodOverlapValidator
validates_with PeriodIntegrityValidator
belongs_to :user, class_name: 'User', foreign_key: 'closed_by'
2019-01-07 12:29:52 +01:00
def delete
false
end
def invoices
2019-03-12 12:15:14 +01:00
Invoice.where('created_at >= :start_date AND CAST(created_at AS DATE) <= :end_date', start_date: start_at, end_date: end_at)
end
def payment_schedules
PaymentSchedule.where('created_at >= :start_date AND CAST(created_at AS DATE) <= :end_date', start_date: start_at, end_date: end_at)
end
2019-03-12 12:15:14 +01:00
def invoices_with_vat(invoices)
vat_service = VatHistoryService.new
2019-03-12 12:15:14 +01:00
invoices.map do |i|
2021-12-24 19:37:43 +01:00
vat_rate_group = {}
i.invoice_items.each do |item|
vat_type = item.invoice_item_type
2021-12-28 19:42:04 +01:00
vat_rate_group[vat_type] = vat_service.invoice_item_vat(item) / 100.0 unless vat_rate_group[vat_type]
2021-12-24 19:37:43 +01:00
end
{ invoice: i, vat_rate: vat_rate_group }
2019-03-12 12:15:14 +01:00
end
end
def archive_folder
dir = "accounting/#{id}"
2020-06-24 15:25:17 +02:00
dir = "test/fixtures/files/accounting/#{id}" if Rails.env.test?
# create directory if it doesn't exists (accounting)
FileUtils.mkdir_p dir
dir
end
def archive_file
"#{archive_folder}/#{start_at.iso8601}_#{end_at.iso8601}.zip"
end
def archive_json_file
"#{start_at.iso8601}_#{end_at.iso8601}.json"
end
def check_footprint
footprint == compute_footprint
end
2019-04-03 17:57:21 +02:00
def previous_period
AccountingPeriod.where('closed_at < ?', closed_at).order(closed_at: :desc).limit(1).last
end
private
def archive_closed_data
2019-04-03 17:57:21 +02:00
ArchiveWorker.perform_async(id)
end
def price_without_taxe(invoice)
2021-12-24 19:37:43 +01:00
invoice[:invoice].invoice_items.map(&:net_amount).sum
end
def compute_totals
period_invoices = invoices_with_vat(invoices.where(type: nil).includes([:invoice_items]))
period_avoirs = invoices_with_vat(invoices.where(type: 'Avoir').includes([:invoice_items]))
self.period_total = (period_invoices.map(&method(:price_without_taxe)).reduce(:+) || 0) -
(period_avoirs.map(&method(:price_without_taxe)).reduce(:+) || 0)
all_invoices = invoices_with_vat(Invoice.where('CAST(created_at AS DATE) <= :end_date AND type IS NULL', end_date: end_at)
.includes([:invoice_items]))
all_avoirs = invoices_with_vat(Invoice.where("CAST(created_at AS DATE) <= :end_date AND type = 'Avoir'", end_date: end_at)
.includes([:invoice_items]))
self.perpetual_total = (all_invoices.map(&method(:price_without_taxe)).reduce(:+) || 0) -
(all_avoirs.map(&method(:price_without_taxe)).reduce(:+) || 0)
self.footprint = compute_footprint
end
def compute_footprint
columns = AccountingPeriod.columns.map(&:name)
.delete_if { |c| %w[id footprint created_at updated_at].include? c }
2021-04-16 10:34:02 +02:00
Integrity::Checksum.text("#{columns.map { |c| self[c] }.join}#{previous_period ? previous_period.footprint : ''}")
end
2019-01-07 12:29:52 +01:00
end