mirror of
https://github.com/bigcapitalhq/bigcapital.git
synced 2026-02-17 05:10:31 +00:00
276 lines
8.3 KiB
JavaScript
276 lines
8.3 KiB
JavaScript
import express from 'express';
|
|
import { omit, sumBy } from 'lodash';
|
|
import moment from 'moment';
|
|
import {
|
|
BillPayment,
|
|
BillPaymentEntry,
|
|
Vendor,
|
|
Bill,
|
|
Account,
|
|
AccountTransaction,
|
|
} from '@/models';
|
|
import ServiceItemsEntries from '@/services/Sales/ServiceItemsEntries';
|
|
import AccountsService from '@/services/Accounts/AccountsService';
|
|
import JournalPoster from '@/services/Accounting/JournalPoster';
|
|
import JournalEntry from '@/services/Accounting/JournalEntry';
|
|
import JournalPosterService from '@/services/Sales/JournalPosterService';
|
|
import { formatDateFields } from '@/utils';
|
|
|
|
/**
|
|
* Bill payments service.
|
|
* @service
|
|
*/
|
|
export default class BillPaymentsService {
|
|
/**
|
|
* Creates a new bill payment transcations and store it to the storage
|
|
* with associated bills entries and journal transactions.
|
|
*
|
|
* Precedures:-
|
|
* ------
|
|
* - Records the bill payment transaction.
|
|
* - Records the bill payment associated entries.
|
|
* - Increment the payment amount of the given vendor bills.
|
|
* - Decrement the vendor balance.
|
|
* - Records payment journal entries.
|
|
*
|
|
* @param {BillPaymentDTO} billPayment
|
|
*/
|
|
static async createBillPayment(billPaymentDTO) {
|
|
const billPayment = {
|
|
amount: sumBy(billPaymentDTO.entries, 'payment_amount'),
|
|
...formatDateFields(billPaymentDTO, ['payment_date']),
|
|
}
|
|
const storedBillPayment = await BillPayment.tenant()
|
|
.query()
|
|
.insert({
|
|
...omit(billPayment, ['entries']),
|
|
});
|
|
const storeOpers = [];
|
|
|
|
billPayment.entries.forEach((entry) => {
|
|
const oper = BillPaymentEntry.tenant()
|
|
.query()
|
|
.insert({
|
|
bill_payment_id: storedBillPayment.id,
|
|
...entry,
|
|
});
|
|
// Increment the bill payment amount.
|
|
const billOper = Bill.changePaymentAmount(
|
|
entry.bill_id,
|
|
entry.payment_amount,
|
|
);
|
|
storeOpers.push(billOper);
|
|
storeOpers.push(oper);
|
|
});
|
|
// Decrement the vendor balance after bills payments.
|
|
const vendorDecrementOper = Vendor.changeBalance(
|
|
billPayment.vendor_id,
|
|
billPayment.amount * -1,
|
|
);
|
|
// Records the journal transactions after bills payment
|
|
// and change diff acoount balance.
|
|
const recordJournalTransaction = this.recordPaymentReceiveJournalEntries({
|
|
id: storedBillPayment.id,
|
|
...billPayment,
|
|
});
|
|
await Promise.all([
|
|
...storeOpers,
|
|
recordJournalTransaction,
|
|
vendorDecrementOper,
|
|
]);
|
|
return storedBillPayment;
|
|
}
|
|
|
|
/**
|
|
* Edits the details of the given bill payment.
|
|
*
|
|
* Preceducres:
|
|
* ------
|
|
* - Update the bill payment transaction.
|
|
* - Insert the new bill payment entries that have no ids.
|
|
* - Update the bill paymeny entries that have ids.
|
|
* - Delete the bill payment entries that not presented.
|
|
* - Re-insert the journal transactions and update the diff accounts balance.
|
|
* - Update the diff vendor balance.
|
|
* - Update the diff bill payment amount.
|
|
*
|
|
* @param {Integer} billPaymentId
|
|
* @param {BillPaymentDTO} billPayment
|
|
* @param {IBillPayment} oldBillPayment
|
|
*/
|
|
static async editBillPayment(billPaymentId, billPaymentDTO, oldBillPayment) {
|
|
const billPayment = {
|
|
amount: sumBy(billPaymentDTO.entries, 'payment_amount'),
|
|
...formatDateFields(billPaymentDTO, ['payment_date']),
|
|
};
|
|
const updateBillPayment = await BillPayment.tenant()
|
|
.query()
|
|
.where('id', billPaymentId)
|
|
.update({
|
|
...omit(billPayment, ['entries']),
|
|
});
|
|
const opers = [];
|
|
const entriesHasIds = billPayment.entries.filter((i) => i.id);
|
|
const entriesHasNoIds = billPayment.entries.filter((e) => !e.id);
|
|
|
|
const entriesIdsShouldDelete = ServiceItemsEntries.entriesShouldDeleted(
|
|
oldBillPayment.entries,
|
|
entriesHasIds
|
|
);
|
|
if (entriesIdsShouldDelete.length > 0) {
|
|
const deleteOper = BillPaymentEntry.tenant()
|
|
.query()
|
|
.bulkDelete(entriesIdsShouldDelete);
|
|
opers.push(deleteOper);
|
|
}
|
|
// Entries that should be update to the storage.
|
|
if (entriesHasIds.length > 0) {
|
|
const updateOper = BillPaymentEntry.tenant()
|
|
.query()
|
|
.bulkUpdate(entriesHasIds, { where: 'id' });
|
|
opers.push(updateOper);
|
|
}
|
|
// Entries that should be inserted to the storage.
|
|
if (entriesHasNoIds.length > 0) {
|
|
const insertOper = BillPaymentEntry.tenant()
|
|
.query()
|
|
.bulkInsert(
|
|
entriesHasNoIds.map((e) => ({ ...e, bill_payment_id: billPaymentId }))
|
|
);
|
|
opers.push(insertOper);
|
|
}
|
|
// Records the journal transactions after bills payment and change
|
|
// different acoount balance.
|
|
const recordJournalTransaction = this.recordPaymentReceiveJournalEntries({
|
|
id: storedBillPayment.id,
|
|
...billPayment,
|
|
});
|
|
// Change the different vendor balance between the new and old one.
|
|
const changeDiffBalance = Vendor.changeDiffBalance(
|
|
billPayment.vendor_id,
|
|
oldBillPayment.vendorId,
|
|
billPayment.amount * -1,
|
|
oldBillPayment.amount * -1,
|
|
);
|
|
await Promise.all([
|
|
...opers,
|
|
recordJournalTransaction,
|
|
changeDiffBalance,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Deletes the bill payment and associated transactions.
|
|
* @param {Integer} billPaymentId - The given bill payment id.
|
|
* @return {Promise}
|
|
*/
|
|
static async deleteBillPayment(billPaymentId) {
|
|
const billPayment = await BillPayment.tenant().query().where('id', billPaymentId).first();
|
|
|
|
await BillPayment.tenant().query()
|
|
.where('id', billPaymentId)
|
|
.delete();
|
|
|
|
await BillPaymentEntry.tenant()
|
|
.query()
|
|
.where('bill_payment_id', billPaymentId)
|
|
.delete();
|
|
|
|
const deleteTransactionsOper = JournalPosterService.deleteJournalTransactions(
|
|
billPaymentId,
|
|
'BillPayment',
|
|
);
|
|
const revertVendorBalanceOper = Vendor.changeBalance(
|
|
billPayment.vendorId,
|
|
billPayment.amount,
|
|
);
|
|
return Promise.all([
|
|
deleteTransactionsOper,
|
|
revertVendorBalanceOper,
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Records bill payment receive journal transactions.
|
|
* @param {BillPayment} billPayment
|
|
* @param {Integer} billPaymentId
|
|
*/
|
|
static async recordPaymentReceiveJournalEntries(billPayment) {
|
|
const paymentAmount = sumBy(billPayment.entries, 'payment_amount');
|
|
const formattedDate = moment(billPayment.payment_date).format('YYYY-MM-DD');
|
|
const payableAccount = await AccountsService.getAccountByType(
|
|
'accounts_payable'
|
|
);
|
|
|
|
const accountsDepGraph = await Account.tenant().depGraph().query();
|
|
const journal = new JournalPoster(accountsDepGraph);
|
|
|
|
const commonJournal = {
|
|
debit: 0,
|
|
credit: 0,
|
|
referenceId: billPayment.id,
|
|
referenceType: 'BillPayment',
|
|
date: formattedDate,
|
|
};
|
|
if (billPayment.id) {
|
|
const transactions = await AccountTransaction.tenant()
|
|
.query()
|
|
.whereIn('reference_type', ['BillPayment'])
|
|
.where('reference_id', billPayment.id)
|
|
.withGraphFetched('account.type');
|
|
|
|
journal.loadEntries(transactions);
|
|
journal.removeEntries();
|
|
}
|
|
const debitReceivable = new JournalEntry({
|
|
...commonJournal,
|
|
debit: paymentAmount,
|
|
contactType: 'Vendor',
|
|
contactId: billPayment.vendor_id,
|
|
account: payableAccount.id,
|
|
});
|
|
const creditPaymentAccount = new JournalEntry({
|
|
...commonJournal,
|
|
credit: paymentAmount,
|
|
account: billPayment.payment_account_id,
|
|
});
|
|
journal.debit(debitReceivable);
|
|
journal.credit(creditPaymentAccount);
|
|
|
|
await Promise.all([
|
|
journal.deleteEntries(),
|
|
journal.saveEntries(),
|
|
journal.saveBalance(),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Retrieve bill payment with associated metadata.
|
|
* @param {number} billPaymentId
|
|
* @return {object}
|
|
*/
|
|
static async getBillPaymentWithMetadata(billPaymentId) {
|
|
const billPayment = await BillPayment.tenant().query()
|
|
.where('id', billPaymentId)
|
|
.withGraphFetched('entries')
|
|
.withGraphFetched('vendor')
|
|
.withGraphFetched('paymentAccount')
|
|
.first();
|
|
|
|
return billPayment;
|
|
}
|
|
|
|
/**
|
|
* Detarmines whether the bill payment exists on the storage.
|
|
* @param {Integer} billPaymentId
|
|
* @return {boolean}
|
|
*/
|
|
static async isBillPaymentExists(billPaymentId) {
|
|
const billPayment = await BillPayment.tenant().query()
|
|
.where('id', billPaymentId)
|
|
.first();
|
|
|
|
return (billPayment.length > 0);
|
|
}
|
|
}
|