mirror of
https://github.com/bigcapitalhq/bigcapital.git
synced 2026-02-23 00:00:31 +00:00
Compare commits
73 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
1989887b25 | ||
|
|
e4fb126d39 | ||
|
|
5fcb2d9cc9 | ||
|
|
06ea631732 | ||
|
|
2f21107a43 | ||
|
|
fb8118bea8 | ||
|
|
4ba1c0aa22 | ||
|
|
169f115fa0 | ||
|
|
61ab2b78d9 | ||
|
|
93732430fc | ||
|
|
0215206220 | ||
|
|
3c8956fedf | ||
|
|
4477ada1ad | ||
|
|
fde9ccc5ca | ||
|
|
bbbd96f159 | ||
|
|
9f4de8115f | ||
|
|
d9f241a2f8 | ||
|
|
ee96dc68cc | ||
|
|
b12f090d13 | ||
|
|
f6ce761a27 | ||
|
|
8c89e04f54 | ||
|
|
f97b127a69 | ||
|
|
5c1fa8f5cd | ||
|
|
64c4d7b5a4 | ||
|
|
ea2fad648b | ||
|
|
9b8b51cb91 | ||
|
|
09b7e74d65 | ||
|
|
7137e06d99 | ||
|
|
fc29b765f7 | ||
|
|
2946475f89 | ||
|
|
d2193fdac0 | ||
|
|
ec2b7e332e | ||
|
|
a3704df6dd | ||
|
|
c3c784e52c | ||
|
|
bbcf695a6c | ||
|
|
038d4dd5a7 | ||
|
|
961e4b99e8 | ||
|
|
9991eebaaf | ||
|
|
cd90fede54 | ||
|
|
a2d28648bd | ||
|
|
3097d05eda | ||
|
|
ff94d8d9b2 | ||
|
|
79cc09fad9 | ||
|
|
c1b29c3f23 | ||
|
|
cf4bb3007e | ||
|
|
193a86cf30 | ||
|
|
7a81f14eb2 | ||
|
|
14d1f0bd1d | ||
|
|
82f8648c59 | ||
|
|
c928940d32 | ||
|
|
0a78d56015 | ||
|
|
1a5716873e | ||
|
|
01b7c86ab9 | ||
|
|
0ca209b195 | ||
|
|
be6f6e3c73 | ||
|
|
cb016be78c | ||
|
|
fc085f2328 | ||
|
|
9a34f3e283 | ||
|
|
7054e862d5 | ||
|
|
faa81abee4 | ||
|
|
6d01f2a323 | ||
|
|
72678bb936 | ||
|
|
9ae5644af9 | ||
|
|
e8830c5911 | ||
|
|
7699889bd6 | ||
|
|
35a061d188 | ||
|
|
c7c021c969 | ||
|
|
be8352654e | ||
|
|
fb58ab8cc1 | ||
|
|
8da89ebe8b | ||
|
|
d43d46ebec | ||
|
|
ac3a514795 | ||
|
|
7147e230de |
@@ -150,6 +150,15 @@
|
|||||||
"contributions": [
|
"contributions": [
|
||||||
"bug"
|
"bug"
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "Champetaman",
|
||||||
|
"name": "Camilo Oviedo",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/64604272?v=4",
|
||||||
|
"profile": "https://www.camilooviedo.com/",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"contributorsPerLine": 7,
|
"contributorsPerLine": 7,
|
||||||
|
|||||||
@@ -129,6 +129,7 @@ Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/d
|
|||||||
<tr>
|
<tr>
|
||||||
<td align="center" valign="top" width="14.28%"><a href="https://github.com/oleynikd"><img src="https://avatars.githubusercontent.com/u/3976868?v=4?s=100" width="100px;" alt="Denis"/><br /><sub><b>Denis</b></sub></a><br /><a href="https://github.com/bigcapitalhq/bigcapital/issues?q=author%3Aoleynikd" title="Bug reports">🐛</a></td>
|
<td align="center" valign="top" width="14.28%"><a href="https://github.com/oleynikd"><img src="https://avatars.githubusercontent.com/u/3976868?v=4?s=100" width="100px;" alt="Denis"/><br /><sub><b>Denis</b></sub></a><br /><a href="https://github.com/bigcapitalhq/bigcapital/issues?q=author%3Aoleynikd" title="Bug reports">🐛</a></td>
|
||||||
<td align="center" valign="top" width="14.28%"><a href="https://myself.vercel.app/"><img src="https://avatars.githubusercontent.com/u/42431274?v=4?s=100" width="100px;" alt="Sachin Mittal"/><br /><sub><b>Sachin Mittal</b></sub></a><br /><a href="https://github.com/bigcapitalhq/bigcapital/issues?q=author%3Amittalsam98" title="Bug reports">🐛</a></td>
|
<td align="center" valign="top" width="14.28%"><a href="https://myself.vercel.app/"><img src="https://avatars.githubusercontent.com/u/42431274?v=4?s=100" width="100px;" alt="Sachin Mittal"/><br /><sub><b>Sachin Mittal</b></sub></a><br /><a href="https://github.com/bigcapitalhq/bigcapital/issues?q=author%3Amittalsam98" title="Bug reports">🐛</a></td>
|
||||||
|
<td align="center" valign="top" width="14.28%"><a href="https://www.camilooviedo.com/"><img src="https://avatars.githubusercontent.com/u/64604272?v=4?s=100" width="100px;" alt="Camilo Oviedo"/><br /><sub><b>Camilo Oviedo</b></sub></a><br /><a href="https://github.com/bigcapitalhq/bigcapital/commits?author=Champetaman" title="Code">💻</a></td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|||||||
@@ -69,9 +69,8 @@
|
|||||||
"is-my-json-valid": "^2.20.5",
|
"is-my-json-valid": "^2.20.5",
|
||||||
"js-money": "^0.6.3",
|
"js-money": "^0.6.3",
|
||||||
"jsonwebtoken": "^8.5.1",
|
"jsonwebtoken": "^8.5.1",
|
||||||
"knex": "^0.95.15",
|
"knex": "^3.1.0",
|
||||||
"knex-cleaner": "^1.3.0",
|
"knex-cleaner": "^1.3.0",
|
||||||
"knex-db-manager": "^0.6.1",
|
|
||||||
"libphonenumber-js": "^1.9.6",
|
"libphonenumber-js": "^1.9.6",
|
||||||
"lodash": "^4.17.15",
|
"lodash": "^4.17.15",
|
||||||
"lru-cache": "^6.0.0",
|
"lru-cache": "^6.0.0",
|
||||||
|
|||||||
@@ -103,24 +103,20 @@ export default class AccountsController extends BaseController {
|
|||||||
check('name')
|
check('name')
|
||||||
.exists()
|
.exists()
|
||||||
.isLength({ min: 3, max: DATATYPES_LENGTH.STRING })
|
.isLength({ min: 3, max: DATATYPES_LENGTH.STRING })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('code')
|
check('code')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isLength({ min: 3, max: 6 })
|
.isLength({ min: 3, max: 6 })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('currency_code').optional(),
|
check('currency_code').optional(),
|
||||||
check('account_type')
|
check('account_type')
|
||||||
.exists()
|
.exists()
|
||||||
.isLength({ min: 3, max: DATATYPES_LENGTH.STRING })
|
.isLength({ min: 3, max: DATATYPES_LENGTH.STRING })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('description')
|
check('description')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isLength({ max: DATATYPES_LENGTH.TEXT })
|
.isLength({ max: DATATYPES_LENGTH.TEXT })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('parent_account_id')
|
check('parent_account_id')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isInt({ min: 0, max: DATATYPES_LENGTH.INT_10 })
|
.isInt({ min: 0, max: DATATYPES_LENGTH.INT_10 })
|
||||||
@@ -136,23 +132,19 @@ export default class AccountsController extends BaseController {
|
|||||||
check('name')
|
check('name')
|
||||||
.exists()
|
.exists()
|
||||||
.isLength({ min: 3, max: DATATYPES_LENGTH.STRING })
|
.isLength({ min: 3, max: DATATYPES_LENGTH.STRING })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('code')
|
check('code')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isLength({ min: 3, max: 6 })
|
.isLength({ min: 3, max: 6 })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('account_type')
|
check('account_type')
|
||||||
.exists()
|
.exists()
|
||||||
.isLength({ min: 3, max: DATATYPES_LENGTH.STRING })
|
.isLength({ min: 3, max: DATATYPES_LENGTH.STRING })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('description')
|
check('description')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isLength({ max: DATATYPES_LENGTH.TEXT })
|
.isLength({ max: DATATYPES_LENGTH.TEXT })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('parent_account_id')
|
check('parent_account_id')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isInt({ min: 0, max: DATATYPES_LENGTH.INT_10 })
|
.isInt({ min: 0, max: DATATYPES_LENGTH.INT_10 })
|
||||||
|
|||||||
@@ -90,27 +90,23 @@ export default class AuthenticationController extends BaseController {
|
|||||||
.exists()
|
.exists()
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('last_name')
|
check('last_name')
|
||||||
.exists()
|
.exists()
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('email')
|
check('email')
|
||||||
.exists()
|
.exists()
|
||||||
.isString()
|
.isString()
|
||||||
.isEmail()
|
.isEmail()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('password')
|
check('password')
|
||||||
.exists()
|
.exists()
|
||||||
.isString()
|
.isString()
|
||||||
.isLength({ min: 6 })
|
.isLength({ min: 6 })
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -150,7 +146,7 @@ export default class AuthenticationController extends BaseController {
|
|||||||
* @returns {ValidationChain[]}
|
* @returns {ValidationChain[]}
|
||||||
*/
|
*/
|
||||||
private get sendResetPasswordSchema(): ValidationChain[] {
|
private get sendResetPasswordSchema(): ValidationChain[] {
|
||||||
return [check('email').exists().isEmail().trim().escape()];
|
return [check('email').exists().isEmail().trim()];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -158,7 +154,11 @@ export default class AuthenticationController extends BaseController {
|
|||||||
* @param {Request} req
|
* @param {Request} req
|
||||||
* @param {Response} res
|
* @param {Response} res
|
||||||
*/
|
*/
|
||||||
private async login(req: Request, res: Response, next: Function): Response {
|
private async login(
|
||||||
|
req: Request,
|
||||||
|
res: Response,
|
||||||
|
next: Function
|
||||||
|
): Promise<Response | null> {
|
||||||
const userDTO: ILoginDTO = this.matchedBodyData(req);
|
const userDTO: ILoginDTO = this.matchedBodyData(req);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import { NextFunction, Request, Response, Router } from 'express';
|
import { NextFunction, Request, Response, Router } from 'express';
|
||||||
|
import { param, query } from 'express-validator';
|
||||||
import BaseController from '@/api/controllers/BaseController';
|
import BaseController from '@/api/controllers/BaseController';
|
||||||
import { GetBankAccountSummary } from '@/services/Banking/BankAccounts/GetBankAccountSummary';
|
import { GetBankAccountSummary } from '@/services/Banking/BankAccounts/GetBankAccountSummary';
|
||||||
import { BankAccountsApplication } from '@/services/Banking/BankAccounts/BankAccountsApplication';
|
import { BankAccountsApplication } from '@/services/Banking/BankAccounts/BankAccountsApplication';
|
||||||
import { param } from 'express-validator';
|
import { GetPendingBankAccountTransactions } from '@/services/Cashflow/GetPendingBankAccountTransaction';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class BankAccountsController extends BaseController {
|
export class BankAccountsController extends BaseController {
|
||||||
@@ -13,6 +14,9 @@ export class BankAccountsController extends BaseController {
|
|||||||
@Inject()
|
@Inject()
|
||||||
private bankAccountsApp: BankAccountsApplication;
|
private bankAccountsApp: BankAccountsApplication;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private getPendingTransactionsService: GetPendingBankAccountTransactions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Router constructor.
|
* Router constructor.
|
||||||
*/
|
*/
|
||||||
@@ -20,6 +24,16 @@ export class BankAccountsController extends BaseController {
|
|||||||
const router = Router();
|
const router = Router();
|
||||||
|
|
||||||
router.get('/:bankAccountId/meta', this.getBankAccountSummary.bind(this));
|
router.get('/:bankAccountId/meta', this.getBankAccountSummary.bind(this));
|
||||||
|
router.get(
|
||||||
|
'/pending_transactions',
|
||||||
|
[
|
||||||
|
query('account_id').optional().isNumeric().toInt(),
|
||||||
|
query('page').optional().isNumeric().toInt(),
|
||||||
|
query('page_size').optional().isNumeric().toInt(),
|
||||||
|
],
|
||||||
|
this.validationResult,
|
||||||
|
this.getBankAccountsPendingTransactions.bind(this)
|
||||||
|
);
|
||||||
router.post(
|
router.post(
|
||||||
'/:bankAccountId/disconnect',
|
'/:bankAccountId/disconnect',
|
||||||
this.disconnectBankAccount.bind(this)
|
this.disconnectBankAccount.bind(this)
|
||||||
@@ -27,17 +41,13 @@ export class BankAccountsController extends BaseController {
|
|||||||
router.post('/:bankAccountId/update', this.refreshBankAccount.bind(this));
|
router.post('/:bankAccountId/update', this.refreshBankAccount.bind(this));
|
||||||
router.post(
|
router.post(
|
||||||
'/:bankAccountId/pause_feeds',
|
'/:bankAccountId/pause_feeds',
|
||||||
[
|
[param('bankAccountId').exists().isNumeric().toInt()],
|
||||||
param('bankAccountId').exists().isNumeric().toInt(),
|
|
||||||
],
|
|
||||||
this.validationResult,
|
this.validationResult,
|
||||||
this.pauseBankAccountFeeds.bind(this)
|
this.pauseBankAccountFeeds.bind(this)
|
||||||
);
|
);
|
||||||
router.post(
|
router.post(
|
||||||
'/:bankAccountId/resume_feeds',
|
'/:bankAccountId/resume_feeds',
|
||||||
[
|
[param('bankAccountId').exists().isNumeric().toInt()],
|
||||||
param('bankAccountId').exists().isNumeric().toInt(),
|
|
||||||
],
|
|
||||||
this.validationResult,
|
this.validationResult,
|
||||||
this.resumeBankAccountFeeds.bind(this)
|
this.resumeBankAccountFeeds.bind(this)
|
||||||
);
|
);
|
||||||
@@ -72,6 +82,32 @@ export class BankAccountsController extends BaseController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the bank account pending transactions.
|
||||||
|
* @param {Request} req
|
||||||
|
* @param {Response} res
|
||||||
|
* @param {NextFunction} next
|
||||||
|
*/
|
||||||
|
async getBankAccountsPendingTransactions(
|
||||||
|
req: Request,
|
||||||
|
res: Response,
|
||||||
|
next: NextFunction
|
||||||
|
) {
|
||||||
|
const { tenantId } = req;
|
||||||
|
const query = this.matchedQueryData(req);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const data =
|
||||||
|
await this.getPendingTransactionsService.getPendingTransactions(
|
||||||
|
tenantId,
|
||||||
|
query
|
||||||
|
);
|
||||||
|
return res.status(200).send(data);
|
||||||
|
} catch (error) {
|
||||||
|
next(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Disonnect the given bank account.
|
* Disonnect the given bank account.
|
||||||
* @param {Request} req
|
* @param {Request} req
|
||||||
|
|||||||
@@ -33,7 +33,16 @@ export class BankingRulesController extends BaseController {
|
|||||||
body('conditions.*.field').exists().isIn(['description', 'amount']),
|
body('conditions.*.field').exists().isIn(['description', 'amount']),
|
||||||
body('conditions.*.comparator')
|
body('conditions.*.comparator')
|
||||||
.exists()
|
.exists()
|
||||||
.isIn(['equals', 'contains', 'not_contain'])
|
.isIn([
|
||||||
|
'equals',
|
||||||
|
'equal',
|
||||||
|
'contains',
|
||||||
|
'not_contain',
|
||||||
|
'bigger',
|
||||||
|
'bigger_or_equal',
|
||||||
|
'smaller',
|
||||||
|
'smaller_or_equal',
|
||||||
|
])
|
||||||
.default('contain')
|
.default('contain')
|
||||||
.trim(),
|
.trim(),
|
||||||
body('conditions.*.value').exists().trim(),
|
body('conditions.*.value').exists().trim(),
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Service, Inject } from 'typedi';
|
import { Service, Inject } from 'typedi';
|
||||||
import { ValidationChain, check, param, query } from 'express-validator';
|
import { ValidationChain, body, check, param, query } from 'express-validator';
|
||||||
import { Router, Request, Response, NextFunction } from 'express';
|
import { Router, Request, Response, NextFunction } from 'express';
|
||||||
import { omit } from 'lodash';
|
import { omit } from 'lodash';
|
||||||
import BaseController from '../BaseController';
|
import BaseController from '../BaseController';
|
||||||
@@ -43,6 +43,16 @@ export default class NewCashflowTransactionController extends BaseController {
|
|||||||
this.asyncMiddleware(this.newCashflowTransaction),
|
this.asyncMiddleware(this.newCashflowTransaction),
|
||||||
this.catchServiceErrors
|
this.catchServiceErrors
|
||||||
);
|
);
|
||||||
|
router.post(
|
||||||
|
'/transactions/uncategorize/bulk',
|
||||||
|
[
|
||||||
|
body('ids').isArray({ min: 1 }),
|
||||||
|
body('ids.*').exists().isNumeric().toInt(),
|
||||||
|
],
|
||||||
|
this.validationResult,
|
||||||
|
this.uncategorizeBulkTransactions.bind(this),
|
||||||
|
this.catchServiceErrors
|
||||||
|
);
|
||||||
router.post(
|
router.post(
|
||||||
'/transactions/:id/uncategorize',
|
'/transactions/:id/uncategorize',
|
||||||
this.revertCategorizedCashflowTransaction,
|
this.revertCategorizedCashflowTransaction,
|
||||||
@@ -112,12 +122,11 @@ export default class NewCashflowTransactionController extends BaseController {
|
|||||||
public get newTransactionValidationSchema() {
|
public get newTransactionValidationSchema() {
|
||||||
return [
|
return [
|
||||||
check('date').exists().isISO8601().toDate(),
|
check('date').exists().isISO8601().toDate(),
|
||||||
check('reference_no').optional({ nullable: true }).trim().escape(),
|
check('reference_no').optional({ nullable: true }).trim(),
|
||||||
check('description')
|
check('description')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isLength({ min: 3 })
|
.isLength({ min: 3 })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('transaction_type').exists(),
|
check('transaction_type').exists(),
|
||||||
|
|
||||||
check('amount').exists().isFloat().toFloat(),
|
check('amount').exists().isFloat().toFloat(),
|
||||||
@@ -185,6 +194,34 @@ export default class NewCashflowTransactionController extends BaseController {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Uncategorize the given transactions in bulk.
|
||||||
|
* @param {Request<{}>} req
|
||||||
|
* @param {Response} res
|
||||||
|
* @param {NextFunction} next
|
||||||
|
* @returns {Promise<Response | null>}
|
||||||
|
*/
|
||||||
|
private uncategorizeBulkTransactions = async (
|
||||||
|
req: Request<{}>,
|
||||||
|
res: Response,
|
||||||
|
next: NextFunction
|
||||||
|
) => {
|
||||||
|
const { tenantId } = req;
|
||||||
|
const { ids: uncategorizedTransactionIds } = this.matchedBodyData(req);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this.cashflowApplication.uncategorizeTransactions(
|
||||||
|
tenantId,
|
||||||
|
uncategorizedTransactionIds
|
||||||
|
);
|
||||||
|
return res.status(200).send({
|
||||||
|
message: 'The given transactions have been uncategorized successfully.',
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
next(error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Categorize the cashflow transaction.
|
* Categorize the cashflow transaction.
|
||||||
* @param {Request} req
|
* @param {Request} req
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ export default class ContactsController extends BaseController {
|
|||||||
*/
|
*/
|
||||||
get autocompleteQuerySchema() {
|
get autocompleteQuerySchema() {
|
||||||
return [
|
return [
|
||||||
query('column_sort_by').optional().trim().escape(),
|
query('column_sort_by').optional().trim(),
|
||||||
query('sort_order').optional().isIn(['desc', 'asc']),
|
query('sort_order').optional().isIn(['desc', 'asc']),
|
||||||
|
|
||||||
query('stringified_filter_roles').optional().isJSON(),
|
query('stringified_filter_roles').optional().isJSON(),
|
||||||
@@ -122,32 +122,27 @@ export default class ContactsController extends BaseController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('first_name')
|
check('first_name')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('last_name')
|
check('last_name')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('company_name')
|
check('company_name')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
|
|
||||||
check('display_name')
|
check('display_name')
|
||||||
.exists()
|
.exists()
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
|
|
||||||
check('email')
|
check('email')
|
||||||
@@ -165,120 +160,101 @@ export default class ContactsController extends BaseController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('personal_phone')
|
check('personal_phone')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
|
|
||||||
check('billing_address_1')
|
check('billing_address_1')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('billing_address_2')
|
check('billing_address_2')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('billing_address_city')
|
check('billing_address_city')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('billing_address_country')
|
check('billing_address_country')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('billing_address_email')
|
check('billing_address_email')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.isEmail()
|
.isEmail()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('billing_address_postcode')
|
check('billing_address_postcode')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('billing_address_phone')
|
check('billing_address_phone')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('billing_address_state')
|
check('billing_address_state')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
|
|
||||||
check('shipping_address_1')
|
check('shipping_address_1')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('shipping_address_2')
|
check('shipping_address_2')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('shipping_address_city')
|
check('shipping_address_city')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('shipping_address_country')
|
check('shipping_address_country')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('shipping_address_email')
|
check('shipping_address_email')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.isEmail()
|
.isEmail()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('shipping_address_postcode')
|
check('shipping_address_postcode')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('shipping_address_phone')
|
check('shipping_address_phone')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('shipping_address_state')
|
check('shipping_address_state')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
|
|
||||||
check('note')
|
check('note')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
||||||
check('active').optional().isBoolean().toBoolean(),
|
check('active').optional().isBoolean().toBoolean(),
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -106,11 +106,7 @@ export default class CustomersController extends ContactsController {
|
|||||||
*/
|
*/
|
||||||
get customerDTOSchema() {
|
get customerDTOSchema() {
|
||||||
return [
|
return [
|
||||||
check('customer_type')
|
check('customer_type').exists().isIn(['business', 'individual']).trim(),
|
||||||
.exists()
|
|
||||||
.isIn(['business', 'individual'])
|
|
||||||
.trim()
|
|
||||||
.escape(),
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -123,7 +119,6 @@ export default class CustomersController extends ContactsController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: 3 }),
|
.isLength({ max: 3 }),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -133,7 +128,7 @@ export default class CustomersController extends ContactsController {
|
|||||||
*/
|
*/
|
||||||
get validateListQuerySchema() {
|
get validateListQuerySchema() {
|
||||||
return [
|
return [
|
||||||
query('column_sort_by').optional().trim().escape(),
|
query('column_sort_by').optional().trim(),
|
||||||
query('sort_order').optional().isIn(['desc', 'asc']),
|
query('sort_order').optional().isIn(['desc', 'asc']),
|
||||||
|
|
||||||
query('page').optional().isNumeric().toInt(),
|
query('page').optional().isNumeric().toInt(),
|
||||||
|
|||||||
@@ -106,7 +106,6 @@ export default class VendorsController extends ContactsController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ min: 3, max: 3 }),
|
.isLength({ min: 3, max: 3 }),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ export default class CurrenciesController extends BaseController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get currencyParamSchema(): ValidationChain[] {
|
get currencyParamSchema(): ValidationChain[] {
|
||||||
return [param('currency_code').exists().trim().escape()];
|
return [param('currency_code').exists().trim()];
|
||||||
}
|
}
|
||||||
|
|
||||||
get listSchema(): ValidationChain[] {
|
get listSchema(): ValidationChain[] {
|
||||||
@@ -187,11 +187,13 @@ export default class CurrenciesController extends BaseController {
|
|||||||
}
|
}
|
||||||
if (error.errorType === 'currency_code_exists') {
|
if (error.errorType === 'currency_code_exists') {
|
||||||
return res.boom.badRequest(null, {
|
return res.boom.badRequest(null, {
|
||||||
errors: [{
|
errors: [
|
||||||
|
{
|
||||||
type: 'CURRENCY_CODE_EXISTS',
|
type: 'CURRENCY_CODE_EXISTS',
|
||||||
message: 'The given currency code is already exists.',
|
message: 'The given currency code is already exists.',
|
||||||
code: 200,
|
code: 200,
|
||||||
}],
|
},
|
||||||
|
],
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (error.errorType === 'CANNOT_DELETE_BASE_CURRENCY') {
|
if (error.errorType === 'CANNOT_DELETE_BASE_CURRENCY') {
|
||||||
|
|||||||
@@ -89,7 +89,6 @@ export class ExpensesController extends BaseController {
|
|||||||
check('reference_no')
|
check('reference_no')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('payment_date').exists().isISO8601().toDate(),
|
check('payment_date').exists().isISO8601().toDate(),
|
||||||
check('payment_account_id')
|
check('payment_account_id')
|
||||||
@@ -123,7 +122,6 @@ export class ExpensesController extends BaseController {
|
|||||||
check('categories.*.description')
|
check('categories.*.description')
|
||||||
.optional()
|
.optional()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('categories.*.landed_cost').optional().isBoolean().toBoolean(),
|
check('categories.*.landed_cost').optional().isBoolean().toBoolean(),
|
||||||
check('categories.*.project_id')
|
check('categories.*.project_id')
|
||||||
@@ -144,7 +142,6 @@ export class ExpensesController extends BaseController {
|
|||||||
check('reference_no')
|
check('reference_no')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('payment_date').exists().isISO8601().toDate(),
|
check('payment_date').exists().isISO8601().toDate(),
|
||||||
check('payment_account_id')
|
check('payment_account_id')
|
||||||
@@ -179,7 +176,6 @@ export class ExpensesController extends BaseController {
|
|||||||
check('categories.*.description')
|
check('categories.*.description')
|
||||||
.optional()
|
.optional()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('categories.*.landed_cost').optional().isBoolean().toBoolean(),
|
check('categories.*.landed_cost').optional().isBoolean().toBoolean(),
|
||||||
check('categories.*.project_id')
|
check('categories.*.project_id')
|
||||||
|
|||||||
@@ -1,9 +1,7 @@
|
|||||||
import { query } from 'express-validator';
|
import { query } from 'express-validator';
|
||||||
import BaseController from "../BaseController";
|
import BaseController from '../BaseController';
|
||||||
|
|
||||||
export default class BaseFinancialReportController extends BaseController {
|
export default class BaseFinancialReportController extends BaseController {
|
||||||
|
|
||||||
|
|
||||||
get sheetNumberFormatValidationSchema() {
|
get sheetNumberFormatValidationSchema() {
|
||||||
return [
|
return [
|
||||||
query('number_format.precision')
|
query('number_format.precision')
|
||||||
@@ -19,8 +17,7 @@ export default class BaseFinancialReportController extends BaseController {
|
|||||||
query('number_format.negative_format')
|
query('number_format.negative_format')
|
||||||
.optional()
|
.optional()
|
||||||
.isIn(['parentheses', 'mines'])
|
.isIn(['parentheses', 'mines'])
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -51,8 +51,7 @@ export default class InventoryDetailsController extends BaseController {
|
|||||||
query('number_format.negative_format')
|
query('number_format.negative_format')
|
||||||
.optional()
|
.optional()
|
||||||
.isIn(['parentheses', 'mines'])
|
.isIn(['parentheses', 'mines'])
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
query('from_date').optional(),
|
query('from_date').optional(),
|
||||||
query('to_date').optional(),
|
query('to_date').optional(),
|
||||||
|
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ export default class JournalSheetController extends BaseFinancialReportControlle
|
|||||||
return [
|
return [
|
||||||
query('from_date').optional().isISO8601(),
|
query('from_date').optional().isISO8601(),
|
||||||
query('to_date').optional().isISO8601(),
|
query('to_date').optional().isISO8601(),
|
||||||
query('transaction_type').optional().trim().escape(),
|
query('transaction_type').optional().trim(),
|
||||||
query('transaction_id').optional().isInt().toInt(),
|
query('transaction_id').optional().isInt().toInt(),
|
||||||
oneOf(
|
oneOf(
|
||||||
[
|
[
|
||||||
|
|||||||
@@ -40,8 +40,7 @@ export default class TransactionsByReferenceController extends BaseController {
|
|||||||
query('number_format.negative_format')
|
query('number_format.negative_format')
|
||||||
.optional()
|
.optional()
|
||||||
.isIn(['parentheses', 'mines'])
|
.isIn(['parentheses', 'mines'])
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -86,7 +86,7 @@ export default class InventoryAdjustmentsController extends BaseController {
|
|||||||
*/
|
*/
|
||||||
get validateListQuerySchema() {
|
get validateListQuerySchema() {
|
||||||
return [
|
return [
|
||||||
query('column_sort_by').optional().trim().escape(),
|
query('column_sort_by').optional().trim(),
|
||||||
query('sort_order').optional().isIn(['desc', 'asc']),
|
query('sort_order').optional().isIn(['desc', 'asc']),
|
||||||
|
|
||||||
query('page').optional().isNumeric().toInt(),
|
query('page').optional().isNumeric().toInt(),
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ export default class InviteUsersController extends BaseController {
|
|||||||
router.post(
|
router.post(
|
||||||
'/send',
|
'/send',
|
||||||
[
|
[
|
||||||
body('email').exists().trim().escape(),
|
body('email').exists().trim(),
|
||||||
body('role_id').exists().isNumeric().toInt(),
|
body('role_id').exists().isNumeric().toInt(),
|
||||||
],
|
],
|
||||||
this.validationResult,
|
this.validationResult,
|
||||||
@@ -57,7 +57,7 @@ export default class InviteUsersController extends BaseController {
|
|||||||
);
|
);
|
||||||
router.get(
|
router.get(
|
||||||
'/invited/:token',
|
'/invited/:token',
|
||||||
[param('token').exists().trim().escape()],
|
[param('token').exists().trim()],
|
||||||
this.validationResult,
|
this.validationResult,
|
||||||
asyncMiddleware(this.invited.bind(this)),
|
asyncMiddleware(this.invited.bind(this)),
|
||||||
this.handleServicesError
|
this.handleServicesError
|
||||||
@@ -72,10 +72,10 @@ export default class InviteUsersController extends BaseController {
|
|||||||
*/
|
*/
|
||||||
private get inviteUserDTO() {
|
private get inviteUserDTO() {
|
||||||
return [
|
return [
|
||||||
check('first_name').exists().trim().escape(),
|
check('first_name').exists().trim(),
|
||||||
check('last_name').exists().trim().escape(),
|
check('last_name').exists().trim(),
|
||||||
check('password').exists().trim().escape().isLength({ min: 5 }),
|
check('password').exists().trim().isLength({ min: 5 }),
|
||||||
param('token').exists().trim().escape(),
|
param('token').exists().trim(),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -73,13 +73,11 @@ export default class ItemsCategoriesController extends BaseController {
|
|||||||
check('name')
|
check('name')
|
||||||
.exists()
|
.exists()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ min: 0, max: DATATYPES_LENGTH.STRING }),
|
.isLength({ min: 0, max: DATATYPES_LENGTH.STRING }),
|
||||||
check('description')
|
check('description')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
||||||
check('sell_account_id')
|
check('sell_account_id')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
@@ -101,9 +99,8 @@ export default class ItemsCategoriesController extends BaseController {
|
|||||||
*/
|
*/
|
||||||
get categoriesListValidationSchema() {
|
get categoriesListValidationSchema() {
|
||||||
return [
|
return [
|
||||||
query('column_sort_by').optional().trim().escape(),
|
query('column_sort_by').optional().trim(),
|
||||||
query('sort_order').optional().trim().escape().isIn(['desc', 'asc']),
|
query('sort_order').optional().trim().isIn(['desc', 'asc']),
|
||||||
|
|
||||||
query('stringified_filter_roles').optional().isJSON(),
|
query('stringified_filter_roles').optional().isJSON(),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -207,10 +204,8 @@ export default class ItemsCategoriesController extends BaseController {
|
|||||||
};
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const {
|
const { itemCategories, filterMeta } =
|
||||||
itemCategories,
|
await this.itemCategoriesService.getItemCategoriesList(
|
||||||
filterMeta,
|
|
||||||
} = await this.itemCategoriesService.getItemCategoriesList(
|
|
||||||
tenantId,
|
tenantId,
|
||||||
itemCategoriesFilter,
|
itemCategoriesFilter,
|
||||||
user
|
user
|
||||||
|
|||||||
@@ -96,13 +96,11 @@ export default class ItemsController extends BaseController {
|
|||||||
.exists()
|
.exists()
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isIn(['service', 'non-inventory', 'inventory']),
|
.isIn(['service', 'non-inventory', 'inventory']),
|
||||||
check('code')
|
check('code')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
// Purchase attributes.
|
// Purchase attributes.
|
||||||
check('purchasable').optional().isBoolean().toBoolean(),
|
check('purchasable').optional().isBoolean().toBoolean(),
|
||||||
@@ -141,13 +139,11 @@ export default class ItemsController extends BaseController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
||||||
check('purchase_description')
|
check('purchase_description')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
||||||
check('sell_tax_rate_id').optional({ nullable: true }).isInt().toInt(),
|
check('sell_tax_rate_id').optional({ nullable: true }).isInt().toInt(),
|
||||||
check('purchase_tax_rate_id')
|
check('purchase_tax_rate_id')
|
||||||
@@ -162,7 +158,6 @@ export default class ItemsController extends BaseController {
|
|||||||
.optional()
|
.optional()
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
||||||
check('active').optional().isBoolean().toBoolean(),
|
check('active').optional().isBoolean().toBoolean(),
|
||||||
|
|
||||||
@@ -184,7 +179,7 @@ export default class ItemsController extends BaseController {
|
|||||||
*/
|
*/
|
||||||
private get validateListQuerySchema() {
|
private get validateListQuerySchema() {
|
||||||
return [
|
return [
|
||||||
query('column_sort_by').optional().trim().escape(),
|
query('column_sort_by').optional().trim(),
|
||||||
query('sort_order').optional().isIn(['desc', 'asc']),
|
query('sort_order').optional().isIn(['desc', 'asc']),
|
||||||
|
|
||||||
query('page').optional().isNumeric().toInt(),
|
query('page').optional().isNumeric().toInt(),
|
||||||
|
|||||||
@@ -94,25 +94,21 @@ export default class ManualJournalsController extends BaseController {
|
|||||||
.optional()
|
.optional()
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('journal_type')
|
check('journal_type')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('reference')
|
check('reference')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
.isLength({ max: DATATYPES_LENGTH.STRING }),
|
||||||
check('description')
|
check('description')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isString()
|
.isString()
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
.isLength({ max: DATATYPES_LENGTH.TEXT }),
|
||||||
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
check('publish').optional().isBoolean().toBoolean(),
|
check('publish').optional().isBoolean().toBoolean(),
|
||||||
@@ -163,7 +159,7 @@ export default class ManualJournalsController extends BaseController {
|
|||||||
query('page_size').optional().isNumeric().toInt(),
|
query('page_size').optional().isNumeric().toInt(),
|
||||||
query('custom_view_id').optional().isNumeric().toInt(),
|
query('custom_view_id').optional().isNumeric().toInt(),
|
||||||
|
|
||||||
query('column_sort_by').optional().trim().escape(),
|
query('column_sort_by').optional().trim(),
|
||||||
query('sort_order').optional().isIn(['desc', 'asc']),
|
query('sort_order').optional().isIn(['desc', 'asc']),
|
||||||
|
|
||||||
query('stringified_filter_roles').optional().isJSON(),
|
query('stringified_filter_roles').optional().isJSON(),
|
||||||
|
|||||||
@@ -61,15 +61,14 @@ export default class MediaController extends BaseController {
|
|||||||
|
|
||||||
get uploadValidationSchema() {
|
get uploadValidationSchema() {
|
||||||
return [
|
return [
|
||||||
// check('attachment'),
|
check('model_name').optional().trim(),
|
||||||
check('model_name').optional().trim().escape(),
|
check('model_id').optional().isNumeric(),
|
||||||
check('model_id').optional().isNumeric().toInt(),
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
get linkValidationSchema() {
|
get linkValidationSchema() {
|
||||||
return [
|
return [
|
||||||
check('model_name').exists().trim().escape(),
|
check('model_name').exists().trim(),
|
||||||
check('model_id').exists().isNumeric().toInt(),
|
check('model_id').exists().isNumeric().toInt(),
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -62,7 +62,7 @@ export default class OrganizationController extends BaseController {
|
|||||||
private get commonOrganizationValidationSchema(): ValidationChain[] {
|
private get commonOrganizationValidationSchema(): ValidationChain[] {
|
||||||
return [
|
return [
|
||||||
check('name').exists().trim(),
|
check('name').exists().trim(),
|
||||||
check('industry').optional({ nullable: true }).isString().trim().escape(),
|
check('industry').optional({ nullable: true }).isString().trim(),
|
||||||
check('location').exists().isString().isISO31661Alpha2(),
|
check('location').exists().isString().isISO31661Alpha2(),
|
||||||
check('base_currency').exists().isISO4217(),
|
check('base_currency').exists().isISO4217(),
|
||||||
check('timezone').exists().isIn(moment.tz.names()),
|
check('timezone').exists().isIn(moment.tz.names()),
|
||||||
@@ -87,11 +87,7 @@ export default class OrganizationController extends BaseController {
|
|||||||
private get updateOrganizationValidationSchema(): ValidationChain[] {
|
private get updateOrganizationValidationSchema(): ValidationChain[] {
|
||||||
return [
|
return [
|
||||||
...this.commonOrganizationValidationSchema,
|
...this.commonOrganizationValidationSchema,
|
||||||
check('tax_number')
|
check('tax_number').optional({ nullable: true }).isString().trim(),
|
||||||
.optional({ nullable: true })
|
|
||||||
.isString()
|
|
||||||
.trim()
|
|
||||||
.escape(),
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -100,8 +100,8 @@ export default class BillsController extends BaseController {
|
|||||||
*/
|
*/
|
||||||
private get billValidationSchema() {
|
private get billValidationSchema() {
|
||||||
return [
|
return [
|
||||||
check('bill_number').exists().trim().escape(),
|
check('bill_number').exists().trim(),
|
||||||
check('reference_no').optional().trim().escape(),
|
check('reference_no').optional().trim(),
|
||||||
check('bill_date').exists().isISO8601(),
|
check('bill_date').exists().isISO8601(),
|
||||||
check('due_date').optional().isISO8601(),
|
check('due_date').optional().isISO8601(),
|
||||||
|
|
||||||
@@ -112,7 +112,7 @@ export default class BillsController extends BaseController {
|
|||||||
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
check('project_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('project_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
|
|
||||||
check('note').optional().trim().escape(),
|
check('note').optional().trim(),
|
||||||
check('open').default(false).isBoolean().toBoolean(),
|
check('open').default(false).isBoolean().toBoolean(),
|
||||||
|
|
||||||
check('is_inclusive_tax').default(false).isBoolean().toBoolean(),
|
check('is_inclusive_tax').default(false).isBoolean().toBoolean(),
|
||||||
@@ -126,10 +126,7 @@ export default class BillsController extends BaseController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
.toFloat(),
|
.toFloat(),
|
||||||
check('entries.*.description')
|
check('entries.*.description').optional({ nullable: true }).trim(),
|
||||||
.optional({ nullable: true })
|
|
||||||
.trim()
|
|
||||||
.escape(),
|
|
||||||
check('entries.*.landed_cost')
|
check('entries.*.landed_cost')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isBoolean()
|
.isBoolean()
|
||||||
@@ -141,7 +138,6 @@ export default class BillsController extends BaseController {
|
|||||||
check('entries.*.tax_code')
|
check('entries.*.tax_code')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isString(),
|
.isString(),
|
||||||
check('entries.*.tax_rate_id')
|
check('entries.*.tax_rate_id')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
@@ -158,8 +154,8 @@ export default class BillsController extends BaseController {
|
|||||||
*/
|
*/
|
||||||
private get billEditValidationSchema() {
|
private get billEditValidationSchema() {
|
||||||
return [
|
return [
|
||||||
check('bill_number').optional().trim().escape(),
|
check('bill_number').optional().trim(),
|
||||||
check('reference_no').optional().trim().escape(),
|
check('reference_no').optional().trim(),
|
||||||
check('bill_date').exists().isISO8601(),
|
check('bill_date').exists().isISO8601(),
|
||||||
check('due_date').optional().isISO8601(),
|
check('due_date').optional().isISO8601(),
|
||||||
|
|
||||||
@@ -170,7 +166,7 @@ export default class BillsController extends BaseController {
|
|||||||
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
check('project_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('project_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
|
|
||||||
check('note').optional().trim().escape(),
|
check('note').optional().trim(),
|
||||||
check('open').default(false).isBoolean().toBoolean(),
|
check('open').default(false).isBoolean().toBoolean(),
|
||||||
|
|
||||||
check('entries').isArray({ min: 1 }),
|
check('entries').isArray({ min: 1 }),
|
||||||
@@ -184,10 +180,7 @@ export default class BillsController extends BaseController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
.toFloat(),
|
.toFloat(),
|
||||||
check('entries.*.description')
|
check('entries.*.description').optional({ nullable: true }).trim(),
|
||||||
.optional({ nullable: true })
|
|
||||||
.trim()
|
|
||||||
.escape(),
|
|
||||||
check('entries.*.landed_cost')
|
check('entries.*.landed_cost')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isBoolean()
|
.isBoolean()
|
||||||
@@ -222,8 +215,8 @@ export default class BillsController extends BaseController {
|
|||||||
|
|
||||||
private get dueBillsListingValidationSchema() {
|
private get dueBillsListingValidationSchema() {
|
||||||
return [
|
return [
|
||||||
query('vendor_id').optional().trim().escape(),
|
query('vendor_id').optional().trim(),
|
||||||
query('payment_made_id').optional().trim().escape(),
|
query('payment_made_id').optional().trim(),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -113,10 +113,10 @@ export default class BillsPayments extends BaseController {
|
|||||||
|
|
||||||
check('amount').exists().isNumeric().toFloat(),
|
check('amount').exists().isNumeric().toFloat(),
|
||||||
check('payment_account_id').exists().isNumeric().toInt(),
|
check('payment_account_id').exists().isNumeric().toInt(),
|
||||||
check('payment_number').optional({ nullable: true }).trim().escape(),
|
check('payment_number').optional({ nullable: true }).trim(),
|
||||||
check('payment_date').exists(),
|
check('payment_date').exists(),
|
||||||
check('statement').optional().trim().escape(),
|
check('statement').optional().trim(),
|
||||||
check('reference').optional().trim().escape(),
|
check('reference').optional().trim(),
|
||||||
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
|
|
||||||
check('entries').exists().isArray(),
|
check('entries').exists().isArray(),
|
||||||
|
|||||||
@@ -156,13 +156,10 @@ export default class VendorCreditController extends BaseController {
|
|||||||
check('vendor_id').exists().isNumeric().toInt(),
|
check('vendor_id').exists().isNumeric().toInt(),
|
||||||
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
||||||
|
|
||||||
check('vendor_credit_number')
|
check('vendor_credit_number').optional({ nullable: true }).trim(),
|
||||||
.optional({ nullable: true })
|
check('reference_no').optional().trim(),
|
||||||
.trim()
|
|
||||||
.escape(),
|
|
||||||
check('reference_no').optional().trim().escape(),
|
|
||||||
check('vendor_credit_date').exists().isISO8601().toDate(),
|
check('vendor_credit_date').exists().isISO8601().toDate(),
|
||||||
check('note').optional().trim().escape(),
|
check('note').optional().trim(),
|
||||||
check('open').default(false).isBoolean().toBoolean(),
|
check('open').default(false).isBoolean().toBoolean(),
|
||||||
|
|
||||||
check('warehouse_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('warehouse_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
@@ -178,10 +175,7 @@ export default class VendorCreditController extends BaseController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
.toFloat(),
|
.toFloat(),
|
||||||
check('entries.*.description')
|
check('entries.*.description').optional({ nullable: true }).trim(),
|
||||||
.optional({ nullable: true })
|
|
||||||
.trim()
|
|
||||||
.escape(),
|
|
||||||
check('entries.*.warehouse_id')
|
check('entries.*.warehouse_id')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
@@ -202,13 +196,10 @@ export default class VendorCreditController extends BaseController {
|
|||||||
check('vendor_id').exists().isNumeric().toInt(),
|
check('vendor_id').exists().isNumeric().toInt(),
|
||||||
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
||||||
|
|
||||||
check('vendor_credit_number')
|
check('vendor_credit_number').optional({ nullable: true }).trim(),
|
||||||
.optional({ nullable: true })
|
check('reference_no').optional().trim(),
|
||||||
.trim()
|
|
||||||
.escape(),
|
|
||||||
check('reference_no').optional().trim().escape(),
|
|
||||||
check('vendor_credit_date').exists().isISO8601().toDate(),
|
check('vendor_credit_date').exists().isISO8601().toDate(),
|
||||||
check('note').optional().trim().escape(),
|
check('note').optional().trim(),
|
||||||
|
|
||||||
check('warehouse_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('warehouse_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
@@ -223,10 +214,7 @@ export default class VendorCreditController extends BaseController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
.toFloat(),
|
.toFloat(),
|
||||||
check('entries.*.description')
|
check('entries.*.description').optional({ nullable: true }).trim(),
|
||||||
.optional({ nullable: true })
|
|
||||||
.trim()
|
|
||||||
.escape(),
|
|
||||||
check('entries.*.warehouse_id')
|
check('entries.*.warehouse_id')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
|
|||||||
@@ -18,9 +18,7 @@ export default class ResourceController extends BaseController {
|
|||||||
|
|
||||||
router.get(
|
router.get(
|
||||||
'/:resource_model/meta',
|
'/:resource_model/meta',
|
||||||
[
|
[param('resource_model').exists().trim()],
|
||||||
param('resource_model').exists().trim().escape()
|
|
||||||
],
|
|
||||||
this.asyncMiddleware(this.resourceMeta.bind(this)),
|
this.asyncMiddleware(this.resourceMeta.bind(this)),
|
||||||
this.handleServiceErrors
|
this.handleServiceErrors
|
||||||
);
|
);
|
||||||
@@ -48,9 +46,7 @@ export default class ResourceController extends BaseController {
|
|||||||
resourceModel
|
resourceModel
|
||||||
);
|
);
|
||||||
return res.status(200).send({
|
return res.status(200).send({
|
||||||
resource_meta: this.transfromToResponse(
|
resource_meta: this.transfromToResponse(resourceMeta),
|
||||||
resourceMeta,
|
|
||||||
),
|
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
next(error);
|
next(error);
|
||||||
|
|||||||
@@ -210,9 +210,9 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
|
|
||||||
check('credit_note_date').exists().isISO8601().toDate(),
|
check('credit_note_date').exists().isISO8601().toDate(),
|
||||||
check('reference_no').optional(),
|
check('reference_no').optional(),
|
||||||
check('credit_note_number').optional({ nullable: true }).trim().escape(),
|
check('credit_note_number').optional({ nullable: true }).trim(),
|
||||||
check('note').optional().trim().escape(),
|
check('note').optional().trim(),
|
||||||
check('terms_conditions').optional().trim().escape(),
|
check('terms_conditions').optional().trim(),
|
||||||
check('open').default(false).isBoolean().toBoolean(),
|
check('open').default(false).isBoolean().toBoolean(),
|
||||||
|
|
||||||
check('warehouse_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('warehouse_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
@@ -228,10 +228,7 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
.toFloat(),
|
.toFloat(),
|
||||||
check('entries.*.description')
|
check('entries.*.description').optional({ nullable: true }).trim(),
|
||||||
.optional({ nullable: true })
|
|
||||||
.trim()
|
|
||||||
.escape(),
|
|
||||||
check('entries.*.warehouse_id')
|
check('entries.*.warehouse_id')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
|
|||||||
@@ -9,9 +9,9 @@ import {
|
|||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import BaseController from '@/api/controllers/BaseController';
|
import BaseController from '@/api/controllers/BaseController';
|
||||||
import asyncMiddleware from '@/api/middleware/asyncMiddleware';
|
import asyncMiddleware from '@/api/middleware/asyncMiddleware';
|
||||||
import PaymentReceivesPages from '@/services/Sales/PaymentReceives/PaymentReceivesPages';
|
import PaymentsReceivedPages from '@/services/Sales/PaymentReceived/PaymentsReceivedPages';
|
||||||
|
import { PaymentReceivesApplication } from '@/services/Sales/PaymentReceived/PaymentReceivedApplication';
|
||||||
import DynamicListingService from '@/services/DynamicListing/DynamicListService';
|
import DynamicListingService from '@/services/DynamicListing/DynamicListService';
|
||||||
import { PaymentReceivesApplication } from '@/services/Sales/PaymentReceives/PaymentReceivesApplication';
|
|
||||||
import CheckPolicies from '@/api/middleware/CheckPolicies';
|
import CheckPolicies from '@/api/middleware/CheckPolicies';
|
||||||
import { ServiceError } from '@/exceptions';
|
import { ServiceError } from '@/exceptions';
|
||||||
import { ACCEPT_TYPE } from '@/interfaces/Http';
|
import { ACCEPT_TYPE } from '@/interfaces/Http';
|
||||||
@@ -22,7 +22,7 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
private paymentReceiveApplication: PaymentReceivesApplication;
|
private paymentReceiveApplication: PaymentReceivesApplication;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private PaymentReceivesPages: PaymentReceivesPages;
|
private PaymentsReceivedPages: PaymentsReceivedPages;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private dynamicListService: DynamicListingService;
|
private dynamicListService: DynamicListingService;
|
||||||
@@ -154,8 +154,8 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
check('payment_date').exists(),
|
check('payment_date').exists(),
|
||||||
check('reference_no').optional(),
|
check('reference_no').optional(),
|
||||||
check('deposit_account_id').exists().isNumeric().toInt(),
|
check('deposit_account_id').exists().isNumeric().toInt(),
|
||||||
check('payment_receive_no').optional({ nullable: true }).trim().escape(),
|
check('payment_receive_no').optional({ nullable: true }).trim(),
|
||||||
check('statement').optional().trim().escape(),
|
check('statement').optional().trim(),
|
||||||
|
|
||||||
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
|
|
||||||
@@ -176,7 +176,6 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
private get validatePaymentReceiveList(): ValidationChain[] {
|
private get validatePaymentReceiveList(): ValidationChain[] {
|
||||||
return [
|
return [
|
||||||
query('stringified_filter_roles').optional().isJSON(),
|
query('stringified_filter_roles').optional().isJSON(),
|
||||||
|
|
||||||
query('view_slug').optional({ nullable: true }).isString().trim(),
|
query('view_slug').optional({ nullable: true }).isString().trim(),
|
||||||
|
|
||||||
query('column_sort_by').optional(),
|
query('column_sort_by').optional(),
|
||||||
@@ -230,7 +229,7 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const storedPaymentReceive =
|
const storedPaymentReceive =
|
||||||
await this.paymentReceiveApplication.createPaymentReceive(
|
await this.paymentReceiveApplication.createPaymentReceived(
|
||||||
tenantId,
|
tenantId,
|
||||||
paymentReceive,
|
paymentReceive,
|
||||||
user
|
user
|
||||||
@@ -377,7 +376,7 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
const { customerId } = this.matchedQueryData(req);
|
const { customerId } = this.matchedQueryData(req);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const entries = await this.PaymentReceivesPages.getNewPageEntries(
|
const entries = await this.PaymentsReceivedPages.getNewPageEntries(
|
||||||
tenantId,
|
tenantId,
|
||||||
customerId
|
customerId
|
||||||
);
|
);
|
||||||
@@ -405,7 +404,7 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const { paymentReceive, entries } =
|
const { paymentReceive, entries } =
|
||||||
await this.PaymentReceivesPages.getPaymentReceiveEditPage(
|
await this.PaymentsReceivedPages.getPaymentReceiveEditPage(
|
||||||
tenantId,
|
tenantId,
|
||||||
paymentReceiveId,
|
paymentReceiveId,
|
||||||
user
|
user
|
||||||
|
|||||||
@@ -155,7 +155,7 @@ export default class SalesEstimatesController extends BaseController {
|
|||||||
check('estimate_date').exists().isISO8601().toDate(),
|
check('estimate_date').exists().isISO8601().toDate(),
|
||||||
check('expiration_date').exists().isISO8601().toDate(),
|
check('expiration_date').exists().isISO8601().toDate(),
|
||||||
check('reference').optional(),
|
check('reference').optional(),
|
||||||
check('estimate_number').optional().trim().escape(),
|
check('estimate_number').optional().trim(),
|
||||||
check('delivered').default(false).isBoolean().toBoolean(),
|
check('delivered').default(false).isBoolean().toBoolean(),
|
||||||
|
|
||||||
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
||||||
@@ -170,8 +170,7 @@ export default class SalesEstimatesController extends BaseController {
|
|||||||
check('entries.*.rate').exists().isNumeric().toFloat(),
|
check('entries.*.rate').exists().isNumeric().toFloat(),
|
||||||
check('entries.*.description')
|
check('entries.*.description')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('entries.*.discount')
|
check('entries.*.discount')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
@@ -181,9 +180,9 @@ export default class SalesEstimatesController extends BaseController {
|
|||||||
.isNumeric()
|
.isNumeric()
|
||||||
.toInt(),
|
.toInt(),
|
||||||
|
|
||||||
check('note').optional().trim().escape(),
|
check('note').optional().trim(),
|
||||||
check('terms_conditions').optional().trim().escape(),
|
check('terms_conditions').optional().trim(),
|
||||||
check('send_to_email').optional().trim().escape(),
|
check('send_to_email').optional().trim(),
|
||||||
|
|
||||||
check('attachments').isArray().optional(),
|
check('attachments').isArray().optional(),
|
||||||
check('attachments.*.key').exists().isString(),
|
check('attachments.*.key').exists().isString(),
|
||||||
|
|||||||
@@ -200,12 +200,12 @@ export default class SaleInvoicesController extends BaseController {
|
|||||||
check('customer_id').exists().isNumeric().toInt(),
|
check('customer_id').exists().isNumeric().toInt(),
|
||||||
check('invoice_date').exists().isISO8601().toDate(),
|
check('invoice_date').exists().isISO8601().toDate(),
|
||||||
check('due_date').exists().isISO8601().toDate(),
|
check('due_date').exists().isISO8601().toDate(),
|
||||||
check('invoice_no').optional().trim().escape(),
|
check('invoice_no').optional().trim(),
|
||||||
check('reference_no').optional().trim().escape(),
|
check('reference_no').optional().trim(),
|
||||||
check('delivered').default(false).isBoolean().toBoolean(),
|
check('delivered').default(false).isBoolean().toBoolean(),
|
||||||
|
|
||||||
check('invoice_message').optional().trim().escape(),
|
check('invoice_message').optional().trim(),
|
||||||
check('terms_conditions').optional().trim().escape(),
|
check('terms_conditions').optional().trim(),
|
||||||
|
|
||||||
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
||||||
|
|
||||||
@@ -226,12 +226,10 @@ export default class SaleInvoicesController extends BaseController {
|
|||||||
.toFloat(),
|
.toFloat(),
|
||||||
check('entries.*.description')
|
check('entries.*.description')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('entries.*.tax_code')
|
check('entries.*.tax_code')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.trim()
|
.trim()
|
||||||
.escape()
|
|
||||||
.isString(),
|
.isString(),
|
||||||
check('entries.*.tax_rate_id')
|
check('entries.*.tax_rate_id')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
|
|||||||
@@ -130,8 +130,8 @@ export default class SalesReceiptsController extends BaseController {
|
|||||||
|
|
||||||
check('deposit_account_id').exists().isNumeric().toInt(),
|
check('deposit_account_id').exists().isNumeric().toInt(),
|
||||||
check('receipt_date').exists().isISO8601(),
|
check('receipt_date').exists().isISO8601(),
|
||||||
check('receipt_number').optional().trim().escape(),
|
check('receipt_number').optional().trim(),
|
||||||
check('reference_no').optional().trim().escape(),
|
check('reference_no').optional().trim(),
|
||||||
check('closed').default(false).isBoolean().toBoolean(),
|
check('closed').default(false).isBoolean().toBoolean(),
|
||||||
|
|
||||||
check('warehouse_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('warehouse_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
@@ -150,14 +150,13 @@ export default class SalesReceiptsController extends BaseController {
|
|||||||
.toInt(),
|
.toInt(),
|
||||||
check('entries.*.description')
|
check('entries.*.description')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.trim()
|
.trim(),
|
||||||
.escape(),
|
|
||||||
check('entries.*.warehouse_id')
|
check('entries.*.warehouse_id')
|
||||||
.optional({ nullable: true })
|
.optional({ nullable: true })
|
||||||
.isNumeric()
|
.isNumeric()
|
||||||
.toInt(),
|
.toInt(),
|
||||||
check('receipt_message').optional().trim().escape(),
|
check('receipt_message').optional().trim(),
|
||||||
check('statement').optional().trim().escape(),
|
check('statement').optional().trim(),
|
||||||
check('attachments').isArray().optional(),
|
check('attachments').isArray().optional(),
|
||||||
check('attachments.*.key').exists().isString(),
|
check('attachments.*.key').exists().isString(),
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -52,10 +52,7 @@ export default class SettingsController extends BaseController {
|
|||||||
* Retrieve the application options from the storage.
|
* Retrieve the application options from the storage.
|
||||||
*/
|
*/
|
||||||
private get getSettingsSchema() {
|
private get getSettingsSchema() {
|
||||||
return [
|
return [query('key').optional().trim(), query('group').optional().trim()];
|
||||||
query('key').optional().trim().escape(),
|
|
||||||
query('group').optional().trim().escape(),
|
|
||||||
];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -32,7 +32,7 @@ export default class ViewsController extends BaseController {
|
|||||||
* Custom views list validation schema.
|
* Custom views list validation schema.
|
||||||
*/
|
*/
|
||||||
get viewsListSchemaValidation() {
|
get viewsListSchemaValidation() {
|
||||||
return [param('resource_model').exists().trim().escape()];
|
return [param('resource_model').exists().trim()];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -32,7 +32,7 @@ module.exports = {
|
|||||||
*/
|
*/
|
||||||
tenant: {
|
tenant: {
|
||||||
db_client: process.env.TENANT_DB_CLIENT || process.env.DB_CLIENT || 'mysql',
|
db_client: process.env.TENANT_DB_CLIENT || process.env.DB_CLIENT || 'mysql',
|
||||||
db_name_prefix: process.env.TENANT_DB_NAME_PERFIX,
|
db_name_prefix: process.env.TENANT_DB_NAME_PERFIX || 'bigcapital_tenant_',
|
||||||
db_host: process.env.TENANT_DB_HOST || process.env.DB_HOST,
|
db_host: process.env.TENANT_DB_HOST || process.env.DB_HOST,
|
||||||
db_user: process.env.TENANT_DB_USER || process.env.DB_USER,
|
db_user: process.env.TENANT_DB_USER || process.env.DB_USER,
|
||||||
db_password: process.env.TENANT_DB_PASSWORD || process.env.DB_PASSWORD,
|
db_password: process.env.TENANT_DB_PASSWORD || process.env.DB_PASSWORD,
|
||||||
|
|||||||
@@ -0,0 +1,18 @@
|
|||||||
|
// This migration changes the precision of the tax_amount_withheld column in the bills and sales_invoices tables from 8, 2 to 13, 2.
|
||||||
|
// This migration is necessary to allow tax_amount_withheld filed to store values bigger than 999,999.99.
|
||||||
|
|
||||||
|
exports.up = function(knex) {
|
||||||
|
return knex.schema.alterTable('bills', function (table) {
|
||||||
|
table.decimal('tax_amount_withheld', 13, 2).alter();
|
||||||
|
}).alterTable('sales_invoices', function (table) {
|
||||||
|
table.decimal('tax_amount_withheld', 13, 2).alter();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
exports.down = function(knex) {
|
||||||
|
return knex.schema.alterTable('bills', function (table) {
|
||||||
|
table.decimal('tax_amount_withheld', 8, 2).alter();
|
||||||
|
}).alterTable('sales_invoices', function (table) {
|
||||||
|
table.decimal('tax_amount_withheld', 8, 2).alter();
|
||||||
|
});
|
||||||
|
};
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
exports.up = function (knex) {
|
||||||
|
return knex.schema.table('uncategorized_cashflow_transactions', (table) => {
|
||||||
|
table.boolean('pending').defaultTo(false);
|
||||||
|
table.string('pending_plaid_transaction_id').nullable();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
exports.down = function (knex) {
|
||||||
|
return knex.schema.table('uncategorized_cashflow_transactions', (table) => {
|
||||||
|
table.dropColumn('pending');
|
||||||
|
table.dropColumn('pending_plaid_transaction_id');
|
||||||
|
});
|
||||||
|
};
|
||||||
@@ -268,6 +268,8 @@ export interface CreateUncategorizedTransactionDTO {
|
|||||||
description?: string;
|
description?: string;
|
||||||
referenceNo?: string | null;
|
referenceNo?: string | null;
|
||||||
plaidTransactionId?: string | null;
|
plaidTransactionId?: string | null;
|
||||||
|
pending?: boolean;
|
||||||
|
pendingPlaidTransactionId?: string | null;
|
||||||
batch?: string;
|
batch?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -283,3 +285,17 @@ export interface IUncategorizedTransactionCreatedEventPayload {
|
|||||||
createUncategorizedTransactionDTO: CreateUncategorizedTransactionDTO;
|
createUncategorizedTransactionDTO: CreateUncategorizedTransactionDTO;
|
||||||
trx: Knex.Transaction;
|
trx: Knex.Transaction;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface IPendingTransactionRemovingEventPayload {
|
||||||
|
tenantId: number;
|
||||||
|
uncategorizedTransactionId: number;
|
||||||
|
pendingTransaction: IUncategorizedCashflowTransaction;
|
||||||
|
trx?: Knex.Transaction;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface IPendingTransactionRemovedEventPayload {
|
||||||
|
tenantId: number;
|
||||||
|
uncategorizedTransactionId: number;
|
||||||
|
pendingTransaction: IUncategorizedCashflowTransaction;
|
||||||
|
trx?: Knex.Transaction;
|
||||||
|
}
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ export interface IGeneralLedgerSheetAccountTransaction {
|
|||||||
currencyCode: string;
|
currencyCode: string;
|
||||||
note?: string;
|
note?: string;
|
||||||
|
|
||||||
transactionType?: string;
|
transactionTypeFormatted: string;
|
||||||
transactionNumber: string;
|
transactionNumber: string;
|
||||||
|
|
||||||
referenceId?: number;
|
referenceId?: number;
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import { ILedgerEntry } from './Ledger';
|
|||||||
import { ISaleInvoice } from './SaleInvoice';
|
import { ISaleInvoice } from './SaleInvoice';
|
||||||
import { AttachmentLinkDTO } from './Attachments';
|
import { AttachmentLinkDTO } from './Attachments';
|
||||||
|
|
||||||
export interface IPaymentReceive {
|
export interface IPaymentReceived {
|
||||||
id?: number;
|
id?: number;
|
||||||
customerId: number;
|
customerId: number;
|
||||||
paymentDate: Date;
|
paymentDate: Date;
|
||||||
@@ -19,14 +19,14 @@ export interface IPaymentReceive {
|
|||||||
depositAccountId: number;
|
depositAccountId: number;
|
||||||
paymentReceiveNo: string;
|
paymentReceiveNo: string;
|
||||||
statement: string;
|
statement: string;
|
||||||
entries: IPaymentReceiveEntry[];
|
entries: IPaymentReceivedEntry[];
|
||||||
userId: number;
|
userId: number;
|
||||||
createdAt: Date;
|
createdAt: Date;
|
||||||
updatedAt: Date;
|
updatedAt: Date;
|
||||||
localAmount?: number;
|
localAmount?: number;
|
||||||
branchId?: number;
|
branchId?: number;
|
||||||
}
|
}
|
||||||
export interface IPaymentReceiveCreateDTO {
|
export interface IPaymentReceivedCreateDTO {
|
||||||
customerId: number;
|
customerId: number;
|
||||||
paymentDate: Date;
|
paymentDate: Date;
|
||||||
amount: number;
|
amount: number;
|
||||||
@@ -35,13 +35,13 @@ export interface IPaymentReceiveCreateDTO {
|
|||||||
depositAccountId: number;
|
depositAccountId: number;
|
||||||
paymentReceiveNo?: string;
|
paymentReceiveNo?: string;
|
||||||
statement: string;
|
statement: string;
|
||||||
entries: IPaymentReceiveEntryDTO[];
|
entries: IPaymentReceivedEntryDTO[];
|
||||||
|
|
||||||
branchId?: number;
|
branchId?: number;
|
||||||
attachments?: AttachmentLinkDTO[];
|
attachments?: AttachmentLinkDTO[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveEditDTO {
|
export interface IPaymentReceivedEditDTO {
|
||||||
customerId: number;
|
customerId: number;
|
||||||
paymentDate: Date;
|
paymentDate: Date;
|
||||||
amount: number;
|
amount: number;
|
||||||
@@ -50,12 +50,12 @@ export interface IPaymentReceiveEditDTO {
|
|||||||
depositAccountId: number;
|
depositAccountId: number;
|
||||||
paymentReceiveNo?: string;
|
paymentReceiveNo?: string;
|
||||||
statement: string;
|
statement: string;
|
||||||
entries: IPaymentReceiveEntryDTO[];
|
entries: IPaymentReceivedEntryDTO[];
|
||||||
branchId?: number;
|
branchId?: number;
|
||||||
attachments?: AttachmentLinkDTO[];
|
attachments?: AttachmentLinkDTO[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveEntry {
|
export interface IPaymentReceivedEntry {
|
||||||
id?: number;
|
id?: number;
|
||||||
paymentReceiveId: number;
|
paymentReceiveId: number;
|
||||||
invoiceId: number;
|
invoiceId: number;
|
||||||
@@ -64,15 +64,15 @@ export interface IPaymentReceiveEntry {
|
|||||||
invoice?: ISaleInvoice;
|
invoice?: ISaleInvoice;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveEntryDTO {
|
export interface IPaymentReceivedEntryDTO {
|
||||||
id?: number;
|
id?: number;
|
||||||
index: number;
|
index: number;
|
||||||
paymentReceiveId: number;
|
paymentReceiveId?: number;
|
||||||
invoiceId: number;
|
invoiceId: number;
|
||||||
paymentAmount: number;
|
paymentAmount: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceivesFilter extends IDynamicListFilterDTO {
|
export interface IPaymentsReceivedFilter extends IDynamicListFilterDTO {
|
||||||
stringifiedFilterRoles?: string;
|
stringifiedFilterRoles?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -88,65 +88,65 @@ export interface IPaymentReceivePageEntry {
|
|||||||
date: Date | string;
|
date: Date | string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveEditPage {
|
export interface IPaymentReceivedEditPage {
|
||||||
paymentReceive: IPaymentReceive;
|
paymentReceive: IPaymentReceived;
|
||||||
entries: IPaymentReceivePageEntry[];
|
entries: IPaymentReceivePageEntry[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentsReceiveService {
|
export interface IPaymentsReceivedService {
|
||||||
validateCustomerHasNoPayments(
|
validateCustomerHasNoPayments(
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
customerId: number
|
customerId: number
|
||||||
): Promise<void>;
|
): Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveSmsDetails {
|
export interface IPaymentReceivedSmsDetails {
|
||||||
customerName: string;
|
customerName: string;
|
||||||
customerPhoneNumber: string;
|
customerPhoneNumber: string;
|
||||||
smsMessage: string;
|
smsMessage: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveCreatingPayload {
|
export interface IPaymentReceivedCreatingPayload {
|
||||||
tenantId: number;
|
tenantId: number;
|
||||||
paymentReceiveDTO: IPaymentReceiveCreateDTO;
|
paymentReceiveDTO: IPaymentReceivedCreateDTO;
|
||||||
trx: Knex.Transaction;
|
trx: Knex.Transaction;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveCreatedPayload {
|
export interface IPaymentReceivedCreatedPayload {
|
||||||
tenantId: number;
|
tenantId: number;
|
||||||
paymentReceive: IPaymentReceive;
|
paymentReceive: IPaymentReceived;
|
||||||
paymentReceiveId: number;
|
paymentReceiveId: number;
|
||||||
authorizedUser: ISystemUser;
|
authorizedUser: ISystemUser;
|
||||||
paymentReceiveDTO: IPaymentReceiveCreateDTO;
|
paymentReceiveDTO: IPaymentReceivedCreateDTO;
|
||||||
trx: Knex.Transaction;
|
trx: Knex.Transaction;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveEditedPayload {
|
export interface IPaymentReceivedEditedPayload {
|
||||||
tenantId: number;
|
tenantId: number;
|
||||||
paymentReceiveId: number;
|
paymentReceiveId: number;
|
||||||
paymentReceive: IPaymentReceive;
|
paymentReceive: IPaymentReceived;
|
||||||
oldPaymentReceive: IPaymentReceive;
|
oldPaymentReceive: IPaymentReceived;
|
||||||
paymentReceiveDTO: IPaymentReceiveEditDTO;
|
paymentReceiveDTO: IPaymentReceivedEditDTO;
|
||||||
authorizedUser: ISystemUser;
|
authorizedUser: ISystemUser;
|
||||||
trx: Knex.Transaction;
|
trx: Knex.Transaction;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveEditingPayload {
|
export interface IPaymentReceivedEditingPayload {
|
||||||
tenantId: number;
|
tenantId: number;
|
||||||
oldPaymentReceive: IPaymentReceive;
|
oldPaymentReceive: IPaymentReceived;
|
||||||
paymentReceiveDTO: IPaymentReceiveEditDTO;
|
paymentReceiveDTO: IPaymentReceivedEditDTO;
|
||||||
trx: Knex.Transaction;
|
trx: Knex.Transaction;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveDeletingPayload {
|
export interface IPaymentReceivedDeletingPayload {
|
||||||
tenantId: number;
|
tenantId: number;
|
||||||
oldPaymentReceive: IPaymentReceive;
|
oldPaymentReceive: IPaymentReceived;
|
||||||
trx: Knex.Transaction;
|
trx: Knex.Transaction;
|
||||||
}
|
}
|
||||||
export interface IPaymentReceiveDeletedPayload {
|
export interface IPaymentReceivedDeletedPayload {
|
||||||
tenantId: number;
|
tenantId: number;
|
||||||
paymentReceiveId: number;
|
paymentReceiveId: number;
|
||||||
oldPaymentReceive: IPaymentReceive;
|
oldPaymentReceive: IPaymentReceived;
|
||||||
authorizedUser: ISystemUser;
|
authorizedUser: ISystemUser;
|
||||||
trx: Knex.Transaction;
|
trx: Knex.Transaction;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -51,5 +51,4 @@ export interface ISystemService {
|
|||||||
cache();
|
cache();
|
||||||
repositories();
|
repositories();
|
||||||
knex();
|
knex();
|
||||||
dbManager();
|
|
||||||
}
|
}
|
||||||
@@ -21,7 +21,7 @@ export default class ComputeItemCostJob {
|
|||||||
|
|
||||||
agenda.define(
|
agenda.define(
|
||||||
'compute-item-cost',
|
'compute-item-cost',
|
||||||
{ priority: 'high', concurrency: 1 },
|
{ priority: 'high', concurrency: 20 },
|
||||||
this.handler.bind(this)
|
this.handler.bind(this)
|
||||||
);
|
);
|
||||||
this.agenda.on('start:compute-item-cost', this.onJobStart.bind(this));
|
this.agenda.on('start:compute-item-cost', this.onJobStart.bind(this));
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ export default class OrganizationSetupJob {
|
|||||||
constructor(agenda) {
|
constructor(agenda) {
|
||||||
agenda.define(
|
agenda.define(
|
||||||
'organization-setup',
|
'organization-setup',
|
||||||
{ priority: 'high', concurrency: 1 },
|
{ priority: 'high', concurrency: 20 },
|
||||||
this.handler
|
this.handler
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ export default class WriteInvoicesJournalEntries {
|
|||||||
|
|
||||||
agenda.define(
|
agenda.define(
|
||||||
eventName,
|
eventName,
|
||||||
{ priority: 'normal', concurrency: 1 },
|
{ priority: 'normal', concurrency: 20 },
|
||||||
this.handler.bind(this)
|
this.handler.bind(this)
|
||||||
);
|
);
|
||||||
agenda.on(`complete:${eventName}`, this.onJobCompleted.bind(this));
|
agenda.on(`complete:${eventName}`, this.onJobCompleted.bind(this));
|
||||||
|
|||||||
@@ -1,7 +0,0 @@
|
|||||||
import knexManager from 'knex-db-manager';
|
|
||||||
import { systemKnexConfig, systemDbManager } from 'config/knexConfig';
|
|
||||||
|
|
||||||
export default () => knexManager.databaseManagerFactory({
|
|
||||||
knex: systemKnexConfig,
|
|
||||||
dbManager: systemDbManager,
|
|
||||||
});
|
|
||||||
@@ -3,7 +3,6 @@ import LoggerInstance from '@/loaders/logger';
|
|||||||
import agendaFactory from '@/loaders/agenda';
|
import agendaFactory from '@/loaders/agenda';
|
||||||
import SmsClientLoader from '@/loaders/smsClient';
|
import SmsClientLoader from '@/loaders/smsClient';
|
||||||
import mailInstance from '@/loaders/mail';
|
import mailInstance from '@/loaders/mail';
|
||||||
import dbManagerFactory from '@/loaders/dbManager';
|
|
||||||
import i18n from '@/loaders/i18n';
|
import i18n from '@/loaders/i18n';
|
||||||
import repositoriesLoader from '@/loaders/systemRepositories';
|
import repositoriesLoader from '@/loaders/systemRepositories';
|
||||||
import Cache from '@/services/Cache';
|
import Cache from '@/services/Cache';
|
||||||
@@ -16,7 +15,6 @@ export default ({ mongoConnection, knex }) => {
|
|||||||
try {
|
try {
|
||||||
const agendaInstance = agendaFactory({ mongoConnection });
|
const agendaInstance = agendaFactory({ mongoConnection });
|
||||||
const smsClientInstance = SmsClientLoader(config.easySMSGateway.api_key);
|
const smsClientInstance = SmsClientLoader(config.easySMSGateway.api_key);
|
||||||
const dbManager = dbManagerFactory(knex);
|
|
||||||
const cacheInstance = new Cache();
|
const cacheInstance = new Cache();
|
||||||
|
|
||||||
Container.set('logger', LoggerInstance);
|
Container.set('logger', LoggerInstance);
|
||||||
@@ -24,7 +22,6 @@ export default ({ mongoConnection, knex }) => {
|
|||||||
Container.set('SMSClient', smsClientInstance);
|
Container.set('SMSClient', smsClientInstance);
|
||||||
Container.set('mail', mailInstance);
|
Container.set('mail', mailInstance);
|
||||||
|
|
||||||
Container.set('dbManager', dbManager);
|
|
||||||
LoggerInstance.info(
|
LoggerInstance.info(
|
||||||
'[DI] Database manager has been injected into container.'
|
'[DI] Database manager has been injected into container.'
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -115,6 +115,7 @@ import { DecrementUncategorizedTransactionOnExclude } from '@/services/Banking/E
|
|||||||
import { DecrementUncategorizedTransactionOnCategorize } from '@/services/Cashflow/subscribers/DecrementUncategorizedTransactionOnCategorize';
|
import { DecrementUncategorizedTransactionOnCategorize } from '@/services/Cashflow/subscribers/DecrementUncategorizedTransactionOnCategorize';
|
||||||
import { DisconnectPlaidItemOnAccountDeleted } from '@/services/Banking/BankAccounts/events/DisconnectPlaidItemOnAccountDeleted';
|
import { DisconnectPlaidItemOnAccountDeleted } from '@/services/Banking/BankAccounts/events/DisconnectPlaidItemOnAccountDeleted';
|
||||||
import { LoopsEventsSubscriber } from '@/services/Loops/LoopsEventsSubscriber';
|
import { LoopsEventsSubscriber } from '@/services/Loops/LoopsEventsSubscriber';
|
||||||
|
import { DeleteUncategorizedTransactionsOnAccountDeleting } from '@/services/Banking/BankAccounts/events/DeleteUncategorizedTransactionsOnAccountDeleting';
|
||||||
|
|
||||||
export default () => {
|
export default () => {
|
||||||
return new EventPublisher();
|
return new EventPublisher();
|
||||||
@@ -277,6 +278,7 @@ export const susbcribers = () => {
|
|||||||
// Plaid
|
// Plaid
|
||||||
RecognizeSyncedBankTranasctions,
|
RecognizeSyncedBankTranasctions,
|
||||||
DisconnectPlaidItemOnAccountDeleted,
|
DisconnectPlaidItemOnAccountDeleted,
|
||||||
|
DeleteUncategorizedTransactionsOnAccountDeleting,
|
||||||
|
|
||||||
// Loops
|
// Loops
|
||||||
LoopsEventsSubscriber
|
LoopsEventsSubscriber
|
||||||
|
|||||||
@@ -34,4 +34,4 @@
|
|||||||
// import 'services/Sales/SaleInvoiceWriteoffSubscriber';
|
// import 'services/Sales/SaleInvoiceWriteoffSubscriber';
|
||||||
// import 'subscribers/SaleInvoices/SendSmsNotificationToCustomer';
|
// import 'subscribers/SaleInvoices/SendSmsNotificationToCustomer';
|
||||||
// import 'subscribers/SaleReceipt/SendNotificationToCustomer';
|
// import 'subscribers/SaleReceipt/SendNotificationToCustomer';
|
||||||
// import 'services/Sales/PaymentReceives/PaymentReceiveSmsSubscriber';
|
// import 'services/Sales/PaymentReceived/PaymentReceiveSmsSubscriber';
|
||||||
@@ -9,7 +9,7 @@ import { SendSaleInvoiceMailJob } from '@/services/Sales/Invoices/SendSaleInvoic
|
|||||||
import { SendSaleInvoiceReminderMailJob } from '@/services/Sales/Invoices/SendSaleInvoiceMailReminderJob';
|
import { SendSaleInvoiceReminderMailJob } from '@/services/Sales/Invoices/SendSaleInvoiceMailReminderJob';
|
||||||
import { SendSaleEstimateMailJob } from '@/services/Sales/Estimates/SendSaleEstimateMailJob';
|
import { SendSaleEstimateMailJob } from '@/services/Sales/Estimates/SendSaleEstimateMailJob';
|
||||||
import { SaleReceiptMailNotificationJob } from '@/services/Sales/Receipts/SaleReceiptMailNotificationJob';
|
import { SaleReceiptMailNotificationJob } from '@/services/Sales/Receipts/SaleReceiptMailNotificationJob';
|
||||||
import { PaymentReceiveMailNotificationJob } from '@/services/Sales/PaymentReceives/PaymentReceiveMailNotificationJob';
|
import { PaymentReceivedMailNotificationJob } from '@/services/Sales/PaymentReceived/PaymentReceivedMailNotificationJob';
|
||||||
import { PlaidFetchTransactionsJob } from '@/services/Banking/Plaid/PlaidFetchTransactionsJob';
|
import { PlaidFetchTransactionsJob } from '@/services/Banking/Plaid/PlaidFetchTransactionsJob';
|
||||||
import { ImportDeleteExpiredFilesJobs } from '@/services/Import/jobs/ImportDeleteExpiredFilesJob';
|
import { ImportDeleteExpiredFilesJobs } from '@/services/Import/jobs/ImportDeleteExpiredFilesJob';
|
||||||
import { SendVerifyMailJob } from '@/services/Authentication/jobs/SendVerifyMailJob';
|
import { SendVerifyMailJob } from '@/services/Authentication/jobs/SendVerifyMailJob';
|
||||||
@@ -28,7 +28,7 @@ export default ({ agenda }: { agenda: Agenda }) => {
|
|||||||
new SendSaleInvoiceReminderMailJob(agenda);
|
new SendSaleInvoiceReminderMailJob(agenda);
|
||||||
new SendSaleEstimateMailJob(agenda);
|
new SendSaleEstimateMailJob(agenda);
|
||||||
new SaleReceiptMailNotificationJob(agenda);
|
new SaleReceiptMailNotificationJob(agenda);
|
||||||
new PaymentReceiveMailNotificationJob(agenda);
|
new PaymentReceivedMailNotificationJob(agenda);
|
||||||
new PlaidFetchTransactionsJob(agenda);
|
new PlaidFetchTransactionsJob(agenda);
|
||||||
new ImportDeleteExpiredFilesJobs(agenda);
|
new ImportDeleteExpiredFilesJobs(agenda);
|
||||||
new SendVerifyMailJob(agenda);
|
new SendVerifyMailJob(agenda);
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import TenantModel from 'models/TenantModel';
|
|||||||
import ModelSetting from './ModelSetting';
|
import ModelSetting from './ModelSetting';
|
||||||
import BillPaymentSettings from './BillPayment.Settings';
|
import BillPaymentSettings from './BillPayment.Settings';
|
||||||
import CustomViewBaseModel from './CustomViewBaseModel';
|
import CustomViewBaseModel from './CustomViewBaseModel';
|
||||||
import { DEFAULT_VIEWS } from '@/services/Sales/PaymentReceives/constants';
|
import { DEFAULT_VIEWS } from '@/services/Sales/PaymentReceived/constants';
|
||||||
import ModelSearchable from './ModelSearchable';
|
import ModelSearchable from './ModelSearchable';
|
||||||
|
|
||||||
export default class BillPayment extends mixin(TenantModel, [
|
export default class BillPayment extends mixin(TenantModel, [
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { Model } from 'objection';
|
import { Model } from 'objection';
|
||||||
|
import { castArray, omit, pick } from 'lodash';
|
||||||
import { isEmpty } from 'lodash';
|
import { isEmpty } from 'lodash';
|
||||||
import { ServiceError } from '@/exceptions';
|
import { ServiceError } from '@/exceptions';
|
||||||
|
|
||||||
@@ -16,7 +17,15 @@ export default class PaginationQueryBuilder extends Model.QueryBuilder {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
queryAndThrowIfHasRelations = ({ type, message }) => {
|
queryAndThrowIfHasRelations = ({
|
||||||
|
type,
|
||||||
|
message,
|
||||||
|
excludeRelations = [],
|
||||||
|
includedRelations = [],
|
||||||
|
}) => {
|
||||||
|
const _excludeRelations = castArray(excludeRelations);
|
||||||
|
const _includedRelations = castArray(includedRelations);
|
||||||
|
|
||||||
const model = this.modelClass();
|
const model = this.modelClass();
|
||||||
const modelRelations = Object.keys(model.relationMappings).filter(
|
const modelRelations = Object.keys(model.relationMappings).filter(
|
||||||
(relation) =>
|
(relation) =>
|
||||||
@@ -25,9 +34,20 @@ export default class PaginationQueryBuilder extends Model.QueryBuilder {
|
|||||||
) !== -1
|
) !== -1
|
||||||
);
|
);
|
||||||
const relations = model.secureDeleteRelations || modelRelations;
|
const relations = model.secureDeleteRelations || modelRelations;
|
||||||
|
const filteredByIncluded = relations.filter((r) =>
|
||||||
|
_includedRelations.includes(r)
|
||||||
|
);
|
||||||
|
const filteredByExcluded = relations.filter(
|
||||||
|
(r) => !excludeRelations.includes(r)
|
||||||
|
);
|
||||||
|
const filteredRelations = !isEmpty(_includedRelations)
|
||||||
|
? filteredByIncluded
|
||||||
|
: !isEmpty(_excludeRelations)
|
||||||
|
? filteredByExcluded
|
||||||
|
: relations;
|
||||||
|
|
||||||
this.runAfter((model, query) => {
|
this.runAfter((model, query) => {
|
||||||
const nonEmptyRelations = relations.filter(
|
const nonEmptyRelations = filteredRelations.filter(
|
||||||
(relation) => !isEmpty(model[relation])
|
(relation) => !isEmpty(model[relation])
|
||||||
);
|
);
|
||||||
if (nonEmptyRelations.length > 0) {
|
if (nonEmptyRelations.length > 0) {
|
||||||
@@ -36,7 +56,7 @@ export default class PaginationQueryBuilder extends Model.QueryBuilder {
|
|||||||
return model;
|
return model;
|
||||||
});
|
});
|
||||||
return this.onBuild((query) => {
|
return this.onBuild((query) => {
|
||||||
relations.forEach((relation) => {
|
filteredRelations.forEach((relation) => {
|
||||||
query.withGraphFetched(`${relation}(selectId)`).modifiers({
|
query.withGraphFetched(`${relation}(selectId)`).modifiers({
|
||||||
selectId(builder) {
|
selectId(builder) {
|
||||||
builder.select('id');
|
builder.select('id');
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import TenantModel from 'models/TenantModel';
|
|||||||
import ModelSetting from './ModelSetting';
|
import ModelSetting from './ModelSetting';
|
||||||
import PaymentReceiveSettings from './PaymentReceive.Settings';
|
import PaymentReceiveSettings from './PaymentReceive.Settings';
|
||||||
import CustomViewBaseModel from './CustomViewBaseModel';
|
import CustomViewBaseModel from './CustomViewBaseModel';
|
||||||
import { DEFAULT_VIEWS } from '@/services/Sales/PaymentReceives/constants';
|
import { DEFAULT_VIEWS } from '@/services/Sales/PaymentReceived/constants';
|
||||||
import ModelSearchable from './ModelSearchable';
|
import ModelSearchable from './ModelSearchable';
|
||||||
|
|
||||||
export default class PaymentReceive extends mixin(TenantModel, [
|
export default class PaymentReceive extends mixin(TenantModel, [
|
||||||
|
|||||||
69
packages/server/src/models/TaxRate.settings.ts
Normal file
69
packages/server/src/models/TaxRate.settings.ts
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
export default {
|
||||||
|
defaultSort: {
|
||||||
|
sortOrder: 'DESC',
|
||||||
|
sortField: 'created_at',
|
||||||
|
},
|
||||||
|
exportable: true,
|
||||||
|
importable: true,
|
||||||
|
print: {
|
||||||
|
pageTitle: 'Tax Rates',
|
||||||
|
},
|
||||||
|
columns: {
|
||||||
|
name: {
|
||||||
|
name: 'Tax Rate Name',
|
||||||
|
type: 'text',
|
||||||
|
accessor: 'name',
|
||||||
|
},
|
||||||
|
code: {
|
||||||
|
name: 'Code',
|
||||||
|
type: 'text',
|
||||||
|
accessor: 'code',
|
||||||
|
},
|
||||||
|
rate: {
|
||||||
|
name: 'Rate',
|
||||||
|
type: 'text',
|
||||||
|
},
|
||||||
|
description: {
|
||||||
|
name: 'Description',
|
||||||
|
type: 'text',
|
||||||
|
},
|
||||||
|
isNonRecoverable: {
|
||||||
|
name: 'Is Non Recoverable',
|
||||||
|
type: 'boolean',
|
||||||
|
},
|
||||||
|
active: {
|
||||||
|
name: 'Active',
|
||||||
|
type: 'boolean',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
field: {},
|
||||||
|
fields2: {
|
||||||
|
name: {
|
||||||
|
name: 'Tax name',
|
||||||
|
fieldType: 'name',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
code: {
|
||||||
|
name: 'Code',
|
||||||
|
fieldType: 'code',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
rate: {
|
||||||
|
name: 'Rate',
|
||||||
|
fieldType: 'number',
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
description: {
|
||||||
|
name: 'Description',
|
||||||
|
fieldType: 'text',
|
||||||
|
},
|
||||||
|
isNonRecoverable: {
|
||||||
|
name: 'Is Non Recoverable',
|
||||||
|
fieldType: 'boolean',
|
||||||
|
},
|
||||||
|
active: {
|
||||||
|
name: 'Active',
|
||||||
|
fieldType: 'boolean',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
@@ -2,8 +2,13 @@ import { mixin, Model, raw } from 'objection';
|
|||||||
import TenantModel from 'models/TenantModel';
|
import TenantModel from 'models/TenantModel';
|
||||||
import ModelSearchable from './ModelSearchable';
|
import ModelSearchable from './ModelSearchable';
|
||||||
import SoftDeleteQueryBuilder from '@/collection/SoftDeleteQueryBuilder';
|
import SoftDeleteQueryBuilder from '@/collection/SoftDeleteQueryBuilder';
|
||||||
|
import TaxRateMeta from './TaxRate.settings';
|
||||||
|
import ModelSetting from './ModelSetting';
|
||||||
|
|
||||||
export default class TaxRate extends mixin(TenantModel, [ModelSearchable]) {
|
export default class TaxRate extends mixin(TenantModel, [
|
||||||
|
ModelSetting,
|
||||||
|
ModelSearchable,
|
||||||
|
]) {
|
||||||
/**
|
/**
|
||||||
* Table name
|
* Table name
|
||||||
*/
|
*/
|
||||||
@@ -25,6 +30,13 @@ export default class TaxRate extends mixin(TenantModel, [ModelSearchable]) {
|
|||||||
return ['createdAt', 'updatedAt'];
|
return ['createdAt', 'updatedAt'];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the tax rate meta.
|
||||||
|
*/
|
||||||
|
static get meta() {
|
||||||
|
return TaxRateMeta;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Virtual attributes.
|
* Virtual attributes.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
/* eslint-disable global-require */
|
/* eslint-disable global-require */
|
||||||
import * as R from 'ramda';
|
import { Model, mixin } from 'objection';
|
||||||
import { Model, ModelOptions, QueryContext, mixin } from 'objection';
|
|
||||||
import TenantModel from 'models/TenantModel';
|
import TenantModel from 'models/TenantModel';
|
||||||
import ModelSettings from './ModelSetting';
|
import ModelSettings from './ModelSetting';
|
||||||
import Account from './Account';
|
import Account from './Account';
|
||||||
@@ -21,6 +20,7 @@ export default class UncategorizedCashflowTransaction extends mixin(
|
|||||||
plaidTransactionId!: string;
|
plaidTransactionId!: string;
|
||||||
recognizedTransactionId!: number;
|
recognizedTransactionId!: number;
|
||||||
excludedAt: Date;
|
excludedAt: Date;
|
||||||
|
pending: boolean;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Table name.
|
* Table name.
|
||||||
@@ -46,7 +46,8 @@ export default class UncategorizedCashflowTransaction extends mixin(
|
|||||||
'isDepositTransaction',
|
'isDepositTransaction',
|
||||||
'isWithdrawalTransaction',
|
'isWithdrawalTransaction',
|
||||||
'isRecognized',
|
'isRecognized',
|
||||||
'isExcluded'
|
'isExcluded',
|
||||||
|
'isPending',
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -99,6 +100,14 @@ export default class UncategorizedCashflowTransaction extends mixin(
|
|||||||
return !!this.excludedAt;
|
return !!this.excludedAt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detarmines whether the transaction is pending.
|
||||||
|
* @returns {boolean}
|
||||||
|
*/
|
||||||
|
public get isPending(): boolean {
|
||||||
|
return !!this.pending;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Model modifiers.
|
* Model modifiers.
|
||||||
*/
|
*/
|
||||||
@@ -143,6 +152,20 @@ export default class UncategorizedCashflowTransaction extends mixin(
|
|||||||
query.whereNull('categorizeRefType');
|
query.whereNull('categorizeRefType');
|
||||||
query.whereNull('categorizeRefId');
|
query.whereNull('categorizeRefId');
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filters the not pending transactions.
|
||||||
|
*/
|
||||||
|
notPending(query) {
|
||||||
|
query.where('pending', false);
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filters the pending transactions.
|
||||||
|
*/
|
||||||
|
pending(query) {
|
||||||
|
query.where('pending', true);
|
||||||
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -249,6 +249,7 @@ export default class Ledger implements ILedger {
|
|||||||
|
|
||||||
transactionId: entry.referenceId,
|
transactionId: entry.referenceId,
|
||||||
transactionType: entry.referenceType,
|
transactionType: entry.referenceType,
|
||||||
|
transactionSubType: entry.transactionType,
|
||||||
|
|
||||||
transactionNumber: entry.transactionNumber,
|
transactionNumber: entry.transactionNumber,
|
||||||
referenceNumber: entry.referenceNumber,
|
referenceNumber: entry.referenceNumber,
|
||||||
@@ -262,6 +263,8 @@ export default class Ledger implements ILedger {
|
|||||||
|
|
||||||
taxRateId: entry.taxRateId,
|
taxRateId: entry.taxRateId,
|
||||||
taxRate: entry.taxRate,
|
taxRate: entry.taxRate,
|
||||||
|
|
||||||
|
note: entry.note,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -73,6 +73,7 @@ export class DeleteAccount {
|
|||||||
.throwIfNotFound()
|
.throwIfNotFound()
|
||||||
.queryAndThrowIfHasRelations({
|
.queryAndThrowIfHasRelations({
|
||||||
type: ERRORS.ACCOUNT_HAS_ASSOCIATED_TRANSACTIONS,
|
type: ERRORS.ACCOUNT_HAS_ASSOCIATED_TRANSACTIONS,
|
||||||
|
excludeRelations: ['uncategorizedTransactions', 'plaidItem']
|
||||||
});
|
});
|
||||||
// Authorize before delete account.
|
// Authorize before delete account.
|
||||||
await this.authorize(tenantId, accountId, oldAccount);
|
await this.authorize(tenantId, accountId, oldAccount);
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import { isEmpty } from 'lodash';
|
import { isEmpty } from 'lodash';
|
||||||
import {
|
import {
|
||||||
IPaymentReceiveCreatedPayload,
|
IPaymentReceivedCreatedPayload,
|
||||||
IPaymentReceiveCreatingPayload,
|
IPaymentReceivedCreatingPayload,
|
||||||
IPaymentReceiveDeletingPayload,
|
IPaymentReceivedDeletingPayload,
|
||||||
IPaymentReceiveEditedPayload,
|
IPaymentReceivedEditedPayload,
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
import { LinkAttachment } from '../LinkAttachment';
|
import { LinkAttachment } from '../LinkAttachment';
|
||||||
@@ -50,13 +50,13 @@ export class AttachmentsOnPaymentsReceived {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Validates the attachment keys on creating payment.
|
* Validates the attachment keys on creating payment.
|
||||||
* @param {IPaymentReceiveCreatingPayload}
|
* @param {IPaymentReceivedCreatingPayload}
|
||||||
* @returns {Promise<void>}
|
* @returns {Promise<void>}
|
||||||
*/
|
*/
|
||||||
private async validateAttachmentsOnPaymentCreate({
|
private async validateAttachmentsOnPaymentCreate({
|
||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
tenantId,
|
tenantId,
|
||||||
}: IPaymentReceiveCreatingPayload): Promise<void> {
|
}: IPaymentReceivedCreatingPayload): Promise<void> {
|
||||||
if (isEmpty(paymentReceiveDTO.attachments)) {
|
if (isEmpty(paymentReceiveDTO.attachments)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -67,7 +67,7 @@ export class AttachmentsOnPaymentsReceived {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Handles linking the attachments of the created payment.
|
* Handles linking the attachments of the created payment.
|
||||||
* @param {IPaymentReceiveCreatedPayload}
|
* @param {IPaymentReceivedCreatedPayload}
|
||||||
* @returns {Promise<void>}
|
* @returns {Promise<void>}
|
||||||
*/
|
*/
|
||||||
private async handleAttachmentsOnPaymentCreated({
|
private async handleAttachmentsOnPaymentCreated({
|
||||||
@@ -75,7 +75,7 @@ export class AttachmentsOnPaymentsReceived {
|
|||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
paymentReceive,
|
paymentReceive,
|
||||||
trx,
|
trx,
|
||||||
}: IPaymentReceiveCreatedPayload): Promise<void> {
|
}: IPaymentReceivedCreatedPayload): Promise<void> {
|
||||||
if (isEmpty(paymentReceiveDTO.attachments)) return;
|
if (isEmpty(paymentReceiveDTO.attachments)) return;
|
||||||
|
|
||||||
const keys = paymentReceiveDTO.attachments?.map(
|
const keys = paymentReceiveDTO.attachments?.map(
|
||||||
@@ -92,14 +92,14 @@ export class AttachmentsOnPaymentsReceived {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Handles unlinking all the unpresented keys of the edited payment.
|
* Handles unlinking all the unpresented keys of the edited payment.
|
||||||
* @param {IPaymentReceiveEditedPayload}
|
* @param {IPaymentReceivedEditedPayload}
|
||||||
*/
|
*/
|
||||||
private async handleUnlinkUnpresentedKeysOnPaymentEdited({
|
private async handleUnlinkUnpresentedKeysOnPaymentEdited({
|
||||||
tenantId,
|
tenantId,
|
||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
oldPaymentReceive,
|
oldPaymentReceive,
|
||||||
trx,
|
trx,
|
||||||
}: IPaymentReceiveEditedPayload) {
|
}: IPaymentReceivedEditedPayload) {
|
||||||
const keys = paymentReceiveDTO.attachments?.map(
|
const keys = paymentReceiveDTO.attachments?.map(
|
||||||
(attachment) => attachment.key
|
(attachment) => attachment.key
|
||||||
);
|
);
|
||||||
@@ -114,7 +114,7 @@ export class AttachmentsOnPaymentsReceived {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Handles linking all the presented keys of the edited payment.
|
* Handles linking all the presented keys of the edited payment.
|
||||||
* @param {IPaymentReceiveEditedPayload}
|
* @param {IPaymentReceivedEditedPayload}
|
||||||
* @returns {Promise<void>}
|
* @returns {Promise<void>}
|
||||||
*/
|
*/
|
||||||
private async handleLinkPresentedKeysOnPaymentEdited({
|
private async handleLinkPresentedKeysOnPaymentEdited({
|
||||||
@@ -122,7 +122,7 @@ export class AttachmentsOnPaymentsReceived {
|
|||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
oldPaymentReceive,
|
oldPaymentReceive,
|
||||||
trx,
|
trx,
|
||||||
}: IPaymentReceiveEditedPayload) {
|
}: IPaymentReceivedEditedPayload) {
|
||||||
if (isEmpty(paymentReceiveDTO.attachments)) return;
|
if (isEmpty(paymentReceiveDTO.attachments)) return;
|
||||||
|
|
||||||
const keys = paymentReceiveDTO.attachments?.map(
|
const keys = paymentReceiveDTO.attachments?.map(
|
||||||
@@ -146,7 +146,7 @@ export class AttachmentsOnPaymentsReceived {
|
|||||||
tenantId,
|
tenantId,
|
||||||
oldPaymentReceive,
|
oldPaymentReceive,
|
||||||
trx,
|
trx,
|
||||||
}: IPaymentReceiveDeletingPayload) {
|
}: IPaymentReceivedDeletingPayload) {
|
||||||
await this.unlinkAttachmentService.unlinkAllModelKeys(
|
await this.unlinkAttachmentService.unlinkAllModelKeys(
|
||||||
tenantId,
|
tenantId,
|
||||||
'PaymentReceive',
|
'PaymentReceive',
|
||||||
|
|||||||
@@ -52,6 +52,9 @@ export class GetBankAccountSummary {
|
|||||||
q.withGraphJoined('matchedBankTransactions');
|
q.withGraphJoined('matchedBankTransactions');
|
||||||
q.whereNull('matchedBankTransactions.id');
|
q.whereNull('matchedBankTransactions.id');
|
||||||
|
|
||||||
|
// Exclude the pending transactions.
|
||||||
|
q.modify('notPending');
|
||||||
|
|
||||||
// Count the results.
|
// Count the results.
|
||||||
q.count('uncategorized_cashflow_transactions.id as total');
|
q.count('uncategorized_cashflow_transactions.id as total');
|
||||||
q.first();
|
q.first();
|
||||||
@@ -65,16 +68,32 @@ export class GetBankAccountSummary {
|
|||||||
q.withGraphJoined('recognizedTransaction');
|
q.withGraphJoined('recognizedTransaction');
|
||||||
q.whereNotNull('recognizedTransaction.id');
|
q.whereNotNull('recognizedTransaction.id');
|
||||||
|
|
||||||
|
// Exclude the pending transactions.
|
||||||
|
q.modify('notPending');
|
||||||
|
|
||||||
// Count the results.
|
// Count the results.
|
||||||
q.count('uncategorized_cashflow_transactions.id as total');
|
q.count('uncategorized_cashflow_transactions.id as total');
|
||||||
q.first();
|
q.first();
|
||||||
});
|
});
|
||||||
|
// Retrieves excluded transactions count.
|
||||||
const excludedTransactionsCount =
|
const excludedTransactionsCount =
|
||||||
await UncategorizedCashflowTransaction.query().onBuild((q) => {
|
await UncategorizedCashflowTransaction.query().onBuild((q) => {
|
||||||
q.where('accountId', bankAccountId);
|
q.where('accountId', bankAccountId);
|
||||||
q.modify('excluded');
|
q.modify('excluded');
|
||||||
|
|
||||||
|
// Exclude the pending transactions.
|
||||||
|
q.modify('notPending');
|
||||||
|
|
||||||
|
// Count the results.
|
||||||
|
q.count('uncategorized_cashflow_transactions.id as total');
|
||||||
|
q.first();
|
||||||
|
});
|
||||||
|
// Retrieves the pending transactions count.
|
||||||
|
const pendingTransactionsCount =
|
||||||
|
await UncategorizedCashflowTransaction.query().onBuild((q) => {
|
||||||
|
q.where('accountId', bankAccountId);
|
||||||
|
q.modify('pending');
|
||||||
|
|
||||||
// Count the results.
|
// Count the results.
|
||||||
q.count('uncategorized_cashflow_transactions.id as total');
|
q.count('uncategorized_cashflow_transactions.id as total');
|
||||||
q.first();
|
q.first();
|
||||||
@@ -83,14 +102,15 @@ export class GetBankAccountSummary {
|
|||||||
const totalUncategorizedTransactions =
|
const totalUncategorizedTransactions =
|
||||||
uncategorizedTranasctionsCount?.total || 0;
|
uncategorizedTranasctionsCount?.total || 0;
|
||||||
const totalRecognizedTransactions = recognizedTransactionsCount?.total || 0;
|
const totalRecognizedTransactions = recognizedTransactionsCount?.total || 0;
|
||||||
|
|
||||||
const totalExcludedTransactions = excludedTransactionsCount?.total || 0;
|
const totalExcludedTransactions = excludedTransactionsCount?.total || 0;
|
||||||
|
const totalPendingTransactions = pendingTransactionsCount?.total || 0;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
name: bankAccount.name,
|
name: bankAccount.name,
|
||||||
totalUncategorizedTransactions,
|
totalUncategorizedTransactions,
|
||||||
totalRecognizedTransactions,
|
totalRecognizedTransactions,
|
||||||
totalExcludedTransactions,
|
totalExcludedTransactions,
|
||||||
|
totalPendingTransactions,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,78 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { initialize } from 'objection';
|
||||||
|
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
||||||
|
import events from '@/subscribers/events';
|
||||||
|
import { IAccountEventDeletePayload } from '@/interfaces';
|
||||||
|
import { DeleteBankRulesService } from '../../Rules/DeleteBankRules';
|
||||||
|
import { RevertRecognizedTransactions } from '../../RegonizeTranasctions/RevertRecognizedTransactions';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class DeleteUncategorizedTransactionsOnAccountDeleting {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private deleteBankRules: DeleteBankRulesService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private revertRecognizedTransactins: RevertRecognizedTransactions;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor method.
|
||||||
|
*/
|
||||||
|
public attach(bus) {
|
||||||
|
bus.subscribe(
|
||||||
|
events.accounts.onDelete,
|
||||||
|
this.handleDeleteBankRulesOnAccountDeleting.bind(this)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handles revert the recognized transactions and delete all the bank rules
|
||||||
|
* associated to the deleted bank account.
|
||||||
|
* @param {IAccountEventDeletePayload}
|
||||||
|
*/
|
||||||
|
private async handleDeleteBankRulesOnAccountDeleting({
|
||||||
|
tenantId,
|
||||||
|
oldAccount,
|
||||||
|
trx,
|
||||||
|
}: IAccountEventDeletePayload) {
|
||||||
|
const knex = this.tenancy.knex(tenantId);
|
||||||
|
const {
|
||||||
|
BankRule,
|
||||||
|
UncategorizedCashflowTransaction,
|
||||||
|
MatchedBankTransaction,
|
||||||
|
RecognizedBankTransaction,
|
||||||
|
} = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
const foundAssociatedRules = await BankRule.query(trx).where(
|
||||||
|
'applyIfAccountId',
|
||||||
|
oldAccount.id
|
||||||
|
);
|
||||||
|
const foundAssociatedRulesIds = foundAssociatedRules.map((rule) => rule.id);
|
||||||
|
|
||||||
|
await initialize(knex, [
|
||||||
|
UncategorizedCashflowTransaction,
|
||||||
|
RecognizedBankTransaction,
|
||||||
|
MatchedBankTransaction,
|
||||||
|
]);
|
||||||
|
// Revert the recognized transactions of the given bank rules.
|
||||||
|
await this.revertRecognizedTransactins.revertRecognizedTransactions(
|
||||||
|
tenantId,
|
||||||
|
foundAssociatedRulesIds,
|
||||||
|
null,
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
// Delete the associated uncategorized transactions.
|
||||||
|
await UncategorizedCashflowTransaction.query(trx)
|
||||||
|
.where('accountId', oldAccount.id)
|
||||||
|
.delete();
|
||||||
|
|
||||||
|
// Delete the given bank rules.
|
||||||
|
await this.deleteBankRules.deleteBankRules(
|
||||||
|
tenantId,
|
||||||
|
foundAssociatedRulesIds,
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -51,6 +51,7 @@ export class DisconnectPlaidItemOnAccountDeleted {
|
|||||||
.findOne('plaidItemId', oldAccount.plaidItemId)
|
.findOne('plaidItemId', oldAccount.plaidItemId)
|
||||||
.delete();
|
.delete();
|
||||||
|
|
||||||
|
// Remove Plaid item once the transaction resolve.
|
||||||
if (oldPlaidItem) {
|
if (oldPlaidItem) {
|
||||||
const plaidInstance = PlaidClientWrapper.getClient();
|
const plaidInstance = PlaidClientWrapper.getClient();
|
||||||
|
|
||||||
|
|||||||
@@ -64,7 +64,7 @@ export class GetMatchedTransactions {
|
|||||||
.whereIn('id', uncategorizedTransactionIds)
|
.whereIn('id', uncategorizedTransactionIds)
|
||||||
.throwIfNotFound();
|
.throwIfNotFound();
|
||||||
|
|
||||||
const totalPending = Math.abs(sumBy(uncategorizedTransactions, 'amount'));
|
const totalPending = sumBy(uncategorizedTransactions, 'amount');
|
||||||
|
|
||||||
const filtered = filter.transactionType
|
const filtered = filter.transactionType
|
||||||
? this.registered.filter((item) => item.type === filter.transactionType)
|
? this.registered.filter((item) => item.type === filter.transactionType)
|
||||||
|
|||||||
@@ -1,18 +1,25 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import { initialize } from 'objection';
|
import { initialize } from 'objection';
|
||||||
|
import { Knex } from 'knex';
|
||||||
|
import { first } from 'lodash';
|
||||||
import { TransformerInjectable } from '@/lib/Transformer/TransformerInjectable';
|
import { TransformerInjectable } from '@/lib/Transformer/TransformerInjectable';
|
||||||
import { GetMatchedTransactionBillsTransformer } from './GetMatchedTransactionBillsTransformer';
|
import { GetMatchedTransactionBillsTransformer } from './GetMatchedTransactionBillsTransformer';
|
||||||
import { GetMatchedTransactionsFilter, MatchedTransactionPOJO } from './types';
|
import {
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
GetMatchedTransactionsFilter,
|
||||||
|
IMatchTransactionDTO,
|
||||||
|
MatchedTransactionPOJO,
|
||||||
|
} from './types';
|
||||||
import { GetMatchedTransactionsByType } from './GetMatchedTransactionsByType';
|
import { GetMatchedTransactionsByType } from './GetMatchedTransactionsByType';
|
||||||
|
import { CreateBillPayment } from '@/services/Purchases/BillPayments/CreateBillPayment';
|
||||||
|
import { IBillPaymentDTO } from '@/interfaces';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class GetMatchedTransactionsByBills extends GetMatchedTransactionsByType {
|
export class GetMatchedTransactionsByBills extends GetMatchedTransactionsByType {
|
||||||
@Inject()
|
@Inject()
|
||||||
private tenancy: HasTenancyService;
|
private transformer: TransformerInjectable;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private transformer: TransformerInjectable;
|
private createPaymentMadeService: CreateBillPayment;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves the matched transactions.
|
* Retrieves the matched transactions.
|
||||||
@@ -71,4 +78,62 @@ export class GetMatchedTransactionsByBills extends GetMatchedTransactionsByType
|
|||||||
new GetMatchedTransactionBillsTransformer()
|
new GetMatchedTransactionBillsTransformer()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates the common matched transaction.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {Array<number>} uncategorizedTransactionIds
|
||||||
|
* @param {IMatchTransactionDTO} matchTransactionDTO
|
||||||
|
* @param {Knex.Transaction} trx
|
||||||
|
*/
|
||||||
|
public async createMatchedTransaction(
|
||||||
|
tenantId: number,
|
||||||
|
uncategorizedTransactionIds: Array<number>,
|
||||||
|
matchTransactionDTO: IMatchTransactionDTO,
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
): Promise<void> {
|
||||||
|
await super.createMatchedTransaction(
|
||||||
|
tenantId,
|
||||||
|
uncategorizedTransactionIds,
|
||||||
|
matchTransactionDTO,
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
const { Bill, UncategorizedCashflowTransaction, MatchedBankTransaction } =
|
||||||
|
this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
const uncategorizedTransactionId = first(uncategorizedTransactionIds);
|
||||||
|
const uncategorizedTransaction =
|
||||||
|
await UncategorizedCashflowTransaction.query(trx)
|
||||||
|
.findById(uncategorizedTransactionId)
|
||||||
|
.throwIfNotFound();
|
||||||
|
|
||||||
|
const bill = await Bill.query(trx)
|
||||||
|
.findById(matchTransactionDTO.referenceId)
|
||||||
|
.throwIfNotFound();
|
||||||
|
|
||||||
|
const createPaymentMadeDTO: IBillPaymentDTO = {
|
||||||
|
vendorId: bill.vendorId,
|
||||||
|
paymentAccountId: uncategorizedTransaction.accountId,
|
||||||
|
paymentDate: uncategorizedTransaction.date,
|
||||||
|
exchangeRate: 1,
|
||||||
|
entries: [
|
||||||
|
{
|
||||||
|
paymentAmount: bill.dueAmount,
|
||||||
|
billId: bill.id,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
branchId: bill.branchId,
|
||||||
|
};
|
||||||
|
// Create a new bill payment associated to the matched bill.
|
||||||
|
const billPayment = await this.createPaymentMadeService.createBillPayment(
|
||||||
|
tenantId,
|
||||||
|
createPaymentMadeDTO,
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
// Link the create bill payment with matched transaction.
|
||||||
|
await super.createMatchedTransaction(tenantId, uncategorizedTransactionIds, {
|
||||||
|
referenceType: 'BillPayment',
|
||||||
|
referenceId: billPayment.id,
|
||||||
|
}, trx);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,22 +1,26 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import { initialize } from 'objection';
|
import { initialize } from 'objection';
|
||||||
|
import { Knex } from 'knex';
|
||||||
|
import { first } from 'lodash';
|
||||||
import { TransformerInjectable } from '@/lib/Transformer/TransformerInjectable';
|
import { TransformerInjectable } from '@/lib/Transformer/TransformerInjectable';
|
||||||
import { GetMatchedTransactionInvoicesTransformer } from './GetMatchedTransactionInvoicesTransformer';
|
import { GetMatchedTransactionInvoicesTransformer } from './GetMatchedTransactionInvoicesTransformer';
|
||||||
import {
|
import {
|
||||||
GetMatchedTransactionsFilter,
|
GetMatchedTransactionsFilter,
|
||||||
|
IMatchTransactionDTO,
|
||||||
MatchedTransactionPOJO,
|
MatchedTransactionPOJO,
|
||||||
MatchedTransactionsPOJO,
|
MatchedTransactionsPOJO,
|
||||||
} from './types';
|
} from './types';
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
|
||||||
import { GetMatchedTransactionsByType } from './GetMatchedTransactionsByType';
|
import { GetMatchedTransactionsByType } from './GetMatchedTransactionsByType';
|
||||||
|
import { CreatePaymentReceived } from '@/services/Sales/PaymentReceived/CreatePaymentReceived';
|
||||||
|
import { IPaymentReceivedCreateDTO } from '@/interfaces';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class GetMatchedTransactionsByInvoices extends GetMatchedTransactionsByType {
|
export class GetMatchedTransactionsByInvoices extends GetMatchedTransactionsByType {
|
||||||
@Inject()
|
@Inject()
|
||||||
protected tenancy: HasTenancyService;
|
protected transformer: TransformerInjectable;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
protected transformer: TransformerInjectable;
|
protected createPaymentReceivedService: CreatePaymentReceived;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves the matched transactions.
|
* Retrieves the matched transactions.
|
||||||
@@ -78,4 +82,64 @@ export class GetMatchedTransactionsByInvoices extends GetMatchedTransactionsByTy
|
|||||||
new GetMatchedTransactionInvoicesTransformer()
|
new GetMatchedTransactionInvoicesTransformer()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates the common matched transaction.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {Array<number>} uncategorizedTransactionIds
|
||||||
|
* @param {IMatchTransactionDTO} matchTransactionDTO
|
||||||
|
* @param {Knex.Transaction} trx
|
||||||
|
*/
|
||||||
|
public async createMatchedTransaction(
|
||||||
|
tenantId: number,
|
||||||
|
uncategorizedTransactionIds: Array<number>,
|
||||||
|
matchTransactionDTO: IMatchTransactionDTO,
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
) {
|
||||||
|
await super.createMatchedTransaction(
|
||||||
|
tenantId,
|
||||||
|
uncategorizedTransactionIds,
|
||||||
|
matchTransactionDTO,
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
const { SaleInvoice, UncategorizedCashflowTransaction, MatchedBankTransaction } =
|
||||||
|
this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
const uncategorizedTransactionId = first(uncategorizedTransactionIds);
|
||||||
|
const uncategorizedTransaction =
|
||||||
|
await UncategorizedCashflowTransaction.query(trx)
|
||||||
|
.findById(uncategorizedTransactionId)
|
||||||
|
.throwIfNotFound();
|
||||||
|
|
||||||
|
const invoice = await SaleInvoice.query(trx)
|
||||||
|
.findById(matchTransactionDTO.referenceId)
|
||||||
|
.throwIfNotFound();
|
||||||
|
|
||||||
|
const createPaymentReceivedDTO: IPaymentReceivedCreateDTO = {
|
||||||
|
customerId: invoice.customerId,
|
||||||
|
paymentDate: uncategorizedTransaction.date,
|
||||||
|
amount: invoice.dueAmount,
|
||||||
|
depositAccountId: uncategorizedTransaction.accountId,
|
||||||
|
entries: [
|
||||||
|
{
|
||||||
|
index: 1,
|
||||||
|
invoiceId: invoice.id,
|
||||||
|
paymentAmount: invoice.dueAmount,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
branchId: invoice.branchId,
|
||||||
|
};
|
||||||
|
// Create a payment received associated to the matched invoice.
|
||||||
|
const paymentReceived = await this.createPaymentReceivedService.createPaymentReceived(
|
||||||
|
tenantId,
|
||||||
|
createPaymentReceivedDTO,
|
||||||
|
{},
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
// Link the create payment received with matched invoice transaction.
|
||||||
|
await super.createMatchedTransaction(tenantId, uncategorizedTransactionIds, {
|
||||||
|
referenceType: 'PaymentReceive',
|
||||||
|
referenceId: paymentReceived.id,
|
||||||
|
}, trx)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import {
|
import {
|
||||||
IBillPaymentEventDeletedPayload,
|
IBillPaymentEventDeletedPayload,
|
||||||
IPaymentReceiveDeletedPayload,
|
IPaymentReceivedDeletedPayload,
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import { ValidateTransactionMatched } from '../ValidateTransactionsMatched';
|
import { ValidateTransactionMatched } from '../ValidateTransactionsMatched';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
@@ -23,7 +23,7 @@ export class ValidateMatchingOnPaymentMadeDelete {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Validates the payment made transaction whether matched with bank transaction on deleting.
|
* Validates the payment made transaction whether matched with bank transaction on deleting.
|
||||||
* @param {IPaymentReceiveDeletedPayload}
|
* @param {IPaymentReceivedDeletedPayload}
|
||||||
*/
|
*/
|
||||||
public async validateMatchingOnPaymentMadeDeleting({
|
public async validateMatchingOnPaymentMadeDeleting({
|
||||||
tenantId,
|
tenantId,
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import { IPaymentReceiveDeletedPayload } from '@/interfaces';
|
import { IPaymentReceivedDeletedPayload } from '@/interfaces';
|
||||||
import { ValidateTransactionMatched } from '../ValidateTransactionsMatched';
|
import { ValidateTransactionMatched } from '../ValidateTransactionsMatched';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
|
|
||||||
@@ -20,13 +20,13 @@ export class ValidateMatchingOnPaymentReceivedDelete {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Validates the payment received transaction whether matched with bank transaction on deleting.
|
* Validates the payment received transaction whether matched with bank transaction on deleting.
|
||||||
* @param {IPaymentReceiveDeletedPayload}
|
* @param {IPaymentReceivedDeletedPayload}
|
||||||
*/
|
*/
|
||||||
public async validateMatchingOnPaymentReceivedDeleting({
|
public async validateMatchingOnPaymentReceivedDeleting({
|
||||||
tenantId,
|
tenantId,
|
||||||
oldPaymentReceive,
|
oldPaymentReceive,
|
||||||
trx,
|
trx,
|
||||||
}: IPaymentReceiveDeletedPayload) {
|
}: IPaymentReceivedDeletedPayload) {
|
||||||
await this.validateNoMatchingLinkedService.validateTransactionNoMatchLinking(
|
await this.validateNoMatchingLinkedService.validateTransactionNoMatchLinking(
|
||||||
tenantId,
|
tenantId,
|
||||||
'PaymentReceive',
|
'PaymentReceive',
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ import { Knex } from 'knex';
|
|||||||
import uniqid from 'uniqid';
|
import uniqid from 'uniqid';
|
||||||
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
|
import { RemovePendingUncategorizedTransaction } from '@/services/Cashflow/RemovePendingUncategorizedTransaction';
|
||||||
|
|
||||||
const CONCURRENCY_ASYNC = 10;
|
const CONCURRENCY_ASYNC = 10;
|
||||||
|
|
||||||
@@ -40,7 +41,7 @@ export class PlaidSyncDb {
|
|||||||
private cashflowApp: CashflowApplication;
|
private cashflowApp: CashflowApplication;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private deleteCashflowTransactionService: DeleteCashflowTransaction;
|
private removePendingTransaction: RemovePendingUncategorizedTransaction;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private eventPublisher: EventPublisher;
|
private eventPublisher: EventPublisher;
|
||||||
@@ -185,21 +186,22 @@ export class PlaidSyncDb {
|
|||||||
plaidTransactionsIds: string[],
|
plaidTransactionsIds: string[],
|
||||||
trx?: Knex.Transaction
|
trx?: Knex.Transaction
|
||||||
) {
|
) {
|
||||||
const { CashflowTransaction } = this.tenancy.models(tenantId);
|
const { UncategorizedCashflowTransaction } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
const cashflowTransactions = await CashflowTransaction.query(trx).whereIn(
|
const uncategorizedTransactions =
|
||||||
|
await UncategorizedCashflowTransaction.query(trx).whereIn(
|
||||||
'plaidTransactionId',
|
'plaidTransactionId',
|
||||||
plaidTransactionsIds
|
plaidTransactionsIds
|
||||||
);
|
);
|
||||||
const cashflowTransactionsIds = cashflowTransactions.map(
|
const uncategorizedTransactionsIds = uncategorizedTransactions.map(
|
||||||
(trans) => trans.id
|
(trans) => trans.id
|
||||||
);
|
);
|
||||||
await bluebird.map(
|
await bluebird.map(
|
||||||
cashflowTransactionsIds,
|
uncategorizedTransactionsIds,
|
||||||
(transactionId: number) =>
|
(uncategorizedTransactionId: number) =>
|
||||||
this.deleteCashflowTransactionService.deleteCashflowTransaction(
|
this.removePendingTransaction.removePendingTransaction(
|
||||||
tenantId,
|
tenantId,
|
||||||
transactionId,
|
uncategorizedTransactionId,
|
||||||
trx
|
trx
|
||||||
),
|
),
|
||||||
{ concurrency: CONCURRENCY_ASYNC }
|
{ concurrency: CONCURRENCY_ASYNC }
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
import { Knex } from 'knex';
|
||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
|
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
||||||
import { PlaidClientWrapper } from '@/lib/Plaid/Plaid';
|
import { PlaidClientWrapper } from '@/lib/Plaid/Plaid';
|
||||||
import { PlaidSyncDb } from './PlaidSyncDB';
|
import { PlaidSyncDb } from './PlaidSyncDB';
|
||||||
import { PlaidFetchedTransactionsUpdates } from '@/interfaces';
|
import { PlaidFetchedTransactionsUpdates } from '@/interfaces';
|
||||||
import UnitOfWork from '@/services/UnitOfWork';
|
import UnitOfWork from '@/services/UnitOfWork';
|
||||||
import { Knex } from 'knex';
|
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class PlaidUpdateTransactions {
|
export class PlaidUpdateTransactions {
|
||||||
@@ -19,8 +19,8 @@ export class PlaidUpdateTransactions {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Handles sync the Plaid item to Bigcaptial under UOW.
|
* Handles sync the Plaid item to Bigcaptial under UOW.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId - Tenant id.
|
||||||
* @param {number} plaidItemId
|
* @param {number} plaidItemId - Plaid item id.
|
||||||
* @returns {Promise<{ addedCount: number; modifiedCount: number; removedCount: number; }>}
|
* @returns {Promise<{ addedCount: number; modifiedCount: number; removedCount: number; }>}
|
||||||
*/
|
*/
|
||||||
public async updateTransactions(tenantId: number, plaidItemId: string) {
|
public async updateTransactions(tenantId: number, plaidItemId: string) {
|
||||||
@@ -73,6 +73,12 @@ export class PlaidUpdateTransactions {
|
|||||||
item,
|
item,
|
||||||
trx
|
trx
|
||||||
);
|
);
|
||||||
|
// Sync removed transactions.
|
||||||
|
await this.plaidSync.syncRemoveTransactions(
|
||||||
|
tenantId,
|
||||||
|
removed?.map((r) => r.transaction_id),
|
||||||
|
trx
|
||||||
|
);
|
||||||
// Sync bank account transactions.
|
// Sync bank account transactions.
|
||||||
await this.plaidSync.syncAccountsTransactions(
|
await this.plaidSync.syncAccountsTransactions(
|
||||||
tenantId,
|
tenantId,
|
||||||
|
|||||||
@@ -83,8 +83,9 @@ export class PlaidWebooks {
|
|||||||
webhookCode: string
|
webhookCode: string
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
const { PlaidItem } = this.tenancy.models(tenantId);
|
const { PlaidItem } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
const plaidItem = await PlaidItem.query()
|
const plaidItem = await PlaidItem.query()
|
||||||
.findById(plaidItemId)
|
.findOne({ plaidItemId })
|
||||||
.throwIfNotFound();
|
.throwIfNotFound();
|
||||||
|
|
||||||
switch (webhookCode) {
|
switch (webhookCode) {
|
||||||
|
|||||||
@@ -3,11 +3,11 @@ import {
|
|||||||
Item as PlaidItem,
|
Item as PlaidItem,
|
||||||
Institution as PlaidInstitution,
|
Institution as PlaidInstitution,
|
||||||
AccountBase as PlaidAccount,
|
AccountBase as PlaidAccount,
|
||||||
|
TransactionBase as PlaidTransactionBase,
|
||||||
} from 'plaid';
|
} from 'plaid';
|
||||||
import {
|
import {
|
||||||
CreateUncategorizedTransactionDTO,
|
CreateUncategorizedTransactionDTO,
|
||||||
IAccountCreateDTO,
|
IAccountCreateDTO,
|
||||||
PlaidTransaction,
|
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -48,7 +48,7 @@ export const transformPlaidAccountToCreateAccount = R.curry(
|
|||||||
export const transformPlaidTrxsToCashflowCreate = R.curry(
|
export const transformPlaidTrxsToCashflowCreate = R.curry(
|
||||||
(
|
(
|
||||||
cashflowAccountId: number,
|
cashflowAccountId: number,
|
||||||
plaidTranasction: PlaidTransaction
|
plaidTranasction: PlaidTransactionBase
|
||||||
): CreateUncategorizedTransactionDTO => {
|
): CreateUncategorizedTransactionDTO => {
|
||||||
return {
|
return {
|
||||||
date: plaidTranasction.date,
|
date: plaidTranasction.date,
|
||||||
@@ -64,6 +64,8 @@ export const transformPlaidTrxsToCashflowCreate = R.curry(
|
|||||||
accountId: cashflowAccountId,
|
accountId: cashflowAccountId,
|
||||||
referenceNo: plaidTranasction.payment_meta?.reference_number,
|
referenceNo: plaidTranasction.payment_meta?.reference_number,
|
||||||
plaidTransactionId: plaidTranasction.transaction_id,
|
plaidTransactionId: plaidTranasction.transaction_id,
|
||||||
|
pending: plaidTranasction.pending,
|
||||||
|
pendingPlaidTransactionId: plaidTranasction.pending_transaction_id,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -33,17 +33,29 @@ const matchNumberCondition = (
|
|||||||
transaction: UncategorizedCashflowTransaction,
|
transaction: UncategorizedCashflowTransaction,
|
||||||
condition: IBankRuleCondition
|
condition: IBankRuleCondition
|
||||||
) => {
|
) => {
|
||||||
|
const conditionValue = parseFloat(condition.value);
|
||||||
|
const transactionAmount =
|
||||||
|
condition.field === 'amount'
|
||||||
|
? Math.abs(transaction[condition.field])
|
||||||
|
: (transaction[condition.field] as unknown as number);
|
||||||
|
|
||||||
switch (condition.comparator) {
|
switch (condition.comparator) {
|
||||||
case BankRuleConditionComparator.Equals:
|
case BankRuleConditionComparator.Equals:
|
||||||
return transaction[condition.field] === condition.value;
|
case BankRuleConditionComparator.Equal:
|
||||||
case BankRuleConditionComparator.Contains:
|
return transactionAmount === conditionValue;
|
||||||
return transaction[condition.field]
|
|
||||||
?.toString()
|
case BankRuleConditionComparator.BiggerOrEqual:
|
||||||
.includes(condition.value.toString());
|
return transactionAmount >= conditionValue;
|
||||||
case BankRuleConditionComparator.NotContain:
|
|
||||||
return !transaction[condition.field]
|
case BankRuleConditionComparator.Bigger:
|
||||||
?.toString()
|
return transactionAmount > conditionValue;
|
||||||
.includes(condition.value.toString());
|
|
||||||
|
case BankRuleConditionComparator.Smaller:
|
||||||
|
return transactionAmount < conditionValue;
|
||||||
|
|
||||||
|
case BankRuleConditionComparator.SmallerOrEqual:
|
||||||
|
return transactionAmount <= conditionValue;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -53,18 +65,19 @@ const matchTextCondition = (
|
|||||||
transaction: UncategorizedCashflowTransaction,
|
transaction: UncategorizedCashflowTransaction,
|
||||||
condition: IBankRuleCondition
|
condition: IBankRuleCondition
|
||||||
): boolean => {
|
): boolean => {
|
||||||
|
const transactionValue = transaction[condition.field] as string;
|
||||||
|
|
||||||
switch (condition.comparator) {
|
switch (condition.comparator) {
|
||||||
case BankRuleConditionComparator.Equals:
|
case BankRuleConditionComparator.Equals:
|
||||||
return transaction[condition.field] === condition.value;
|
case BankRuleConditionComparator.Equal:
|
||||||
|
return transactionValue === condition.value;
|
||||||
case BankRuleConditionComparator.Contains:
|
case BankRuleConditionComparator.Contains:
|
||||||
const fieldValue = lowerCase(transaction[condition.field]);
|
const fieldValue = lowerCase(transactionValue);
|
||||||
const conditionValue = lowerCase(condition.value);
|
const conditionValue = lowerCase(condition.value);
|
||||||
|
|
||||||
return fieldValue.includes(conditionValue);
|
return fieldValue.includes(conditionValue);
|
||||||
case BankRuleConditionComparator.NotContain:
|
case BankRuleConditionComparator.NotContain:
|
||||||
return !transaction[condition.field]?.includes(
|
return !transactionValue?.includes(condition.value.toString());
|
||||||
condition.value.toString()
|
|
||||||
);
|
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -101,8 +114,8 @@ const determineFieldType = (field: string): string => {
|
|||||||
case 'amount':
|
case 'amount':
|
||||||
return 'number';
|
return 'number';
|
||||||
case 'description':
|
case 'description':
|
||||||
return 'text';
|
case 'payee':
|
||||||
default:
|
default:
|
||||||
return 'unknown';
|
return 'text';
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { isEqual, omit } from 'lodash';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
import {
|
import {
|
||||||
IBankRuleEventCreatedPayload,
|
IBankRuleEventCreatedPayload,
|
||||||
@@ -55,10 +56,22 @@ export class TriggerRecognizedTransactions {
|
|||||||
private async recognizedTransactionsOnRuleEdited({
|
private async recognizedTransactionsOnRuleEdited({
|
||||||
tenantId,
|
tenantId,
|
||||||
editRuleDTO,
|
editRuleDTO,
|
||||||
|
oldBankRule,
|
||||||
|
bankRule,
|
||||||
ruleId,
|
ruleId,
|
||||||
}: IBankRuleEventEditedPayload) {
|
}: IBankRuleEventEditedPayload) {
|
||||||
const payload = { tenantId, ruleId };
|
const payload = { tenantId, ruleId };
|
||||||
|
|
||||||
|
// Cannot continue if the new and old bank rule values are the same,
|
||||||
|
// after excluding `createdAt` and `updatedAt` dates.
|
||||||
|
if (
|
||||||
|
isEqual(
|
||||||
|
omit(bankRule, ['createdAt', 'updatedAt']),
|
||||||
|
omit(oldBankRule, ['createdAt', 'updatedAt'])
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
await this.agenda.now(
|
await this.agenda.now(
|
||||||
'rerecognize-uncategorized-transactions-job',
|
'rerecognize-uncategorized-transactions-job',
|
||||||
payload
|
payload
|
||||||
|
|||||||
@@ -26,14 +26,20 @@ export class DeleteBankRuleSerivce {
|
|||||||
* @param {number} ruleId
|
* @param {number} ruleId
|
||||||
* @returns {Promise<void>}
|
* @returns {Promise<void>}
|
||||||
*/
|
*/
|
||||||
public async deleteBankRule(tenantId: number, ruleId: number): Promise<void> {
|
public async deleteBankRule(
|
||||||
|
tenantId: number,
|
||||||
|
ruleId: number,
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
): Promise<void> {
|
||||||
const { BankRule, BankRuleCondition } = this.tenancy.models(tenantId);
|
const { BankRule, BankRuleCondition } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
const oldBankRule = await BankRule.query()
|
const oldBankRule = await BankRule.query()
|
||||||
.findById(ruleId)
|
.findById(ruleId)
|
||||||
.throwIfNotFound();
|
.throwIfNotFound();
|
||||||
|
|
||||||
return this.uow.withTransaction(tenantId, async (trx: Knex.Transaction) => {
|
return this.uow.withTransaction(
|
||||||
|
tenantId,
|
||||||
|
async (trx: Knex.Transaction) => {
|
||||||
// Triggers `onBankRuleDeleting` event.
|
// Triggers `onBankRuleDeleting` event.
|
||||||
await this.eventPublisher.emitAsync(events.bankRules.onDeleting, {
|
await this.eventPublisher.emitAsync(events.bankRules.onDeleting, {
|
||||||
tenantId,
|
tenantId,
|
||||||
@@ -42,7 +48,7 @@ export class DeleteBankRuleSerivce {
|
|||||||
trx,
|
trx,
|
||||||
} as IBankRuleEventDeletingPayload);
|
} as IBankRuleEventDeletingPayload);
|
||||||
|
|
||||||
await BankRuleCondition.query(trx).where('ruleId', ruleId).delete();
|
await BankRuleCondition.query(trx).where('ruleId', ruleId).delete()
|
||||||
await BankRule.query(trx).findById(ruleId).delete();
|
await BankRule.query(trx).findById(ruleId).delete();
|
||||||
|
|
||||||
// Triggers `onBankRuleDeleted` event.
|
// Triggers `onBankRuleDeleted` event.
|
||||||
@@ -51,6 +57,8 @@ export class DeleteBankRuleSerivce {
|
|||||||
ruleId,
|
ruleId,
|
||||||
trx,
|
trx,
|
||||||
} as IBankRuleEventDeletedPayload);
|
} as IBankRuleEventDeletedPayload);
|
||||||
});
|
},
|
||||||
|
trx
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,34 @@
|
|||||||
|
import { Knex } from 'knex';
|
||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import PromisePool from '@supercharge/promise-pool';
|
||||||
|
import { castArray, uniq } from 'lodash';
|
||||||
|
import { DeleteBankRuleSerivce } from './DeleteBankRule';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class DeleteBankRulesService {
|
||||||
|
@Inject()
|
||||||
|
private deleteBankRuleService: DeleteBankRuleSerivce;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete bank rules.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number | Array<number>} bankRuleId
|
||||||
|
*/
|
||||||
|
async deleteBankRules(
|
||||||
|
tenantId: number,
|
||||||
|
bankRuleId: number | Array<number>,
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
) {
|
||||||
|
const bankRulesIds = uniq(castArray(bankRuleId));
|
||||||
|
|
||||||
|
const results = await PromisePool.withConcurrency(1)
|
||||||
|
.for(bankRulesIds)
|
||||||
|
.process(async (bankRuleId: number) => {
|
||||||
|
await this.deleteBankRuleService.deleteBankRule(
|
||||||
|
tenantId,
|
||||||
|
bankRuleId,
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -47,6 +47,7 @@ export class EditBankRuleService {
|
|||||||
|
|
||||||
const oldBankRule = await BankRule.query()
|
const oldBankRule = await BankRule.query()
|
||||||
.findById(ruleId)
|
.findById(ruleId)
|
||||||
|
.withGraphFetched('conditions')
|
||||||
.throwIfNotFound();
|
.throwIfNotFound();
|
||||||
|
|
||||||
const tranformDTO = this.transformDTO(editRuleDTO);
|
const tranformDTO = this.transformDTO(editRuleDTO);
|
||||||
@@ -64,15 +65,15 @@ export class EditBankRuleService {
|
|||||||
} as IBankRuleEventEditingPayload);
|
} as IBankRuleEventEditingPayload);
|
||||||
|
|
||||||
// Updates the given bank rule.
|
// Updates the given bank rule.
|
||||||
await BankRule.query(trx).upsertGraphAndFetch({
|
const bankRule = await BankRule.query(trx).upsertGraphAndFetch({
|
||||||
...tranformDTO,
|
...tranformDTO,
|
||||||
id: ruleId,
|
id: ruleId,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Triggers `onBankRuleEdited` event.
|
// Triggers `onBankRuleEdited` event.
|
||||||
await this.eventPublisher.emitAsync(events.bankRules.onEdited, {
|
await this.eventPublisher.emitAsync(events.bankRules.onEdited, {
|
||||||
tenantId,
|
tenantId,
|
||||||
oldBankRule,
|
oldBankRule,
|
||||||
|
bankRule,
|
||||||
ruleId,
|
ruleId,
|
||||||
editRuleDTO,
|
editRuleDTO,
|
||||||
trx,
|
trx,
|
||||||
|
|||||||
@@ -1,15 +1,20 @@
|
|||||||
import { Knex } from 'knex';
|
import { Knex } from 'knex';
|
||||||
|
|
||||||
export enum BankRuleConditionField {
|
export enum BankRuleConditionField {
|
||||||
Amount = 'Amount',
|
Amount = 'amount',
|
||||||
Description = 'Description',
|
Description = 'description',
|
||||||
Payee = 'Payee',
|
Payee = 'payee',
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum BankRuleConditionComparator {
|
export enum BankRuleConditionComparator {
|
||||||
Contains = 'contains',
|
Contains = 'contains',
|
||||||
Equals = 'equals',
|
Equals = 'equals',
|
||||||
|
Equal = 'equal',
|
||||||
NotContain = 'not_contain',
|
NotContain = 'not_contain',
|
||||||
|
Bigger = 'bigger',
|
||||||
|
BiggerOrEqual = 'bigger_or_equal',
|
||||||
|
Smaller = 'smaller',
|
||||||
|
SmallerOrEqual = 'smaller_or_equal',
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IBankRuleCondition {
|
export interface IBankRuleCondition {
|
||||||
@@ -56,7 +61,15 @@ export enum BankRuleAssignCategory {
|
|||||||
export interface IBankRuleConditionDTO {
|
export interface IBankRuleConditionDTO {
|
||||||
id?: number;
|
id?: number;
|
||||||
field: string;
|
field: string;
|
||||||
comparator: string;
|
comparator:
|
||||||
|
| 'contains'
|
||||||
|
| 'equals'
|
||||||
|
| 'not_contains'
|
||||||
|
| 'equal'
|
||||||
|
| 'bigger'
|
||||||
|
| 'bigger_or_equal'
|
||||||
|
| 'smaller'
|
||||||
|
| 'smaller_or_equal';
|
||||||
value: number;
|
value: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -99,6 +112,8 @@ export interface IBankRuleEventEditingPayload {
|
|||||||
export interface IBankRuleEventEditedPayload {
|
export interface IBankRuleEventEditedPayload {
|
||||||
tenantId: number;
|
tenantId: number;
|
||||||
ruleId: number;
|
ruleId: number;
|
||||||
|
oldBankRule: IBankRule;
|
||||||
|
bankRule: IBankRule;
|
||||||
editRuleDTO: IEditBankRuleDTO;
|
editRuleDTO: IEditBankRuleDTO;
|
||||||
trx?: Knex.Transaction;
|
trx?: Knex.Transaction;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
import {
|
import {
|
||||||
IPaymentReceiveCreatingPayload,
|
IPaymentReceivedCreatingPayload,
|
||||||
IPaymentReceiveEditingPayload,
|
IPaymentReceivedEditingPayload,
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import { ValidateBranchExistance } from '../../Integrations/ValidateBranchExistance';
|
import { ValidateBranchExistance } from '../../Integrations/ValidateBranchExistance';
|
||||||
|
|
||||||
@@ -28,12 +28,12 @@ export class PaymentReceiveBranchValidateSubscriber {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Validate branch existance on estimate creating.
|
* Validate branch existance on estimate creating.
|
||||||
* @param {IPaymentReceiveCreatingPayload} payload
|
* @param {IPaymentReceivedCreatingPayload} payload
|
||||||
*/
|
*/
|
||||||
private validateBranchExistanceOnPaymentCreating = async ({
|
private validateBranchExistanceOnPaymentCreating = async ({
|
||||||
tenantId,
|
tenantId,
|
||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
}: IPaymentReceiveCreatingPayload) => {
|
}: IPaymentReceivedCreatingPayload) => {
|
||||||
await this.validateBranchExistance.validateTransactionBranchWhenActive(
|
await this.validateBranchExistance.validateTransactionBranchWhenActive(
|
||||||
tenantId,
|
tenantId,
|
||||||
paymentReceiveDTO.branchId
|
paymentReceiveDTO.branchId
|
||||||
@@ -42,12 +42,12 @@ export class PaymentReceiveBranchValidateSubscriber {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Validate branch existance once estimate editing.
|
* Validate branch existance once estimate editing.
|
||||||
* @param {IPaymentReceiveEditingPayload} payload
|
* @param {IPaymentReceivedEditingPayload} payload
|
||||||
*/
|
*/
|
||||||
private validateBranchExistanceOnPaymentEditing = async ({
|
private validateBranchExistanceOnPaymentEditing = async ({
|
||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
tenantId,
|
tenantId,
|
||||||
}: IPaymentReceiveEditingPayload) => {
|
}: IPaymentReceivedEditingPayload) => {
|
||||||
await this.validateBranchExistance.validateTransactionBranchWhenActive(
|
await this.validateBranchExistance.validateTransactionBranchWhenActive(
|
||||||
tenantId,
|
tenantId,
|
||||||
paymentReceiveDTO.branchId
|
paymentReceiveDTO.branchId
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ import GetCashflowAccountsService from './GetCashflowAccountsService';
|
|||||||
import { GetCashflowTransactionService } from './GetCashflowTransactionsService';
|
import { GetCashflowTransactionService } from './GetCashflowTransactionsService';
|
||||||
import { GetRecognizedTransactionsService } from './GetRecongizedTransactions';
|
import { GetRecognizedTransactionsService } from './GetRecongizedTransactions';
|
||||||
import { GetRecognizedTransactionService } from './GetRecognizedTransaction';
|
import { GetRecognizedTransactionService } from './GetRecognizedTransaction';
|
||||||
|
import { UncategorizeCashflowTransactionsBulk } from './UncategorizeCashflowTransactionsBulk';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class CashflowApplication {
|
export class CashflowApplication {
|
||||||
@@ -39,6 +40,9 @@ export class CashflowApplication {
|
|||||||
@Inject()
|
@Inject()
|
||||||
private uncategorizeTransactionService: UncategorizeCashflowTransaction;
|
private uncategorizeTransactionService: UncategorizeCashflowTransaction;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private uncategorizeTransasctionsService: UncategorizeCashflowTransactionsBulk;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private categorizeTransactionService: CategorizeCashflowTransaction;
|
private categorizeTransactionService: CategorizeCashflowTransaction;
|
||||||
|
|
||||||
@@ -155,6 +159,22 @@ export class CashflowApplication {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Uncategorize the given transactions in bulk.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number | Array<number>} transactionId
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
public uncategorizeTransactions(
|
||||||
|
tenantId: number,
|
||||||
|
transactionId: number | Array<number>
|
||||||
|
) {
|
||||||
|
return this.uncategorizeTransasctionsService.uncategorizeBulk(
|
||||||
|
tenantId,
|
||||||
|
transactionId
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Categorize the given cashflow transaction.
|
* Categorize the given cashflow transaction.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
|
|||||||
@@ -0,0 +1,53 @@
|
|||||||
|
import { Inject } from 'typedi';
|
||||||
|
import { TransformerInjectable } from '@/lib/Transformer/TransformerInjectable';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import { GetPendingBankAccountTransactionTransformer } from './GetPendingBankAccountTransactionTransformer';
|
||||||
|
|
||||||
|
export class GetPendingBankAccountTransactions {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private transformer: TransformerInjectable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the given bank accounts pending transaction.
|
||||||
|
* @param {number} tenantId - Tenant id.
|
||||||
|
* @param {GetPendingTransactionsQuery} filter - Pending transactions query.
|
||||||
|
*/
|
||||||
|
async getPendingTransactions(
|
||||||
|
tenantId: number,
|
||||||
|
filter?: GetPendingTransactionsQuery
|
||||||
|
) {
|
||||||
|
const { UncategorizedCashflowTransaction } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
const _filter = {
|
||||||
|
page: 1,
|
||||||
|
pageSize: 20,
|
||||||
|
...filter,
|
||||||
|
};
|
||||||
|
const { results, pagination } =
|
||||||
|
await UncategorizedCashflowTransaction.query()
|
||||||
|
.onBuild((q) => {
|
||||||
|
q.modify('pending');
|
||||||
|
|
||||||
|
if (_filter?.accountId) {
|
||||||
|
q.where('accountId', _filter.accountId);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.pagination(_filter.page - 1, _filter.pageSize);
|
||||||
|
|
||||||
|
const data = await this.transformer.transform(
|
||||||
|
tenantId,
|
||||||
|
results,
|
||||||
|
new GetPendingBankAccountTransactionTransformer()
|
||||||
|
);
|
||||||
|
return { data, pagination };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
interface GetPendingTransactionsQuery {
|
||||||
|
page?: number;
|
||||||
|
pageSize?: number;
|
||||||
|
accountId?: number;
|
||||||
|
}
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
import { Transformer } from '@/lib/Transformer/Transformer';
|
||||||
|
import { formatNumber } from '@/utils';
|
||||||
|
|
||||||
|
export class GetPendingBankAccountTransactionTransformer extends Transformer {
|
||||||
|
/**
|
||||||
|
* Include these attributes to sale invoice object.
|
||||||
|
* @returns {string[]}
|
||||||
|
*/
|
||||||
|
public includeAttributes = (): string[] => {
|
||||||
|
return [
|
||||||
|
'formattedAmount',
|
||||||
|
'formattedDate',
|
||||||
|
'formattedDepositAmount',
|
||||||
|
'formattedWithdrawalAmount',
|
||||||
|
];
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exclude all attributes.
|
||||||
|
* @returns {Array<string>}
|
||||||
|
*/
|
||||||
|
public excludeAttributes = (): string[] => {
|
||||||
|
return [];
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Formattes the transaction date.
|
||||||
|
* @param transaction
|
||||||
|
* @returns {string}
|
||||||
|
*/
|
||||||
|
public formattedDate(transaction) {
|
||||||
|
return this.formatDate(transaction.date);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Formatted amount.
|
||||||
|
* @param transaction
|
||||||
|
* @returns {string}
|
||||||
|
*/
|
||||||
|
public formattedAmount(transaction) {
|
||||||
|
return formatNumber(transaction.amount, {
|
||||||
|
currencyCode: transaction.currencyCode,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Formatted deposit amount.
|
||||||
|
* @param transaction
|
||||||
|
* @returns {string}
|
||||||
|
*/
|
||||||
|
protected formattedDepositAmount(transaction) {
|
||||||
|
if (transaction.isDepositTransaction) {
|
||||||
|
return formatNumber(transaction.deposit, {
|
||||||
|
currencyCode: transaction.currencyCode,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Formatted withdrawal amount.
|
||||||
|
* @param transaction
|
||||||
|
* @returns {string}
|
||||||
|
*/
|
||||||
|
protected formattedWithdrawalAmount(transaction) {
|
||||||
|
if (transaction.isWithdrawalTransaction) {
|
||||||
|
return formatNumber(transaction.withdrawal, {
|
||||||
|
currencyCode: transaction.currencyCode,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -34,8 +34,13 @@ export class GetRecognizedTransactionsService {
|
|||||||
q.withGraphFetched('recognizedTransaction.assignAccount');
|
q.withGraphFetched('recognizedTransaction.assignAccount');
|
||||||
q.withGraphFetched('recognizedTransaction.bankRule');
|
q.withGraphFetched('recognizedTransaction.bankRule');
|
||||||
q.whereNotNull('recognizedTransactionId');
|
q.whereNotNull('recognizedTransactionId');
|
||||||
|
|
||||||
|
// Exclude the excluded transactions.
|
||||||
q.modify('notExcluded');
|
q.modify('notExcluded');
|
||||||
|
|
||||||
|
// Exclude the pending transactions.
|
||||||
|
q.modify('notPending');
|
||||||
|
|
||||||
if (_filter.accountId) {
|
if (_filter.accountId) {
|
||||||
q.where('accountId', _filter.accountId);
|
q.where('accountId', _filter.accountId);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -51,7 +51,9 @@ export class GetUncategorizedTransactions {
|
|||||||
.onBuild((q) => {
|
.onBuild((q) => {
|
||||||
q.where('accountId', accountId);
|
q.where('accountId', accountId);
|
||||||
q.where('categorized', false);
|
q.where('categorized', false);
|
||||||
|
|
||||||
q.modify('notExcluded');
|
q.modify('notExcluded');
|
||||||
|
q.modify('notPending');
|
||||||
|
|
||||||
q.withGraphFetched('account');
|
q.withGraphFetched('account');
|
||||||
q.withGraphFetched('recognizedTransaction.assignAccount');
|
q.withGraphFetched('recognizedTransaction.assignAccount');
|
||||||
|
|||||||
@@ -0,0 +1,72 @@
|
|||||||
|
import { Knex } from 'knex';
|
||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import UnitOfWork from '../UnitOfWork';
|
||||||
|
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
||||||
|
import events from '@/subscribers/events';
|
||||||
|
import { ServiceError } from '@/exceptions';
|
||||||
|
import { ERRORS } from './constants';
|
||||||
|
import {
|
||||||
|
IPendingTransactionRemovedEventPayload,
|
||||||
|
IPendingTransactionRemovingEventPayload,
|
||||||
|
} from '@/interfaces';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class RemovePendingUncategorizedTransaction {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private uow: UnitOfWork;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private eventPublisher: EventPublisher;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* REmoves the pending uncategorized transaction.
|
||||||
|
* @param {number} tenantId -
|
||||||
|
* @param {number} uncategorizedTransactionId -
|
||||||
|
* @param {Knex.Transaction} trx -
|
||||||
|
* @returns {Promise<void>}
|
||||||
|
*/
|
||||||
|
public async removePendingTransaction(
|
||||||
|
tenantId: number,
|
||||||
|
uncategorizedTransactionId: number,
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
): Promise<void> {
|
||||||
|
const { UncategorizedCashflowTransaction } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
const pendingTransaction = await UncategorizedCashflowTransaction.query(trx)
|
||||||
|
.findById(uncategorizedTransactionId)
|
||||||
|
.throwIfNotFound();
|
||||||
|
|
||||||
|
if (!pendingTransaction.isPending) {
|
||||||
|
throw new ServiceError(ERRORS.TRANSACTION_NOT_PENDING);
|
||||||
|
}
|
||||||
|
return this.uow.withTransaction(tenantId, async (trx: Knex.Transaction) => {
|
||||||
|
await this.eventPublisher.emitAsync(
|
||||||
|
events.bankTransactions.onPendingRemoving,
|
||||||
|
{
|
||||||
|
tenantId,
|
||||||
|
uncategorizedTransactionId,
|
||||||
|
pendingTransaction,
|
||||||
|
trx,
|
||||||
|
} as IPendingTransactionRemovingEventPayload
|
||||||
|
);
|
||||||
|
// Removes the pending uncategorized transaction.
|
||||||
|
await UncategorizedCashflowTransaction.query(trx)
|
||||||
|
.findById(uncategorizedTransactionId)
|
||||||
|
.delete();
|
||||||
|
|
||||||
|
await this.eventPublisher.emitAsync(
|
||||||
|
events.bankTransactions.onPendingRemoved,
|
||||||
|
{
|
||||||
|
tenantId,
|
||||||
|
uncategorizedTransactionId,
|
||||||
|
pendingTransaction,
|
||||||
|
trx,
|
||||||
|
} as IPendingTransactionRemovedEventPayload
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
import PromisePool from '@supercharge/promise-pool';
|
||||||
|
import { castArray } from 'lodash';
|
||||||
|
import { Service, Inject } from 'typedi';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import { UncategorizeCashflowTransaction } from './UncategorizeCashflowTransaction';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class UncategorizeCashflowTransactionsBulk {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private uncategorizeTransaction: UncategorizeCashflowTransaction;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Uncategorize the given bank transactions in bulk.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} uncategorizedTransactionId
|
||||||
|
*/
|
||||||
|
public async uncategorizeBulk(
|
||||||
|
tenantId: number,
|
||||||
|
uncategorizedTransactionId: number | Array<number>
|
||||||
|
) {
|
||||||
|
const uncategorizedTransactionIds = castArray(uncategorizedTransactionId);
|
||||||
|
|
||||||
|
const result = await PromisePool.withConcurrency(MIGRATION_CONCURRENCY)
|
||||||
|
.for(uncategorizedTransactionIds)
|
||||||
|
.process(async (_uncategorizedTransactionId: number, index, pool) => {
|
||||||
|
await this.uncategorizeTransaction.uncategorize(
|
||||||
|
tenantId,
|
||||||
|
_uncategorizedTransactionId
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const MIGRATION_CONCURRENCY = 1;
|
||||||
@@ -15,10 +15,10 @@ export const ERRORS = {
|
|||||||
'UNCATEGORIZED_TRANSACTION_TYPE_INVALID',
|
'UNCATEGORIZED_TRANSACTION_TYPE_INVALID',
|
||||||
CANNOT_DELETE_TRANSACTION_CONVERTED_FROM_UNCATEGORIZED:
|
CANNOT_DELETE_TRANSACTION_CONVERTED_FROM_UNCATEGORIZED:
|
||||||
'CANNOT_DELETE_TRANSACTION_CONVERTED_FROM_UNCATEGORIZED',
|
'CANNOT_DELETE_TRANSACTION_CONVERTED_FROM_UNCATEGORIZED',
|
||||||
|
CANNOT_CATEGORIZE_EXCLUDED_TRANSACTION:
|
||||||
CANNOT_CATEGORIZE_EXCLUDED_TRANSACTION: 'CANNOT_CATEGORIZE_EXCLUDED_TRANSACTION',
|
'CANNOT_CATEGORIZE_EXCLUDED_TRANSACTION',
|
||||||
TRANSACTION_NOT_CATEGORIZED: 'TRANSACTION_NOT_CATEGORIZED'
|
TRANSACTION_NOT_CATEGORIZED: 'TRANSACTION_NOT_CATEGORIZED',
|
||||||
|
TRANSACTION_NOT_PENDING: 'TRANSACTION_NOT_PENDING',
|
||||||
};
|
};
|
||||||
|
|
||||||
export enum CASHFLOW_DIRECTION {
|
export enum CASHFLOW_DIRECTION {
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
|
import PromisePool from '@supercharge/promise-pool';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
||||||
import {
|
import {
|
||||||
ICashflowTransactionCategorizedPayload,
|
ICashflowTransactionCategorizedPayload,
|
||||||
ICashflowTransactionUncategorizedPayload,
|
ICashflowTransactionUncategorizedPayload,
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import PromisePool from '@supercharge/promise-pool';
|
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class DecrementUncategorizedTransactionOnCategorize {
|
export class DecrementUncategorizedTransactionOnCategorize {
|
||||||
@@ -36,13 +36,17 @@ export class DecrementUncategorizedTransactionOnCategorize {
|
|||||||
public async decrementUnCategorizedTransactionsOnCategorized({
|
public async decrementUnCategorizedTransactionsOnCategorized({
|
||||||
tenantId,
|
tenantId,
|
||||||
uncategorizedTransactions,
|
uncategorizedTransactions,
|
||||||
trx
|
trx,
|
||||||
}: ICashflowTransactionCategorizedPayload) {
|
}: ICashflowTransactionCategorizedPayload) {
|
||||||
const { Account } = this.tenancy.models(tenantId);
|
const { Account } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
await PromisePool.withConcurrency(1)
|
await PromisePool.withConcurrency(1)
|
||||||
.for(uncategorizedTransactions)
|
.for(uncategorizedTransactions)
|
||||||
.process(async (uncategorizedTransaction) => {
|
.process(async (uncategorizedTransaction) => {
|
||||||
|
// Cannot continue if the transaction is still pending.
|
||||||
|
if (uncategorizedTransaction.isPending) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
await Account.query(trx)
|
await Account.query(trx)
|
||||||
.findById(uncategorizedTransaction.accountId)
|
.findById(uncategorizedTransaction.accountId)
|
||||||
.decrement('uncategorizedTransactions', 1);
|
.decrement('uncategorizedTransactions', 1);
|
||||||
@@ -56,13 +60,17 @@ export class DecrementUncategorizedTransactionOnCategorize {
|
|||||||
public async incrementUnCategorizedTransactionsOnUncategorized({
|
public async incrementUnCategorizedTransactionsOnUncategorized({
|
||||||
tenantId,
|
tenantId,
|
||||||
uncategorizedTransactions,
|
uncategorizedTransactions,
|
||||||
trx
|
trx,
|
||||||
}: ICashflowTransactionUncategorizedPayload) {
|
}: ICashflowTransactionUncategorizedPayload) {
|
||||||
const { Account } = this.tenancy.models(tenantId);
|
const { Account } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
await PromisePool.withConcurrency(1)
|
await PromisePool.withConcurrency(1)
|
||||||
.for(uncategorizedTransactions)
|
.for(uncategorizedTransactions)
|
||||||
.process(async (uncategorizedTransaction) => {
|
.process(async (uncategorizedTransaction) => {
|
||||||
|
// Cannot continue if the transaction is still pending.
|
||||||
|
if (uncategorizedTransaction.isPending) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
await Account.query(trx)
|
await Account.query(trx)
|
||||||
.findById(uncategorizedTransaction.accountId)
|
.findById(uncategorizedTransaction.accountId)
|
||||||
.increment('uncategorizedTransactions', 1);
|
.increment('uncategorizedTransactions', 1);
|
||||||
@@ -82,6 +90,9 @@ export class DecrementUncategorizedTransactionOnCategorize {
|
|||||||
|
|
||||||
if (!uncategorizedTransaction.accountId) return;
|
if (!uncategorizedTransaction.accountId) return;
|
||||||
|
|
||||||
|
// Cannot continue if the transaction is still pending.
|
||||||
|
if (uncategorizedTransaction.isPending) return;
|
||||||
|
|
||||||
await Account.query(trx)
|
await Account.query(trx)
|
||||||
.findById(uncategorizedTransaction.accountId)
|
.findById(uncategorizedTransaction.accountId)
|
||||||
.increment('uncategorizedTransactions', 1);
|
.increment('uncategorizedTransactions', 1);
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import {
|
|||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
||||||
import UnitOfWork from '@/services/UnitOfWork';
|
import UnitOfWork from '@/services/UnitOfWork';
|
||||||
import { PaymentReceiveValidators } from '../Sales/PaymentReceives/PaymentReceiveValidators';
|
import { PaymentReceivedValidators } from '../Sales/PaymentReceived/PaymentReceivedValidators';
|
||||||
import BaseCreditNotes from './CreditNotes';
|
import BaseCreditNotes from './CreditNotes';
|
||||||
import {
|
import {
|
||||||
IApplyCreditToInvoicesDTO,
|
IApplyCreditToInvoicesDTO,
|
||||||
@@ -26,7 +26,7 @@ export default class CreditNoteApplyToInvoices extends BaseCreditNotes {
|
|||||||
private tenancy: HasTenancyService;
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private paymentReceiveValidators: PaymentReceiveValidators;
|
private paymentReceiveValidators: PaymentReceivedValidators;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private uow: UnitOfWork;
|
private uow: UnitOfWork;
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ export class ExpenseTransfromer extends Transformer {
|
|||||||
'formattedAllocatedCostAmount',
|
'formattedAllocatedCostAmount',
|
||||||
'formattedDate',
|
'formattedDate',
|
||||||
'formattedCreatedAt',
|
'formattedCreatedAt',
|
||||||
|
'formattedPublishedAt',
|
||||||
'categories',
|
'categories',
|
||||||
'attachments',
|
'attachments',
|
||||||
];
|
];
|
||||||
@@ -91,4 +92,13 @@ export class ExpenseTransfromer extends Transformer {
|
|||||||
protected attachments = (expense: IExpense) => {
|
protected attachments = (expense: IExpense) => {
|
||||||
return this.item(expense.attachments, new AttachmentTransformer());
|
return this.item(expense.attachments, new AttachmentTransformer());
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve formatted published at date.
|
||||||
|
* @param {IExpense} expense
|
||||||
|
* @returns {string}
|
||||||
|
*/
|
||||||
|
protected formattedPublishedAt = (expense: IExpense): string => {
|
||||||
|
return this.formatDate(expense.publishedAt);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,12 +9,13 @@ import { SaleInvoicesExportable } from '../Sales/Invoices/SaleInvoicesExportable
|
|||||||
import { SaleEstimatesExportable } from '../Sales/Estimates/SaleEstimatesExportable';
|
import { SaleEstimatesExportable } from '../Sales/Estimates/SaleEstimatesExportable';
|
||||||
import { SaleReceiptsExportable } from '../Sales/Receipts/SaleReceiptsExportable';
|
import { SaleReceiptsExportable } from '../Sales/Receipts/SaleReceiptsExportable';
|
||||||
import { BillsExportable } from '../Purchases/Bills/BillsExportable';
|
import { BillsExportable } from '../Purchases/Bills/BillsExportable';
|
||||||
import { PaymentsReceivedExportable } from '../Sales/PaymentReceives/PaymentsReceivedExportable';
|
import { PaymentsReceivedExportable } from '../Sales/PaymentReceived/PaymentsReceivedExportable';
|
||||||
import { BillPaymentExportable } from '../Purchases/BillPayments/BillPaymentExportable';
|
import { BillPaymentExportable } from '../Purchases/BillPayments/BillPaymentExportable';
|
||||||
import { ManualJournalsExportable } from '../ManualJournals/ManualJournalExportable';
|
import { ManualJournalsExportable } from '../ManualJournals/ManualJournalExportable';
|
||||||
import { CreditNotesExportable } from '../CreditNotes/CreditNotesExportable';
|
import { CreditNotesExportable } from '../CreditNotes/CreditNotesExportable';
|
||||||
import { VendorCreditsExportable } from '../Purchases/VendorCredits/VendorCreditsExportable';
|
import { VendorCreditsExportable } from '../Purchases/VendorCredits/VendorCreditsExportable';
|
||||||
import { ItemCategoriesExportable } from '../ItemCategories/ItemCategoriesExportable';
|
import { ItemCategoriesExportable } from '../ItemCategories/ItemCategoriesExportable';
|
||||||
|
import { TaxRatesExportable } from '../TaxRates/TaxRatesExportable';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class ExportableResources {
|
export class ExportableResources {
|
||||||
@@ -46,6 +47,7 @@ export class ExportableResources {
|
|||||||
{ resource: 'ManualJournal', exportable: ManualJournalsExportable },
|
{ resource: 'ManualJournal', exportable: ManualJournalsExportable },
|
||||||
{ resource: 'CreditNote', exportable: CreditNotesExportable },
|
{ resource: 'CreditNote', exportable: CreditNotesExportable },
|
||||||
{ resource: 'VendorCredit', exportable: VendorCreditsExportable },
|
{ resource: 'VendorCredit', exportable: VendorCreditsExportable },
|
||||||
|
{ resource: 'TaxRate', exportable: TaxRatesExportable },
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ import { FinancialSheetStructure } from '../FinancialSheetStructure';
|
|||||||
import { flatToNestedArray } from '@/utils';
|
import { flatToNestedArray } from '@/utils';
|
||||||
import Ledger from '@/services/Accounting/Ledger';
|
import Ledger from '@/services/Accounting/Ledger';
|
||||||
import { calculateRunningBalance } from './_utils';
|
import { calculateRunningBalance } from './_utils';
|
||||||
|
import { getTransactionTypeLabel } from '@/utils/transactions-types';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* General ledger sheet.
|
* General ledger sheet.
|
||||||
@@ -90,11 +91,13 @@ export default class GeneralLedgerSheet extends R.compose(
|
|||||||
date: entry.date,
|
date: entry.date,
|
||||||
dateFormatted: moment(entry.date).format('YYYY MMM DD'),
|
dateFormatted: moment(entry.date).format('YYYY MMM DD'),
|
||||||
|
|
||||||
transactionNumber: entry.transactionNumber,
|
referenceType: entry.transactionType,
|
||||||
referenceType: entry.referenceType,
|
referenceId: entry.transactionId,
|
||||||
referenceId: entry.referenceId,
|
|
||||||
referenceTypeFormatted: this.i18n.__(entry.referenceTypeFormatted),
|
|
||||||
|
|
||||||
|
transactionNumber: entry.transactionNumber,
|
||||||
|
transactionTypeFormatted: this.i18n.__(
|
||||||
|
getTransactionTypeLabel(entry.transactionType, entry.transactionSubType)
|
||||||
|
),
|
||||||
contactName: get(contact, 'displayName'),
|
contactName: get(contact, 'displayName'),
|
||||||
contactType: get(contact, 'contactService'),
|
contactType: get(contact, 'contactService'),
|
||||||
|
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ export class GeneralLedgerTable extends R.compose(
|
|||||||
return [
|
return [
|
||||||
{ key: 'date', accessor: 'dateFormatted' },
|
{ key: 'date', accessor: 'dateFormatted' },
|
||||||
{ key: 'account_name', accessor: 'account.name' },
|
{ key: 'account_name', accessor: 'account.name' },
|
||||||
{ key: 'reference_type', accessor: 'referenceTypeFormatted' },
|
{ key: 'reference_type', accessor: 'transactionTypeFormatted' },
|
||||||
{ key: 'reference_number', accessor: 'transactionNumber' },
|
{ key: 'reference_number', accessor: 'transactionNumber' },
|
||||||
{ key: 'description', accessor: 'note' },
|
{ key: 'description', accessor: 'note' },
|
||||||
{ key: 'credit', accessor: 'formattedCredit' },
|
{ key: 'credit', accessor: 'formattedCredit' },
|
||||||
|
|||||||
@@ -13,9 +13,10 @@ import { SaleInvoicesImportable } from '../Sales/Invoices/SaleInvoicesImportable
|
|||||||
import { SaleEstimatesImportable } from '../Sales/Estimates/SaleEstimatesImportable';
|
import { SaleEstimatesImportable } from '../Sales/Estimates/SaleEstimatesImportable';
|
||||||
import { BillPaymentsImportable } from '../Purchases/BillPayments/BillPaymentsImportable';
|
import { BillPaymentsImportable } from '../Purchases/BillPayments/BillPaymentsImportable';
|
||||||
import { VendorCreditsImportable } from '../Purchases/VendorCredits/VendorCreditsImportable';
|
import { VendorCreditsImportable } from '../Purchases/VendorCredits/VendorCreditsImportable';
|
||||||
import { PaymentReceivesImportable } from '../Sales/PaymentReceives/PaymentReceivesImportable';
|
import { PaymentsReceivedImportable } from '../Sales/PaymentReceived/PaymentsReceivedImportable';
|
||||||
import { CreditNotesImportable } from '../CreditNotes/CreditNotesImportable';
|
import { CreditNotesImportable } from '../CreditNotes/CreditNotesImportable';
|
||||||
import { SaleReceiptsImportable } from '../Sales/Receipts/SaleReceiptsImportable';
|
import { SaleReceiptsImportable } from '../Sales/Receipts/SaleReceiptsImportable';
|
||||||
|
import { TaxRatesImportable } from '../TaxRates/TaxRatesImportable';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class ImportableResources {
|
export class ImportableResources {
|
||||||
@@ -44,10 +45,11 @@ export class ImportableResources {
|
|||||||
{ resource: 'SaleInvoice', importable: SaleInvoicesImportable },
|
{ resource: 'SaleInvoice', importable: SaleInvoicesImportable },
|
||||||
{ resource: 'SaleEstimate', importable: SaleEstimatesImportable },
|
{ resource: 'SaleEstimate', importable: SaleEstimatesImportable },
|
||||||
{ resource: 'BillPayment', importable: BillPaymentsImportable },
|
{ resource: 'BillPayment', importable: BillPaymentsImportable },
|
||||||
{ resource: 'PaymentReceive', importable: PaymentReceivesImportable },
|
{ resource: 'PaymentReceive', importable: PaymentsReceivedImportable },
|
||||||
{ resource: 'VendorCredit', importable: VendorCreditsImportable },
|
{ resource: 'VendorCredit', importable: VendorCreditsImportable },
|
||||||
{ resource: 'CreditNote', importable: CreditNotesImportable },
|
{ resource: 'CreditNote', importable: CreditNotesImportable },
|
||||||
{ resource: 'SaleReceipt', importable: SaleReceiptsImportable }
|
{ resource: 'SaleReceipt', importable: SaleReceiptsImportable },
|
||||||
|
{ resource: 'TaxRate', importable: TaxRatesImportable },
|
||||||
];
|
];
|
||||||
|
|
||||||
public get registry() {
|
public get registry() {
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ import SaleNotifyBySms from '../SaleNotifyBySms';
|
|||||||
import SmsNotificationsSettingsService from '@/services/Settings/SmsNotificationsSettings';
|
import SmsNotificationsSettingsService from '@/services/Settings/SmsNotificationsSettings';
|
||||||
import {
|
import {
|
||||||
ICustomer,
|
ICustomer,
|
||||||
IPaymentReceiveSmsDetails,
|
IPaymentReceivedSmsDetails,
|
||||||
ISaleEstimate,
|
ISaleEstimate,
|
||||||
SMS_NOTIFICATION_KEY,
|
SMS_NOTIFICATION_KEY,
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
@@ -173,12 +173,12 @@ export class SaleEstimateNotifyBySms {
|
|||||||
* Retrieve the SMS details of the given payment receive transaction.
|
* Retrieve the SMS details of the given payment receive transaction.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
* @param {number} saleEstimateId
|
* @param {number} saleEstimateId
|
||||||
* @returns {Promise<IPaymentReceiveSmsDetails>}
|
* @returns {Promise<IPaymentReceivedSmsDetails>}
|
||||||
*/
|
*/
|
||||||
public smsDetails = async (
|
public smsDetails = async (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
saleEstimateId: number
|
saleEstimateId: number
|
||||||
): Promise<IPaymentReceiveSmsDetails> => {
|
): Promise<IPaymentReceivedSmsDetails> => {
|
||||||
const { SaleEstimate } = this.tenancy.models(tenantId);
|
const { SaleEstimate } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
// Retrieve the sale invoice or throw not found service error.
|
// Retrieve the sale invoice or throw not found service error.
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import { CreateSaleEstimate } from './CreateSaleEstimate';
|
|||||||
import {
|
import {
|
||||||
IFilterMeta,
|
IFilterMeta,
|
||||||
IPaginationMeta,
|
IPaginationMeta,
|
||||||
IPaymentReceiveSmsDetails,
|
IPaymentReceivedSmsDetails,
|
||||||
ISaleEstimate,
|
ISaleEstimate,
|
||||||
ISaleEstimateDTO,
|
ISaleEstimateDTO,
|
||||||
ISalesEstimatesFilter,
|
ISalesEstimatesFilter,
|
||||||
@@ -191,12 +191,12 @@ export class SaleEstimatesApplication {
|
|||||||
* Retrieve the SMS details of the given payment receive transaction.
|
* Retrieve the SMS details of the given payment receive transaction.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
* @param {number} saleEstimateId
|
* @param {number} saleEstimateId
|
||||||
* @returns {Promise<IPaymentReceiveSmsDetails>}
|
* @returns {Promise<IPaymentReceivedSmsDetails>}
|
||||||
*/
|
*/
|
||||||
public getSaleEstimateSmsDetails = (
|
public getSaleEstimateSmsDetails = (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
saleEstimateId: number
|
saleEstimateId: number
|
||||||
): Promise<IPaymentReceiveSmsDetails> => {
|
): Promise<IPaymentReceivedSmsDetails> => {
|
||||||
return this.saleEstimateNotifyBySmsService.smsDetails(
|
return this.saleEstimateNotifyBySmsService.smsDetails(
|
||||||
tenantId,
|
tenantId,
|
||||||
saleEstimateId
|
saleEstimateId
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import { Knex } from 'knex';
|
|||||||
import async from 'async';
|
import async from 'async';
|
||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
||||||
import { PaymentReceiveGLEntries } from '../PaymentReceives/PaymentReceiveGLEntries';
|
import { PaymentReceivedGLEntries } from '../PaymentReceived/PaymentReceivedGLEntries';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class InvoicePaymentsGLEntriesRewrite {
|
export class InvoicePaymentsGLEntriesRewrite {
|
||||||
@@ -10,7 +10,7 @@ export class InvoicePaymentsGLEntriesRewrite {
|
|||||||
public tenancy: HasTenancyService;
|
public tenancy: HasTenancyService;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
public paymentGLEntries: PaymentReceiveGLEntries;
|
public paymentGLEntries: PaymentReceivedGLEntries;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Rewrites the payment GL entries task.
|
* Rewrites the payment GL entries task.
|
||||||
|
|||||||
@@ -2,23 +2,23 @@ import { Inject, Service } from 'typedi';
|
|||||||
import { Knex } from 'knex';
|
import { Knex } from 'knex';
|
||||||
import {
|
import {
|
||||||
ICustomer,
|
ICustomer,
|
||||||
IPaymentReceiveCreateDTO,
|
IPaymentReceivedCreateDTO,
|
||||||
IPaymentReceiveCreatedPayload,
|
IPaymentReceivedCreatedPayload,
|
||||||
IPaymentReceiveCreatingPayload,
|
IPaymentReceivedCreatingPayload,
|
||||||
ISystemUser,
|
ISystemUser,
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import { PaymentReceiveValidators } from './PaymentReceiveValidators';
|
import { PaymentReceivedValidators } from './PaymentReceivedValidators';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
||||||
import UnitOfWork from '@/services/UnitOfWork';
|
import UnitOfWork from '@/services/UnitOfWork';
|
||||||
import { PaymentReceiveDTOTransformer } from './PaymentReceiveDTOTransformer';
|
import { PaymentReceiveDTOTransformer } from './PaymentReceivedDTOTransformer';
|
||||||
import { TenantMetadata } from '@/system/models';
|
import { TenantMetadata } from '@/system/models';
|
||||||
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class CreatePaymentReceive {
|
export class CreatePaymentReceived {
|
||||||
@Inject()
|
@Inject()
|
||||||
private validators: PaymentReceiveValidators;
|
private validators: PaymentReceivedValidators;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private eventPublisher: EventPublisher;
|
private eventPublisher: EventPublisher;
|
||||||
@@ -37,11 +37,11 @@ export class CreatePaymentReceive {
|
|||||||
* with associated invoices payment and journal transactions.
|
* with associated invoices payment and journal transactions.
|
||||||
* @async
|
* @async
|
||||||
* @param {number} tenantId - Tenant id.
|
* @param {number} tenantId - Tenant id.
|
||||||
* @param {IPaymentReceive} paymentReceive
|
* @param {IPaymentReceived} paymentReceive
|
||||||
*/
|
*/
|
||||||
public async createPaymentReceive(
|
public async createPaymentReceived(
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
paymentReceiveDTO: IPaymentReceiveCreateDTO,
|
paymentReceiveDTO: IPaymentReceivedCreateDTO,
|
||||||
authorizedUser: ISystemUser,
|
authorizedUser: ISystemUser,
|
||||||
trx?: Knex.Transaction
|
trx?: Knex.Transaction
|
||||||
) {
|
) {
|
||||||
@@ -97,7 +97,7 @@ export class CreatePaymentReceive {
|
|||||||
trx,
|
trx,
|
||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
tenantId,
|
tenantId,
|
||||||
} as IPaymentReceiveCreatingPayload);
|
} as IPaymentReceivedCreatingPayload);
|
||||||
|
|
||||||
// Inserts the payment receive transaction.
|
// Inserts the payment receive transaction.
|
||||||
const paymentReceive = await PaymentReceive.query(
|
const paymentReceive = await PaymentReceive.query(
|
||||||
@@ -113,7 +113,7 @@ export class CreatePaymentReceive {
|
|||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
authorizedUser,
|
authorizedUser,
|
||||||
trx,
|
trx,
|
||||||
} as IPaymentReceiveCreatedPayload);
|
} as IPaymentReceivedCreatedPayload);
|
||||||
|
|
||||||
return paymentReceive;
|
return paymentReceive;
|
||||||
},
|
},
|
||||||
@@ -125,13 +125,13 @@ export class CreatePaymentReceive {
|
|||||||
* Transform the create payment receive DTO.
|
* Transform the create payment receive DTO.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
* @param {ICustomer} customer
|
* @param {ICustomer} customer
|
||||||
* @param {IPaymentReceiveCreateDTO} paymentReceiveDTO
|
* @param {IPaymentReceivedCreateDTO} paymentReceiveDTO
|
||||||
* @returns
|
* @returns
|
||||||
*/
|
*/
|
||||||
private transformCreateDTOToModel = async (
|
private transformCreateDTOToModel = async (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
customer: ICustomer,
|
customer: ICustomer,
|
||||||
paymentReceiveDTO: IPaymentReceiveCreateDTO
|
paymentReceiveDTO: IPaymentReceivedCreateDTO
|
||||||
) => {
|
) => {
|
||||||
return this.transformer.transformPaymentReceiveDTOToModel(
|
return this.transformer.transformPaymentReceiveDTOToModel(
|
||||||
tenantId,
|
tenantId,
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import { Knex } from 'knex';
|
import { Knex } from 'knex';
|
||||||
import {
|
import {
|
||||||
IPaymentReceiveDeletedPayload,
|
IPaymentReceivedDeletedPayload,
|
||||||
IPaymentReceiveDeletingPayload,
|
IPaymentReceivedDeletingPayload,
|
||||||
ISystemUser,
|
ISystemUser,
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import UnitOfWork from '@/services/UnitOfWork';
|
import UnitOfWork from '@/services/UnitOfWork';
|
||||||
@@ -11,7 +11,7 @@ import events from '@/subscribers/events';
|
|||||||
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class DeletePaymentReceive {
|
export class DeletePaymentReceived {
|
||||||
@Inject()
|
@Inject()
|
||||||
private eventPublisher: EventPublisher;
|
private eventPublisher: EventPublisher;
|
||||||
|
|
||||||
@@ -33,7 +33,7 @@ export class DeletePaymentReceive {
|
|||||||
* @async
|
* @async
|
||||||
* @param {number} tenantId - Tenant id.
|
* @param {number} tenantId - Tenant id.
|
||||||
* @param {Integer} paymentReceiveId - Payment receive id.
|
* @param {Integer} paymentReceiveId - Payment receive id.
|
||||||
* @param {IPaymentReceive} paymentReceive - Payment receive object.
|
* @param {IPaymentReceived} paymentReceive - Payment receive object.
|
||||||
*/
|
*/
|
||||||
public async deletePaymentReceive(
|
public async deletePaymentReceive(
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
@@ -56,7 +56,7 @@ export class DeletePaymentReceive {
|
|||||||
tenantId,
|
tenantId,
|
||||||
oldPaymentReceive,
|
oldPaymentReceive,
|
||||||
trx,
|
trx,
|
||||||
} as IPaymentReceiveDeletingPayload);
|
} as IPaymentReceivedDeletingPayload);
|
||||||
|
|
||||||
// Deletes the payment receive associated entries.
|
// Deletes the payment receive associated entries.
|
||||||
await PaymentReceiveEntry.query(trx)
|
await PaymentReceiveEntry.query(trx)
|
||||||
@@ -73,7 +73,7 @@ export class DeletePaymentReceive {
|
|||||||
oldPaymentReceive,
|
oldPaymentReceive,
|
||||||
authorizedUser,
|
authorizedUser,
|
||||||
trx,
|
trx,
|
||||||
} as IPaymentReceiveDeletedPayload);
|
} as IPaymentReceivedDeletedPayload);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -2,14 +2,14 @@ import { Inject, Service } from 'typedi';
|
|||||||
import { Knex } from 'knex';
|
import { Knex } from 'knex';
|
||||||
import {
|
import {
|
||||||
ICustomer,
|
ICustomer,
|
||||||
IPaymentReceive,
|
IPaymentReceived,
|
||||||
IPaymentReceiveEditDTO,
|
IPaymentReceivedEditDTO,
|
||||||
IPaymentReceiveEditedPayload,
|
IPaymentReceivedEditedPayload,
|
||||||
IPaymentReceiveEditingPayload,
|
IPaymentReceivedEditingPayload,
|
||||||
ISystemUser,
|
ISystemUser,
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import { PaymentReceiveDTOTransformer } from './PaymentReceiveDTOTransformer';
|
import { PaymentReceiveDTOTransformer } from './PaymentReceivedDTOTransformer';
|
||||||
import { PaymentReceiveValidators } from './PaymentReceiveValidators';
|
import { PaymentReceivedValidators } from './PaymentReceivedValidators';
|
||||||
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
import UnitOfWork from '@/services/UnitOfWork';
|
import UnitOfWork from '@/services/UnitOfWork';
|
||||||
@@ -17,12 +17,12 @@ import HasTenancyService from '@/services/Tenancy/TenancyService';
|
|||||||
import { TenantMetadata } from '@/system/models';
|
import { TenantMetadata } from '@/system/models';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class EditPaymentReceive {
|
export class EditPaymentReceived {
|
||||||
@Inject()
|
@Inject()
|
||||||
private transformer: PaymentReceiveDTOTransformer;
|
private transformer: PaymentReceiveDTOTransformer;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private validators: PaymentReceiveValidators;
|
private validators: PaymentReceivedValidators;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
private eventPublisher: EventPublisher;
|
private eventPublisher: EventPublisher;
|
||||||
@@ -46,12 +46,12 @@ export class EditPaymentReceive {
|
|||||||
* @async
|
* @async
|
||||||
* @param {number} tenantId -
|
* @param {number} tenantId -
|
||||||
* @param {Integer} paymentReceiveId -
|
* @param {Integer} paymentReceiveId -
|
||||||
* @param {IPaymentReceive} paymentReceive -
|
* @param {IPaymentReceived} paymentReceive -
|
||||||
*/
|
*/
|
||||||
public async editPaymentReceive(
|
public async editPaymentReceive(
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
paymentReceiveId: number,
|
paymentReceiveId: number,
|
||||||
paymentReceiveDTO: IPaymentReceiveEditDTO,
|
paymentReceiveDTO: IPaymentReceivedEditDTO,
|
||||||
authorizedUser: ISystemUser
|
authorizedUser: ISystemUser
|
||||||
) {
|
) {
|
||||||
const { PaymentReceive, Contact } = this.tenancy.models(tenantId);
|
const { PaymentReceive, Contact } = this.tenancy.models(tenantId);
|
||||||
@@ -131,7 +131,7 @@ export class EditPaymentReceive {
|
|||||||
tenantId,
|
tenantId,
|
||||||
oldPaymentReceive,
|
oldPaymentReceive,
|
||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
} as IPaymentReceiveEditingPayload);
|
} as IPaymentReceivedEditingPayload);
|
||||||
|
|
||||||
// Update the payment receive transaction.
|
// Update the payment receive transaction.
|
||||||
const paymentReceive = await PaymentReceive.query(
|
const paymentReceive = await PaymentReceive.query(
|
||||||
@@ -149,7 +149,7 @@ export class EditPaymentReceive {
|
|||||||
paymentReceiveDTO,
|
paymentReceiveDTO,
|
||||||
authorizedUser,
|
authorizedUser,
|
||||||
trx,
|
trx,
|
||||||
} as IPaymentReceiveEditedPayload);
|
} as IPaymentReceivedEditedPayload);
|
||||||
|
|
||||||
return paymentReceive;
|
return paymentReceive;
|
||||||
});
|
});
|
||||||
@@ -159,15 +159,15 @@ export class EditPaymentReceive {
|
|||||||
* Transform the edit payment receive DTO.
|
* Transform the edit payment receive DTO.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
* @param {ICustomer} customer
|
* @param {ICustomer} customer
|
||||||
* @param {IPaymentReceiveEditDTO} paymentReceiveDTO
|
* @param {IPaymentReceivedEditDTO} paymentReceiveDTO
|
||||||
* @param {IPaymentReceive} oldPaymentReceive
|
* @param {IPaymentReceived} oldPaymentReceive
|
||||||
* @returns
|
* @returns
|
||||||
*/
|
*/
|
||||||
private transformEditDTOToModel = async (
|
private transformEditDTOToModel = async (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
customer: ICustomer,
|
customer: ICustomer,
|
||||||
paymentReceiveDTO: IPaymentReceiveEditDTO,
|
paymentReceiveDTO: IPaymentReceivedEditDTO,
|
||||||
oldPaymentReceive: IPaymentReceive
|
oldPaymentReceive: IPaymentReceived
|
||||||
) => {
|
) => {
|
||||||
return this.transformer.transformPaymentReceiveDTOToModel(
|
return this.transformer.transformPaymentReceiveDTOToModel(
|
||||||
tenantId,
|
tenantId,
|
||||||
@@ -1,13 +1,13 @@
|
|||||||
import { ServiceError } from '@/exceptions';
|
import { ServiceError } from '@/exceptions';
|
||||||
import { IPaymentReceive } from '@/interfaces';
|
import { IPaymentReceived } from '@/interfaces';
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import { ERRORS } from './constants';
|
import { ERRORS } from './constants';
|
||||||
import { PaymentReceiveTransfromer } from './PaymentReceiveTransformer';
|
import { PaymentReceiveTransfromer } from './PaymentReceivedTransformer';
|
||||||
import { TransformerInjectable } from '@/lib/Transformer/TransformerInjectable';
|
import { TransformerInjectable } from '@/lib/Transformer/TransformerInjectable';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class GetPaymentReceive {
|
export class GetPaymentReceived {
|
||||||
@Inject()
|
@Inject()
|
||||||
private tenancy: HasTenancyService;
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
@@ -18,12 +18,12 @@ export class GetPaymentReceive {
|
|||||||
* Retrieve payment receive details.
|
* Retrieve payment receive details.
|
||||||
* @param {number} tenantId - Tenant id.
|
* @param {number} tenantId - Tenant id.
|
||||||
* @param {number} paymentReceiveId - Payment receive id.
|
* @param {number} paymentReceiveId - Payment receive id.
|
||||||
* @return {Promise<IPaymentReceive>}
|
* @return {Promise<IPaymentReceived>}
|
||||||
*/
|
*/
|
||||||
public async getPaymentReceive(
|
public async getPaymentReceive(
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
paymentReceiveId: number
|
paymentReceiveId: number
|
||||||
): Promise<IPaymentReceive> {
|
): Promise<IPaymentReceived> {
|
||||||
const { PaymentReceive } = this.tenancy.models(tenantId);
|
const { PaymentReceive } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
const paymentReceive = await PaymentReceive.query()
|
const paymentReceive = await PaymentReceive.query()
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user