fix: save payee contact to expense transaction.

This commit is contained in:
a.bouhuolia
2020-12-16 12:23:07 +02:00
parent 76488008c1
commit ddf31feb72
6 changed files with 45 additions and 14 deletions

View File

@@ -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);
} }

View File

@@ -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[],
} }

View File

@@ -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)
} }
/** /**

View File

@@ -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.

View File

@@ -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) {

View File

@@ -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);