mirror of
https://github.com/bigcapitalhq/bigcapital.git
synced 2026-02-22 15:50:32 +00:00
fix: save payee contact to expense transaction.
This commit is contained in:
@@ -104,6 +104,7 @@ export default class ExpensesController extends BaseController {
|
|||||||
check('currency_code').optional().isString().isLength({ max: 3 }),
|
check('currency_code').optional().isString().isLength({ max: 3 }),
|
||||||
check('exchange_rate').optional({ nullable: true }).isNumeric().toFloat(),
|
check('exchange_rate').optional({ nullable: true }).isNumeric().toFloat(),
|
||||||
check('publish').optional().isBoolean().toBoolean(),
|
check('publish').optional().isBoolean().toBoolean(),
|
||||||
|
check('payee_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
|
|
||||||
check('categories').exists().isArray({ min: 1 }),
|
check('categories').exists().isArray({ min: 1 }),
|
||||||
check('categories.*.index')
|
check('categories.*.index')
|
||||||
@@ -392,6 +393,11 @@ export default class ExpensesController extends BaseController {
|
|||||||
errors: [{ type: 'EXPENSE_ALREADY_PUBLISHED', code: 700 }],
|
errors: [{ type: 'EXPENSE_ALREADY_PUBLISHED', code: 700 }],
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
if (error.errorType === 'contact_not_found') {
|
||||||
|
return res.boom.badRequest(null, {
|
||||||
|
errors: [{ type: 'CONTACT_NOT_FOUND', code: 800 }],
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
next(error);
|
next(error);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ export interface IExpense {
|
|||||||
publishedAt: Date|null,
|
publishedAt: Date|null,
|
||||||
userId: number,
|
userId: number,
|
||||||
paymentDate: Date,
|
paymentDate: Date,
|
||||||
|
payeeId: number,
|
||||||
categories: IExpenseCategory[],
|
categories: IExpenseCategory[],
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -43,7 +43,7 @@ export interface IExpenseDTO {
|
|||||||
publish: boolean,
|
publish: boolean,
|
||||||
userId: number,
|
userId: number,
|
||||||
paymentDate: Date,
|
paymentDate: Date,
|
||||||
|
payeeId: number,
|
||||||
categories: IExpenseCategoryDTO[],
|
categories: IExpenseCategoryDTO[],
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -205,14 +205,15 @@ export default class EntityRepository {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
* Arbitrary relation graphs can be upserted (insert + update + delete)
|
||||||
|
* using the upsertGraph method.
|
||||||
* @param graph
|
* @param graph
|
||||||
* @param options
|
* @param options
|
||||||
*/
|
*/
|
||||||
upsertGraph(graph, options) {
|
upsertGraph(graph, options) {
|
||||||
// Keep the input grpah immutable
|
// Keep the input grpah immutable
|
||||||
const graphCloned = cloneDeep(graph);
|
const graphCloned = cloneDeep(graph);
|
||||||
return this.model.upsertGraph(graphCloned)
|
return this.model.query().upsertGraph(graphCloned, options)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import TenantRepository from "./TenantRepository";
|
import TenantRepository from "./TenantRepository";
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import { Expense } from 'models';
|
import { Expense } from 'models';
|
||||||
|
|
||||||
export default class ExpenseRepository extends TenantRepository {
|
export default class ExpenseRepository extends TenantRepository {
|
||||||
/**
|
/**
|
||||||
* Constructor method.
|
* Constructor method.
|
||||||
|
|||||||
@@ -31,13 +31,17 @@ export default class ContactsService {
|
|||||||
* @param {TContactService} contactService
|
* @param {TContactService} contactService
|
||||||
* @return {Promise<IContact>}
|
* @return {Promise<IContact>}
|
||||||
*/
|
*/
|
||||||
public async getContactByIdOrThrowError(tenantId: number, contactId: number, contactService: TContactService) {
|
public async getContactByIdOrThrowError(
|
||||||
|
tenantId: number,
|
||||||
|
contactId: number,
|
||||||
|
contactService?: TContactService
|
||||||
|
) {
|
||||||
const { contactRepository } = this.tenancy.repositories(tenantId);
|
const { contactRepository } = this.tenancy.repositories(tenantId);
|
||||||
|
|
||||||
this.logger.info('[contact] trying to validate contact existance.', { tenantId, contactId });
|
this.logger.info('[contact] trying to validate contact existance.', { tenantId, contactId });
|
||||||
const contact = await contactRepository.findOne({
|
const contact = await contactRepository.findOne({
|
||||||
id: contactId,
|
id: contactId,
|
||||||
contactService: contactService,
|
...(contactService) && ({ contactService }),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!contact) {
|
if (!contact) {
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import JournalCommands from 'services/Accounting/JournalCommands';
|
|||||||
import { IExpense, IExpensesFilter, IAccount, IExpenseDTO, IExpensesService, ISystemUser, IPaginationMeta } from 'interfaces';
|
import { IExpense, IExpensesFilter, IAccount, IExpenseDTO, IExpensesService, ISystemUser, IPaginationMeta } from 'interfaces';
|
||||||
import DynamicListingService from 'services/DynamicListing/DynamicListService';
|
import DynamicListingService from 'services/DynamicListing/DynamicListService';
|
||||||
import events from 'subscribers/events';
|
import events from 'subscribers/events';
|
||||||
|
import ContactsService from "services/Contacts/ContactsService";
|
||||||
|
|
||||||
const ERRORS = {
|
const ERRORS = {
|
||||||
EXPENSE_NOT_FOUND: 'expense_not_found',
|
EXPENSE_NOT_FOUND: 'expense_not_found',
|
||||||
@@ -38,6 +39,9 @@ export default class ExpensesService implements IExpensesService {
|
|||||||
@EventDispatcher()
|
@EventDispatcher()
|
||||||
eventDispatcher: EventDispatcherInterface;
|
eventDispatcher: EventDispatcherInterface;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
contactsService: ContactsService;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve the payment account details or returns not found server error in case the
|
* Retrieve the payment account details or returns not found server error in case the
|
||||||
* given account not found on the storage.
|
* given account not found on the storage.
|
||||||
@@ -323,6 +327,13 @@ export default class ExpensesService implements IExpensesService {
|
|||||||
// - Validate expenses accounts type.
|
// - Validate expenses accounts type.
|
||||||
await this.validateExpensesAccountsType(tenantId, expensesAccounts);
|
await this.validateExpensesAccountsType(tenantId, expensesAccounts);
|
||||||
|
|
||||||
|
// - Validate the expense payee contact id existance on storage.
|
||||||
|
if (expenseDTO.payeeId) {
|
||||||
|
await this.contactsService.getContactByIdOrThrowError(
|
||||||
|
tenantId,
|
||||||
|
expenseDTO.payeeId,
|
||||||
|
)
|
||||||
|
}
|
||||||
// - Validate the given expense categories not equal zero.
|
// - Validate the given expense categories not equal zero.
|
||||||
this.validateCategoriesNotEqualZero(expenseDTO);
|
this.validateCategoriesNotEqualZero(expenseDTO);
|
||||||
|
|
||||||
@@ -346,8 +357,9 @@ export default class ExpensesService implements IExpensesService {
|
|||||||
* 2. Validate expense accounts exist on the storage.
|
* 2. Validate expense accounts exist on the storage.
|
||||||
* 3. Validate payment account type.
|
* 3. Validate payment account type.
|
||||||
* 4. Validate expenses accounts type.
|
* 4. Validate expenses accounts type.
|
||||||
* 5. Validate the given expense categories not equal zero.
|
* 5. Validate the expense payee contact id existance on storage.
|
||||||
* 6. Stores the expense to the storage.
|
* 6. Validate the given expense categories not equal zero.
|
||||||
|
* 7. Stores the expense to the storage.
|
||||||
* ---------
|
* ---------
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
* @param {IExpenseDTO} expenseDTO
|
* @param {IExpenseDTO} expenseDTO
|
||||||
@@ -359,26 +371,33 @@ export default class ExpensesService implements IExpensesService {
|
|||||||
): Promise<IExpense> {
|
): Promise<IExpense> {
|
||||||
const { expenseRepository } = this.tenancy.repositories(tenantId);
|
const { expenseRepository } = this.tenancy.repositories(tenantId);
|
||||||
|
|
||||||
// 1. Validate payment account existance on the storage.
|
// - Validate payment account existance on the storage.
|
||||||
const paymentAccount = await this.getPaymentAccountOrThrowError(
|
const paymentAccount = await this.getPaymentAccountOrThrowError(
|
||||||
tenantId,
|
tenantId,
|
||||||
expenseDTO.paymentAccountId,
|
expenseDTO.paymentAccountId,
|
||||||
);
|
);
|
||||||
// 2. Validate expense accounts exist on the storage.
|
// - Validate expense accounts exist on the storage.
|
||||||
const expensesAccounts = await this.getExpensesAccountsOrThrowError(
|
const expensesAccounts = await this.getExpensesAccountsOrThrowError(
|
||||||
tenantId,
|
tenantId,
|
||||||
this.mapExpensesAccountsIdsFromDTO(expenseDTO),
|
this.mapExpensesAccountsIdsFromDTO(expenseDTO),
|
||||||
);
|
);
|
||||||
// 3. Validate payment account type.
|
// - Validate payment account type.
|
||||||
await this.validatePaymentAccountType(tenantId, paymentAccount);
|
await this.validatePaymentAccountType(tenantId, paymentAccount);
|
||||||
|
|
||||||
// 4. Validate expenses accounts type.
|
// - Validate expenses accounts type.
|
||||||
await this.validateExpensesAccountsType(tenantId, expensesAccounts);
|
await this.validateExpensesAccountsType(tenantId, expensesAccounts);
|
||||||
|
|
||||||
// 5. Validate the given expense categories not equal zero.
|
// - Validate the expense payee contact id existance on storage.
|
||||||
|
if (expenseDTO.payeeId) {
|
||||||
|
await this.contactsService.getContactByIdOrThrowError(
|
||||||
|
tenantId,
|
||||||
|
expenseDTO.payeeId,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
// - Validate the given expense categories not equal zero.
|
||||||
this.validateCategoriesNotEqualZero(expenseDTO);
|
this.validateCategoriesNotEqualZero(expenseDTO);
|
||||||
|
|
||||||
// 6. Save the expense to the storage.
|
// - Save the expense to the storage.
|
||||||
const expenseObj = this.expenseDTOToModel(expenseDTO, authorizedUser);
|
const expenseObj = this.expenseDTOToModel(expenseDTO, authorizedUser);
|
||||||
const expenseModel = await expenseRepository.upsertGraph(expenseObj);
|
const expenseModel = await expenseRepository.upsertGraph(expenseObj);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user