module Credere # Entries are the recording of debits and credits to various accounts. # This table can be thought of as a traditional accounting Journal. # # Posting to a Ledger can be considered to happen automatically, since # Accounts have the reverse 'has_many' relationship to either it's credit or # debit entries # # @example # cash = Credere::Asset.find_by_name('Cash') # accounts_receivable = Credere::Asset.find_by_name('Accounts Receivable') # # debit_amount = Credere::DebitAmount.new(:account => cash, :amount => 1000) # credit_amount = Credere::CreditAmount.new(:account => accounts_receivable, :amount => 1000) # # entry = Credere::Entry.new(:description => "Receiving payment on an invoice") # entry.debit_amounts << debit_amount # entry.credit_amounts << credit_amount # entry.save # # @see http://en.wikipedia.org/wiki/Journal_entry Journal Entry # class Entry < ActiveRecord::Base before_save :default_date if ActiveRecord::VERSION::MAJOR > 4 belongs_to :commercial_document, :polymorphic => true, optional: true else belongs_to :commercial_document, :polymorphic => true end has_many :credit_amounts, :extend => AmountsExtension, :class_name => 'Credere::CreditAmount', :inverse_of => :entry has_many :debit_amounts, :extend => AmountsExtension, :class_name => 'Credere::DebitAmount', :inverse_of => :entry has_many :credit_accounts, :through => :credit_amounts, :source => :account, :class_name => 'Credere::Account' has_many :debit_accounts, :through => :debit_amounts, :source => :account, :class_name => 'Credere::Account' validates_presence_of :description validate :has_credit_amounts? validate :has_debit_amounts? validate :amounts_cancel? # Support construction using 'credits' and 'debits' keys accepts_nested_attributes_for :credit_amounts, :debit_amounts, allow_destroy: true alias_method :credits=, :credit_amounts_attributes= alias_method :debits=, :debit_amounts_attributes= # attr_accessible :credits, :debits # Support the deprecated .build method def self.build(hash) ActiveSupport::Deprecation.warn('Credere::Transaction.build() is deprecated (use new instead)', caller) new(hash) end def initialize(*args) super end private def default_date todays_date = ActiveRecord::Base.default_timezone == :utc ? Time.now.utc : Time.now self.date ||= todays_date end def has_credit_amounts? errors[:base] << "Entry must have at least one credit amount" if self.credit_amounts.blank? end def has_debit_amounts? errors[:base] << "Entry must have at least one debit amount" if self.debit_amounts.blank? end def amounts_cancel? errors[:base] << "The credit and debit amounts are not equal" if credit_amounts.balance_for_new_record != debit_amounts.balance_for_new_record end end end