mirror of
https://github.com/bigcapitalhq/bigcapital.git
synced 2026-02-24 08:39:49 +00:00
Compare commits
20 Commits
big-152-un
...
accounts-b
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f9e5028e0d | ||
|
|
7a3e121942 | ||
|
|
fc1d123c6b | ||
|
|
ad4e51d81d | ||
|
|
973d1832bd | ||
|
|
858e3541cb | ||
|
|
a5ab535d3b | ||
|
|
1d8cec5069 | ||
|
|
aba06991d4 | ||
|
|
ff5730d8a7 | ||
|
|
a27c877321 | ||
|
|
c5063fc5b5 | ||
|
|
ab4c0ab7a7 | ||
|
|
084d9d3d10 | ||
|
|
daa1e3a6bd | ||
|
|
4270d66928 | ||
|
|
90b4f3ef6d | ||
|
|
1fc6445123 | ||
|
|
b1d5390bfc | ||
|
|
1ba26a3b85 |
@@ -25,6 +25,7 @@
|
|||||||
"@types/i18n": "^0.8.7",
|
"@types/i18n": "^0.8.7",
|
||||||
"@types/knex": "^0.16.1",
|
"@types/knex": "^0.16.1",
|
||||||
"@types/mathjs": "^6.0.12",
|
"@types/mathjs": "^6.0.12",
|
||||||
|
"@types/yup": "^0.29.13",
|
||||||
"accepts": "^1.3.7",
|
"accepts": "^1.3.7",
|
||||||
"accounting": "^0.4.1",
|
"accounting": "^0.4.1",
|
||||||
"agenda": "^4.2.1",
|
"agenda": "^4.2.1",
|
||||||
@@ -53,7 +54,6 @@
|
|||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
"express-basic-auth": "^1.2.0",
|
"express-basic-auth": "^1.2.0",
|
||||||
"express-boom": "^3.0.0",
|
"express-boom": "^3.0.0",
|
||||||
"express-fileupload": "^1.1.7-alpha.3",
|
|
||||||
"express-oauth-server": "^2.0.0",
|
"express-oauth-server": "^2.0.0",
|
||||||
"express-validator": "^6.12.2",
|
"express-validator": "^6.12.2",
|
||||||
"form-data": "^4.0.0",
|
"form-data": "^4.0.0",
|
||||||
@@ -77,6 +77,7 @@
|
|||||||
"moment-timezone": "^0.5.43",
|
"moment-timezone": "^0.5.43",
|
||||||
"mongodb": "^6.1.0",
|
"mongodb": "^6.1.0",
|
||||||
"mongoose": "^5.10.0",
|
"mongoose": "^5.10.0",
|
||||||
|
"multer": "1.4.5-lts.1",
|
||||||
"mustache": "^3.0.3",
|
"mustache": "^3.0.3",
|
||||||
"mysql": "^2.17.1",
|
"mysql": "^2.17.1",
|
||||||
"mysql2": "^1.6.5",
|
"mysql2": "^1.6.5",
|
||||||
@@ -105,7 +106,8 @@
|
|||||||
"typedi": "^0.8.0",
|
"typedi": "^0.8.0",
|
||||||
"uniqid": "^5.2.0",
|
"uniqid": "^5.2.0",
|
||||||
"winston": "^3.2.1",
|
"winston": "^3.2.1",
|
||||||
"xlsx": "^0.18.5"
|
"xlsx": "^0.18.5",
|
||||||
|
"yup": "^0.28.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/lodash": "^4.14.158",
|
"@types/lodash": "^4.14.158",
|
||||||
|
|||||||
@@ -242,7 +242,7 @@
|
|||||||
"account.field.normal.credit": "دائن",
|
"account.field.normal.credit": "دائن",
|
||||||
"account.field.normal.debit": "مدين",
|
"account.field.normal.debit": "مدين",
|
||||||
"account.field.type": "نوع الحساب",
|
"account.field.type": "نوع الحساب",
|
||||||
"account.field.active": "Activity",
|
"account.field.active": "Active",
|
||||||
"account.field.balance": "الرصيد",
|
"account.field.balance": "الرصيد",
|
||||||
"account.field.created_at": "أنشئت في",
|
"account.field.created_at": "أنشئت في",
|
||||||
"item.field.type": "نوع الصنف",
|
"item.field.type": "نوع الصنف",
|
||||||
|
|||||||
@@ -241,7 +241,8 @@
|
|||||||
"account.field.normal.credit": "Credit",
|
"account.field.normal.credit": "Credit",
|
||||||
"account.field.normal.debit": "Debit",
|
"account.field.normal.debit": "Debit",
|
||||||
"account.field.type": "Type",
|
"account.field.type": "Type",
|
||||||
"account.field.active": "Activity",
|
"account.field.active": "Active",
|
||||||
|
"account.field.currency": "Currency",
|
||||||
"account.field.balance": "Balance",
|
"account.field.balance": "Balance",
|
||||||
"account.field.created_at": "Created at",
|
"account.field.created_at": "Created at",
|
||||||
"item.field.type": "Item type",
|
"item.field.type": "Item type",
|
||||||
@@ -376,8 +377,8 @@
|
|||||||
"customer.field.last_name": "Last name",
|
"customer.field.last_name": "Last name",
|
||||||
"customer.field.display_name": "Display name",
|
"customer.field.display_name": "Display name",
|
||||||
"customer.field.email": "Email",
|
"customer.field.email": "Email",
|
||||||
"customer.field.work_phone": "Work phone",
|
"customer.field.work_phone": "Work Phone Number",
|
||||||
"customer.field.personal_phone": "Personal phone",
|
"customer.field.personal_phone": "Personal Phone Number",
|
||||||
"customer.field.company_name": "Company name",
|
"customer.field.company_name": "Company name",
|
||||||
"customer.field.website": "Website",
|
"customer.field.website": "Website",
|
||||||
"customer.field.opening_balance_at": "Opening balance at",
|
"customer.field.opening_balance_at": "Opening balance at",
|
||||||
@@ -385,7 +386,7 @@
|
|||||||
"customer.field.created_at": "Created at",
|
"customer.field.created_at": "Created at",
|
||||||
"customer.field.balance": "Balance",
|
"customer.field.balance": "Balance",
|
||||||
"customer.field.status": "Status",
|
"customer.field.status": "Status",
|
||||||
"customer.field.currency": "Curreny",
|
"customer.field.currency": "Currency",
|
||||||
"customer.field.status.active": "Active",
|
"customer.field.status.active": "Active",
|
||||||
"customer.field.status.inactive": "Inactive",
|
"customer.field.status.inactive": "Inactive",
|
||||||
"customer.field.status.overdue": "Overdue",
|
"customer.field.status.overdue": "Overdue",
|
||||||
@@ -394,8 +395,8 @@
|
|||||||
"vendor.field.last_name": "Last name",
|
"vendor.field.last_name": "Last name",
|
||||||
"vendor.field.display_name": "Display name",
|
"vendor.field.display_name": "Display name",
|
||||||
"vendor.field.email": "Email",
|
"vendor.field.email": "Email",
|
||||||
"vendor.field.work_phone": "Work phone",
|
"vendor.field.work_phone": "Work Phone Number",
|
||||||
"vendor.field.personal_phone": "Personal phone",
|
"vendor.field.personal_phone": "Personal Phone Number",
|
||||||
"vendor.field.company_name": "Company name",
|
"vendor.field.company_name": "Company name",
|
||||||
"vendor.field.website": "Website",
|
"vendor.field.website": "Website",
|
||||||
"vendor.field.opening_balance_at": "Opening balance at",
|
"vendor.field.opening_balance_at": "Opening balance at",
|
||||||
@@ -403,7 +404,7 @@
|
|||||||
"vendor.field.created_at": "Created at",
|
"vendor.field.created_at": "Created at",
|
||||||
"vendor.field.balance": "Balance",
|
"vendor.field.balance": "Balance",
|
||||||
"vendor.field.status": "Status",
|
"vendor.field.status": "Status",
|
||||||
"vendor.field.currency": "Curreny",
|
"vendor.field.currency": "Currency",
|
||||||
"vendor.field.status.active": "Active",
|
"vendor.field.status.active": "Active",
|
||||||
"vendor.field.status.inactive": "Inactive",
|
"vendor.field.status.inactive": "Inactive",
|
||||||
"vendor.field.status.overdue": "Overdue",
|
"vendor.field.status.overdue": "Overdue",
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ export default class AccountsController extends BaseController {
|
|||||||
/**
|
/**
|
||||||
* Router constructor method.
|
* Router constructor method.
|
||||||
*/
|
*/
|
||||||
router() {
|
public router() {
|
||||||
const router = Router();
|
const router = Router();
|
||||||
|
|
||||||
router.get(
|
router.get(
|
||||||
@@ -98,7 +98,7 @@ export default class AccountsController extends BaseController {
|
|||||||
/**
|
/**
|
||||||
* Create account DTO Schema validation.
|
* Create account DTO Schema validation.
|
||||||
*/
|
*/
|
||||||
get createAccountDTOSchema() {
|
private get createAccountDTOSchema() {
|
||||||
return [
|
return [
|
||||||
check('name')
|
check('name')
|
||||||
.exists()
|
.exists()
|
||||||
@@ -131,7 +131,7 @@ export default class AccountsController extends BaseController {
|
|||||||
/**
|
/**
|
||||||
* Account DTO Schema validation.
|
* Account DTO Schema validation.
|
||||||
*/
|
*/
|
||||||
get editAccountDTOSchema() {
|
private get editAccountDTOSchema() {
|
||||||
return [
|
return [
|
||||||
check('name')
|
check('name')
|
||||||
.exists()
|
.exists()
|
||||||
@@ -160,14 +160,14 @@ export default class AccountsController extends BaseController {
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
get accountParamSchema() {
|
private get accountParamSchema() {
|
||||||
return [param('id').exists().isNumeric().toInt()];
|
return [param('id').exists().isNumeric().toInt()];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Accounts list validation schema.
|
* Accounts list validation schema.
|
||||||
*/
|
*/
|
||||||
get accountsListSchema() {
|
private get accountsListSchema() {
|
||||||
return [
|
return [
|
||||||
query('view_slug').optional({ nullable: true }).isString().trim(),
|
query('view_slug').optional({ nullable: true }).isString().trim(),
|
||||||
query('stringified_filter_roles').optional().isJSON(),
|
query('stringified_filter_roles').optional().isJSON(),
|
||||||
@@ -349,7 +349,7 @@ export default class AccountsController extends BaseController {
|
|||||||
// Filter query.
|
// Filter query.
|
||||||
const filter = {
|
const filter = {
|
||||||
sortOrder: 'desc',
|
sortOrder: 'desc',
|
||||||
columnSortBy: 'created_at',
|
columnSortBy: 'createdAt',
|
||||||
inactiveMode: false,
|
inactiveMode: false,
|
||||||
structure: IAccountsStructureType.Tree,
|
structure: IAccountsStructureType.Tree,
|
||||||
...this.matchedQueryData(req),
|
...this.matchedQueryData(req),
|
||||||
|
|||||||
@@ -160,10 +160,8 @@ export default class CustomersController extends ContactsController {
|
|||||||
try {
|
try {
|
||||||
const contact = await this.customersApplication.createCustomer(
|
const contact = await this.customersApplication.createCustomer(
|
||||||
tenantId,
|
tenantId,
|
||||||
contactDTO,
|
contactDTO
|
||||||
user
|
|
||||||
);
|
);
|
||||||
|
|
||||||
return res.status(200).send({
|
return res.status(200).send({
|
||||||
id: contact.id,
|
id: contact.id,
|
||||||
message: 'The customer has been created successfully.',
|
message: 'The customer has been created successfully.',
|
||||||
@@ -291,7 +289,7 @@ export default class CustomersController extends ContactsController {
|
|||||||
const filter = {
|
const filter = {
|
||||||
inactiveMode: false,
|
inactiveMode: false,
|
||||||
sortOrder: 'desc',
|
sortOrder: 'desc',
|
||||||
columnSortBy: 'created_at',
|
columnSortBy: 'createdAt',
|
||||||
page: 1,
|
page: 1,
|
||||||
pageSize: 12,
|
pageSize: 12,
|
||||||
...this.matchedQueryData(req),
|
...this.matchedQueryData(req),
|
||||||
|
|||||||
@@ -272,7 +272,7 @@ export default class VendorsController extends ContactsController {
|
|||||||
const vendorsFilter: IVendorsFilter = {
|
const vendorsFilter: IVendorsFilter = {
|
||||||
inactiveMode: false,
|
inactiveMode: false,
|
||||||
sortOrder: 'desc',
|
sortOrder: 'desc',
|
||||||
columnSortBy: 'created_at',
|
columnSortBy: 'createdAt',
|
||||||
page: 1,
|
page: 1,
|
||||||
pageSize: 12,
|
pageSize: 12,
|
||||||
...this.matchedQueryData(req),
|
...this.matchedQueryData(req),
|
||||||
|
|||||||
248
packages/server/src/api/controllers/Import/ImportController.ts
Normal file
248
packages/server/src/api/controllers/Import/ImportController.ts
Normal file
@@ -0,0 +1,248 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { Router, Request, Response, NextFunction } from 'express';
|
||||||
|
import { body, param, query } from 'express-validator';
|
||||||
|
import { defaultTo } from 'lodash';
|
||||||
|
import BaseController from '@/api/controllers/BaseController';
|
||||||
|
import { ServiceError } from '@/exceptions';
|
||||||
|
import { ImportResourceApplication } from '@/services/Import/ImportResourceApplication';
|
||||||
|
import { uploadImportFile } from './_utils';
|
||||||
|
import { parseJsonSafe } from '@/utils/parse-json-safe';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportController extends BaseController {
|
||||||
|
@Inject()
|
||||||
|
private importResourceApp: ImportResourceApplication;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Router constructor method.
|
||||||
|
*/
|
||||||
|
router() {
|
||||||
|
const router = Router();
|
||||||
|
|
||||||
|
router.post(
|
||||||
|
'/file',
|
||||||
|
uploadImportFile.single('file'),
|
||||||
|
this.importValidationSchema,
|
||||||
|
this.validationResult,
|
||||||
|
this.asyncMiddleware(this.fileUpload.bind(this)),
|
||||||
|
this.catchServiceErrors
|
||||||
|
);
|
||||||
|
router.post(
|
||||||
|
'/:import_id/import',
|
||||||
|
this.asyncMiddleware(this.import.bind(this)),
|
||||||
|
this.catchServiceErrors
|
||||||
|
);
|
||||||
|
router.post(
|
||||||
|
'/:import_id/mapping',
|
||||||
|
[
|
||||||
|
param('import_id').exists().isString(),
|
||||||
|
body('mapping').exists().isArray({ min: 1 }),
|
||||||
|
body('mapping.*.from').exists(),
|
||||||
|
body('mapping.*.to').exists(),
|
||||||
|
],
|
||||||
|
this.validationResult,
|
||||||
|
this.asyncMiddleware(this.mapping.bind(this)),
|
||||||
|
this.catchServiceErrors
|
||||||
|
);
|
||||||
|
router.get(
|
||||||
|
'/sample',
|
||||||
|
[query('resource').exists(), query('format').optional()],
|
||||||
|
this.downloadImportSample.bind(this),
|
||||||
|
this.catchServiceErrors
|
||||||
|
);
|
||||||
|
router.get(
|
||||||
|
'/:import_id',
|
||||||
|
this.asyncMiddleware(this.getImportFileMeta.bind(this)),
|
||||||
|
this.catchServiceErrors
|
||||||
|
);
|
||||||
|
router.get(
|
||||||
|
'/:import_id/preview',
|
||||||
|
this.asyncMiddleware(this.preview.bind(this)),
|
||||||
|
this.catchServiceErrors
|
||||||
|
);
|
||||||
|
return router;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Import validation schema.
|
||||||
|
* @returns {ValidationSchema[]}
|
||||||
|
*/
|
||||||
|
private get importValidationSchema() {
|
||||||
|
return [body('resource').exists(), body('params').optional()];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Imports xlsx/csv to the given resource type.
|
||||||
|
* @param {Request} req -
|
||||||
|
* @param {Response} res -
|
||||||
|
* @param {NextFunction} next -
|
||||||
|
*/
|
||||||
|
private async fileUpload(req: Request, res: Response, next: NextFunction) {
|
||||||
|
const { tenantId } = req;
|
||||||
|
const body = this.matchedBodyData(req);
|
||||||
|
const params = defaultTo(parseJsonSafe(body.params), {});
|
||||||
|
|
||||||
|
try {
|
||||||
|
const data = await this.importResourceApp.import(
|
||||||
|
tenantId,
|
||||||
|
body.resource,
|
||||||
|
req.file.filename,
|
||||||
|
params
|
||||||
|
);
|
||||||
|
return res.status(200).send(data);
|
||||||
|
} catch (error) {
|
||||||
|
next(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maps the columns of the imported file.
|
||||||
|
* @param {Request} req
|
||||||
|
* @param {Response} res
|
||||||
|
* @param {NextFunction} next
|
||||||
|
*/
|
||||||
|
private async mapping(req: Request, res: Response, next: NextFunction) {
|
||||||
|
const { tenantId } = req;
|
||||||
|
const { import_id: importId } = req.params;
|
||||||
|
const body = this.matchedBodyData(req);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const mapping = await this.importResourceApp.mapping(
|
||||||
|
tenantId,
|
||||||
|
importId,
|
||||||
|
body?.mapping
|
||||||
|
);
|
||||||
|
return res.status(200).send(mapping);
|
||||||
|
} catch (error) {
|
||||||
|
next(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Preview the imported file before actual importing.
|
||||||
|
* @param {Request} req
|
||||||
|
* @param {Response} res
|
||||||
|
* @param {NextFunction} next
|
||||||
|
*/
|
||||||
|
private async preview(req: Request, res: Response, next: NextFunction) {
|
||||||
|
const { tenantId } = req;
|
||||||
|
const { import_id: importId } = req.params;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const preview = await this.importResourceApp.preview(tenantId, importId);
|
||||||
|
|
||||||
|
return res.status(200).send(preview);
|
||||||
|
} catch (error) {
|
||||||
|
next(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Importing the imported file to the application storage.
|
||||||
|
* @param {Request} req
|
||||||
|
* @param {Response} res
|
||||||
|
* @param {NextFunction} next
|
||||||
|
*/
|
||||||
|
private async import(req: Request, res: Response, next: NextFunction) {
|
||||||
|
const { tenantId } = req;
|
||||||
|
const { import_id: importId } = req.params;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await this.importResourceApp.process(tenantId, importId);
|
||||||
|
|
||||||
|
return res.status(200).send(result);
|
||||||
|
} catch (error) {
|
||||||
|
next(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the csv/xlsx sample sheet of the given resource name.
|
||||||
|
* @param {Request} req
|
||||||
|
* @param {Response} res
|
||||||
|
* @param {NextFunction} next
|
||||||
|
*/
|
||||||
|
private async downloadImportSample(
|
||||||
|
req: Request,
|
||||||
|
res: Response,
|
||||||
|
next: NextFunction
|
||||||
|
) {
|
||||||
|
const { tenantId } = req;
|
||||||
|
const { format, resource } = this.matchedQueryData(req);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = this.importResourceApp.sample(tenantId, resource, format);
|
||||||
|
|
||||||
|
return res.status(200).send(result);
|
||||||
|
} catch (error) {
|
||||||
|
next(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the import file meta.
|
||||||
|
* @param {Request} req
|
||||||
|
* @param {Response} res
|
||||||
|
* @param {NextFunction} next
|
||||||
|
*/
|
||||||
|
private async getImportFileMeta(
|
||||||
|
req: Request,
|
||||||
|
res: Response,
|
||||||
|
next: NextFunction
|
||||||
|
) {
|
||||||
|
const { tenantId } = req;
|
||||||
|
const { import_id: importId } = req.params;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await this.importResourceApp.importMeta(
|
||||||
|
tenantId,
|
||||||
|
importId
|
||||||
|
);
|
||||||
|
return res.status(200).send(result);
|
||||||
|
} catch (error) {
|
||||||
|
next(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transforms service errors to response.
|
||||||
|
* @param {Error}
|
||||||
|
* @param {Request} req
|
||||||
|
* @param {Response} res
|
||||||
|
* @param {ServiceError} error
|
||||||
|
*/
|
||||||
|
private catchServiceErrors(
|
||||||
|
error,
|
||||||
|
req: Request,
|
||||||
|
res: Response,
|
||||||
|
next: NextFunction
|
||||||
|
) {
|
||||||
|
if (error instanceof ServiceError) {
|
||||||
|
if (error.errorType === 'INVALID_MAP_ATTRS') {
|
||||||
|
return res.status(400).send({
|
||||||
|
errors: [{ type: 'INVALID_MAP_ATTRS' }],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (error.errorType === 'DUPLICATED_FROM_MAP_ATTR') {
|
||||||
|
return res.status(400).send({
|
||||||
|
errors: [{ type: 'DUPLICATED_FROM_MAP_ATTR' }],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (error.errorType === 'DUPLICATED_TO_MAP_ATTR') {
|
||||||
|
return res.status(400).send({
|
||||||
|
errors: [{ type: 'DUPLICATED_TO_MAP_ATTR' }],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (error.errorType === 'IMPORTED_FILE_EXTENSION_INVALID') {
|
||||||
|
return res.status(400).send({
|
||||||
|
errors: [{ type: 'IMPORTED_FILE_EXTENSION_INVALID' }],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return res.status(400).send({
|
||||||
|
errors: [{ type: error.errorType }],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
next(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
21
packages/server/src/api/controllers/Import/_utils.ts
Normal file
21
packages/server/src/api/controllers/Import/_utils.ts
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
import Multer from 'multer';
|
||||||
|
import { ServiceError } from '@/exceptions';
|
||||||
|
|
||||||
|
export function allowSheetExtensions(req, file, cb) {
|
||||||
|
if (
|
||||||
|
file.mimetype !== 'text/csv' &&
|
||||||
|
file.mimetype !== 'application/vnd.ms-excel' &&
|
||||||
|
file.mimetype !== 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
|
||||||
|
) {
|
||||||
|
cb(new ServiceError('IMPORTED_FILE_EXTENSION_INVALID'));
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
cb(null, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
export const uploadImportFile = Multer({
|
||||||
|
dest: './public/imports',
|
||||||
|
limits: { fileSize: 5 * 1024 * 1024 },
|
||||||
|
fileFilter: allowSheetExtensions,
|
||||||
|
});
|
||||||
@@ -56,6 +56,7 @@ import { ProjectsController } from './controllers/Projects/Projects';
|
|||||||
import { ProjectTasksController } from './controllers/Projects/Tasks';
|
import { ProjectTasksController } from './controllers/Projects/Tasks';
|
||||||
import { ProjectTimesController } from './controllers/Projects/Times';
|
import { ProjectTimesController } from './controllers/Projects/Times';
|
||||||
import { TaxRatesController } from './controllers/TaxRates/TaxRates';
|
import { TaxRatesController } from './controllers/TaxRates/TaxRates';
|
||||||
|
import { ImportController } from './controllers/Import/ImportController';
|
||||||
import { BankingController } from './controllers/Banking/BankingController';
|
import { BankingController } from './controllers/Banking/BankingController';
|
||||||
import { Webhooks } from './controllers/Webhooks/Webhooks';
|
import { Webhooks } from './controllers/Webhooks/Webhooks';
|
||||||
|
|
||||||
@@ -135,6 +136,9 @@ export default () => {
|
|||||||
dashboard.use('/warehouses', Container.get(WarehousesController).router());
|
dashboard.use('/warehouses', Container.get(WarehousesController).router());
|
||||||
dashboard.use('/projects', Container.get(ProjectsController).router());
|
dashboard.use('/projects', Container.get(ProjectsController).router());
|
||||||
dashboard.use('/tax-rates', Container.get(TaxRatesController).router());
|
dashboard.use('/tax-rates', Container.get(TaxRatesController).router());
|
||||||
|
|
||||||
|
dashboard.use('/import', Container.get(ImportController).router());
|
||||||
|
|
||||||
dashboard.use('/', Container.get(ProjectTasksController).router());
|
dashboard.use('/', Container.get(ProjectTasksController).router());
|
||||||
dashboard.use('/', Container.get(ProjectTimesController).router());
|
dashboard.use('/', Container.get(ProjectTimesController).router());
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,16 @@
|
|||||||
|
exports.up = function (knex) {
|
||||||
|
return knex.schema.createTable('imports', (table) => {
|
||||||
|
table.increments();
|
||||||
|
table.string('filename');
|
||||||
|
table.string('import_id');
|
||||||
|
table.string('resource');
|
||||||
|
table.json('columns');
|
||||||
|
table.json('mapping');
|
||||||
|
table.json('params');
|
||||||
|
table.timestamps();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
exports.down = function (knex) {
|
||||||
|
return knex.schema.dropTableIfExists('imports');
|
||||||
|
};
|
||||||
@@ -34,20 +34,40 @@ export interface IModelMetaFieldCommon {
|
|||||||
columnable?: boolean;
|
columnable?: boolean;
|
||||||
fieldType: IModelColumnType;
|
fieldType: IModelColumnType;
|
||||||
customQuery?: Function;
|
customQuery?: Function;
|
||||||
|
required?: boolean;
|
||||||
|
importHint?: string;
|
||||||
|
order?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IModelMetaFieldNumber {
|
export interface IModelMetaFieldText {
|
||||||
fieldType: 'number';
|
fieldType: 'text';
|
||||||
minLength?: number;
|
minLength?: number;
|
||||||
maxLength?: number;
|
maxLength?: number;
|
||||||
}
|
}
|
||||||
|
export interface IModelMetaFieldBoolean {
|
||||||
export interface IModelMetaFieldOther {
|
fieldType: 'boolean';
|
||||||
fieldType: 'text' | 'boolean';
|
}
|
||||||
|
export interface IModelMetaFieldNumber {
|
||||||
|
fieldType: 'number';
|
||||||
|
min?: number;
|
||||||
|
max?: number;
|
||||||
|
}
|
||||||
|
export interface IModelMetaFieldDate {
|
||||||
|
fieldType: 'date';
|
||||||
|
}
|
||||||
|
export interface IModelMetaFieldUrl {
|
||||||
|
fieldType: 'url';
|
||||||
}
|
}
|
||||||
|
|
||||||
export type IModelMetaField = IModelMetaFieldCommon &
|
export type IModelMetaField = IModelMetaFieldCommon &
|
||||||
(IModelMetaFieldOther | IModelMetaEnumerationField | IModelMetaRelationField);
|
(
|
||||||
|
| IModelMetaFieldText
|
||||||
|
| IModelMetaFieldNumber
|
||||||
|
| IModelMetaFieldBoolean
|
||||||
|
| IModelMetaFieldDate
|
||||||
|
| IModelMetaFieldUrl
|
||||||
|
| IModelMetaEnumerationField
|
||||||
|
| IModelMetaRelationField
|
||||||
|
);
|
||||||
|
|
||||||
export interface IModelMetaEnumerationOption {
|
export interface IModelMetaEnumerationOption {
|
||||||
key: string;
|
key: string;
|
||||||
@@ -70,12 +90,12 @@ export interface IModelMetaRelationEnumerationField {
|
|||||||
relationEntityKey: string;
|
relationEntityKey: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type IModelMetaRelationField = IModelMetaRelationFieldCommon & (
|
export type IModelMetaRelationField = IModelMetaRelationFieldCommon &
|
||||||
IModelMetaRelationEnumerationField
|
IModelMetaRelationEnumerationField;
|
||||||
);
|
|
||||||
|
|
||||||
export interface IModelMeta {
|
export interface IModelMeta {
|
||||||
defaultFilterField: string;
|
defaultFilterField: string;
|
||||||
defaultSort: IModelMetaDefaultSort;
|
defaultSort: IModelMetaDefaultSort;
|
||||||
|
importable?: boolean;
|
||||||
fields: { [key: string]: IModelMetaField };
|
fields: { [key: string]: IModelMetaField };
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ import helmet from 'helmet';
|
|||||||
import boom from 'express-boom';
|
import boom from 'express-boom';
|
||||||
import errorHandler from 'errorhandler';
|
import errorHandler from 'errorhandler';
|
||||||
import bodyParser from 'body-parser';
|
import bodyParser from 'body-parser';
|
||||||
import fileUpload from 'express-fileupload';
|
|
||||||
import { Server } from 'socket.io';
|
import { Server } from 'socket.io';
|
||||||
import Container from 'typedi';
|
import Container from 'typedi';
|
||||||
import routes from 'api';
|
import routes from 'api';
|
||||||
@@ -47,13 +46,6 @@ export default ({ app }) => {
|
|||||||
|
|
||||||
app.use('/public', express.static(path.join(global.__storage_dir)));
|
app.use('/public', express.static(path.join(global.__storage_dir)));
|
||||||
|
|
||||||
// Handle multi-media requests.
|
|
||||||
app.use(
|
|
||||||
fileUpload({
|
|
||||||
createParentPath: true,
|
|
||||||
})
|
|
||||||
);
|
|
||||||
|
|
||||||
// Logger middleware.
|
// Logger middleware.
|
||||||
app.use(LoggerMiddleware);
|
app.use(LoggerMiddleware);
|
||||||
|
|
||||||
|
|||||||
@@ -61,6 +61,7 @@ import Task from 'models/Task';
|
|||||||
import TaxRate from 'models/TaxRate';
|
import TaxRate from 'models/TaxRate';
|
||||||
import TaxRateTransaction from 'models/TaxRateTransaction';
|
import TaxRateTransaction from 'models/TaxRateTransaction';
|
||||||
import Attachment from 'models/Attachment';
|
import Attachment from 'models/Attachment';
|
||||||
|
import Import from 'models/Import';
|
||||||
import PlaidItem from 'models/PlaidItem';
|
import PlaidItem from 'models/PlaidItem';
|
||||||
import UncategorizedCashflowTransaction from 'models/UncategorizedCashflowTransaction';
|
import UncategorizedCashflowTransaction from 'models/UncategorizedCashflowTransaction';
|
||||||
|
|
||||||
@@ -127,6 +128,7 @@ export default (knex) => {
|
|||||||
TaxRate,
|
TaxRate,
|
||||||
TaxRateTransaction,
|
TaxRateTransaction,
|
||||||
Attachment,
|
Attachment,
|
||||||
|
Import,
|
||||||
PlaidItem,
|
PlaidItem,
|
||||||
UncategorizedCashflowTransaction
|
UncategorizedCashflowTransaction
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -6,16 +6,24 @@ export default {
|
|||||||
sortOrder: 'DESC',
|
sortOrder: 'DESC',
|
||||||
sortField: 'name',
|
sortField: 'name',
|
||||||
},
|
},
|
||||||
|
importable: true,
|
||||||
fields: {
|
fields: {
|
||||||
name: {
|
name: {
|
||||||
name: 'account.field.name',
|
name: 'account.field.name',
|
||||||
column: 'name',
|
column: 'name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
unique: true,
|
||||||
|
required: true,
|
||||||
|
importable: true,
|
||||||
|
exportable: true,
|
||||||
|
order: 1,
|
||||||
},
|
},
|
||||||
description: {
|
description: {
|
||||||
name: 'account.field.description',
|
name: 'account.field.description',
|
||||||
column: 'description',
|
column: 'description',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
exportable: true,
|
||||||
},
|
},
|
||||||
slug: {
|
slug: {
|
||||||
name: 'account.field.slug',
|
name: 'account.field.slug',
|
||||||
@@ -23,13 +31,19 @@ export default {
|
|||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
columnable: false,
|
columnable: false,
|
||||||
filterable: false,
|
filterable: false,
|
||||||
|
importable: false,
|
||||||
},
|
},
|
||||||
code: {
|
code: {
|
||||||
name: 'account.field.code',
|
name: 'account.field.code',
|
||||||
column: 'code',
|
column: 'code',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
exportable: true,
|
||||||
|
importable: true,
|
||||||
|
minLength: 3,
|
||||||
|
maxLength: 6,
|
||||||
|
importHint: 'Unique number to identify the account.',
|
||||||
},
|
},
|
||||||
root_type: {
|
rootType: {
|
||||||
name: 'account.field.root_type',
|
name: 'account.field.root_type',
|
||||||
fieldType: 'enumeration',
|
fieldType: 'enumeration',
|
||||||
options: [
|
options: [
|
||||||
@@ -41,6 +55,7 @@ export default {
|
|||||||
],
|
],
|
||||||
filterCustomQuery: RootTypeFieldFilterQuery,
|
filterCustomQuery: RootTypeFieldFilterQuery,
|
||||||
sortable: false,
|
sortable: false,
|
||||||
|
importable: false,
|
||||||
},
|
},
|
||||||
normal: {
|
normal: {
|
||||||
name: 'account.field.normal',
|
name: 'account.field.normal',
|
||||||
@@ -51,37 +66,56 @@ export default {
|
|||||||
],
|
],
|
||||||
filterCustomQuery: NormalTypeFieldFilterQuery,
|
filterCustomQuery: NormalTypeFieldFilterQuery,
|
||||||
sortable: false,
|
sortable: false,
|
||||||
|
importable: false,
|
||||||
},
|
},
|
||||||
type: {
|
accountType: {
|
||||||
name: 'account.field.type',
|
name: 'account.field.type',
|
||||||
column: 'account_type',
|
column: 'account_type',
|
||||||
fieldType: 'enumeration',
|
fieldType: 'enumeration',
|
||||||
options: ACCOUNT_TYPES.map((accountType) => ({
|
options: ACCOUNT_TYPES.map((accountType) => ({
|
||||||
label: accountType.label,
|
label: accountType.label,
|
||||||
key: accountType.key
|
key: accountType.key,
|
||||||
})),
|
})),
|
||||||
|
required: true,
|
||||||
|
importable: true,
|
||||||
|
exportable: true,
|
||||||
|
order: 2,
|
||||||
},
|
},
|
||||||
active: {
|
active: {
|
||||||
name: 'account.field.active',
|
name: 'account.field.active',
|
||||||
column: 'active',
|
column: 'active',
|
||||||
fieldType: 'boolean',
|
fieldType: 'boolean',
|
||||||
filterable: false,
|
filterable: false,
|
||||||
|
exportable: true,
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
balance: {
|
balance: {
|
||||||
name: 'account.field.balance',
|
name: 'account.field.balance',
|
||||||
column: 'amount',
|
column: 'amount',
|
||||||
fieldType: 'number',
|
fieldType: 'number',
|
||||||
|
importable: false,
|
||||||
},
|
},
|
||||||
currency: {
|
currencyCode: {
|
||||||
name: 'account.field.currency',
|
name: 'account.field.currency',
|
||||||
column: 'currency_code',
|
column: 'currency_code',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
filterable: false,
|
filterable: false,
|
||||||
|
importable: true,
|
||||||
|
exportable: true,
|
||||||
},
|
},
|
||||||
created_at: {
|
parentAccount: {
|
||||||
|
name: 'account.field.parent_account',
|
||||||
|
column: 'parent_account_id',
|
||||||
|
fieldType: 'relation',
|
||||||
|
to: { model: 'Account', to: 'id' },
|
||||||
|
importable: false,
|
||||||
|
},
|
||||||
|
createdAt: {
|
||||||
name: 'account.field.created_at',
|
name: 'account.field.created_at',
|
||||||
column: 'created_at',
|
column: 'created_at',
|
||||||
fieldType: 'date',
|
fieldType: 'date',
|
||||||
|
importable: false,
|
||||||
|
exportable: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,70 +1,112 @@
|
|||||||
export default {
|
export default {
|
||||||
|
importable: true,
|
||||||
|
defaultFilterField: 'displayName',
|
||||||
|
defaultSort: {
|
||||||
|
sortOrder: 'DESC',
|
||||||
|
sortField: 'createdAt',
|
||||||
|
},
|
||||||
fields: {
|
fields: {
|
||||||
first_name: {
|
customerType: {
|
||||||
|
name: 'Customer Type',
|
||||||
|
column: 'contact_type',
|
||||||
|
fieldType: 'enumeration',
|
||||||
|
options: [
|
||||||
|
{ key: 'business', label: 'Business' },
|
||||||
|
{ key: 'individual', label: 'Individual' },
|
||||||
|
],
|
||||||
|
importable: true,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
firstName: {
|
||||||
name: 'customer.field.first_name',
|
name: 'customer.field.first_name',
|
||||||
column: 'first_name',
|
column: 'first_name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
last_name: {
|
lastName: {
|
||||||
name: 'customer.field.last_name',
|
name: 'customer.field.last_name',
|
||||||
column: 'last_name',
|
column: 'last_name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
display_name: {
|
displayName: {
|
||||||
name: 'customer.field.display_name',
|
name: 'customer.field.display_name',
|
||||||
column: 'display_name',
|
column: 'display_name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
required: true,
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
email: {
|
email: {
|
||||||
name: 'customer.field.email',
|
name: 'customer.field.email',
|
||||||
column: 'email',
|
column: 'email',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
work_phone: {
|
workPhone: {
|
||||||
name: 'customer.field.work_phone',
|
name: 'customer.field.work_phone',
|
||||||
column: 'work_phone',
|
column: 'work_phone',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
personal_phone: {
|
personalPhone: {
|
||||||
name: 'customer.field.personal_phone',
|
name: 'customer.field.personal_phone',
|
||||||
column: 'personal_phone',
|
column: 'personal_phone',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
company_name: {
|
companyName: {
|
||||||
name: 'customer.field.company_name',
|
name: 'customer.field.company_name',
|
||||||
column: 'company_name',
|
column: 'company_name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
website: {
|
website: {
|
||||||
name: 'customer.field.website',
|
name: 'customer.field.website',
|
||||||
column: 'website',
|
column: 'website',
|
||||||
fieldType: 'text',
|
fieldType: 'url',
|
||||||
},
|
importable: true,
|
||||||
created_at: {
|
|
||||||
name: 'customer.field.created_at',
|
|
||||||
column: 'created_at',
|
|
||||||
fieldType: 'date',
|
|
||||||
},
|
},
|
||||||
balance: {
|
balance: {
|
||||||
name: 'customer.field.balance',
|
name: 'customer.field.balance',
|
||||||
column: 'balance',
|
column: 'balance',
|
||||||
fieldType: 'number',
|
fieldType: 'number',
|
||||||
},
|
},
|
||||||
opening_balance: {
|
openingBalance: {
|
||||||
name: 'customer.field.opening_balance',
|
name: 'customer.field.opening_balance',
|
||||||
column: 'opening_balance',
|
column: 'opening_balance',
|
||||||
fieldType: 'number',
|
fieldType: 'number',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
opening_balance_at: {
|
openingBalanceAt: {
|
||||||
name: 'customer.field.opening_balance_at',
|
name: 'customer.field.opening_balance_at',
|
||||||
column: 'opening_balance_at',
|
column: 'opening_balance_at',
|
||||||
filterable: false,
|
filterable: false,
|
||||||
fieldType: 'date',
|
fieldType: 'date',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
currency_code: {
|
openingBalanceExchangeRate: {
|
||||||
|
name: 'Opening Balance Ex. Rate',
|
||||||
|
column: 'opening_balance_exchange_rate',
|
||||||
|
fieldType: 'number',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
currencyCode: {
|
||||||
name: 'customer.field.currency',
|
name: 'customer.field.currency',
|
||||||
column: 'currency_code',
|
column: 'currency_code',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
note: {
|
||||||
|
name: 'Note',
|
||||||
|
column: 'note',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
active: {
|
||||||
|
name: 'Active',
|
||||||
|
column: 'active',
|
||||||
|
fieldType: 'boolean',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
status: {
|
status: {
|
||||||
name: 'customer.field.status',
|
name: 'customer.field.status',
|
||||||
@@ -77,6 +119,98 @@ export default {
|
|||||||
],
|
],
|
||||||
filterCustomQuery: statusFieldFilterQuery,
|
filterCustomQuery: statusFieldFilterQuery,
|
||||||
},
|
},
|
||||||
|
// Billing Address
|
||||||
|
billingAddress1: {
|
||||||
|
name: 'Billing Address 1',
|
||||||
|
column: 'billing_address1',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddress2: {
|
||||||
|
name: 'Billing Address 2',
|
||||||
|
column: 'billing_address2',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressCity: {
|
||||||
|
name: 'Billing Address City',
|
||||||
|
column: 'billing_address_city',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressCountry: {
|
||||||
|
name: 'Billing Address Country',
|
||||||
|
column: 'billing_address_country',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressPostcode: {
|
||||||
|
name: 'Billing Address Postcode',
|
||||||
|
column: 'billing_address_postcode',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressState: {
|
||||||
|
name: 'Billing Address State',
|
||||||
|
column: 'billing_address_state',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressPhone: {
|
||||||
|
name: 'Billing Address Phone',
|
||||||
|
column: 'billing_address_phone',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
// Shipping Address
|
||||||
|
shippingAddress1: {
|
||||||
|
name: 'Shipping Address 1',
|
||||||
|
column: 'shipping_address1',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddress2: {
|
||||||
|
name: 'Shipping Address 2',
|
||||||
|
column: 'shipping_address2',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressCity: {
|
||||||
|
name: 'Shipping Address City',
|
||||||
|
column: 'shipping_address_city',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressCountry: {
|
||||||
|
name: 'Shipping Address Country',
|
||||||
|
column: 'shipping_address_country',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressPostcode: {
|
||||||
|
name: 'Shipping Address Postcode',
|
||||||
|
column: 'shipping_address_postcode',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressPhone: {
|
||||||
|
name: 'Shipping Address Phone',
|
||||||
|
column: 'shipping_address_phone',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressState: {
|
||||||
|
name: 'Shipping Address State',
|
||||||
|
column: 'shipping_address_state',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
//
|
||||||
|
createdAt: {
|
||||||
|
name: 'customer.field.created_at',
|
||||||
|
column: 'created_at',
|
||||||
|
fieldType: 'date',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
69
packages/server/src/models/Import.ts
Normal file
69
packages/server/src/models/Import.ts
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
import TenantModel from 'models/TenantModel';
|
||||||
|
|
||||||
|
export default class Import extends TenantModel {
|
||||||
|
resource!: string;
|
||||||
|
mapping!: string;
|
||||||
|
columns!: string;
|
||||||
|
params!: Record<string, any>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Table name.
|
||||||
|
*/
|
||||||
|
static get tableName() {
|
||||||
|
return 'imports';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Virtual attributes.
|
||||||
|
*/
|
||||||
|
static get virtualAttributes() {
|
||||||
|
return ['mappingParsed'];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Timestamps columns.
|
||||||
|
*/
|
||||||
|
get timestamps() {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Relationship mapping.
|
||||||
|
*/
|
||||||
|
static get relationMappings() {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detarmines whether the import is mapped.
|
||||||
|
* @returns {boolean}
|
||||||
|
*/
|
||||||
|
public get isMapped() {
|
||||||
|
return Boolean(this.mapping);
|
||||||
|
}
|
||||||
|
|
||||||
|
public get columnsParsed() {
|
||||||
|
try {
|
||||||
|
return JSON.parse(this.columns);
|
||||||
|
} catch {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public get paramsParsed() {
|
||||||
|
try {
|
||||||
|
return JSON.parse(this.params);
|
||||||
|
} catch {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public get mappingParsed() {
|
||||||
|
try {
|
||||||
|
return JSON.parse(this.mapping);
|
||||||
|
} catch {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,51 +1,59 @@
|
|||||||
export default {
|
export default {
|
||||||
|
importable: true,
|
||||||
defaultFilterField: 'name',
|
defaultFilterField: 'name',
|
||||||
defaultSort: {
|
defaultSort: {
|
||||||
sortField: 'name',
|
sortField: 'name',
|
||||||
sortOrder: 'DESC',
|
sortOrder: 'DESC',
|
||||||
},
|
},
|
||||||
fields: {
|
fields: {
|
||||||
'type': {
|
type: {
|
||||||
name: 'item.field.type',
|
name: 'item.field.type',
|
||||||
column: 'type',
|
column: 'type',
|
||||||
fieldType: 'enumeration',
|
fieldType: 'enumeration',
|
||||||
options: [
|
options: [
|
||||||
{ key: 'inventory', label: 'item.field.type.inventory', },
|
{ key: 'inventory', label: 'item.field.type.inventory' },
|
||||||
{ key: 'service', label: 'item.field.type.service' },
|
{ key: 'service', label: 'item.field.type.service' },
|
||||||
{ key: 'non-inventory', label: 'item.field.type.non-inventory', },
|
{ key: 'non-inventory', label: 'item.field.type.non-inventory' },
|
||||||
],
|
],
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'name': {
|
name: {
|
||||||
name: 'item.field.name',
|
name: 'item.field.name',
|
||||||
column: 'name',
|
column: 'name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'code': {
|
code: {
|
||||||
name: 'item.field.code',
|
name: 'item.field.code',
|
||||||
column: 'code',
|
column: 'code',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'sellable': {
|
sellable: {
|
||||||
name: 'item.field.sellable',
|
name: 'item.field.sellable',
|
||||||
column: 'sellable',
|
column: 'sellable',
|
||||||
fieldType: 'boolean',
|
fieldType: 'boolean',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'purchasable': {
|
purchasable: {
|
||||||
name: 'item.field.purchasable',
|
name: 'item.field.purchasable',
|
||||||
column: 'purchasable',
|
column: 'purchasable',
|
||||||
fieldType: 'boolean',
|
fieldType: 'boolean',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'sell_price': {
|
sellPrice: {
|
||||||
name: 'item.field.cost_price',
|
name: 'item.field.cost_price',
|
||||||
column: 'sell_price',
|
column: 'sell_price',
|
||||||
fieldType: 'number',
|
fieldType: 'number',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'cost_price': {
|
costPrice: {
|
||||||
name: 'item.field.cost_account',
|
name: 'item.field.cost_account',
|
||||||
column: 'cost_price',
|
column: 'cost_price',
|
||||||
fieldType: 'number',
|
fieldType: 'number',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'cost_account': {
|
costAccount: {
|
||||||
name: 'item.field.sell_account',
|
name: 'item.field.sell_account',
|
||||||
column: 'cost_account_id',
|
column: 'cost_account_id',
|
||||||
fieldType: 'relation',
|
fieldType: 'relation',
|
||||||
@@ -55,8 +63,10 @@ export default {
|
|||||||
|
|
||||||
relationEntityLabel: 'name',
|
relationEntityLabel: 'name',
|
||||||
relationEntityKey: 'slug',
|
relationEntityKey: 'slug',
|
||||||
|
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'sell_account': {
|
sellAccount: {
|
||||||
name: 'item.field.sell_description',
|
name: 'item.field.sell_description',
|
||||||
column: 'sell_account_id',
|
column: 'sell_account_id',
|
||||||
fieldType: 'relation',
|
fieldType: 'relation',
|
||||||
@@ -66,8 +76,10 @@ export default {
|
|||||||
|
|
||||||
relationEntityLabel: 'name',
|
relationEntityLabel: 'name',
|
||||||
relationEntityKey: 'slug',
|
relationEntityKey: 'slug',
|
||||||
|
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'inventory_account': {
|
inventoryAccount: {
|
||||||
name: 'item.field.inventory_account',
|
name: 'item.field.inventory_account',
|
||||||
column: 'inventory_account_id',
|
column: 'inventory_account_id',
|
||||||
|
|
||||||
@@ -76,28 +88,34 @@ export default {
|
|||||||
|
|
||||||
relationEntityLabel: 'name',
|
relationEntityLabel: 'name',
|
||||||
relationEntityKey: 'slug',
|
relationEntityKey: 'slug',
|
||||||
|
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'sell_description': {
|
sellDescription: {
|
||||||
name: 'Sell description',
|
name: 'Sell description',
|
||||||
column: 'sell_description',
|
column: 'sell_description',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'purchase_description': {
|
purchaseDescription: {
|
||||||
name: 'Purchase description',
|
name: 'Purchase description',
|
||||||
column: 'purchase_description',
|
column: 'purchase_description',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'quantity_on_hand': {
|
quantityOnHand: {
|
||||||
name: 'item.field.quantity_on_hand',
|
name: 'item.field.quantity_on_hand',
|
||||||
column: 'quantity_on_hand',
|
column: 'quantity_on_hand',
|
||||||
fieldType: 'number',
|
fieldType: 'number',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'note': {
|
note: {
|
||||||
name: 'item.field.note',
|
name: 'item.field.note',
|
||||||
column: 'note',
|
column: 'note',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'category': {
|
category: {
|
||||||
name: 'item.field.category',
|
name: 'item.field.category',
|
||||||
column: 'category_id',
|
column: 'category_id',
|
||||||
|
|
||||||
@@ -106,14 +124,15 @@ export default {
|
|||||||
|
|
||||||
relationEntityLabel: 'name',
|
relationEntityLabel: 'name',
|
||||||
relationEntityKey: 'id',
|
relationEntityKey: 'id',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
'active': {
|
active: {
|
||||||
name: 'item.field.active',
|
name: 'item.field.active',
|
||||||
column: 'active',
|
column: 'active',
|
||||||
fieldType: 'boolean',
|
fieldType: 'boolean',
|
||||||
filterable: false,
|
importable: true,
|
||||||
},
|
},
|
||||||
'created_at': {
|
createdAt: {
|
||||||
name: 'item.field.created_at',
|
name: 'item.field.created_at',
|
||||||
column: 'created_at',
|
column: 'created_at',
|
||||||
columnType: 'date',
|
columnType: 'date',
|
||||||
|
|||||||
@@ -0,0 +1,54 @@
|
|||||||
|
export default {
|
||||||
|
defaultFilterField: 'createdAt',
|
||||||
|
defaultSort: {
|
||||||
|
sortOrder: 'DESC',
|
||||||
|
sortField: 'createdAt',
|
||||||
|
},
|
||||||
|
importable: true,
|
||||||
|
fields: {
|
||||||
|
date: {
|
||||||
|
name: 'Date',
|
||||||
|
column: 'date',
|
||||||
|
fieldType: 'date',
|
||||||
|
importable: true,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
payee: {
|
||||||
|
name: 'Payee',
|
||||||
|
column: 'payee',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
description: {
|
||||||
|
name: 'Description',
|
||||||
|
column: 'description',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
referenceNo: {
|
||||||
|
name: 'Reference No.',
|
||||||
|
column: 'reference_no',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
amount: {
|
||||||
|
name: 'Amount',
|
||||||
|
column: 'Amount',
|
||||||
|
fieldType: 'numeric',
|
||||||
|
required: true,
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
account: {
|
||||||
|
name: 'Account',
|
||||||
|
column: 'account_id',
|
||||||
|
fieldType: 'relation',
|
||||||
|
to: { model: 'Account', to: 'id' },
|
||||||
|
},
|
||||||
|
createdAt: {
|
||||||
|
name: 'Created At',
|
||||||
|
column: 'createdAt',
|
||||||
|
fieldType: 'date',
|
||||||
|
importable: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
@@ -1,9 +1,15 @@
|
|||||||
/* eslint-disable global-require */
|
/* eslint-disable global-require */
|
||||||
|
import * as R from 'ramda';
|
||||||
|
import { Model, ModelOptions, QueryContext, mixin } from 'objection';
|
||||||
import TenantModel from 'models/TenantModel';
|
import TenantModel from 'models/TenantModel';
|
||||||
import { Model, ModelOptions, QueryContext } from 'objection';
|
import ModelSettings from './ModelSetting';
|
||||||
import Account from './Account';
|
import Account from './Account';
|
||||||
|
import UncategorizedCashflowTransactionMeta from './UncategorizedCashflowTransaction.meta';
|
||||||
|
|
||||||
export default class UncategorizedCashflowTransaction extends TenantModel {
|
export default class UncategorizedCashflowTransaction extends mixin(
|
||||||
|
TenantModel,
|
||||||
|
[ModelSettings]
|
||||||
|
) {
|
||||||
id!: number;
|
id!: number;
|
||||||
amount!: number;
|
amount!: number;
|
||||||
categorized!: boolean;
|
categorized!: boolean;
|
||||||
@@ -35,6 +41,10 @@ export default class UncategorizedCashflowTransaction extends TenantModel {
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static get meta() {
|
||||||
|
return UncategorizedCashflowTransactionMeta;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves the withdrawal amount.
|
* Retrieves the withdrawal amount.
|
||||||
* @returns {number}
|
* @returns {number}
|
||||||
|
|||||||
@@ -1,74 +1,100 @@
|
|||||||
export default {
|
export default {
|
||||||
defaultFilterField: 'display_name',
|
defaultFilterField: 'displayName',
|
||||||
defaultSort: {
|
defaultSort: {
|
||||||
sortOrder: 'DESC',
|
sortOrder: 'DESC',
|
||||||
sortField: 'created_at',
|
sortField: 'createdAt',
|
||||||
},
|
},
|
||||||
|
importable: true,
|
||||||
fields: {
|
fields: {
|
||||||
first_name: {
|
firstName: {
|
||||||
name: 'vendor.field.first_name',
|
name: 'vendor.field.first_name',
|
||||||
column: 'first_name',
|
column: 'first_name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
last_name: {
|
lastName: {
|
||||||
name: 'vendor.field.last_name',
|
name: 'vendor.field.last_name',
|
||||||
column: 'last_name',
|
column: 'last_name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
display_name: {
|
displayName: {
|
||||||
name: 'vendor.field.display_name',
|
name: 'vendor.field.display_name',
|
||||||
column: 'display_name',
|
column: 'display_name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
required: true,
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
email: {
|
email: {
|
||||||
name: 'vendor.field.email',
|
name: 'vendor.field.email',
|
||||||
column: 'email',
|
column: 'email',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
work_phone: {
|
workPhone: {
|
||||||
name: 'vendor.field.work_phone',
|
name: 'vendor.field.work_phone',
|
||||||
column: 'work_phone',
|
column: 'work_phone',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
personal_phone: {
|
personalPhone: {
|
||||||
name: 'vendor.field.personal_pone',
|
name: 'vendor.field.personal_phone',
|
||||||
column: 'personal_phone',
|
column: 'personal_phone',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
company_name: {
|
companyName: {
|
||||||
name: 'vendor.field.company_name',
|
name: 'vendor.field.company_name',
|
||||||
column: 'company_name',
|
column: 'company_name',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
website: {
|
website: {
|
||||||
name: 'vendor.field.website',
|
name: 'vendor.field.website',
|
||||||
column: 'website',
|
column: 'website',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
},
|
importable: true,
|
||||||
created_at: {
|
|
||||||
name: 'vendor.field.created_at',
|
|
||||||
column: 'created_at',
|
|
||||||
fieldType: 'date',
|
|
||||||
},
|
},
|
||||||
balance: {
|
balance: {
|
||||||
name: 'vendor.field.balance',
|
name: 'vendor.field.balance',
|
||||||
column: 'balance',
|
column: 'balance',
|
||||||
fieldType: 'number',
|
fieldType: 'number',
|
||||||
},
|
},
|
||||||
opening_balance: {
|
openingBalance: {
|
||||||
name: 'vendor.field.opening_balance',
|
name: 'vendor.field.opening_balance',
|
||||||
column: 'opening_balance',
|
column: 'opening_balance',
|
||||||
fieldType: 'number',
|
fieldType: 'number',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
opening_balance_at: {
|
openingBalanceAt: {
|
||||||
name: 'vendor.field.opening_balance_at',
|
name: 'vendor.field.opening_balance_at',
|
||||||
column: 'opening_balance_at',
|
column: 'opening_balance_at',
|
||||||
fieldType: 'date',
|
fieldType: 'date',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
currency_code: {
|
openingBalanceExchangeRate: {
|
||||||
|
name: 'Opening Balance Ex. Rate',
|
||||||
|
column: 'opening_balance_exchange_rate',
|
||||||
|
fieldType: 'number',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
currencyCode: {
|
||||||
name: 'vendor.field.currency',
|
name: 'vendor.field.currency',
|
||||||
column: 'currency_code',
|
column: 'currency_code',
|
||||||
fieldType: 'text',
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
note: {
|
||||||
|
name: 'Note',
|
||||||
|
column: 'note',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
active: {
|
||||||
|
name: 'Active',
|
||||||
|
column: 'active',
|
||||||
|
fieldType: 'boolean',
|
||||||
|
importable: true,
|
||||||
},
|
},
|
||||||
status: {
|
status: {
|
||||||
name: 'vendor.field.status',
|
name: 'vendor.field.status',
|
||||||
@@ -88,5 +114,96 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
// Billing Address
|
||||||
|
billingAddress1: {
|
||||||
|
name: 'Billing Address 1',
|
||||||
|
column: 'billing_address1',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddress2: {
|
||||||
|
name: 'Billing Address 2',
|
||||||
|
column: 'billing_address2',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressCity: {
|
||||||
|
name: 'Billing Address City',
|
||||||
|
column: 'billing_address_city',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressCountry: {
|
||||||
|
name: 'Billing Address Country',
|
||||||
|
column: 'billing_address_country',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressPostcode: {
|
||||||
|
name: 'Billing Address Postcode',
|
||||||
|
column: 'billing_address_postcode',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressState: {
|
||||||
|
name: 'Billing Address State',
|
||||||
|
column: 'billing_address_state',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
billingAddressPhone: {
|
||||||
|
name: 'Billing Address Phone',
|
||||||
|
column: 'billing_address_phone',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
// Shipping Address
|
||||||
|
shippingAddress1: {
|
||||||
|
name: 'Shipping Address 1',
|
||||||
|
column: 'shipping_address1',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddress2: {
|
||||||
|
name: 'Shipping Address 2',
|
||||||
|
column: 'shipping_address2',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressCity: {
|
||||||
|
name: 'Shipping Address City',
|
||||||
|
column: 'shipping_address_city',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressCountry: {
|
||||||
|
name: 'Shipping Address Country',
|
||||||
|
column: 'shipping_address_country',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressPostcode: {
|
||||||
|
name: 'Shipping Address Postcode',
|
||||||
|
column: 'shipping_address_postcode',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressState: {
|
||||||
|
name: 'Shipping Address State',
|
||||||
|
column: 'shipping_address_state',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
shippingAddressPhone: {
|
||||||
|
name: 'Shipping Address Phone',
|
||||||
|
column: 'shipping_address_phone',
|
||||||
|
fieldType: 'text',
|
||||||
|
importable: true,
|
||||||
|
},
|
||||||
|
createdAt: {
|
||||||
|
name: 'vendor.field.created_at',
|
||||||
|
column: 'created_at',
|
||||||
|
fieldType: 'date',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ import { ActivateAccount } from './ActivateAccount';
|
|||||||
import { GetAccounts } from './GetAccounts';
|
import { GetAccounts } from './GetAccounts';
|
||||||
import { GetAccount } from './GetAccount';
|
import { GetAccount } from './GetAccount';
|
||||||
import { GetAccountTransactions } from './GetAccountTransactions';
|
import { GetAccountTransactions } from './GetAccountTransactions';
|
||||||
|
import { Knex } from 'knex';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class AccountsApplication {
|
export class AccountsApplication {
|
||||||
@@ -48,9 +49,10 @@ export class AccountsApplication {
|
|||||||
*/
|
*/
|
||||||
public createAccount = (
|
public createAccount = (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
accountDTO: IAccountCreateDTO
|
accountDTO: IAccountCreateDTO,
|
||||||
|
trx?: Knex.Transaction
|
||||||
): Promise<IAccount> => {
|
): Promise<IAccount> => {
|
||||||
return this.createAccountService.createAccount(tenantId, accountDTO);
|
return this.createAccountService.createAccount(tenantId, accountDTO, trx);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -0,0 +1,50 @@
|
|||||||
|
export const AccountsSampleData = [
|
||||||
|
{
|
||||||
|
'Account Name': 'Utilities Expense',
|
||||||
|
'Account Code': 9000,
|
||||||
|
Type: 'Expense',
|
||||||
|
Description: 'Omnis voluptatum consequatur.',
|
||||||
|
Active: 'T',
|
||||||
|
'Currency Code': '',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'Account Name': 'Unearned Revenue',
|
||||||
|
'Account Code': 9010,
|
||||||
|
Type: 'Long Term Liability',
|
||||||
|
Description: 'Autem odit voluptas nihil unde.',
|
||||||
|
Active: 'T',
|
||||||
|
'Currency Code': '',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'Account Name': 'Long-Term Debt',
|
||||||
|
'Account Code': 9020,
|
||||||
|
Type: 'Long Term Liability',
|
||||||
|
Description: 'In voluptas cumque exercitationem.',
|
||||||
|
Active: 'T',
|
||||||
|
'Currency Code': '',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'Account Name': 'Salaries and Wages Expense',
|
||||||
|
'Account Code': 9030,
|
||||||
|
Type: 'Expense',
|
||||||
|
Description: 'Assumenda aspernatur soluta aliquid perspiciatis quasi.',
|
||||||
|
Active: 'T',
|
||||||
|
'Currency Code': '',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'Account Name': 'Rental Income',
|
||||||
|
'Account Code': 9040,
|
||||||
|
Type: 'Income',
|
||||||
|
Description: 'Omnis possimus amet occaecati inventore.',
|
||||||
|
Active: 'T',
|
||||||
|
'Currency Code': '',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'Account Name': 'Paypal',
|
||||||
|
'Account Code': 9050,
|
||||||
|
Type: 'Bank',
|
||||||
|
Description: 'In voluptas cumque exercitationem.',
|
||||||
|
Active: 'T',
|
||||||
|
'Currency Code': '',
|
||||||
|
},
|
||||||
|
];
|
||||||
45
packages/server/src/services/Accounts/AccountsImportable.ts
Normal file
45
packages/server/src/services/Accounts/AccountsImportable.ts
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { Knex } from 'knex';
|
||||||
|
import { IAccountCreateDTO } from '@/interfaces';
|
||||||
|
import { CreateAccount } from './CreateAccount';
|
||||||
|
import { Importable } from '../Import/Importable';
|
||||||
|
import { AccountsSampleData } from './AccountsImportable.SampleData';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class AccountsImportable extends Importable {
|
||||||
|
@Inject()
|
||||||
|
private createAccountService: CreateAccount;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Importing to account service.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {IAccountCreateDTO} createAccountDTO
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
public importable(
|
||||||
|
tenantId: number,
|
||||||
|
createAccountDTO: IAccountCreateDTO,
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
) {
|
||||||
|
return this.createAccountService.createAccount(
|
||||||
|
tenantId,
|
||||||
|
createAccountDTO,
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Concurrrency controlling of the importing process.
|
||||||
|
* @returns {number}
|
||||||
|
*/
|
||||||
|
public get concurrency() {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the sample data that used to download accounts sample sheet.
|
||||||
|
*/
|
||||||
|
public sampleData(): any[] {
|
||||||
|
return AccountsSampleData;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -97,13 +97,14 @@ export class CreateAccount {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new account on the storage.
|
* Creates a new account on the storage.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
* @param {IAccountCreateDTO} accountDTO
|
* @param {IAccountCreateDTO} accountDTO
|
||||||
* @returns {Promise<IAccount>}
|
* @returns {Promise<IAccount>}
|
||||||
*/
|
*/
|
||||||
public createAccount = async (
|
public createAccount = async (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
accountDTO: IAccountCreateDTO
|
accountDTO: IAccountCreateDTO,
|
||||||
|
trx?: Knex.Transaction
|
||||||
): Promise<IAccount> => {
|
): Promise<IAccount> => {
|
||||||
const { Account } = this.tenancy.models(tenantId);
|
const { Account } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
@@ -119,27 +120,31 @@ export class CreateAccount {
|
|||||||
tenantMeta.baseCurrency
|
tenantMeta.baseCurrency
|
||||||
);
|
);
|
||||||
// Creates a new account with associated transactions under unit-of-work envirement.
|
// Creates a new account with associated transactions under unit-of-work envirement.
|
||||||
return this.uow.withTransaction(tenantId, async (trx: Knex.Transaction) => {
|
return this.uow.withTransaction(
|
||||||
// Triggers `onAccountCreating` event.
|
tenantId,
|
||||||
await this.eventPublisher.emitAsync(events.accounts.onCreating, {
|
async (trx: Knex.Transaction) => {
|
||||||
tenantId,
|
// Triggers `onAccountCreating` event.
|
||||||
accountDTO,
|
await this.eventPublisher.emitAsync(events.accounts.onCreating, {
|
||||||
trx,
|
tenantId,
|
||||||
} as IAccountEventCreatingPayload);
|
accountDTO,
|
||||||
|
trx,
|
||||||
|
} as IAccountEventCreatingPayload);
|
||||||
|
|
||||||
// Inserts account to the storage.
|
// Inserts account to the storage.
|
||||||
const account = await Account.query(trx).insertAndFetch({
|
const account = await Account.query(trx).insertAndFetch({
|
||||||
...accountInputModel,
|
...accountInputModel,
|
||||||
});
|
});
|
||||||
// Triggers `onAccountCreated` event.
|
// Triggers `onAccountCreated` event.
|
||||||
await this.eventPublisher.emitAsync(events.accounts.onCreated, {
|
await this.eventPublisher.emitAsync(events.accounts.onCreated, {
|
||||||
tenantId,
|
tenantId,
|
||||||
account,
|
account,
|
||||||
accountId: account.id,
|
accountId: account.id,
|
||||||
trx,
|
trx,
|
||||||
} as IAccountEventCreatedPayload);
|
} as IAccountEventCreatedPayload);
|
||||||
|
|
||||||
return account;
|
return account;
|
||||||
});
|
},
|
||||||
|
trx
|
||||||
|
);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
|
import { Knex } from 'knex';
|
||||||
import { Inject, Service } from 'typedi';
|
import { Inject, Service } from 'typedi';
|
||||||
import HasTenancyService from '../Tenancy/TenancyService';
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
import UnitOfWork, { IsolationLevel } from '../UnitOfWork';
|
import UnitOfWork from '../UnitOfWork';
|
||||||
import { Knex } from 'knex';
|
|
||||||
import { CreateUncategorizedTransactionDTO } from '@/interfaces';
|
import { CreateUncategorizedTransactionDTO } from '@/interfaces';
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
@@ -19,10 +19,10 @@ export class CreateUncategorizedTransaction {
|
|||||||
*/
|
*/
|
||||||
public create(
|
public create(
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
createDTO: CreateUncategorizedTransactionDTO
|
createDTO: CreateUncategorizedTransactionDTO,
|
||||||
|
trx?: Knex.Transaction
|
||||||
) {
|
) {
|
||||||
const { UncategorizedCashflowTransaction, Account } =
|
const { UncategorizedCashflowTransaction } = this.tenancy.models(tenantId);
|
||||||
this.tenancy.models(tenantId);
|
|
||||||
|
|
||||||
return this.uow.withTransaction(
|
return this.uow.withTransaction(
|
||||||
tenantId,
|
tenantId,
|
||||||
@@ -32,9 +32,9 @@ export class CreateUncategorizedTransaction {
|
|||||||
).insertAndFetch({
|
).insertAndFetch({
|
||||||
...createDTO,
|
...createDTO,
|
||||||
});
|
});
|
||||||
|
|
||||||
return transaction;
|
return transaction;
|
||||||
},
|
},
|
||||||
|
trx
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,82 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { Knex } from 'knex';
|
||||||
|
import * as yup from 'yup';
|
||||||
|
import { Importable } from '../Import/Importable';
|
||||||
|
import { CreateUncategorizedTransaction } from './CreateUncategorizedTransaction';
|
||||||
|
import { CreateUncategorizedTransactionDTO } from '@/interfaces';
|
||||||
|
import { ImportableContext } from '../Import/interfaces';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import { BankTransactionsSampleData } from './constants';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class UncategorizedTransactionsImportable extends Importable {
|
||||||
|
@Inject()
|
||||||
|
private createUncategorizedTransaction: CreateUncategorizedTransaction;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
/**
|
||||||
|
* Passing the sheet DTO to create uncategorized transaction.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {any} createDTO
|
||||||
|
* @param {Knex.Transaction} trx
|
||||||
|
*/
|
||||||
|
public async importable(
|
||||||
|
tenantId: number,
|
||||||
|
createDTO: CreateUncategorizedTransactionDTO,
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
) {
|
||||||
|
return this.createUncategorizedTransaction.create(tenantId, createDTO, trx);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transformes the DTO before validating and importing.
|
||||||
|
* @param {CreateUncategorizedTransactionDTO} createDTO
|
||||||
|
* @param {ImportableContext} context
|
||||||
|
* @returns {CreateUncategorizedTransactionDTO}
|
||||||
|
*/
|
||||||
|
public transform(
|
||||||
|
createDTO: CreateUncategorizedTransactionDTO,
|
||||||
|
context?: ImportableContext
|
||||||
|
): CreateUncategorizedTransactionDTO {
|
||||||
|
return {
|
||||||
|
...createDTO,
|
||||||
|
accountId: context.import.paramsParsed.accountId,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sample data used to download sample sheet.
|
||||||
|
* @returns {Record<string, any>[]}
|
||||||
|
*/
|
||||||
|
public sampleData(): Record<string, any>[] {
|
||||||
|
return BankTransactionsSampleData;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Params validation schema.
|
||||||
|
* @returns {ValidationSchema[]}
|
||||||
|
*/
|
||||||
|
public paramsValidationSchema() {
|
||||||
|
return yup.object().shape({
|
||||||
|
accountId: yup.number().required(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates the params existance asyncly.
|
||||||
|
* @param {number} tenantId -
|
||||||
|
* @param {Record<string, any>} params -
|
||||||
|
*/
|
||||||
|
public async validateParams(
|
||||||
|
tenantId: number,
|
||||||
|
params: Record<string, any>
|
||||||
|
): Promise<void> {
|
||||||
|
const { Account } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
if (params.accountId) {
|
||||||
|
await Account.query().findById(params.accountId).throwIfNotFound({});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -11,8 +11,10 @@ export const ERRORS = {
|
|||||||
ACCOUNT_HAS_ASSOCIATED_TRANSACTIONS: 'account_has_associated_transactions',
|
ACCOUNT_HAS_ASSOCIATED_TRANSACTIONS: 'account_has_associated_transactions',
|
||||||
TRANSACTION_ALREADY_CATEGORIZED: 'TRANSACTION_ALREADY_CATEGORIZED',
|
TRANSACTION_ALREADY_CATEGORIZED: 'TRANSACTION_ALREADY_CATEGORIZED',
|
||||||
TRANSACTION_ALREADY_UNCATEGORIZED: 'TRANSACTION_ALREADY_UNCATEGORIZED',
|
TRANSACTION_ALREADY_UNCATEGORIZED: 'TRANSACTION_ALREADY_UNCATEGORIZED',
|
||||||
UNCATEGORIZED_TRANSACTION_TYPE_INVALID: 'UNCATEGORIZED_TRANSACTION_TYPE_INVALID',
|
UNCATEGORIZED_TRANSACTION_TYPE_INVALID:
|
||||||
CANNOT_DELETE_TRANSACTION_CONVERTED_FROM_UNCATEGORIZED: 'CANNOT_DELETE_TRANSACTION_CONVERTED_FROM_UNCATEGORIZED'
|
'UNCATEGORIZED_TRANSACTION_TYPE_INVALID',
|
||||||
|
CANNOT_DELETE_TRANSACTION_CONVERTED_FROM_UNCATEGORIZED:
|
||||||
|
'CANNOT_DELETE_TRANSACTION_CONVERTED_FROM_UNCATEGORIZED',
|
||||||
};
|
};
|
||||||
|
|
||||||
export enum CASHFLOW_DIRECTION {
|
export enum CASHFLOW_DIRECTION {
|
||||||
@@ -75,3 +77,29 @@ export interface ICashflowTransactionTypeMeta {
|
|||||||
direction: CASHFLOW_DIRECTION;
|
direction: CASHFLOW_DIRECTION;
|
||||||
creditType: string[];
|
creditType: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const BankTransactionsSampleData = [
|
||||||
|
[
|
||||||
|
{
|
||||||
|
Amount: '6,410.19',
|
||||||
|
Date: '2024-03-26',
|
||||||
|
Payee: 'MacGyver and Sons',
|
||||||
|
'Reference No.': 'REF-1',
|
||||||
|
Description: 'Commodi quo labore.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Amount: '8,914.17',
|
||||||
|
Date: '2024-01-05',
|
||||||
|
Payee: 'Eichmann - Bergnaum',
|
||||||
|
'Reference No.': 'REF-1',
|
||||||
|
Description: 'Quia enim et.',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Amount: '6,200.88',
|
||||||
|
Date: '2024-02-17',
|
||||||
|
Payee: 'Luettgen, Mraz and Legros',
|
||||||
|
'Reference No.': 'REF-1',
|
||||||
|
Description: 'Occaecati consequuntur cum impedit illo.',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ export class CreateCustomer {
|
|||||||
public async createCustomer(
|
public async createCustomer(
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
customerDTO: ICustomerNewDTO,
|
customerDTO: ICustomerNewDTO,
|
||||||
authorizedUser: ISystemUser
|
trx?: Knex.Transaction
|
||||||
): Promise<ICustomer> {
|
): Promise<ICustomer> {
|
||||||
const { Contact } = this.tenancy.models(tenantId);
|
const { Contact } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
@@ -46,28 +46,31 @@ export class CreateCustomer {
|
|||||||
customerDTO
|
customerDTO
|
||||||
);
|
);
|
||||||
// Creates a new customer under unit-of-work envirement.
|
// Creates a new customer under unit-of-work envirement.
|
||||||
return this.uow.withTransaction(tenantId, async (trx: Knex.Transaction) => {
|
return this.uow.withTransaction(
|
||||||
// Triggers `onCustomerCreating` event.
|
tenantId,
|
||||||
await this.eventPublisher.emitAsync(events.customers.onCreating, {
|
async (trx: Knex.Transaction) => {
|
||||||
tenantId,
|
// Triggers `onCustomerCreating` event.
|
||||||
customerDTO,
|
await this.eventPublisher.emitAsync(events.customers.onCreating, {
|
||||||
trx,
|
tenantId,
|
||||||
} as ICustomerEventCreatingPayload);
|
customerDTO,
|
||||||
|
trx,
|
||||||
|
} as ICustomerEventCreatingPayload);
|
||||||
|
|
||||||
// Creates a new contact as customer.
|
// Creates a new contact as customer.
|
||||||
const customer = await Contact.query(trx).insertAndFetch({
|
const customer = await Contact.query(trx).insertAndFetch({
|
||||||
...customerObj,
|
...customerObj,
|
||||||
});
|
});
|
||||||
// Triggers `onCustomerCreated` event.
|
// Triggers `onCustomerCreated` event.
|
||||||
await this.eventPublisher.emitAsync(events.customers.onCreated, {
|
await this.eventPublisher.emitAsync(events.customers.onCreated, {
|
||||||
customer,
|
customer,
|
||||||
tenantId,
|
tenantId,
|
||||||
customerId: customer.id,
|
customerId: customer.id,
|
||||||
authorizedUser,
|
trx,
|
||||||
trx,
|
} as ICustomerEventCreatedPayload);
|
||||||
} as ICustomerEventCreatedPayload);
|
|
||||||
|
|
||||||
return customer;
|
return customer;
|
||||||
});
|
},
|
||||||
|
trx
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
import { defaultTo, omit, isEmpty } from 'lodash';
|
import { defaultTo, omit, isEmpty } from 'lodash';
|
||||||
import { Service, Inject } from 'typedi';
|
import { Service } from 'typedi';
|
||||||
import {
|
import {
|
||||||
ContactService,
|
ContactService,
|
||||||
ICustomer,
|
ICustomer,
|
||||||
@@ -51,6 +51,10 @@ export class CreateEditCustomerDTO {
|
|||||||
).toMySqlDateTime(),
|
).toMySqlDateTime(),
|
||||||
}
|
}
|
||||||
: {}),
|
: {}),
|
||||||
|
openingBalanceExchangeRate: defaultTo(
|
||||||
|
customerDTO.openingBalanceExchangeRate,
|
||||||
|
1
|
||||||
|
),
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ import {
|
|||||||
ICustomerEditDTO,
|
ICustomerEditDTO,
|
||||||
ICustomerEventEditedPayload,
|
ICustomerEventEditedPayload,
|
||||||
ICustomerEventEditingPayload,
|
ICustomerEventEditingPayload,
|
||||||
ISystemUser,
|
|
||||||
} 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';
|
||||||
|
|||||||
@@ -53,13 +53,8 @@ export class CustomersApplication {
|
|||||||
public createCustomer = (
|
public createCustomer = (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
customerDTO: ICustomerNewDTO,
|
customerDTO: ICustomerNewDTO,
|
||||||
authorizedUser: ISystemUser
|
|
||||||
) => {
|
) => {
|
||||||
return this.createCustomerService.createCustomer(
|
return this.createCustomerService.createCustomer(tenantId, customerDTO);
|
||||||
tenantId,
|
|
||||||
customerDTO,
|
|
||||||
authorizedUser
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -0,0 +1,34 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { Importable } from '@/services/Import/Importable';
|
||||||
|
import { CreateCustomer } from './CRUD/CreateCustomer';
|
||||||
|
import { Knex } from 'knex';
|
||||||
|
import { ICustomer, ICustomerNewDTO } from '@/interfaces';
|
||||||
|
import { CustomersSampleData } from './_SampleData';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class CustomersImportable extends Importable {
|
||||||
|
@Inject()
|
||||||
|
private createCustomerService: CreateCustomer;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Mapps the imported data to create a new customer service.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {ICustomerNewDTO} createDTO
|
||||||
|
* @param {Knex.Transaction} trx
|
||||||
|
* @returns {Promise<void>}
|
||||||
|
*/
|
||||||
|
public async importable(
|
||||||
|
tenantId: number,
|
||||||
|
createDTO: ICustomerNewDTO,
|
||||||
|
trx?: Knex.Transaction<any, any[]>
|
||||||
|
): Promise<void> {
|
||||||
|
await this.createCustomerService.createCustomer(tenantId, createDTO, trx);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the sample data of customers used to download sample sheet.
|
||||||
|
*/
|
||||||
|
public sampleData(): any[] {
|
||||||
|
return CustomersSampleData;
|
||||||
|
}
|
||||||
|
}
|
||||||
158
packages/server/src/services/Contacts/Customers/_SampleData.ts
Normal file
158
packages/server/src/services/Contacts/Customers/_SampleData.ts
Normal file
@@ -0,0 +1,158 @@
|
|||||||
|
|
||||||
|
export const CustomersSampleData = [
|
||||||
|
{
|
||||||
|
"Customer Type": "Business",
|
||||||
|
"First Name": "Nicolette",
|
||||||
|
"Last Name": "Schamberger",
|
||||||
|
"Company Name": "Homenick - Hane",
|
||||||
|
"Display Name": "Rowland Rowe",
|
||||||
|
"Email": "cicero86@yahoo.com",
|
||||||
|
"Personal Phone Number": "811-603-2235",
|
||||||
|
"Work Phone Number": "906-993-5190",
|
||||||
|
"Website": "http://google.com",
|
||||||
|
"Opening Balance": 54302.23,
|
||||||
|
"Opening Balance At": "2022-02-02",
|
||||||
|
"Opening Balance Ex. Rate": 2,
|
||||||
|
"Currency": "LYD",
|
||||||
|
"Active": "F",
|
||||||
|
"Note": "Doloribus autem optio temporibus dolores mollitia sit.",
|
||||||
|
"Billing Address 1": "862 Jessika Well",
|
||||||
|
"Billing Address 2": "1091 Dorthy Mount",
|
||||||
|
"Billing Address City": "Deckowfort",
|
||||||
|
"Billing Address Country": "Ghana",
|
||||||
|
"Billing Address Phone": "825-011-5207",
|
||||||
|
"Billing Address Postcode": "38228",
|
||||||
|
"Billing Address State": "Oregon",
|
||||||
|
"Shipping Address 1": "37626 Thiel Villages",
|
||||||
|
"Shipping Address 2": "132 Batz Avenue",
|
||||||
|
"Shipping Address City": "Pagacburgh",
|
||||||
|
"Shipping Address Country": "Albania",
|
||||||
|
"Shipping Address Phone": "171-546-3701",
|
||||||
|
"Shipping Address Postcode": "13709",
|
||||||
|
"Shipping Address State": "Georgia"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Customer Type": "Business",
|
||||||
|
"First Name": "Hermann",
|
||||||
|
"Last Name": "Crooks",
|
||||||
|
"Company Name": "Veum - Schaefer",
|
||||||
|
"Display Name": "Harley Veum",
|
||||||
|
"Email": "immanuel56@hotmail.com",
|
||||||
|
"Personal Phone Number": "449-780-9999",
|
||||||
|
"Work Phone Number": "970-473-5785",
|
||||||
|
"Website": "http://google.com",
|
||||||
|
"Opening Balance": 54302.23,
|
||||||
|
"Opening Balance At": "2022-02-02",
|
||||||
|
"Opening Balance Ex. Rate": 2,
|
||||||
|
"Currency": "LYD",
|
||||||
|
"Active": "T",
|
||||||
|
"Note": "Doloribus dolore dolor dicta vitae in fugit nisi quibusdam.",
|
||||||
|
"Billing Address 1": "532 Simonis Spring",
|
||||||
|
"Billing Address 2": "3122 Nicolas Inlet",
|
||||||
|
"Billing Address City": "East Matteofort",
|
||||||
|
"Billing Address Country": "Holy See (Vatican City State)",
|
||||||
|
"Billing Address Phone": "366-084-8629",
|
||||||
|
"Billing Address Postcode": "41607",
|
||||||
|
"Billing Address State": "Montana",
|
||||||
|
"Shipping Address 1": "2889 Tremblay Plaza",
|
||||||
|
"Shipping Address 2": "71355 Kutch Isle",
|
||||||
|
"Shipping Address City": "D'Amorehaven",
|
||||||
|
"Shipping Address Country": "Monaco",
|
||||||
|
"Shipping Address Phone": "614-189-3328",
|
||||||
|
"Shipping Address Postcode": "09634-0435",
|
||||||
|
"Shipping Address State": "Nevada"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Customer Type": "Business",
|
||||||
|
"First Name": "Nellie",
|
||||||
|
"Last Name": "Gulgowski",
|
||||||
|
"Company Name": "Boyle, Heller and Jones",
|
||||||
|
"Display Name": "Randall Kohler",
|
||||||
|
"Email": "anibal_frami@yahoo.com",
|
||||||
|
"Personal Phone Number": "498-578-0740",
|
||||||
|
"Work Phone Number": "394-550-6827",
|
||||||
|
"Website": "http://google.com",
|
||||||
|
"Opening Balance": 54302.23,
|
||||||
|
"Opening Balance At": "2022-02-02",
|
||||||
|
"Opening Balance Ex. Rate": 2,
|
||||||
|
"Currency": "LYD",
|
||||||
|
"Active": "T",
|
||||||
|
"Note": "Vero quibusdam rem fugit aperiam est modi.",
|
||||||
|
"Billing Address 1": "214 Sauer Villages",
|
||||||
|
"Billing Address 2": "30687 Kacey Square",
|
||||||
|
"Billing Address City": "Jayceborough",
|
||||||
|
"Billing Address Country": "Benin",
|
||||||
|
"Billing Address Phone": "332-820-1127",
|
||||||
|
"Billing Address Postcode": "16425-3887",
|
||||||
|
"Billing Address State": "Mississippi",
|
||||||
|
"Shipping Address 1": "562 Diamond Loaf",
|
||||||
|
"Shipping Address 2": "9595 Satterfield Trafficway",
|
||||||
|
"Shipping Address City": "Alexandrinefort",
|
||||||
|
"Shipping Address Country": "Puerto Rico",
|
||||||
|
"Shipping Address Phone": "776-500-8456",
|
||||||
|
"Shipping Address Postcode": "30258",
|
||||||
|
"Shipping Address State": "South Dakota"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Customer Type": "Business",
|
||||||
|
"First Name": "Stone",
|
||||||
|
"Last Name": "Jerde",
|
||||||
|
"Company Name": "Cassin, Casper and Maggio",
|
||||||
|
"Display Name": "Clint McLaughlin",
|
||||||
|
"Email": "nathanael22@yahoo.com",
|
||||||
|
"Personal Phone Number": "562-790-6059",
|
||||||
|
"Work Phone Number": "686-838-0027",
|
||||||
|
"Website": "http://google.com",
|
||||||
|
"Opening Balance": 54302.23,
|
||||||
|
"Opening Balance At": "2022-02-02",
|
||||||
|
"Opening Balance Ex. Rate": 2,
|
||||||
|
"Currency": "LYD",
|
||||||
|
"Active": "F",
|
||||||
|
"Note": "Quis cumque molestias rerum.",
|
||||||
|
"Billing Address 1": "22590 Cathy Harbor",
|
||||||
|
"Billing Address 2": "24493 Brycen Brooks",
|
||||||
|
"Billing Address City": "Elnorashire",
|
||||||
|
"Billing Address Country": "Andorra",
|
||||||
|
"Billing Address Phone": "701-852-8005",
|
||||||
|
"Billing Address Postcode": "5680",
|
||||||
|
"Billing Address State": "Nevada",
|
||||||
|
"Shipping Address 1": "5355 Erdman Bridge",
|
||||||
|
"Shipping Address 2": "421 Jeanette Camp",
|
||||||
|
"Shipping Address City": "East Philip",
|
||||||
|
"Shipping Address Country": "Venezuela",
|
||||||
|
"Shipping Address Phone": "426-119-0858",
|
||||||
|
"Shipping Address Postcode": "34929-0501",
|
||||||
|
"Shipping Address State": "Tennessee"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Customer Type": "Individual",
|
||||||
|
"First Name": "Lempi",
|
||||||
|
"Last Name": "Kling",
|
||||||
|
"Company Name": "Schamberger, O'Connell and Bechtelar",
|
||||||
|
"Display Name": "Alexie Barton",
|
||||||
|
"Email": "eulah.kreiger@hotmail.com",
|
||||||
|
"Personal Phone Number": "745-756-1063",
|
||||||
|
"Work Phone Number": "965-150-1945",
|
||||||
|
"Website": "http://google.com",
|
||||||
|
"Opening Balance": 54302.23,
|
||||||
|
"Opening Balance At": "2022-02-02",
|
||||||
|
"Opening Balance Ex. Rate": 2,
|
||||||
|
"Currency": "LYD",
|
||||||
|
"Active": "F",
|
||||||
|
"Note": "Maxime laboriosam hic voluptate maiores est officia.",
|
||||||
|
"Billing Address 1": "0851 Jones Flat",
|
||||||
|
"Billing Address 2": "845 Bailee Drives",
|
||||||
|
"Billing Address City": "Kamrenport",
|
||||||
|
"Billing Address Country": "Niger",
|
||||||
|
"Billing Address Phone": "220-125-0608",
|
||||||
|
"Billing Address Postcode": "30311",
|
||||||
|
"Billing Address State": "Delaware",
|
||||||
|
"Shipping Address 1": "929 Ferry Row",
|
||||||
|
"Shipping Address 2": "020 Adam Plaza",
|
||||||
|
"Shipping Address City": "West Carmellaside",
|
||||||
|
"Shipping Address Country": "Ghana",
|
||||||
|
"Shipping Address Phone": "053-333-6679",
|
||||||
|
"Shipping Address Postcode": "79221-4681",
|
||||||
|
"Shipping Address State": "Illinois"
|
||||||
|
}
|
||||||
|
]
|
||||||
@@ -49,6 +49,10 @@ export class CreateEditVendorDTO {
|
|||||||
).toMySqlDateTime(),
|
).toMySqlDateTime(),
|
||||||
}
|
}
|
||||||
: {}),
|
: {}),
|
||||||
|
openingBalanceExchangeRate: defaultTo(
|
||||||
|
vendorDTO.openingBalanceExchangeRate,
|
||||||
|
1
|
||||||
|
),
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ export class CreateVendor {
|
|||||||
public async createVendor(
|
public async createVendor(
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
vendorDTO: IVendorNewDTO,
|
vendorDTO: IVendorNewDTO,
|
||||||
authorizedUser: ISystemUser
|
trx?: Knex.Transaction
|
||||||
) {
|
) {
|
||||||
const { Contact } = this.tenancy.models(tenantId);
|
const { Contact } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
@@ -45,28 +45,31 @@ export class CreateVendor {
|
|||||||
vendorDTO
|
vendorDTO
|
||||||
);
|
);
|
||||||
// Creates vendor contact under unit-of-work evnirement.
|
// Creates vendor contact under unit-of-work evnirement.
|
||||||
return this.uow.withTransaction(tenantId, async (trx: Knex.Transaction) => {
|
return this.uow.withTransaction(
|
||||||
// Triggers `onVendorCreating` event.
|
tenantId,
|
||||||
await this.eventPublisher.emitAsync(events.vendors.onCreating, {
|
async (trx: Knex.Transaction) => {
|
||||||
tenantId,
|
// Triggers `onVendorCreating` event.
|
||||||
vendorDTO,
|
await this.eventPublisher.emitAsync(events.vendors.onCreating, {
|
||||||
trx,
|
tenantId,
|
||||||
} as IVendorEventCreatingPayload);
|
vendorDTO,
|
||||||
|
trx,
|
||||||
|
} as IVendorEventCreatingPayload);
|
||||||
|
|
||||||
// Creates a new contact as vendor.
|
// Creates a new contact as vendor.
|
||||||
const vendor = await Contact.query(trx).insertAndFetch({
|
const vendor = await Contact.query(trx).insertAndFetch({
|
||||||
...vendorObject,
|
...vendorObject,
|
||||||
});
|
});
|
||||||
// Triggers `onVendorCreated` event.
|
// Triggers `onVendorCreated` event.
|
||||||
await this.eventPublisher.emitAsync(events.vendors.onCreated, {
|
await this.eventPublisher.emitAsync(events.vendors.onCreated, {
|
||||||
tenantId,
|
tenantId,
|
||||||
vendorId: vendor.id,
|
vendorId: vendor.id,
|
||||||
vendor,
|
vendor,
|
||||||
authorizedUser,
|
trx,
|
||||||
trx,
|
} as IVendorEventCreatedPayload);
|
||||||
} as IVendorEventCreatedPayload);
|
|
||||||
|
|
||||||
return vendor;
|
return vendor;
|
||||||
});
|
},
|
||||||
|
trx
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
import { Importable } from '@/services/Import/Importable';
|
||||||
|
import { CreateVendor } from './CRUD/CreateVendor';
|
||||||
|
import { Knex } from 'knex';
|
||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { VendorsSampleData } from './_SampleData';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class VendorsImportable extends Importable {
|
||||||
|
@Inject()
|
||||||
|
private createVendorService: CreateVendor;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maps the imported data to create a new vendor service.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {} createDTO
|
||||||
|
* @param {Knex.Transaction} trx
|
||||||
|
*/
|
||||||
|
public async importable(
|
||||||
|
tenantId: number,
|
||||||
|
createDTO: any,
|
||||||
|
trx?: Knex.Transaction<any, any[]>
|
||||||
|
): Promise<void> {
|
||||||
|
await this.createVendorService.createVendor(tenantId, createDTO, trx);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the sample data of vendors sample sheet.
|
||||||
|
*/
|
||||||
|
public sampleData(): any[] {
|
||||||
|
return VendorsSampleData;
|
||||||
|
}
|
||||||
|
}
|
||||||
122
packages/server/src/services/Contacts/Vendors/_SampleData.ts
Normal file
122
packages/server/src/services/Contacts/Vendors/_SampleData.ts
Normal file
@@ -0,0 +1,122 @@
|
|||||||
|
export const VendorsSampleData = [
|
||||||
|
{
|
||||||
|
"First Name": "Nicolette",
|
||||||
|
"Last Name": "Schamberger",
|
||||||
|
"Company Name": "Homenick - Hane",
|
||||||
|
"Display Name": "Rowland Rowe",
|
||||||
|
"Email": "cicero86@yahoo.com",
|
||||||
|
"Personal Phone Number": "811-603-2235",
|
||||||
|
"Work Phone Number": "906-993-5190",
|
||||||
|
"Website": "http://google.com",
|
||||||
|
"Opening Balance": 54302.23,
|
||||||
|
"Opening Balance At": "2022-02-02",
|
||||||
|
"Opening Balance Ex. Rate": 2,
|
||||||
|
"Currency": "LYD",
|
||||||
|
"Active": "F",
|
||||||
|
"Note": "Doloribus autem optio temporibus dolores mollitia sit.",
|
||||||
|
"Billing Address 1": "862 Jessika Well",
|
||||||
|
"Billing Address 2": "1091 Dorthy Mount",
|
||||||
|
"Billing Address City": "Deckowfort",
|
||||||
|
"Billing Address Country": "Ghana",
|
||||||
|
"Billing Address Phone": "825-011-5207",
|
||||||
|
"Billing Address Postcode": "38228",
|
||||||
|
"Billing Address State": "Oregon",
|
||||||
|
"Shipping Address 1": "37626 Thiel Villages",
|
||||||
|
"Shipping Address 2": "132 Batz Avenue",
|
||||||
|
"Shipping Address City": "Pagacburgh",
|
||||||
|
"Shipping Address Country": "Albania",
|
||||||
|
"Shipping Address Phone": "171-546-3701",
|
||||||
|
"Shipping Address Postcode": "13709",
|
||||||
|
"Shipping Address State": "Georgia"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"First Name": "Hermann",
|
||||||
|
"Last Name": "Crooks",
|
||||||
|
"Company Name": "Veum - Schaefer",
|
||||||
|
"Display Name": "Harley Veum",
|
||||||
|
"Email": "immanuel56@hotmail.com",
|
||||||
|
"Personal Phone Number": "449-780-9999",
|
||||||
|
"Work Phone Number": "970-473-5785",
|
||||||
|
"Website": "http://google.com",
|
||||||
|
"Opening Balance": 54302.23,
|
||||||
|
"Opening Balance At": "2022-02-02",
|
||||||
|
"Opening Balance Ex. Rate": 2,
|
||||||
|
"Currency": "LYD",
|
||||||
|
"Active": "F",
|
||||||
|
"Note": "Doloribus dolore dolor dicta vitae in fugit nisi quibusdam.",
|
||||||
|
"Billing Address 1": "532 Simonis Spring",
|
||||||
|
"Billing Address 2": "3122 Nicolas Inlet",
|
||||||
|
"Billing Address City": "East Matteofort",
|
||||||
|
"Billing Address Country": "Holy See (Vatican City State)",
|
||||||
|
"Billing Address Phone": "366-084-8629",
|
||||||
|
"Billing Address Postcode": "41607",
|
||||||
|
"Billing Address State": "Montana",
|
||||||
|
"Shipping Address 1": "2889 Tremblay Plaza",
|
||||||
|
"Shipping Address 2": "71355 Kutch Isle",
|
||||||
|
"Shipping Address City": "D'Amorehaven",
|
||||||
|
"Shipping Address Country": "Monaco",
|
||||||
|
"Shipping Address Phone": "614-189-3328",
|
||||||
|
"Shipping Address Postcode": "09634-0435",
|
||||||
|
"Shipping Address State": "Nevada"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"First Name": "Nellie",
|
||||||
|
"Last Name": "Gulgowski",
|
||||||
|
"Company Name": "Boyle, Heller and Jones",
|
||||||
|
"Display Name": "Randall Kohler",
|
||||||
|
"Email": "anibal_frami@yahoo.com",
|
||||||
|
"Personal Phone Number": "498-578-0740",
|
||||||
|
"Work Phone Number": "394-550-6827",
|
||||||
|
"Website": "http://google.com",
|
||||||
|
"Opening Balance": 54302.23,
|
||||||
|
"Opening Balance At": "2022-02-02",
|
||||||
|
"Opening Balance Ex. Rate": 2,
|
||||||
|
"Currency": "LYD",
|
||||||
|
"Active": "F",
|
||||||
|
"Note": "Vero quibusdam rem fugit aperiam est modi.",
|
||||||
|
"Billing Address 1": "214 Sauer Villages",
|
||||||
|
"Billing Address 2": "30687 Kacey Square",
|
||||||
|
"Billing Address City": "Jayceborough",
|
||||||
|
"Billing Address Country": "Benin",
|
||||||
|
"Billing Address Phone": "332-820-1127",
|
||||||
|
"Billing Address Postcode": "16425-3887",
|
||||||
|
"Billing Address State": "Mississippi",
|
||||||
|
"Shipping Address 1": "562 Diamond Loaf",
|
||||||
|
"Shipping Address 2": "9595 Satterfield Trafficway",
|
||||||
|
"Shipping Address City": "Alexandrinefort",
|
||||||
|
"Shipping Address Country": "Puerto Rico",
|
||||||
|
"Shipping Address Phone": "776-500-8456",
|
||||||
|
"Shipping Address Postcode": "30258",
|
||||||
|
"Shipping Address State": "South Dakota"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"First Name": "Stone",
|
||||||
|
"Last Name": "Jerde",
|
||||||
|
"Company Name": "Cassin, Casper and Maggio",
|
||||||
|
"Display Name": "Clint McLaughlin",
|
||||||
|
"Email": "nathanael22@yahoo.com",
|
||||||
|
"Personal Phone Number": "562-790-6059",
|
||||||
|
"Work Phone Number": "686-838-0027",
|
||||||
|
"Website": "http://google.com",
|
||||||
|
"Opening Balance": 54302.23,
|
||||||
|
"Opening Balance At": "2022-02-02",
|
||||||
|
"Opening Balance Ex. Rate": 2,
|
||||||
|
"Currency": "LYD",
|
||||||
|
"Active": "F",
|
||||||
|
"Note": "Quis cumque molestias rerum.",
|
||||||
|
"Billing Address 1": "22590 Cathy Harbor",
|
||||||
|
"Billing Address 2": "24493 Brycen Brooks",
|
||||||
|
"Billing Address City": "Elnorashire",
|
||||||
|
"Billing Address Country": "Andorra",
|
||||||
|
"Billing Address Phone": "701-852-8005",
|
||||||
|
"Billing Address Postcode": "5680",
|
||||||
|
"Billing Address State": "Nevada",
|
||||||
|
"Shipping Address 1": "5355 Erdman Bridge",
|
||||||
|
"Shipping Address 2": "421 Jeanette Camp",
|
||||||
|
"Shipping Address City": "East Philip",
|
||||||
|
"Shipping Address Country": "Venezuela",
|
||||||
|
"Shipping Address Phone": "426-119-0858",
|
||||||
|
"Shipping Address Postcode": "34929-0501",
|
||||||
|
"Shipping Address State": "Tennessee"
|
||||||
|
}
|
||||||
|
]
|
||||||
205
packages/server/src/services/Import/ImportFileCommon.ts
Normal file
205
packages/server/src/services/Import/ImportFileCommon.ts
Normal file
@@ -0,0 +1,205 @@
|
|||||||
|
import fs from 'fs/promises';
|
||||||
|
import XLSX from 'xlsx';
|
||||||
|
import bluebird from 'bluebird';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { first } from 'lodash';
|
||||||
|
import { ImportFileDataValidator } from './ImportFileDataValidator';
|
||||||
|
import { Knex } from 'knex';
|
||||||
|
import {
|
||||||
|
ImportOperError,
|
||||||
|
ImportOperSuccess,
|
||||||
|
ImportableContext,
|
||||||
|
} from './interfaces';
|
||||||
|
import { ServiceError } from '@/exceptions';
|
||||||
|
import { trimObject } from './_utils';
|
||||||
|
import { ImportableResources } from './ImportableResources';
|
||||||
|
import ResourceService from '../Resource/ResourceService';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import Import from '@/models/Import';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportFileCommon {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importFileValidator: ImportFileDataValidator;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importable: ImportableResources;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private resource: ResourceService;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maps the columns of the imported data based on the provided mapping attributes.
|
||||||
|
* @param {Record<string, any>[]} body - The array of data objects to map.
|
||||||
|
* @param {ImportMappingAttr[]} map - The mapping attributes.
|
||||||
|
* @returns {Record<string, any>[]} - The mapped data objects.
|
||||||
|
*/
|
||||||
|
public parseXlsxSheet(buffer: Buffer): Record<string, unknown>[] {
|
||||||
|
const workbook = XLSX.read(buffer, { type: 'buffer', raw: true });
|
||||||
|
|
||||||
|
const firstSheetName = workbook.SheetNames[0];
|
||||||
|
const worksheet = workbook.Sheets[firstSheetName];
|
||||||
|
|
||||||
|
return XLSX.utils.sheet_to_json(worksheet, {});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the import file.
|
||||||
|
* @param {string} filename
|
||||||
|
* @returns {Promise<Buffer>}
|
||||||
|
*/
|
||||||
|
public readImportFile(filename: string) {
|
||||||
|
return fs.readFile(`public/imports/${filename}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Imports the given parsed data to the resource storage through registered importable service.
|
||||||
|
* @param {number} tenantId -
|
||||||
|
* @param {string} resourceName - Resource name.
|
||||||
|
* @param {Record<string, any>} parsedData - Parsed data.
|
||||||
|
* @param {Knex.Transaction} trx - Knex transaction.
|
||||||
|
* @returns {Promise<[ImportOperSuccess[], ImportOperError[]]>}
|
||||||
|
*/
|
||||||
|
public async import(
|
||||||
|
tenantId: number,
|
||||||
|
importFile: Import,
|
||||||
|
parsedData: Record<string, any>[],
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
): Promise<[ImportOperSuccess[], ImportOperError[]]> {
|
||||||
|
const importableFields = this.resource.getResourceImportableFields(
|
||||||
|
tenantId,
|
||||||
|
importFile.resource
|
||||||
|
);
|
||||||
|
const ImportableRegistry = this.importable.registry;
|
||||||
|
const importable = ImportableRegistry.getImportable(importFile.resource);
|
||||||
|
|
||||||
|
const concurrency = importable.concurrency || 10;
|
||||||
|
|
||||||
|
const success: ImportOperSuccess[] = [];
|
||||||
|
const failed: ImportOperError[] = [];
|
||||||
|
|
||||||
|
const importAsync = async (objectDTO, index: number): Promise<void> => {
|
||||||
|
const context: ImportableContext = {
|
||||||
|
rowIndex: index,
|
||||||
|
import: importFile,
|
||||||
|
};
|
||||||
|
const transformedDTO = importable.transform(objectDTO, context);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Validate the DTO object before passing it to the service layer.
|
||||||
|
await this.importFileValidator.validateData(
|
||||||
|
importableFields,
|
||||||
|
transformedDTO
|
||||||
|
);
|
||||||
|
try {
|
||||||
|
// Run the importable function and listen to the errors.
|
||||||
|
const data = await importable.importable(
|
||||||
|
tenantId,
|
||||||
|
transformedDTO,
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
success.push({ index, data });
|
||||||
|
} catch (err) {
|
||||||
|
if (err instanceof ServiceError) {
|
||||||
|
const error = [
|
||||||
|
{
|
||||||
|
errorCode: 'ValidationError',
|
||||||
|
errorMessage: err.message || err.errorType,
|
||||||
|
rowNumber: index + 1,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
failed.push({ index, error });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (errors) {
|
||||||
|
const error = errors.map((er) => ({ ...er, rowNumber: index + 1 }));
|
||||||
|
failed.push({ index, error });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
await bluebird.map(parsedData, importAsync, { concurrency });
|
||||||
|
|
||||||
|
return [success, failed];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {string} resourceName
|
||||||
|
* @param {Record<string, any>} params
|
||||||
|
*/
|
||||||
|
public async validateParamsSchema(
|
||||||
|
resourceName: string,
|
||||||
|
params: Record<string, any>
|
||||||
|
) {
|
||||||
|
const ImportableRegistry = this.importable.registry;
|
||||||
|
const importable = ImportableRegistry.getImportable(resourceName);
|
||||||
|
|
||||||
|
const yupSchema = importable.paramsValidationSchema();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await yupSchema.validate(params, { abortEarly: false });
|
||||||
|
} catch (validationError) {
|
||||||
|
const errors = validationError.inner.map((error) => ({
|
||||||
|
errorCode: 'ParamsValidationError',
|
||||||
|
errorMessage: error.errors,
|
||||||
|
}));
|
||||||
|
throw errors;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {string} resourceName
|
||||||
|
* @param {Record<string, any>} params
|
||||||
|
*/
|
||||||
|
public async validateParams(
|
||||||
|
tenantId: number,
|
||||||
|
resourceName: string,
|
||||||
|
params: Record<string, any>
|
||||||
|
) {
|
||||||
|
const ImportableRegistry = this.importable.registry;
|
||||||
|
const importable = ImportableRegistry.getImportable(resourceName);
|
||||||
|
|
||||||
|
await importable.validateParams(tenantId, params);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {string} resourceName
|
||||||
|
* @param {Record<string, any>} params
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
public transformParams(resourceName: string, params: Record<string, any>) {
|
||||||
|
const ImportableRegistry = this.importable.registry;
|
||||||
|
const importable = ImportableRegistry.getImportable(resourceName);
|
||||||
|
|
||||||
|
return importable.transformParams(params);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the sheet columns from the given sheet data.
|
||||||
|
* @param {unknown[]} json
|
||||||
|
* @returns {string[]}
|
||||||
|
*/
|
||||||
|
public parseSheetColumns(json: unknown[]): string[] {
|
||||||
|
return R.compose(Object.keys, trimObject, first)(json);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Deletes the imported file from the storage and database.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {} importFile
|
||||||
|
*/
|
||||||
|
public async deleteImportFile(tenantId: number, importFile: any) {
|
||||||
|
const { Import } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
// Deletes the import row.
|
||||||
|
await Import.query().findById(importFile.id).delete();
|
||||||
|
|
||||||
|
// Deletes the imported file.
|
||||||
|
await fs.unlink(`public/imports/${importFile.filename}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
101
packages/server/src/services/Import/ImportFileDataTransformer.ts
Normal file
101
packages/server/src/services/Import/ImportFileDataTransformer.ts
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
import { Service } from 'typedi';
|
||||||
|
import * as R from 'ramda';
|
||||||
|
import { isUndefined, get, chain } from 'lodash';
|
||||||
|
import { ImportMappingAttr, ResourceMetaFieldsMap } from './interfaces';
|
||||||
|
import { parseBoolean } from '@/utils';
|
||||||
|
import { trimObject } from './_utils';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportFileDataTransformer {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {number} tenantId -
|
||||||
|
* @param {}
|
||||||
|
*/
|
||||||
|
public parseSheetData(
|
||||||
|
importFile: any,
|
||||||
|
importableFields: any,
|
||||||
|
data: Record<string, unknown>[]
|
||||||
|
) {
|
||||||
|
// Sanitize the sheet data.
|
||||||
|
const sanitizedData = this.sanitizeSheetData(data);
|
||||||
|
|
||||||
|
// Map the sheet columns key with the given map.
|
||||||
|
const mappedDTOs = this.mapSheetColumns(
|
||||||
|
sanitizedData,
|
||||||
|
importFile.mappingParsed
|
||||||
|
);
|
||||||
|
// Parse the mapped sheet values.
|
||||||
|
const parsedValues = this.parseExcelValues(importableFields, mappedDTOs);
|
||||||
|
|
||||||
|
return parsedValues;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sanitizes the data in the imported sheet by trimming object keys.
|
||||||
|
* @param json - The JSON data representing the imported sheet.
|
||||||
|
* @returns {string[][]} - The sanitized data with trimmed object keys.
|
||||||
|
*/
|
||||||
|
public sanitizeSheetData(json) {
|
||||||
|
return R.compose(R.map(trimObject))(json);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maps the columns of the imported data based on the provided mapping attributes.
|
||||||
|
* @param {Record<string, any>[]} body - The array of data objects to map.
|
||||||
|
* @param {ImportMappingAttr[]} map - The mapping attributes.
|
||||||
|
* @returns {Record<string, any>[]} - The mapped data objects.
|
||||||
|
*/
|
||||||
|
public mapSheetColumns(
|
||||||
|
body: Record<string, any>[],
|
||||||
|
map: ImportMappingAttr[]
|
||||||
|
): Record<string, any>[] {
|
||||||
|
return body.map((item) => {
|
||||||
|
const newItem = {};
|
||||||
|
map
|
||||||
|
.filter((mapping) => !isUndefined(item[mapping.from]))
|
||||||
|
.forEach((mapping) => {
|
||||||
|
newItem[mapping.to] = item[mapping.from];
|
||||||
|
});
|
||||||
|
return newItem;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses sheet values before passing to the service layer.
|
||||||
|
* @param {ResourceMetaFieldsMap} fields -
|
||||||
|
* @param {Record<string, any>} valueDTOS -
|
||||||
|
* @returns {Record<string, any>}
|
||||||
|
*/
|
||||||
|
public parseExcelValues(
|
||||||
|
fields: ResourceMetaFieldsMap,
|
||||||
|
valueDTOs: Record<string, any>[]
|
||||||
|
): Record<string, any> {
|
||||||
|
const parser = (value, key) => {
|
||||||
|
let _value = value;
|
||||||
|
|
||||||
|
// Parses the boolean value.
|
||||||
|
if (fields[key].fieldType === 'boolean') {
|
||||||
|
_value = parseBoolean(value, false);
|
||||||
|
|
||||||
|
// Parses the enumeration value.
|
||||||
|
} else if (fields[key].fieldType === 'enumeration') {
|
||||||
|
const field = fields[key];
|
||||||
|
const option = get(field, 'options', []).find(
|
||||||
|
(option) => option.label === value
|
||||||
|
);
|
||||||
|
_value = get(option, 'key');
|
||||||
|
// Prases the numeric value.
|
||||||
|
} else if (fields[key].fieldType === 'number') {
|
||||||
|
_value = parseFloat(value);
|
||||||
|
}
|
||||||
|
return _value;
|
||||||
|
};
|
||||||
|
return valueDTOs.map((DTO) => {
|
||||||
|
return chain(DTO)
|
||||||
|
.pickBy((value, key) => !isUndefined(fields[key]))
|
||||||
|
.mapValues(parser)
|
||||||
|
.value();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
import { Service } from 'typedi';
|
||||||
|
import { ImportInsertError, ResourceMetaFieldsMap } from './interfaces';
|
||||||
|
import { ERRORS, convertFieldsToYupValidation } from './_utils';
|
||||||
|
import { IModelMeta } from '@/interfaces';
|
||||||
|
import { ServiceError } from '@/exceptions';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportFileDataValidator {
|
||||||
|
/**
|
||||||
|
* Validates the given resource is importable.
|
||||||
|
* @param {IModelMeta} resourceMeta
|
||||||
|
*/
|
||||||
|
public validateResourceImportable(resourceMeta: IModelMeta) {
|
||||||
|
// Throw service error if the resource does not support importing.
|
||||||
|
if (!resourceMeta.importable) {
|
||||||
|
throw new ServiceError(ERRORS.RESOURCE_NOT_IMPORTABLE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates the given mapped DTOs and returns errors with their index.
|
||||||
|
* @param {Record<string, any>} mappedDTOs
|
||||||
|
* @returns {Promise<void | ImportInsertError[]>}
|
||||||
|
*/
|
||||||
|
public async validateData(
|
||||||
|
importableFields: ResourceMetaFieldsMap,
|
||||||
|
data: Record<string, any>
|
||||||
|
): Promise<void | ImportInsertError[]> {
|
||||||
|
const YupSchema = convertFieldsToYupValidation(importableFields);
|
||||||
|
const _data = { ...data };
|
||||||
|
|
||||||
|
try {
|
||||||
|
await YupSchema.validate(_data, { abortEarly: false });
|
||||||
|
} catch (validationError) {
|
||||||
|
const errors = validationError.inner.map((error) => ({
|
||||||
|
errorCode: 'ValidationError',
|
||||||
|
errorMessage: error.errors,
|
||||||
|
}));
|
||||||
|
throw errors;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
145
packages/server/src/services/Import/ImportFileMapping.ts
Normal file
145
packages/server/src/services/Import/ImportFileMapping.ts
Normal file
@@ -0,0 +1,145 @@
|
|||||||
|
import { fromPairs } from 'lodash';
|
||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import {
|
||||||
|
ImportDateFormats,
|
||||||
|
ImportFileMapPOJO,
|
||||||
|
ImportMappingAttr,
|
||||||
|
} from './interfaces';
|
||||||
|
import ResourceService from '../Resource/ResourceService';
|
||||||
|
import { ServiceError } from '@/exceptions';
|
||||||
|
import { ERRORS } from './_utils';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportFileMapping {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private resource: ResourceService;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Mapping the excel sheet columns with resource columns.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} importId
|
||||||
|
* @param {ImportMappingAttr} maps
|
||||||
|
*/
|
||||||
|
public async mapping(
|
||||||
|
tenantId: number,
|
||||||
|
importId: number,
|
||||||
|
maps: ImportMappingAttr[]
|
||||||
|
): Promise<ImportFileMapPOJO> {
|
||||||
|
const { Import } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
const importFile = await Import.query()
|
||||||
|
.findOne('filename', importId)
|
||||||
|
.throwIfNotFound();
|
||||||
|
|
||||||
|
// Invalidate the from/to map attributes.
|
||||||
|
this.validateMapsAttrs(tenantId, importFile, maps);
|
||||||
|
|
||||||
|
// Validate the diplicated relations of map attrs.
|
||||||
|
this.validateDuplicatedMapAttrs(maps);
|
||||||
|
|
||||||
|
// Validate the date format mapping.
|
||||||
|
this.validateDateFormatMapping(tenantId, importFile.resource, maps);
|
||||||
|
|
||||||
|
const mappingStringified = JSON.stringify(maps);
|
||||||
|
|
||||||
|
await Import.query().findById(importFile.id).patch({
|
||||||
|
mapping: mappingStringified,
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
import: {
|
||||||
|
importId: importFile.importId,
|
||||||
|
resource: importFile.resource,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validate the mapping attributes.
|
||||||
|
* @param {number} tenantId -
|
||||||
|
* @param {} importFile -
|
||||||
|
* @param {ImportMappingAttr[]} maps
|
||||||
|
* @throws {ServiceError(ERRORS.INVALID_MAP_ATTRS)}
|
||||||
|
*/
|
||||||
|
private validateMapsAttrs(
|
||||||
|
tenantId: number,
|
||||||
|
importFile: any,
|
||||||
|
maps: ImportMappingAttr[]
|
||||||
|
) {
|
||||||
|
const fields = this.resource.getResourceImportableFields(
|
||||||
|
tenantId,
|
||||||
|
importFile.resource
|
||||||
|
);
|
||||||
|
const columnsMap = fromPairs(
|
||||||
|
importFile.columnsParsed.map((field) => [field, ''])
|
||||||
|
);
|
||||||
|
const invalid = [];
|
||||||
|
|
||||||
|
maps.forEach((map) => {
|
||||||
|
if (
|
||||||
|
'undefined' === typeof fields[map.to] ||
|
||||||
|
'undefined' === typeof columnsMap[map.from]
|
||||||
|
) {
|
||||||
|
invalid.push(map);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if (invalid.length > 0) {
|
||||||
|
throw new ServiceError(ERRORS.INVALID_MAP_ATTRS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validate the map attrs relation should be one-to-one relation only.
|
||||||
|
* @param {ImportMappingAttr[]} maps
|
||||||
|
*/
|
||||||
|
private validateDuplicatedMapAttrs(maps: ImportMappingAttr[]) {
|
||||||
|
const fromMap = {};
|
||||||
|
const toMap = {};
|
||||||
|
|
||||||
|
maps.forEach((map) => {
|
||||||
|
if (fromMap[map.from]) {
|
||||||
|
throw new ServiceError(ERRORS.DUPLICATED_FROM_MAP_ATTR);
|
||||||
|
} else {
|
||||||
|
fromMap[map.from] = true;
|
||||||
|
}
|
||||||
|
if (toMap[map.to]) {
|
||||||
|
throw new ServiceError(ERRORS.DUPLICATED_TO_MAP_ATTR);
|
||||||
|
} else {
|
||||||
|
toMap[map.to] = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates the date format mapping.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {string} resource
|
||||||
|
* @param {ImportMappingAttr[]} maps
|
||||||
|
*/
|
||||||
|
private validateDateFormatMapping(
|
||||||
|
tenantId: number,
|
||||||
|
resource: string,
|
||||||
|
maps: ImportMappingAttr[]
|
||||||
|
) {
|
||||||
|
const fields = this.resource.getResourceImportableFields(
|
||||||
|
tenantId,
|
||||||
|
resource
|
||||||
|
);
|
||||||
|
maps.forEach((map) => {
|
||||||
|
if (
|
||||||
|
typeof fields[map.to] !== 'undefined' &&
|
||||||
|
fields[map.to].fieldType === 'date'
|
||||||
|
) {
|
||||||
|
if (
|
||||||
|
typeof map.dateFormat !== 'undefined' &&
|
||||||
|
ImportDateFormats.indexOf(map.dateFormat) === -1
|
||||||
|
) {
|
||||||
|
throw new ServiceError(ERRORS.INVALID_MAP_DATE_FORMAT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
32
packages/server/src/services/Import/ImportFileMeta.ts
Normal file
32
packages/server/src/services/Import/ImportFileMeta.ts
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import { TransformerInjectable } from '@/lib/Transformer/TransformerInjectable';
|
||||||
|
import { ImportFileMetaTransformer } from './ImportFileMetaTransformer';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportFileMeta {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private transformer: TransformerInjectable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} importId
|
||||||
|
* @returns {}
|
||||||
|
*/
|
||||||
|
async getImportMeta(tenantId: number, importId: string) {
|
||||||
|
const { Import } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
const importFile = await Import.query().findOne('importId', importId);
|
||||||
|
|
||||||
|
// Retrieves the transformed accounts collection.
|
||||||
|
return this.transformer.transform(
|
||||||
|
tenantId,
|
||||||
|
importFile,
|
||||||
|
new ImportFileMetaTransformer()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
import { Transformer } from '@/lib/Transformer/Transformer';
|
||||||
|
|
||||||
|
export class ImportFileMetaTransformer extends Transformer {
|
||||||
|
/**
|
||||||
|
* Include these attributes to sale invoice object.
|
||||||
|
* @returns {Array}
|
||||||
|
*/
|
||||||
|
public includeAttributes = (): string[] => {
|
||||||
|
return ['map'];
|
||||||
|
};
|
||||||
|
|
||||||
|
public excludeAttributes = (): string[] => {
|
||||||
|
return ['id', 'filename', 'columns', 'mappingParsed', 'mapping'];
|
||||||
|
}
|
||||||
|
|
||||||
|
map(importFile) {
|
||||||
|
return importFile.mappingParsed;
|
||||||
|
}
|
||||||
|
}
|
||||||
34
packages/server/src/services/Import/ImportFilePreview.ts
Normal file
34
packages/server/src/services/Import/ImportFilePreview.ts
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import { ImportFilePreviewPOJO } from './interfaces';
|
||||||
|
import { ImportFileProcess } from './ImportFileProcess';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportFilePreview {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importFile: ImportFileProcess;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Preview the imported file results before commiting the transactions.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} importId
|
||||||
|
* @returns {Promise<ImportFilePreviewPOJO>}
|
||||||
|
*/
|
||||||
|
public async preview(
|
||||||
|
tenantId: number,
|
||||||
|
importId: number
|
||||||
|
): Promise<ImportFilePreviewPOJO> {
|
||||||
|
const knex = this.tenancy.knex(tenantId);
|
||||||
|
const trx = await knex.transaction({ isolationLevel: 'read uncommitted' });
|
||||||
|
|
||||||
|
const meta = await this.importFile.import(tenantId, importId, trx);
|
||||||
|
|
||||||
|
// Rollback the successed transaction.
|
||||||
|
await trx.rollback();
|
||||||
|
|
||||||
|
return meta;
|
||||||
|
}
|
||||||
|
}
|
||||||
96
packages/server/src/services/Import/ImportFileProcess.ts
Normal file
96
packages/server/src/services/Import/ImportFileProcess.ts
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { chain } from 'lodash';
|
||||||
|
import { Knex } from 'knex';
|
||||||
|
import { ServiceError } from '@/exceptions';
|
||||||
|
import { ERRORS, getSheetColumns, getUnmappedSheetColumns } from './_utils';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import { ImportFileCommon } from './ImportFileCommon';
|
||||||
|
import { ImportFileDataTransformer } from './ImportFileDataTransformer';
|
||||||
|
import ResourceService from '../Resource/ResourceService';
|
||||||
|
import UnitOfWork from '../UnitOfWork';
|
||||||
|
import { ImportFilePreviewPOJO } from './interfaces';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportFileProcess {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private resource: ResourceService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importCommon: ImportFileCommon;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importParser: ImportFileDataTransformer;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private uow: UnitOfWork;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Preview the imported file results before commiting the transactions.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} importId
|
||||||
|
* @returns {Promise<ImportFilePreviewPOJO>}
|
||||||
|
*/
|
||||||
|
public async import(
|
||||||
|
tenantId: number,
|
||||||
|
importId: number,
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
): Promise<ImportFilePreviewPOJO> {
|
||||||
|
const { Import } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
const importFile = await Import.query()
|
||||||
|
.findOne('importId', importId)
|
||||||
|
.throwIfNotFound();
|
||||||
|
|
||||||
|
// Throw error if the import file is not mapped yet.
|
||||||
|
if (!importFile.isMapped) {
|
||||||
|
throw new ServiceError(ERRORS.IMPORT_FILE_NOT_MAPPED);
|
||||||
|
}
|
||||||
|
// Read the imported file.
|
||||||
|
const buffer = await this.importCommon.readImportFile(importFile.filename);
|
||||||
|
const sheetData = this.importCommon.parseXlsxSheet(buffer);
|
||||||
|
const header = getSheetColumns(sheetData);
|
||||||
|
|
||||||
|
const importableFields = this.resource.getResourceImportableFields(
|
||||||
|
tenantId,
|
||||||
|
importFile.resource
|
||||||
|
);
|
||||||
|
// Prases the sheet json data.
|
||||||
|
const parsedData = this.importParser.parseSheetData(
|
||||||
|
importFile,
|
||||||
|
importableFields,
|
||||||
|
sheetData
|
||||||
|
);
|
||||||
|
// Runs the importing operation with ability to return errors that will happen.
|
||||||
|
const [successedImport, failedImport] = await this.uow.withTransaction(
|
||||||
|
tenantId,
|
||||||
|
(trx: Knex.Transaction) =>
|
||||||
|
this.importCommon.import(tenantId, importFile, parsedData, trx),
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
const mapping = importFile.mappingParsed;
|
||||||
|
const errors = chain(failedImport)
|
||||||
|
.map((oper) => oper.error)
|
||||||
|
.flatten()
|
||||||
|
.value();
|
||||||
|
|
||||||
|
const unmappedColumns = getUnmappedSheetColumns(header, mapping);
|
||||||
|
const totalCount = parsedData.length;
|
||||||
|
|
||||||
|
const createdCount = successedImport.length;
|
||||||
|
const errorsCount = failedImport.length;
|
||||||
|
const skippedCount = errorsCount;
|
||||||
|
|
||||||
|
return {
|
||||||
|
createdCount,
|
||||||
|
skippedCount,
|
||||||
|
totalCount,
|
||||||
|
errorsCount,
|
||||||
|
errors,
|
||||||
|
unmappedColumns: unmappedColumns,
|
||||||
|
unmappedColumnsCount: unmappedColumns.length,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
111
packages/server/src/services/Import/ImportFileUpload.ts
Normal file
111
packages/server/src/services/Import/ImportFileUpload.ts
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import HasTenancyService from '../Tenancy/TenancyService';
|
||||||
|
import { sanitizeResourceName } from './_utils';
|
||||||
|
import ResourceService from '../Resource/ResourceService';
|
||||||
|
import { IModelMetaField } from '@/interfaces';
|
||||||
|
import { ImportFileCommon } from './ImportFileCommon';
|
||||||
|
import { ImportFileDataValidator } from './ImportFileDataValidator';
|
||||||
|
import { ImportFileUploadPOJO } from './interfaces';
|
||||||
|
import { ServiceError } from '@/exceptions';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportFileUploadService {
|
||||||
|
@Inject()
|
||||||
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private resourceService: ResourceService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importFileCommon: ImportFileCommon;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importValidator: ImportFileDataValidator;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the imported file and stores the import file meta under unqiue id.
|
||||||
|
* @param {number} tenantId - Tenant id.
|
||||||
|
* @param {string} resource - Resource name.
|
||||||
|
* @param {string} filePath - File path.
|
||||||
|
* @param {string} fileName - File name.
|
||||||
|
* @returns {Promise<ImportFileUploadPOJO>}
|
||||||
|
*/
|
||||||
|
public async import(
|
||||||
|
tenantId: number,
|
||||||
|
resourceName: string,
|
||||||
|
filename: string,
|
||||||
|
params: Record<string, number | string>
|
||||||
|
): Promise<ImportFileUploadPOJO> {
|
||||||
|
const { Import } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
const resource = sanitizeResourceName(resourceName);
|
||||||
|
const resourceMeta = this.resourceService.getResourceMeta(
|
||||||
|
tenantId,
|
||||||
|
resource
|
||||||
|
);
|
||||||
|
// Throw service error if the resource does not support importing.
|
||||||
|
this.importValidator.validateResourceImportable(resourceMeta);
|
||||||
|
|
||||||
|
// Reads the imported file into buffer.
|
||||||
|
const buffer = await this.importFileCommon.readImportFile(filename);
|
||||||
|
|
||||||
|
// Parse the buffer file to array data.
|
||||||
|
const sheetData = this.importFileCommon.parseXlsxSheet(buffer);
|
||||||
|
const sheetColumns = this.importFileCommon.parseSheetColumns(sheetData);
|
||||||
|
const coumnsStringified = JSON.stringify(sheetColumns);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Validates the params Yup schema.
|
||||||
|
await this.importFileCommon.validateParamsSchema(resource, params);
|
||||||
|
|
||||||
|
// Validates importable params asyncly.
|
||||||
|
await this.importFileCommon.validateParams(tenantId, resource, params);
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
const _params = this.importFileCommon.transformParams(resource, params);
|
||||||
|
const paramsStringified = JSON.stringify(_params);
|
||||||
|
|
||||||
|
// Store the import model with related metadata.
|
||||||
|
const importFile = await Import.query().insert({
|
||||||
|
filename,
|
||||||
|
resource,
|
||||||
|
importId: filename,
|
||||||
|
columns: coumnsStringified,
|
||||||
|
params: paramsStringified,
|
||||||
|
});
|
||||||
|
const resourceColumnsMap = this.resourceService.getResourceImportableFields(
|
||||||
|
tenantId,
|
||||||
|
resource
|
||||||
|
);
|
||||||
|
const resourceColumns = this.getResourceColumns(resourceColumnsMap);
|
||||||
|
|
||||||
|
return {
|
||||||
|
import: {
|
||||||
|
importId: importFile.importId,
|
||||||
|
resource: importFile.resource,
|
||||||
|
},
|
||||||
|
sheetColumns,
|
||||||
|
resourceColumns,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
getResourceColumns(resourceColumns: { [key: string]: IModelMetaField }) {
|
||||||
|
return Object.entries(resourceColumns)
|
||||||
|
.map(
|
||||||
|
([key, { name, importHint, required, order }]: [
|
||||||
|
string,
|
||||||
|
IModelMetaField
|
||||||
|
]) => ({
|
||||||
|
key,
|
||||||
|
name,
|
||||||
|
required,
|
||||||
|
hint: importHint,
|
||||||
|
order,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
.sort((a, b) =>
|
||||||
|
a.order && b.order ? a.order - b.order : a.order ? -1 : b.order ? 1 : 0
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
102
packages/server/src/services/Import/ImportResourceApplication.ts
Normal file
102
packages/server/src/services/Import/ImportResourceApplication.ts
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
import { Inject } from 'typedi';
|
||||||
|
import { ImportFileUploadService } from './ImportFileUpload';
|
||||||
|
import { ImportFileMapping } from './ImportFileMapping';
|
||||||
|
import { ImportMappingAttr } from './interfaces';
|
||||||
|
import { ImportFileProcess } from './ImportFileProcess';
|
||||||
|
import { ImportFilePreview } from './ImportFilePreview';
|
||||||
|
import { ImportSampleService } from './ImportSample';
|
||||||
|
import { ImportFileMeta } from './ImportFileMeta';
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
export class ImportResourceApplication {
|
||||||
|
@Inject()
|
||||||
|
private importFileService: ImportFileUploadService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importMappingService: ImportFileMapping;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importProcessService: ImportFileProcess;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private ImportFilePreviewService: ImportFilePreview;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importSampleService: ImportSampleService;
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
private importMetaService: ImportFileMeta;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the imported file and stores the import file meta under unqiue id.
|
||||||
|
* @param {number} tenantId -
|
||||||
|
* @param {string} resource - Resource name.
|
||||||
|
* @param {string} fileName - File name.
|
||||||
|
* @returns {Promise<ImportFileUploadPOJO>}
|
||||||
|
*/
|
||||||
|
public async import(
|
||||||
|
tenantId: number,
|
||||||
|
resource: string,
|
||||||
|
filename: string,
|
||||||
|
params: Record<string, any>
|
||||||
|
) {
|
||||||
|
return this.importFileService.import(tenantId, resource, filename, params);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Mapping the excel sheet columns with resource columns.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} importId - Import id.
|
||||||
|
* @param {ImportMappingAttr} maps
|
||||||
|
*/
|
||||||
|
public async mapping(
|
||||||
|
tenantId: number,
|
||||||
|
importId: number,
|
||||||
|
maps: ImportMappingAttr[]
|
||||||
|
) {
|
||||||
|
return this.importMappingService.mapping(tenantId, importId, maps);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Preview the mapped results before process importing.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} importId - Import id.
|
||||||
|
* @returns {Promise<ImportFilePreviewPOJO>}
|
||||||
|
*/
|
||||||
|
public async preview(tenantId: number, importId: number) {
|
||||||
|
return this.ImportFilePreviewService.preview(tenantId, importId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Process the import file sheet through service for creating entities.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} importId
|
||||||
|
* @returns {Promise<ImportFilePreviewPOJO>}
|
||||||
|
*/
|
||||||
|
public async process(tenantId: number, importId: number) {
|
||||||
|
return this.importProcessService.import(tenantId, importId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the import meta of the given import id.
|
||||||
|
* @param {number} tenantId -
|
||||||
|
* @param {string} importId - Import id.
|
||||||
|
* @returns {}
|
||||||
|
*/
|
||||||
|
public importMeta(tenantId: number, importId: string) {
|
||||||
|
return this.importMetaService.getImportMeta(tenantId, importId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the csv/xlsx sample sheet of the given
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} resource - Resource name.
|
||||||
|
*/
|
||||||
|
public sample(
|
||||||
|
tenantId: number,
|
||||||
|
resource: string,
|
||||||
|
format: 'csv' | 'xlsx' = 'csv'
|
||||||
|
) {
|
||||||
|
return this.importSampleService.sample(tenantId, resource, format);
|
||||||
|
}
|
||||||
|
}
|
||||||
46
packages/server/src/services/Import/ImportSample.ts
Normal file
46
packages/server/src/services/Import/ImportSample.ts
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
import XLSX from 'xlsx';
|
||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { ImportableResources } from './ImportableResources';
|
||||||
|
import { sanitizeResourceName } from './_utils';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportSampleService {
|
||||||
|
@Inject()
|
||||||
|
private importable: ImportableResources;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the sample sheet of the given resource.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {string} resource
|
||||||
|
* @param {string} format
|
||||||
|
* @returns {Buffer | string}
|
||||||
|
*/
|
||||||
|
public sample(
|
||||||
|
tenantId: number,
|
||||||
|
resource: string,
|
||||||
|
format: 'csv' | 'xlsx'
|
||||||
|
): Buffer | string {
|
||||||
|
const _resource = sanitizeResourceName(resource);
|
||||||
|
|
||||||
|
const ImportableRegistry = this.importable.registry;
|
||||||
|
const importable = ImportableRegistry.getImportable(_resource);
|
||||||
|
|
||||||
|
const data = importable.sampleData();
|
||||||
|
|
||||||
|
const workbook = XLSX.utils.book_new();
|
||||||
|
const worksheet = XLSX.utils.json_to_sheet(data);
|
||||||
|
XLSX.utils.book_append_sheet(workbook, worksheet, 'Sheet1');
|
||||||
|
|
||||||
|
// Determine the output format
|
||||||
|
if (format === 'csv') {
|
||||||
|
const csvOutput = XLSX.utils.sheet_to_csv(worksheet);
|
||||||
|
return csvOutput;
|
||||||
|
} else {
|
||||||
|
const xlsxOutput = XLSX.write(workbook, {
|
||||||
|
bookType: 'xlsx',
|
||||||
|
type: 'buffer',
|
||||||
|
});
|
||||||
|
return xlsxOutput;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
72
packages/server/src/services/Import/Importable.ts
Normal file
72
packages/server/src/services/Import/Importable.ts
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
import { Knex } from 'knex';
|
||||||
|
import * as Yup from 'yup';
|
||||||
|
import { ImportableContext } from './interfaces';
|
||||||
|
|
||||||
|
export abstract class Importable {
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {any} createDTO
|
||||||
|
* @param {Knex.Transaction} trx
|
||||||
|
*/
|
||||||
|
public importable(tenantId: number, createDTO: any, trx?: Knex.Transaction) {
|
||||||
|
throw new Error(
|
||||||
|
'The `importable` function is not defined in service importable.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transformes the DTO before passing it to importable and validation.
|
||||||
|
* @param {Record<string, any>} createDTO
|
||||||
|
* @param {ImportableContext} context
|
||||||
|
* @returns {Record<string, any>}
|
||||||
|
*/
|
||||||
|
public transform(createDTO: Record<string, any>, context: ImportableContext) {
|
||||||
|
return createDTO;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Concurrency controlling of the importing process.
|
||||||
|
* @returns {number}
|
||||||
|
*/
|
||||||
|
public get concurrency() {
|
||||||
|
return 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the sample data of importable.
|
||||||
|
* @returns {Array<any>}
|
||||||
|
*/
|
||||||
|
public sampleData(): Array<any> {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
// ------------------
|
||||||
|
// # Params
|
||||||
|
// ------------------
|
||||||
|
/**
|
||||||
|
* Params Yup validation schema.
|
||||||
|
* @returns {Yup.ObjectSchema<object, object>}
|
||||||
|
*/
|
||||||
|
public paramsValidationSchema(): Yup.ObjectSchema<object, object> {
|
||||||
|
return Yup.object().nullable();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates the params of the importable service.
|
||||||
|
* @param {Record<string, any>}
|
||||||
|
* @returns {Promise<boolean>} - True means passed and false failed.
|
||||||
|
*/
|
||||||
|
public async validateParams(
|
||||||
|
tenantId: number,
|
||||||
|
params: Record<string, any>
|
||||||
|
): Promise<void> {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transformes the import params before storing them.
|
||||||
|
* @param {Record<string, any>} parmas
|
||||||
|
*/
|
||||||
|
public transformParams(parmas: Record<string, any>) {
|
||||||
|
return parmas;
|
||||||
|
}
|
||||||
|
}
|
||||||
46
packages/server/src/services/Import/ImportableRegistry.ts
Normal file
46
packages/server/src/services/Import/ImportableRegistry.ts
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
import { camelCase, upperFirst } from 'lodash';
|
||||||
|
import { Importable } from './Importable';
|
||||||
|
|
||||||
|
export class ImportableRegistry {
|
||||||
|
private static instance: ImportableRegistry;
|
||||||
|
private importables: Record<string, Importable>;
|
||||||
|
|
||||||
|
private constructor() {
|
||||||
|
this.importables = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets singleton instance of registry.
|
||||||
|
* @returns {ImportableRegistry}
|
||||||
|
*/
|
||||||
|
public static getInstance(): ImportableRegistry {
|
||||||
|
if (!ImportableRegistry.instance) {
|
||||||
|
ImportableRegistry.instance = new ImportableRegistry();
|
||||||
|
}
|
||||||
|
return ImportableRegistry.instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers the given importable service.
|
||||||
|
* @param {string} resource
|
||||||
|
* @param {Importable} importable
|
||||||
|
*/
|
||||||
|
public registerImportable(resource: string, importable: Importable): void {
|
||||||
|
const _resource = this.sanitizeResourceName(resource);
|
||||||
|
this.importables[_resource] = importable;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the importable service instance of the given resource name.
|
||||||
|
* @param {string} name
|
||||||
|
* @returns {Importable}
|
||||||
|
*/
|
||||||
|
public getImportable(name: string): Importable {
|
||||||
|
const _name = this.sanitizeResourceName(name);
|
||||||
|
return this.importables[_name];
|
||||||
|
}
|
||||||
|
|
||||||
|
private sanitizeResourceName(resource: string) {
|
||||||
|
return upperFirst(camelCase(resource));
|
||||||
|
}
|
||||||
|
}
|
||||||
47
packages/server/src/services/Import/ImportableResources.ts
Normal file
47
packages/server/src/services/Import/ImportableResources.ts
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
import Container, { Service } from 'typedi';
|
||||||
|
import { AccountsImportable } from '../Accounts/AccountsImportable';
|
||||||
|
import { ImportableRegistry } from './ImportableRegistry';
|
||||||
|
import { UncategorizedTransactionsImportable } from '../Cashflow/UncategorizedTransactionsImportable';
|
||||||
|
import { CustomersImportable } from '../Contacts/Customers/CustomersImportable';
|
||||||
|
import { VendorsImportable } from '../Contacts/Vendors/VendorsImportable';
|
||||||
|
|
||||||
|
@Service()
|
||||||
|
export class ImportableResources {
|
||||||
|
private static registry: ImportableRegistry;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.boot();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Importable instances.
|
||||||
|
*/
|
||||||
|
private importables = [
|
||||||
|
{ resource: 'Account', importable: AccountsImportable },
|
||||||
|
{
|
||||||
|
resource: 'UncategorizedCashflowTransaction',
|
||||||
|
importable: UncategorizedTransactionsImportable,
|
||||||
|
},
|
||||||
|
{ resource: 'Customer', importable: CustomersImportable },
|
||||||
|
{ resource: 'Vendor', importable: VendorsImportable },
|
||||||
|
];
|
||||||
|
|
||||||
|
public get registry() {
|
||||||
|
return ImportableResources.registry;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Boots all the registered importables.
|
||||||
|
*/
|
||||||
|
public boot() {
|
||||||
|
if (!ImportableResources.registry) {
|
||||||
|
const instance = ImportableRegistry.getInstance();
|
||||||
|
|
||||||
|
this.importables.forEach((importable) => {
|
||||||
|
const importableInstance = Container.get(importable.importable);
|
||||||
|
instance.registerImportable(importable.resource, importableInstance);
|
||||||
|
});
|
||||||
|
ImportableResources.registry = instance;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
103
packages/server/src/services/Import/_utils.ts
Normal file
103
packages/server/src/services/Import/_utils.ts
Normal file
@@ -0,0 +1,103 @@
|
|||||||
|
import * as Yup from 'yup';
|
||||||
|
import { upperFirst, camelCase, first, isUndefined } from 'lodash';
|
||||||
|
import pluralize from 'pluralize';
|
||||||
|
import { ResourceMetaFieldsMap } from './interfaces';
|
||||||
|
import { IModelMetaField } from '@/interfaces';
|
||||||
|
import moment from 'moment';
|
||||||
|
|
||||||
|
export const ERRORS = {
|
||||||
|
RESOURCE_NOT_IMPORTABLE: 'RESOURCE_NOT_IMPORTABLE',
|
||||||
|
INVALID_MAP_ATTRS: 'INVALID_MAP_ATTRS',
|
||||||
|
DUPLICATED_FROM_MAP_ATTR: 'DUPLICATED_FROM_MAP_ATTR',
|
||||||
|
DUPLICATED_TO_MAP_ATTR: 'DUPLICATED_TO_MAP_ATTR',
|
||||||
|
IMPORT_FILE_NOT_MAPPED: 'IMPORT_FILE_NOT_MAPPED',
|
||||||
|
INVALID_MAP_DATE_FORMAT: 'INVALID_MAP_DATE_FORMAT',
|
||||||
|
MAP_DATE_FORMAT_NOT_DEFINED: 'MAP_DATE_FORMAT_NOT_DEFINED',
|
||||||
|
};
|
||||||
|
|
||||||
|
export function trimObject(obj) {
|
||||||
|
return Object.entries(obj).reduce((acc, [key, value]) => {
|
||||||
|
// Trim the key
|
||||||
|
const trimmedKey = key.trim();
|
||||||
|
|
||||||
|
// Trim the value if it's a string, otherwise leave it as is
|
||||||
|
const trimmedValue = typeof value === 'string' ? value.trim() : value;
|
||||||
|
|
||||||
|
// Assign the trimmed key and value to the accumulator object
|
||||||
|
return { ...acc, [trimmedKey]: trimmedValue };
|
||||||
|
}, {});
|
||||||
|
}
|
||||||
|
|
||||||
|
export const convertFieldsToYupValidation = (fields: ResourceMetaFieldsMap) => {
|
||||||
|
const yupSchema = {};
|
||||||
|
Object.keys(fields).forEach((fieldName: string) => {
|
||||||
|
const field = fields[fieldName] as IModelMetaField;
|
||||||
|
let fieldSchema;
|
||||||
|
fieldSchema = Yup.string().label(field.name);
|
||||||
|
|
||||||
|
if (field.fieldType === 'text') {
|
||||||
|
if (!isUndefined(field.minLength)) {
|
||||||
|
fieldSchema = fieldSchema.min(
|
||||||
|
field.minLength,
|
||||||
|
`Minimum length is ${field.minLength} characters`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (!isUndefined(field.maxLength)) {
|
||||||
|
fieldSchema = fieldSchema.max(
|
||||||
|
field.maxLength,
|
||||||
|
`Maximum length is ${field.maxLength} characters`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else if (field.fieldType === 'number') {
|
||||||
|
fieldSchema = Yup.number().label(field.name);
|
||||||
|
|
||||||
|
if (!isUndefined(field.max)) {
|
||||||
|
fieldSchema = fieldSchema.max(field.max);
|
||||||
|
}
|
||||||
|
if (!isUndefined(field.min)) {
|
||||||
|
fieldSchema = fieldSchema.min(field.min);
|
||||||
|
}
|
||||||
|
} else if (field.fieldType === 'boolean') {
|
||||||
|
fieldSchema = Yup.boolean().label(field.name);
|
||||||
|
} else if (field.fieldType === 'enumeration') {
|
||||||
|
const options = field.options.reduce((acc, option) => {
|
||||||
|
acc[option.key] = option.label;
|
||||||
|
return acc;
|
||||||
|
}, {});
|
||||||
|
fieldSchema = Yup.string().oneOf(Object.keys(options)).label(field.name);
|
||||||
|
// Validate date field type.
|
||||||
|
} else if (field.fieldType === 'date') {
|
||||||
|
fieldSchema = fieldSchema.test(
|
||||||
|
'date validation',
|
||||||
|
'Invalid date or format. The string should be a valid YYYY-MM-DD format.',
|
||||||
|
(val) => {
|
||||||
|
if (!val) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return moment(val, 'YYYY-MM-DD', true).isValid();
|
||||||
|
}
|
||||||
|
);
|
||||||
|
} else if (field.fieldType === 'url') {
|
||||||
|
fieldSchema = fieldSchema.url();
|
||||||
|
}
|
||||||
|
if (field.required) {
|
||||||
|
fieldSchema = fieldSchema.required();
|
||||||
|
}
|
||||||
|
yupSchema[fieldName] = fieldSchema;
|
||||||
|
});
|
||||||
|
return Yup.object().shape(yupSchema);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getUnmappedSheetColumns = (columns, mapping) => {
|
||||||
|
return columns.filter(
|
||||||
|
(column) => !mapping.some((map) => map.from === column)
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const sanitizeResourceName = (resourceName: string) => {
|
||||||
|
return upperFirst(camelCase(pluralize.singular(resourceName)));
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getSheetColumns = (sheetData: unknown[]) => {
|
||||||
|
return Object.keys(first(sheetData));
|
||||||
|
};
|
||||||
76
packages/server/src/services/Import/interfaces.ts
Normal file
76
packages/server/src/services/Import/interfaces.ts
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
import { IModelMetaField } from '@/interfaces';
|
||||||
|
import Import from '@/models/Import';
|
||||||
|
|
||||||
|
export interface ImportMappingAttr {
|
||||||
|
from: string;
|
||||||
|
to: string;
|
||||||
|
dateFormat?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ImportValidationError {
|
||||||
|
index: number;
|
||||||
|
property: string;
|
||||||
|
constraints: Record<string, string>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ResourceMetaFieldsMap = { [key: string]: IModelMetaField };
|
||||||
|
|
||||||
|
export interface ImportInsertError {
|
||||||
|
rowNumber: number;
|
||||||
|
errorCode: string;
|
||||||
|
errorMessage: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ImportFileUploadPOJO {
|
||||||
|
import: {
|
||||||
|
importId: string;
|
||||||
|
resource: string;
|
||||||
|
};
|
||||||
|
sheetColumns: string[];
|
||||||
|
resourceColumns: {
|
||||||
|
key: string;
|
||||||
|
name: string;
|
||||||
|
required?: boolean;
|
||||||
|
hint?: string;
|
||||||
|
}[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ImportFileMapPOJO {
|
||||||
|
import: {
|
||||||
|
importId: string;
|
||||||
|
resource: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ImportFilePreviewPOJO {
|
||||||
|
createdCount: number;
|
||||||
|
skippedCount: number;
|
||||||
|
totalCount: number;
|
||||||
|
errorsCount: number;
|
||||||
|
errors: ImportInsertError[];
|
||||||
|
unmappedColumns: string[];
|
||||||
|
unmappedColumnsCount: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ImportOperSuccess {
|
||||||
|
data: unknown;
|
||||||
|
index: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ImportOperError {
|
||||||
|
error: ImportInsertError;
|
||||||
|
index: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ImportableContext {
|
||||||
|
import: Import,
|
||||||
|
rowIndex: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export const ImportDateFormats = [
|
||||||
|
'yyyy-MM-dd',
|
||||||
|
'dd.MM.yy',
|
||||||
|
'MM/dd/yy',
|
||||||
|
'dd/MMM/yyyy'
|
||||||
|
]
|
||||||
@@ -21,25 +21,19 @@ import { ERRORS } from './constants';
|
|||||||
@Service()
|
@Service()
|
||||||
export default class OrganizationService {
|
export default class OrganizationService {
|
||||||
@Inject()
|
@Inject()
|
||||||
eventPublisher: EventPublisher;
|
private eventPublisher: EventPublisher;
|
||||||
|
|
||||||
@Inject('logger')
|
|
||||||
logger: any;
|
|
||||||
|
|
||||||
@Inject('repositories')
|
|
||||||
sysRepositories: any;
|
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
tenantsManager: TenantsManager;
|
private tenantsManager: TenantsManager;
|
||||||
|
|
||||||
@Inject('agenda')
|
@Inject('agenda')
|
||||||
agenda: any;
|
private agenda: any;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
baseCurrencyMutateLocking: OrganizationBaseCurrencyLocking;
|
private baseCurrencyMutateLocking: OrganizationBaseCurrencyLocking;
|
||||||
|
|
||||||
@Inject()
|
@Inject()
|
||||||
tenancy: HasTenancyService;
|
private tenancy: HasTenancyService;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Builds the database schema and seed data of the given organization id.
|
* Builds the database schema and seed data of the given organization id.
|
||||||
|
|||||||
@@ -1,12 +1,11 @@
|
|||||||
import { Service, Inject } from 'typedi';
|
import { Service, Inject } from 'typedi';
|
||||||
import { camelCase, upperFirst } from 'lodash';
|
import { camelCase, upperFirst, pickBy } from 'lodash';
|
||||||
import * as qim from 'qim';
|
import * as qim from 'qim';
|
||||||
import pluralize from 'pluralize';
|
import pluralize from 'pluralize';
|
||||||
import { IModelMeta } from '@/interfaces';
|
import { IModelMeta, IModelMetaField } from '@/interfaces';
|
||||||
import TenancyService from '@/services/Tenancy/TenancyService';
|
import TenancyService from '@/services/Tenancy/TenancyService';
|
||||||
import { ServiceError } from '@/exceptions';
|
import { ServiceError } from '@/exceptions';
|
||||||
import I18nService from '@/services/I18n/I18nService';
|
import I18nService from '@/services/I18n/I18nService';
|
||||||
import { tenantKnexConfig } from 'config/knexConfig';
|
|
||||||
|
|
||||||
const ERRORS = {
|
const ERRORS = {
|
||||||
RESOURCE_MODEL_NOT_FOUND: 'RESOURCE_MODEL_NOT_FOUND',
|
RESOURCE_MODEL_NOT_FOUND: 'RESOURCE_MODEL_NOT_FOUND',
|
||||||
@@ -24,7 +23,7 @@ export default class ResourceService {
|
|||||||
* Transform resource to model name.
|
* Transform resource to model name.
|
||||||
* @param {string} resourceName
|
* @param {string} resourceName
|
||||||
*/
|
*/
|
||||||
private resourceToModelName(resourceName: string): string {
|
public resourceToModelName(resourceName: string): string {
|
||||||
return upperFirst(camelCase(pluralize.singular(resourceName)));
|
return upperFirst(camelCase(pluralize.singular(resourceName)));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -63,6 +62,33 @@ export default class ResourceService {
|
|||||||
return this.getResourceMetaLocalized(resourceMeta, tenantId);
|
return this.getResourceMetaLocalized(resourceMeta, tenantId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public getResourceFields(
|
||||||
|
tenantId: number,
|
||||||
|
modelName: string
|
||||||
|
): { [key: string]: IModelMetaField } {
|
||||||
|
const meta = this.getResourceMeta(tenantId, modelName);
|
||||||
|
|
||||||
|
return meta.fields;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {string} modelName
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
public getResourceImportableFields(
|
||||||
|
tenantId: number,
|
||||||
|
modelName: string
|
||||||
|
): { [key: string]: IModelMetaField } {
|
||||||
|
const fields = this.getResourceFields(tenantId, modelName);
|
||||||
|
|
||||||
|
return pickBy(fields, (field) => field.importable);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve the resource meta localized based on the current user language.
|
* Retrieve the resource meta localized based on the current user language.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { Service, Inject } from 'typedi';
|
import { Service, Inject } from 'typedi';
|
||||||
import TenancyService from '@/services/Tenancy/TenancyService';
|
import TenancyService from '@/services/Tenancy/TenancyService';
|
||||||
|
import { Transaction } from 'objection';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Enumeration that represents transaction isolation levels for use with the {@link Transactional} annotation
|
* Enumeration that represents transaction isolation levels for use with the {@link Transactional} annotation
|
||||||
@@ -38,18 +39,26 @@ export default class UnitOfWork {
|
|||||||
public withTransaction = async (
|
public withTransaction = async (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
work,
|
work,
|
||||||
|
trx?: Transaction,
|
||||||
isolationLevel: IsolationLevel = IsolationLevel.READ_UNCOMMITTED
|
isolationLevel: IsolationLevel = IsolationLevel.READ_UNCOMMITTED
|
||||||
) => {
|
) => {
|
||||||
const knex = this.tenancy.knex(tenantId);
|
const knex = this.tenancy.knex(tenantId);
|
||||||
const trx = await knex.transaction({ isolationLevel });
|
let _trx = trx;
|
||||||
|
|
||||||
|
if (!_trx) {
|
||||||
|
_trx = await knex.transaction({ isolationLevel });
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
const result = await work(trx);
|
const result = await work(_trx);
|
||||||
trx.commit();
|
|
||||||
|
|
||||||
|
if (!trx) {
|
||||||
|
_trx.commit();
|
||||||
|
}
|
||||||
return result;
|
return result;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
trx.rollback();
|
if (!trx) {
|
||||||
|
_trx.rollback();
|
||||||
|
}
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
7
packages/server/src/utils/parse-json-safe.ts
Normal file
7
packages/server/src/utils/parse-json-safe.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
export const parseJsonSafe = (value: string) => {
|
||||||
|
try {
|
||||||
|
return JSON.parse(value);
|
||||||
|
} catch {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -20,11 +20,11 @@
|
|||||||
"@testing-library/jest-dom": "^4.2.4",
|
"@testing-library/jest-dom": "^4.2.4",
|
||||||
"@testing-library/react": "^9.4.0",
|
"@testing-library/react": "^9.4.0",
|
||||||
"@testing-library/user-event": "^7.2.1",
|
"@testing-library/user-event": "^7.2.1",
|
||||||
"@tiptap/extension-color": "latest",
|
|
||||||
"@tiptap/extension-text-style": "2.1.13",
|
|
||||||
"@tiptap/core": "2.1.13",
|
"@tiptap/core": "2.1.13",
|
||||||
"@tiptap/pm": "2.1.13",
|
"@tiptap/extension-color": "latest",
|
||||||
"@tiptap/extension-list-item": "2.1.13",
|
"@tiptap/extension-list-item": "2.1.13",
|
||||||
|
"@tiptap/extension-text-style": "2.1.13",
|
||||||
|
"@tiptap/pm": "2.1.13",
|
||||||
"@tiptap/react": "2.1.13",
|
"@tiptap/react": "2.1.13",
|
||||||
"@tiptap/starter-kit": "2.1.13",
|
"@tiptap/starter-kit": "2.1.13",
|
||||||
"@types/jest": "^26.0.15",
|
"@types/jest": "^26.0.15",
|
||||||
@@ -38,9 +38,9 @@
|
|||||||
"@types/react-redux": "^7.1.24",
|
"@types/react-redux": "^7.1.24",
|
||||||
"@types/react-router-dom": "^5.3.3",
|
"@types/react-router-dom": "^5.3.3",
|
||||||
"@types/react-transition-group": "^4.4.5",
|
"@types/react-transition-group": "^4.4.5",
|
||||||
|
"@types/socket.io-client": "^3.0.0",
|
||||||
"@types/styled-components": "^5.1.25",
|
"@types/styled-components": "^5.1.25",
|
||||||
"@types/yup": "^0.29.13",
|
"@types/yup": "^0.29.13",
|
||||||
"@types/socket.io-client": "^3.0.0",
|
|
||||||
"@typescript-eslint/eslint-plugin": "^2.10.0",
|
"@typescript-eslint/eslint-plugin": "^2.10.0",
|
||||||
"@typescript-eslint/parser": "^2.10.0",
|
"@typescript-eslint/parser": "^2.10.0",
|
||||||
"@welldone-software/why-did-you-render": "^6.0.0-rc.1",
|
"@welldone-software/why-did-you-render": "^6.0.0-rc.1",
|
||||||
@@ -69,10 +69,9 @@
|
|||||||
"moment": "^2.24.0",
|
"moment": "^2.24.0",
|
||||||
"moment-timezone": "^0.5.33",
|
"moment-timezone": "^0.5.33",
|
||||||
"path-browserify": "^1.0.1",
|
"path-browserify": "^1.0.1",
|
||||||
"prop-types": "15.8.1",
|
|
||||||
"plaid": "^9.3.0",
|
"plaid": "^9.3.0",
|
||||||
"plaid-threads": "^11.4.3",
|
"plaid-threads": "^11.4.3",
|
||||||
"react-plaid-link": "^3.2.1",
|
"prop-types": "15.8.1",
|
||||||
"query-string": "^7.1.1",
|
"query-string": "^7.1.1",
|
||||||
"ramda": "^0.27.1",
|
"ramda": "^0.27.1",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
@@ -82,11 +81,13 @@
|
|||||||
"react-dev-utils": "^11.0.4",
|
"react-dev-utils": "^11.0.4",
|
||||||
"react-dom": "^18.2.0",
|
"react-dom": "^18.2.0",
|
||||||
"react-dropzone": "^11.0.1",
|
"react-dropzone": "^11.0.1",
|
||||||
|
"react-dropzone-esm": "^15.0.1",
|
||||||
"react-error-boundary": "^3.0.2",
|
"react-error-boundary": "^3.0.2",
|
||||||
"react-error-overlay": "^6.0.9",
|
"react-error-overlay": "^6.0.9",
|
||||||
"react-hotkeys-hook": "^3.0.3",
|
"react-hotkeys-hook": "^3.0.3",
|
||||||
"react-intl-universal": "^2.4.7",
|
"react-intl-universal": "^2.4.7",
|
||||||
"react-loadable": "^5.5.0",
|
"react-loadable": "^5.5.0",
|
||||||
|
"react-plaid-link": "^3.2.1",
|
||||||
"react-query": "^3.6.0",
|
"react-query": "^3.6.0",
|
||||||
"react-query-devtools": "^2.1.1",
|
"react-query-devtools": "^2.1.1",
|
||||||
"react-redux": "^7.2.9",
|
"react-redux": "^7.2.9",
|
||||||
@@ -112,10 +113,10 @@
|
|||||||
"rtl-detect": "^1.0.3",
|
"rtl-detect": "^1.0.3",
|
||||||
"sass": "^1.68.0",
|
"sass": "^1.68.0",
|
||||||
"semver": "6.3.0",
|
"semver": "6.3.0",
|
||||||
|
"socket.io-client": "^4.7.4",
|
||||||
"style-loader": "0.23.1",
|
"style-loader": "0.23.1",
|
||||||
"styled-components": "^5.3.1",
|
"styled-components": "^5.3.1",
|
||||||
"stylis-rtlcss": "^2.1.1",
|
"stylis-rtlcss": "^2.1.1",
|
||||||
"socket.io-client": "^4.7.4",
|
|
||||||
"typescript": "^4.8.3",
|
"typescript": "^4.8.3",
|
||||||
"yup": "^0.28.1"
|
"yup": "^0.28.1"
|
||||||
},
|
},
|
||||||
|
|||||||
12
packages/webapp/src/components/Dropzone/Dropzone.module.css
Normal file
12
packages/webapp/src/components/Dropzone/Dropzone.module.css
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
|
||||||
|
|
||||||
|
.root {
|
||||||
|
padding: 20px;
|
||||||
|
border: 2px dotted #c5cbd3;
|
||||||
|
border-radius: 6px;
|
||||||
|
min-height: 200px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
background: #fff;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
291
packages/webapp/src/components/Dropzone/Dropzone.tsx
Normal file
291
packages/webapp/src/components/Dropzone/Dropzone.tsx
Normal file
@@ -0,0 +1,291 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import React from 'react';
|
||||||
|
import { Ref, useCallback } from 'react';
|
||||||
|
import clsx from 'classnames';
|
||||||
|
import {
|
||||||
|
Accept,
|
||||||
|
DropEvent,
|
||||||
|
FileError,
|
||||||
|
FileRejection,
|
||||||
|
FileWithPath,
|
||||||
|
useDropzone,
|
||||||
|
} from 'react-dropzone-esm';
|
||||||
|
import { DropzoneProvider } from './DropzoneProvider';
|
||||||
|
import { DropzoneAccept, DropzoneIdle, DropzoneReject } from './DropzoneStatus';
|
||||||
|
import { Box } from '../Layout';
|
||||||
|
import styles from './Dropzone.module.css';
|
||||||
|
import { CloudLoadingIndicator } from '../Indicator';
|
||||||
|
|
||||||
|
export type DropzoneStylesNames = 'root' | 'inner';
|
||||||
|
export type DropzoneVariant = 'filled' | 'light';
|
||||||
|
export type DropzoneCssVariables = {
|
||||||
|
root:
|
||||||
|
| '--dropzone-radius'
|
||||||
|
| '--dropzone-accept-color'
|
||||||
|
| '--dropzone-accept-bg'
|
||||||
|
| '--dropzone-reject-color'
|
||||||
|
| '--dropzone-reject-bg';
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface DropzoneProps {
|
||||||
|
/** Key of `theme.colors` or any valid CSS color to set colors of `Dropzone.Accept`, `theme.primaryColor` by default */
|
||||||
|
acceptColor?: MantineColor;
|
||||||
|
|
||||||
|
/** Key of `theme.colors` or any valid CSS color to set colors of `Dropzone.Reject`, `'red'` by default */
|
||||||
|
rejectColor?: MantineColor;
|
||||||
|
|
||||||
|
/** Key of `theme.radius` or any valid CSS value to set `border-radius`, numbers are converted to rem, `theme.defaultRadius` by default */
|
||||||
|
radius?: MantineRadius;
|
||||||
|
|
||||||
|
/** Determines whether files capturing should be disabled, `false` by default */
|
||||||
|
disabled?: boolean;
|
||||||
|
|
||||||
|
/** Called when any files are dropped to the dropzone */
|
||||||
|
onDropAny?: (files: FileWithPath[], fileRejections: FileRejection[]) => void;
|
||||||
|
|
||||||
|
/** Called when valid files are dropped to the dropzone */
|
||||||
|
onDrop: (files: FileWithPath[]) => void;
|
||||||
|
|
||||||
|
/** Called when dropped files do not meet file restrictions */
|
||||||
|
onReject?: (fileRejections: FileRejection[]) => void;
|
||||||
|
|
||||||
|
/** Determines whether a loading overlay should be displayed over the dropzone, `false` by default */
|
||||||
|
loading?: boolean;
|
||||||
|
|
||||||
|
/** Mime types of the files that dropzone can accepts. By default, dropzone accepts all file types. */
|
||||||
|
accept?: Accept | string[];
|
||||||
|
|
||||||
|
/** A ref function which when called opens the file system file picker */
|
||||||
|
openRef?: React.ForwardedRef<() => void | undefined>;
|
||||||
|
|
||||||
|
/** Determines whether multiple files can be dropped to the dropzone or selected from file system picker, `true` by default */
|
||||||
|
multiple?: boolean;
|
||||||
|
|
||||||
|
/** Maximum file size in bytes */
|
||||||
|
maxSize?: number;
|
||||||
|
|
||||||
|
/** Name of the form control. Submitted with the form as part of a name/value pair. */
|
||||||
|
name?: string;
|
||||||
|
|
||||||
|
/** Maximum number of files that can be picked at once */
|
||||||
|
maxFiles?: number;
|
||||||
|
|
||||||
|
/** Set to autofocus the root element */
|
||||||
|
autoFocus?: boolean;
|
||||||
|
|
||||||
|
/** If `false`, disables click to open the native file selection dialog */
|
||||||
|
activateOnClick?: boolean;
|
||||||
|
|
||||||
|
/** If `false`, disables drag 'n' drop */
|
||||||
|
activateOnDrag?: boolean;
|
||||||
|
|
||||||
|
/** If `false`, disables Space/Enter to open the native file selection dialog. Note that it also stops tracking the focus state. */
|
||||||
|
activateOnKeyboard?: boolean;
|
||||||
|
|
||||||
|
/** If `false`, stops drag event propagation to parents */
|
||||||
|
dragEventsBubbling?: boolean;
|
||||||
|
|
||||||
|
/** Called when the `dragenter` event occurs */
|
||||||
|
onDragEnter?: (event: React.DragEvent<HTMLElement>) => void;
|
||||||
|
|
||||||
|
/** Called when the `dragleave` event occurs */
|
||||||
|
onDragLeave?: (event: React.DragEvent<HTMLElement>) => void;
|
||||||
|
|
||||||
|
/** Called when the `dragover` event occurs */
|
||||||
|
onDragOver?: (event: React.DragEvent<HTMLElement>) => void;
|
||||||
|
|
||||||
|
/** Called when user closes the file selection dialog with no selection */
|
||||||
|
onFileDialogCancel?: () => void;
|
||||||
|
|
||||||
|
/** Called when user opens the file selection dialog */
|
||||||
|
onFileDialogOpen?: () => void;
|
||||||
|
|
||||||
|
/** If `false`, allow dropped items to take over the current browser window */
|
||||||
|
preventDropOnDocument?: boolean;
|
||||||
|
|
||||||
|
/** Set to true to use the File System Access API to open the file picker instead of using an <input type="file"> click event, defaults to true */
|
||||||
|
useFsAccessApi?: boolean;
|
||||||
|
|
||||||
|
/** Use this to provide a custom file aggregator */
|
||||||
|
getFilesFromEvent?: (
|
||||||
|
event: DropEvent,
|
||||||
|
) => Promise<Array<File | DataTransferItem>>;
|
||||||
|
|
||||||
|
/** Custom validation function. It must return null if there's no errors. */
|
||||||
|
validator?: <T extends File>(file: T) => FileError | FileError[] | null;
|
||||||
|
|
||||||
|
/** Determines whether pointer events should be enabled on the inner element, `false` by default */
|
||||||
|
enablePointerEvents?: boolean;
|
||||||
|
|
||||||
|
/** Props passed down to the Loader component */
|
||||||
|
loaderProps?: LoaderProps;
|
||||||
|
|
||||||
|
/** Props passed down to the internal Input component */
|
||||||
|
inputProps?: React.InputHTMLAttributes<HTMLInputElement>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type DropzoneFactory = Factory<{
|
||||||
|
props: DropzoneProps;
|
||||||
|
ref: HTMLDivElement;
|
||||||
|
stylesNames: DropzoneStylesNames;
|
||||||
|
vars: DropzoneCssVariables;
|
||||||
|
staticComponents: {
|
||||||
|
Accept: typeof DropzoneAccept;
|
||||||
|
Idle: typeof DropzoneIdle;
|
||||||
|
Reject: typeof DropzoneReject;
|
||||||
|
};
|
||||||
|
}>;
|
||||||
|
|
||||||
|
const defaultProps: Partial<DropzoneProps> = {
|
||||||
|
loading: false,
|
||||||
|
multiple: true,
|
||||||
|
maxSize: Infinity,
|
||||||
|
autoFocus: false,
|
||||||
|
activateOnClick: true,
|
||||||
|
activateOnDrag: true,
|
||||||
|
dragEventsBubbling: true,
|
||||||
|
activateOnKeyboard: true,
|
||||||
|
useFsAccessApi: true,
|
||||||
|
variant: 'light',
|
||||||
|
rejectColor: 'red',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Dropzone = (_props: DropzoneProps) => {
|
||||||
|
const {
|
||||||
|
// classNames,
|
||||||
|
// className,
|
||||||
|
// style,
|
||||||
|
// styles,
|
||||||
|
// unstyled,
|
||||||
|
// vars,
|
||||||
|
radius,
|
||||||
|
disabled,
|
||||||
|
loading,
|
||||||
|
multiple,
|
||||||
|
maxSize,
|
||||||
|
accept,
|
||||||
|
children,
|
||||||
|
onDropAny,
|
||||||
|
onDrop,
|
||||||
|
onReject,
|
||||||
|
openRef,
|
||||||
|
name,
|
||||||
|
maxFiles,
|
||||||
|
autoFocus,
|
||||||
|
activateOnClick,
|
||||||
|
activateOnDrag,
|
||||||
|
dragEventsBubbling,
|
||||||
|
activateOnKeyboard,
|
||||||
|
onDragEnter,
|
||||||
|
onDragLeave,
|
||||||
|
onDragOver,
|
||||||
|
onFileDialogCancel,
|
||||||
|
onFileDialogOpen,
|
||||||
|
preventDropOnDocument,
|
||||||
|
useFsAccessApi,
|
||||||
|
getFilesFromEvent,
|
||||||
|
validator,
|
||||||
|
rejectColor,
|
||||||
|
acceptColor,
|
||||||
|
enablePointerEvents,
|
||||||
|
loaderProps,
|
||||||
|
inputProps,
|
||||||
|
// mod,
|
||||||
|
classNames,
|
||||||
|
...others
|
||||||
|
} = {
|
||||||
|
...defaultProps,
|
||||||
|
..._props,
|
||||||
|
};
|
||||||
|
|
||||||
|
const { getRootProps, getInputProps, isDragAccept, isDragReject, open } =
|
||||||
|
useDropzone({
|
||||||
|
onDrop: onDropAny,
|
||||||
|
onDropAccepted: onDrop,
|
||||||
|
onDropRejected: onReject,
|
||||||
|
disabled: disabled || loading,
|
||||||
|
accept: Array.isArray(accept)
|
||||||
|
? accept.reduce((r, key) => ({ ...r, [key]: [] }), {})
|
||||||
|
: accept,
|
||||||
|
multiple,
|
||||||
|
maxSize,
|
||||||
|
maxFiles,
|
||||||
|
autoFocus,
|
||||||
|
noClick: !activateOnClick,
|
||||||
|
noDrag: !activateOnDrag,
|
||||||
|
noDragEventsBubbling: !dragEventsBubbling,
|
||||||
|
noKeyboard: !activateOnKeyboard,
|
||||||
|
onDragEnter,
|
||||||
|
onDragLeave,
|
||||||
|
onDragOver,
|
||||||
|
onFileDialogCancel,
|
||||||
|
onFileDialogOpen,
|
||||||
|
preventDropOnDocument,
|
||||||
|
useFsAccessApi,
|
||||||
|
validator,
|
||||||
|
...(getFilesFromEvent ? { getFilesFromEvent } : null),
|
||||||
|
});
|
||||||
|
|
||||||
|
const isIdle = !isDragAccept && !isDragReject;
|
||||||
|
assignRef(openRef, open);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<DropzoneProvider
|
||||||
|
value={{ accept: isDragAccept, reject: isDragReject, idle: isIdle }}
|
||||||
|
>
|
||||||
|
<Box
|
||||||
|
{...getRootProps({
|
||||||
|
className: clsx(styles.root, classNames?.root),
|
||||||
|
})}
|
||||||
|
// {...getStyles('root', { focusable: true })}
|
||||||
|
{...others}
|
||||||
|
mod={[
|
||||||
|
{
|
||||||
|
accept: isDragAccept,
|
||||||
|
reject: isDragReject,
|
||||||
|
idle: isIdle,
|
||||||
|
loading,
|
||||||
|
'activate-on-click': activateOnClick,
|
||||||
|
},
|
||||||
|
// mod,
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<input {...getInputProps(inputProps)} name={name} />
|
||||||
|
<div
|
||||||
|
data-enable-pointer-events={enablePointerEvents || undefined}
|
||||||
|
className={classNames?.content}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</Box>
|
||||||
|
</DropzoneProvider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
Dropzone.displayName = '@mantine/dropzone/Dropzone';
|
||||||
|
Dropzone.Accept = DropzoneAccept;
|
||||||
|
Dropzone.Idle = DropzoneIdle;
|
||||||
|
Dropzone.Reject = DropzoneReject;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
type PossibleRef<T> = Ref<T> | undefined;
|
||||||
|
|
||||||
|
export function assignRef<T>(ref: PossibleRef<T>, value: T) {
|
||||||
|
if (typeof ref === 'function') {
|
||||||
|
ref(value);
|
||||||
|
} else if (typeof ref === 'object' && ref !== null && 'current' in ref) {
|
||||||
|
(ref as React.MutableRefObject<T>).current = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function mergeRefs<T>(...refs: PossibleRef<T>[]) {
|
||||||
|
return (node: T | null) => {
|
||||||
|
refs.forEach((ref) => assignRef(ref, node));
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useMergedRef<T>(...refs: PossibleRef<T>[]) {
|
||||||
|
return useCallback(mergeRefs(...refs), refs);
|
||||||
|
}
|
||||||
12
packages/webapp/src/components/Dropzone/DropzoneProvider.tsx
Normal file
12
packages/webapp/src/components/Dropzone/DropzoneProvider.tsx
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
import { createSafeContext } from './create-safe-context';
|
||||||
|
|
||||||
|
export interface DropzoneContextValue {
|
||||||
|
idle: boolean;
|
||||||
|
accept: boolean;
|
||||||
|
reject: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const [DropzoneProvider, useDropzoneContext] =
|
||||||
|
createSafeContext<DropzoneContextValue>(
|
||||||
|
'Dropzone component was not found in tree',
|
||||||
|
);
|
||||||
36
packages/webapp/src/components/Dropzone/DropzoneStatus.tsx
Normal file
36
packages/webapp/src/components/Dropzone/DropzoneStatus.tsx
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
import React, { cloneElement } from 'react';
|
||||||
|
import { upperFirst } from 'lodash';
|
||||||
|
import { DropzoneContextValue, useDropzoneContext } from './DropzoneProvider';
|
||||||
|
import { isElement } from '@/utils/is-element';
|
||||||
|
|
||||||
|
export interface DropzoneStatusProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
type DropzoneStatusComponent = React.FC<DropzoneStatusProps>;
|
||||||
|
|
||||||
|
function createDropzoneStatus(status: keyof DropzoneContextValue) {
|
||||||
|
const Component: DropzoneStatusComponent = (props) => {
|
||||||
|
const { children, ...others } = props;
|
||||||
|
|
||||||
|
const ctx = useDropzoneContext();
|
||||||
|
const _children = isElement(children) ? children : <span>{children}</span>;
|
||||||
|
|
||||||
|
if (ctx[status]) {
|
||||||
|
return cloneElement(_children as JSX.Element, others);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
Component.displayName = `@bigcapital/core/dropzone/${upperFirst(status)}`;
|
||||||
|
|
||||||
|
return Component;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const DropzoneAccept = createDropzoneStatus('accept');
|
||||||
|
export const DropzoneReject = createDropzoneStatus('reject');
|
||||||
|
export const DropzoneIdle = createDropzoneStatus('idle');
|
||||||
|
|
||||||
|
export type DropzoneAcceptProps = DropzoneStatusProps;
|
||||||
|
export type DropzoneRejectProps = DropzoneStatusProps;
|
||||||
|
export type DropzoneIdleProps = DropzoneStatusProps;
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
import React, { createContext, useContext } from 'react';
|
||||||
|
|
||||||
|
export function createSafeContext<ContextValue>(errorMessage: string) {
|
||||||
|
const Context = createContext<ContextValue | null>(null);
|
||||||
|
|
||||||
|
const useSafeContext = () => {
|
||||||
|
const ctx = useContext(Context);
|
||||||
|
|
||||||
|
if (ctx === null) {
|
||||||
|
throw new Error(errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ctx;
|
||||||
|
};
|
||||||
|
|
||||||
|
const Provider = ({
|
||||||
|
children,
|
||||||
|
value,
|
||||||
|
}: {
|
||||||
|
value: ContextValue;
|
||||||
|
children: React.ReactNode;
|
||||||
|
}) => <Context.Provider value={value}>{children}</Context.Provider>;
|
||||||
|
|
||||||
|
return [Provider, useSafeContext] as const;
|
||||||
|
}
|
||||||
1
packages/webapp/src/components/Dropzone/index.ts
Normal file
1
packages/webapp/src/components/Dropzone/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './Dropzone';
|
||||||
39
packages/webapp/src/components/Dropzone/mine-types.ts
Normal file
39
packages/webapp/src/components/Dropzone/mine-types.ts
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
export const MIME_TYPES = {
|
||||||
|
// Images
|
||||||
|
png: 'image/png',
|
||||||
|
gif: 'image/gif',
|
||||||
|
jpeg: 'image/jpeg',
|
||||||
|
svg: 'image/svg+xml',
|
||||||
|
webp: 'image/webp',
|
||||||
|
avif: 'image/avif',
|
||||||
|
heic: 'image/heic',
|
||||||
|
|
||||||
|
// Documents
|
||||||
|
mp4: 'video/mp4',
|
||||||
|
zip: 'application/zip',
|
||||||
|
csv: 'text/csv',
|
||||||
|
pdf: 'application/pdf',
|
||||||
|
doc: 'application/msword',
|
||||||
|
docx: 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
|
||||||
|
xls: 'application/vnd.ms-excel',
|
||||||
|
xlsx: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
|
||||||
|
ppt: 'application/vnd.ms-powerpoint',
|
||||||
|
pptx: 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
|
||||||
|
exe: 'application/vnd.microsoft.portable-executable',
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
export const IMAGE_MIME_TYPE = [
|
||||||
|
MIME_TYPES.png,
|
||||||
|
MIME_TYPES.gif,
|
||||||
|
MIME_TYPES.jpeg,
|
||||||
|
MIME_TYPES.svg,
|
||||||
|
MIME_TYPES.webp,
|
||||||
|
MIME_TYPES.avif,
|
||||||
|
MIME_TYPES.heic,
|
||||||
|
];
|
||||||
|
|
||||||
|
export const PDF_MIME_TYPE = [MIME_TYPES.pdf];
|
||||||
|
export const MS_WORD_MIME_TYPE = [MIME_TYPES.doc, MIME_TYPES.docx];
|
||||||
|
export const MS_EXCEL_MIME_TYPE = [MIME_TYPES.xls, MIME_TYPES.xlsx];
|
||||||
|
export const MS_POWERPOINT_MIME_TYPE = [MIME_TYPES.ppt, MIME_TYPES.pptx];
|
||||||
|
export const EXE_MIME_TYPE = [MIME_TYPES.exe];
|
||||||
@@ -1,14 +1,27 @@
|
|||||||
// @ts-nocheck
|
// @ts-nocheck
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Tooltip } from '@blueprintjs/core';
|
import { Position, Tooltip } from '@blueprintjs/core';
|
||||||
import { Icon } from '../Icon';
|
import { Icon } from '../Icon';
|
||||||
|
|
||||||
import '@/style/components/Hint.scss';
|
import '@/style/components/Hint.scss';
|
||||||
|
import { Tooltip2Props } from '@blueprintjs/popover2';
|
||||||
|
|
||||||
|
interface HintProps {
|
||||||
|
content: string;
|
||||||
|
position?: Position;
|
||||||
|
iconSize?: number;
|
||||||
|
tooltipProps?: Partial<Tooltip2Props>;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Field hint.
|
* Field hint.
|
||||||
*/
|
*/
|
||||||
export function FieldHint({ content, position, iconSize = 12, tooltipProps }) {
|
export function FieldHint({
|
||||||
|
content,
|
||||||
|
position,
|
||||||
|
iconSize = 12,
|
||||||
|
tooltipProps,
|
||||||
|
}: HintProps) {
|
||||||
return (
|
return (
|
||||||
<span class="hint">
|
<span class="hint">
|
||||||
<Tooltip content={content} position={position} {...tooltipProps}>
|
<Tooltip content={content} position={position} {...tooltipProps}>
|
||||||
|
|||||||
@@ -17,11 +17,20 @@
|
|||||||
|
|
||||||
import classNames from 'classnames';
|
import classNames from 'classnames';
|
||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
import { Classes } from '@blueprintjs/core';
|
import { Classes, Props } from '@blueprintjs/core';
|
||||||
import IconSvgPaths from '@/static/json/icons';
|
import IconSvgPaths from '@/static/json/icons';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
|
export interface IconProps extends Props {
|
||||||
|
color?: string;
|
||||||
|
htmlTitle?: string;
|
||||||
|
icon: IconName | MaybeElement;
|
||||||
|
iconSize?: number;
|
||||||
|
style?: object;
|
||||||
|
tagName?: keyof JSX.IntrinsicElements;
|
||||||
|
title?: string;
|
||||||
|
}
|
||||||
|
|
||||||
export class Icon extends React.Component {
|
export class Icon extends React.Component<IconProps> {
|
||||||
static displayName = `af.Icon`;
|
static displayName = `af.Icon`;
|
||||||
|
|
||||||
static SIZE_STANDARD = 16;
|
static SIZE_STANDARD = 16;
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import { HTMLDivProps, Props } from '@blueprintjs/core';
|
||||||
|
|
||||||
export interface BoxProps {
|
export interface BoxProps extends Props, HTMLDivProps {
|
||||||
className?: string;
|
className?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ export function Stack(props: StackProps) {
|
|||||||
const StackStyled = styled(Box)`
|
const StackStyled = styled(Box)`
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: align;
|
align-items: ${(props: StackProps) => props.align};
|
||||||
justify-content: justify;
|
justify-content: justify;
|
||||||
gap: ${(props: StackProps) => props.spacing}px;
|
gap: ${(props: StackProps) => props.spacing}px;
|
||||||
`;
|
`;
|
||||||
|
|||||||
248
packages/webapp/src/components/Section/Section.tsx
Normal file
248
packages/webapp/src/components/Section/Section.tsx
Normal file
@@ -0,0 +1,248 @@
|
|||||||
|
import classNames from 'classnames';
|
||||||
|
import React from 'react';
|
||||||
|
import {
|
||||||
|
Card,
|
||||||
|
Collapse,
|
||||||
|
type CollapseProps,
|
||||||
|
Elevation,
|
||||||
|
Utils,
|
||||||
|
DISPLAYNAME_PREFIX,
|
||||||
|
type HTMLDivProps,
|
||||||
|
type MaybeElement,
|
||||||
|
type Props,
|
||||||
|
IconName,
|
||||||
|
} from '@blueprintjs/core';
|
||||||
|
import { H6 } from '@blueprintjs/core';
|
||||||
|
import { CLASSES } from '@/constants';
|
||||||
|
import { Icon } from '../Icon';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Subset of {@link Elevation} options which are visually supported by the {@link Section} component.
|
||||||
|
*
|
||||||
|
* Note that an elevation greater than 1 creates too much visual clutter/noise in the UI, especially when
|
||||||
|
* multiple Sections are shown on a single page.
|
||||||
|
*/
|
||||||
|
export type SectionElevation = typeof Elevation.ZERO | typeof Elevation.ONE;
|
||||||
|
|
||||||
|
export interface SectionCollapseProps
|
||||||
|
extends Pick<
|
||||||
|
CollapseProps,
|
||||||
|
'className' | 'isOpen' | 'keepChildrenMounted' | 'transitionDuration'
|
||||||
|
> {
|
||||||
|
/**
|
||||||
|
* Whether the component is initially open or closed.
|
||||||
|
*
|
||||||
|
* This prop has no effect if `collapsible={false}` or the component is in controlled mode,
|
||||||
|
* i.e. when `isOpen` is **not** `undefined`.
|
||||||
|
*
|
||||||
|
* @default true
|
||||||
|
*/
|
||||||
|
defaultIsOpen?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether the component is open or closed.
|
||||||
|
*
|
||||||
|
* Passing a boolean value to `isOpen` will enabled controlled mode for the component.
|
||||||
|
*/
|
||||||
|
isOpen?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callback invoked in controlled mode when the collapse toggle element is clicked.
|
||||||
|
*/
|
||||||
|
onToggle?: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SectionProps extends Props, Omit<HTMLDivProps, 'title'> {
|
||||||
|
/**
|
||||||
|
* Whether this section's contents should be collapsible.
|
||||||
|
*
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
|
collapsible?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Subset of props to forward to the underlying {@link Collapse} component, with the addition of a
|
||||||
|
* `defaultIsOpen` option which sets the default open state of the component when in uncontrolled mode.
|
||||||
|
*/
|
||||||
|
collapseProps?: SectionCollapseProps;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether this section should use compact styles.
|
||||||
|
*
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
|
compact?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Visual elevation of this container element.
|
||||||
|
*
|
||||||
|
* @default Elevation.ZERO
|
||||||
|
*/
|
||||||
|
elevation?: SectionElevation;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Name of a Blueprint UI icon (or an icon element) to render in the section's header.
|
||||||
|
* Note that the header will only be rendered if `title` is provided.
|
||||||
|
*/
|
||||||
|
icon?: IconName | MaybeElement;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Element to render on the right side of the section header.
|
||||||
|
* Note that the header will only be rendered if `title` is provided.
|
||||||
|
*/
|
||||||
|
rightElement?: JSX.Element;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sub-title of the section.
|
||||||
|
* Note that the header will only be rendered if `title` is provided.
|
||||||
|
*/
|
||||||
|
subtitle?: JSX.Element | string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Title of the section.
|
||||||
|
* Note that the header will only be rendered if `title` is provided.
|
||||||
|
*/
|
||||||
|
title?: JSX.Element | string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optional title renderer function. If provided, it is recommended to include a Blueprint `<H6>` element
|
||||||
|
* as part of the title. The render function is supplied with `className` and `id` attributes which you must
|
||||||
|
* forward to the DOM. The `title` prop is also passed along to this renderer via `props.children`.
|
||||||
|
*
|
||||||
|
* @default H6
|
||||||
|
*/
|
||||||
|
titleRenderer?: React.FC<React.HTMLAttributes<HTMLElement>>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Section component.
|
||||||
|
*
|
||||||
|
* @see https://blueprintjs.com/docs/#core/components/section
|
||||||
|
*/
|
||||||
|
export const Section: React.FC<SectionProps> = React.forwardRef(
|
||||||
|
(props, ref) => {
|
||||||
|
const {
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
collapseProps,
|
||||||
|
collapsible,
|
||||||
|
compact,
|
||||||
|
elevation,
|
||||||
|
icon,
|
||||||
|
rightElement,
|
||||||
|
subtitle,
|
||||||
|
title,
|
||||||
|
titleRenderer = H6,
|
||||||
|
...htmlProps
|
||||||
|
} = props;
|
||||||
|
// Determine whether to use controlled or uncontrolled state.
|
||||||
|
const isControlled = collapseProps?.isOpen != null;
|
||||||
|
|
||||||
|
// The initial useState value is negated in order to conform to the `isCollapsed` expectation.
|
||||||
|
const [isCollapsedUncontrolled, setIsCollapsed] = React.useState<boolean>(
|
||||||
|
!(collapseProps?.defaultIsOpen ?? true),
|
||||||
|
);
|
||||||
|
|
||||||
|
const isCollapsed = isControlled
|
||||||
|
? !collapseProps?.isOpen
|
||||||
|
: isCollapsedUncontrolled;
|
||||||
|
|
||||||
|
const toggleIsCollapsed = React.useCallback(() => {
|
||||||
|
if (isControlled) {
|
||||||
|
collapseProps?.onToggle?.();
|
||||||
|
} else {
|
||||||
|
setIsCollapsed(!isCollapsed);
|
||||||
|
}
|
||||||
|
}, [collapseProps, isCollapsed, isControlled]);
|
||||||
|
|
||||||
|
const isHeaderRightContainerVisible = rightElement != null || collapsible;
|
||||||
|
|
||||||
|
const sectionId = Utils.uniqueId('section');
|
||||||
|
const sectionTitleId = title ? Utils.uniqueId('section-title') : undefined;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card
|
||||||
|
className={classNames(className, CLASSES.SECTION, {
|
||||||
|
[CLASSES.COMPACT]: compact,
|
||||||
|
[CLASSES.SECTION_COLLAPSED]:
|
||||||
|
(collapsible && isCollapsed) || Utils.isReactNodeEmpty(children),
|
||||||
|
})}
|
||||||
|
elevation={elevation}
|
||||||
|
aria-labelledby={sectionTitleId}
|
||||||
|
{...htmlProps}
|
||||||
|
id={sectionId}
|
||||||
|
>
|
||||||
|
{title && (
|
||||||
|
<div
|
||||||
|
role={collapsible ? 'button' : undefined}
|
||||||
|
aria-pressed={collapsible ? isCollapsed : undefined}
|
||||||
|
aria-expanded={collapsible ? isCollapsed : undefined}
|
||||||
|
aria-controls={collapsible ? sectionId : undefined}
|
||||||
|
className={classNames(CLASSES.SECTION_HEADER, {
|
||||||
|
[CLASSES.INTERACTIVE]: collapsible,
|
||||||
|
})}
|
||||||
|
onClick={collapsible ? toggleIsCollapsed : undefined}
|
||||||
|
>
|
||||||
|
<div className={CLASSES.SECTION_HEADER_LEFT}>
|
||||||
|
{/* {icon && (
|
||||||
|
<Icon
|
||||||
|
icon={icon}
|
||||||
|
aria-hidden={true}
|
||||||
|
tabIndex={-1}
|
||||||
|
className={CLASSES.TEXT_MUTED}
|
||||||
|
/>
|
||||||
|
)} */}
|
||||||
|
<div>
|
||||||
|
{React.createElement(
|
||||||
|
titleRenderer,
|
||||||
|
{
|
||||||
|
className: CLASSES.SECTION_HEADER_TITLE,
|
||||||
|
id: sectionTitleId,
|
||||||
|
},
|
||||||
|
title,
|
||||||
|
)}
|
||||||
|
{subtitle && (
|
||||||
|
<div
|
||||||
|
className={classNames(
|
||||||
|
CLASSES.TEXT_MUTED,
|
||||||
|
CLASSES.SECTION_HEADER_SUB_TITLE,
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{subtitle}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{isHeaderRightContainerVisible && (
|
||||||
|
<div className={CLASSES.SECTION_HEADER_RIGHT}>
|
||||||
|
{rightElement}
|
||||||
|
{collapsible &&
|
||||||
|
(isCollapsed ? (
|
||||||
|
<Icon
|
||||||
|
icon={'chevron-down'}
|
||||||
|
className={CLASSES.TEXT_MUTED}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Icon icon={'chevron-up'} className={CLASSES.TEXT_MUTED} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{collapsible ? (
|
||||||
|
// @ts-ignore
|
||||||
|
<Collapse isOpen={!isCollapsed}>
|
||||||
|
{children}
|
||||||
|
</Collapse>
|
||||||
|
) : (
|
||||||
|
children
|
||||||
|
)}
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
Section.defaultProps = {
|
||||||
|
compact: false,
|
||||||
|
elevation: Elevation.ZERO,
|
||||||
|
};
|
||||||
|
Section.displayName = `${DISPLAYNAME_PREFIX}.Section`;
|
||||||
41
packages/webapp/src/components/Section/SectionCard.tsx
Normal file
41
packages/webapp/src/components/Section/SectionCard.tsx
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
import classNames from 'classnames';
|
||||||
|
import * as React from 'react';
|
||||||
|
import { DISPLAYNAME_PREFIX, HTMLDivProps, Props } from '@blueprintjs/core';
|
||||||
|
import { CLASSES } from '@/constants';
|
||||||
|
|
||||||
|
export interface SectionCardProps
|
||||||
|
extends Props,
|
||||||
|
HTMLDivProps,
|
||||||
|
React.RefAttributes<HTMLDivElement> {
|
||||||
|
/**
|
||||||
|
* Whether to apply visual padding inside the content container element.
|
||||||
|
*
|
||||||
|
* @default true
|
||||||
|
*/
|
||||||
|
padded?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Section card component.
|
||||||
|
*
|
||||||
|
* @see https://blueprintjs.com/docs/#core/components/section.section-card
|
||||||
|
*/
|
||||||
|
export const SectionCard: React.FC<SectionCardProps> = React.forwardRef(
|
||||||
|
(props, ref) => {
|
||||||
|
const { className, children, padded, ...htmlProps } = props;
|
||||||
|
const classes = classNames(
|
||||||
|
CLASSES.SECTION_CARD,
|
||||||
|
{ [CLASSES.PADDED]: padded },
|
||||||
|
className,
|
||||||
|
);
|
||||||
|
return (
|
||||||
|
<div className={classes} ref={ref} {...htmlProps}>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
SectionCard.defaultProps = {
|
||||||
|
padded: true,
|
||||||
|
};
|
||||||
|
SectionCard.displayName = `${DISPLAYNAME_PREFIX}.SectionCard`;
|
||||||
2
packages/webapp/src/components/Section/index.ts
Normal file
2
packages/webapp/src/components/Section/index.ts
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
export * from './Section';
|
||||||
|
export * from './SectionCard';
|
||||||
111
packages/webapp/src/components/Stepper/Stepper.tsx
Normal file
111
packages/webapp/src/components/Stepper/Stepper.tsx
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { cloneElement } from 'react';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
import { toArray } from 'lodash';
|
||||||
|
import { Box } from '../Layout';
|
||||||
|
import { StepperCompleted } from './StepperCompleted';
|
||||||
|
import { StepperStep } from './StepperStep';
|
||||||
|
import { StepperStepState } from './types';
|
||||||
|
|
||||||
|
export interface StepperProps {
|
||||||
|
/** <Stepper.Step /> components */
|
||||||
|
children: React.ReactNode;
|
||||||
|
|
||||||
|
/** Index of the active step */
|
||||||
|
active: number;
|
||||||
|
|
||||||
|
/** Called when step is clicked */
|
||||||
|
onStepClick?: (stepIndex: number) => void;
|
||||||
|
|
||||||
|
/** Determines whether next steps can be selected, `true` by default **/
|
||||||
|
allowNextStepsSelect?: boolean;
|
||||||
|
|
||||||
|
classNames?: Record<string, string>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Stepper({
|
||||||
|
active,
|
||||||
|
onStepClick,
|
||||||
|
children,
|
||||||
|
classNames,
|
||||||
|
}: StepperProps) {
|
||||||
|
const convertedChildren = toArray(children) as React.ReactElement[];
|
||||||
|
const _children = convertedChildren.filter(
|
||||||
|
(child) => child.type !== StepperCompleted,
|
||||||
|
);
|
||||||
|
const completedStep = convertedChildren.find(
|
||||||
|
(item) => item.type === StepperCompleted,
|
||||||
|
);
|
||||||
|
const items = _children.reduce<React.ReactElement[]>((acc, item, index) => {
|
||||||
|
const state =
|
||||||
|
active === index
|
||||||
|
? StepperStepState.Progress
|
||||||
|
: active > index
|
||||||
|
? StepperStepState.Completed
|
||||||
|
: StepperStepState.Inactive;
|
||||||
|
|
||||||
|
const shouldAllowSelect = () => {
|
||||||
|
if (typeof onStepClick !== 'function') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (typeof item.props.allowStepSelect === 'boolean') {
|
||||||
|
return item.props.allowStepSelect;
|
||||||
|
}
|
||||||
|
return state === 'stepCompleted' || allowNextStepsSelect;
|
||||||
|
};
|
||||||
|
const isStepSelectionEnabled = shouldAllowSelect();
|
||||||
|
|
||||||
|
acc.push(
|
||||||
|
cloneElement(item, {
|
||||||
|
key: index,
|
||||||
|
step: index + 1,
|
||||||
|
state,
|
||||||
|
onClick: () => isStepSelectionEnabled && onStepClick?.(index),
|
||||||
|
allowStepClick: isStepSelectionEnabled,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
if (index !== _children.length - 1) {
|
||||||
|
acc.push(
|
||||||
|
<StepSeparator
|
||||||
|
data-active={index < active || undefined}
|
||||||
|
key={`separator-${index}`}
|
||||||
|
/>,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return acc;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const stepContent = _children[active]?.props?.children;
|
||||||
|
const completedContent = completedStep?.props?.children;
|
||||||
|
const content =
|
||||||
|
active > _children.length - 1 ? completedContent : stepContent;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box className={classNames?.root}>
|
||||||
|
<StepsItems className={classNames?.items}>{items}</StepsItems>
|
||||||
|
<StepsContent className={classNames?.content}>{content} </StepsContent>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Stepper.Step = StepperStep;
|
||||||
|
Stepper.Completed = StepperCompleted;
|
||||||
|
Stepper.displayName = '@bigcapital/core/stepper';
|
||||||
|
|
||||||
|
const StepsItems = styled(Box)`
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
align-items: center;
|
||||||
|
gap: 16px;
|
||||||
|
`;
|
||||||
|
const StepsContent = styled(Box)`
|
||||||
|
margin-top: 16px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
`;
|
||||||
|
const StepSeparator = styled.div`
|
||||||
|
flex: 1;
|
||||||
|
display: block;
|
||||||
|
border-color: #c5cbd3;
|
||||||
|
border-top-style: solid;
|
||||||
|
border-top-width: 1px;
|
||||||
|
`;
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
export interface StepperCompletedProps {
|
||||||
|
/** Label content */
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const StepperCompleted: React.FC<StepperCompletedProps> = () => null;
|
||||||
|
StepperCompleted.displayName = '@bigcapital/core/StepperCompleted';
|
||||||
102
packages/webapp/src/components/Stepper/StepperStep.tsx
Normal file
102
packages/webapp/src/components/Stepper/StepperStep.tsx
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { StepperStepState } from './types';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
import { Icon } from '../Icon';
|
||||||
|
|
||||||
|
interface StepperStepProps {
|
||||||
|
label: string;
|
||||||
|
description?: string;
|
||||||
|
children: React.ReactNode;
|
||||||
|
step?: number;
|
||||||
|
active?: boolean;
|
||||||
|
state?: StepperStepState;
|
||||||
|
allowStepClick?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function StepperStep({
|
||||||
|
label,
|
||||||
|
description,
|
||||||
|
step,
|
||||||
|
active,
|
||||||
|
state,
|
||||||
|
children,
|
||||||
|
}: StepperStepProps) {
|
||||||
|
return (
|
||||||
|
<StepButton>
|
||||||
|
<StepIconWrap>
|
||||||
|
<StepIcon
|
||||||
|
isCompleted={state === StepperStepState.Completed}
|
||||||
|
isActive={state === StepperStepState.Progress}
|
||||||
|
>
|
||||||
|
{state === StepperStepState.Completed && (
|
||||||
|
<Icon icon={'done'} iconSize={24} />
|
||||||
|
)}
|
||||||
|
<StepIconText>{step}</StepIconText>
|
||||||
|
</StepIcon>
|
||||||
|
</StepIconWrap>
|
||||||
|
|
||||||
|
<StepTextWrap>
|
||||||
|
<StepTitle
|
||||||
|
isCompleted={state === StepperStepState.Completed}
|
||||||
|
isActive={state === StepperStepState.Progress}
|
||||||
|
>
|
||||||
|
{label}
|
||||||
|
</StepTitle>
|
||||||
|
{description && (
|
||||||
|
<StepDescription
|
||||||
|
isCompleted={state === StepperStepState.Completed}
|
||||||
|
isActive={state === StepperStepState.Progress}
|
||||||
|
>
|
||||||
|
{description}
|
||||||
|
</StepDescription>
|
||||||
|
)}
|
||||||
|
</StepTextWrap>
|
||||||
|
</StepButton>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const StepButton = styled.button`
|
||||||
|
background: transparent;
|
||||||
|
color: inherit;
|
||||||
|
border: 0;
|
||||||
|
align-items: center;
|
||||||
|
display: flex;
|
||||||
|
gap: 10px;
|
||||||
|
text-align: left;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const StepIcon = styled.span`
|
||||||
|
display: block;
|
||||||
|
height: 24px;
|
||||||
|
width: 24px;
|
||||||
|
display: block;
|
||||||
|
line-height: 24px;
|
||||||
|
border-radius: 24px;
|
||||||
|
text-align: center;
|
||||||
|
background-color: ${(props) =>
|
||||||
|
props.isCompleted || props.isActive ? 'rgb(0, 82, 204)' : '#9e9e9e'};
|
||||||
|
color: #fff;
|
||||||
|
margin: auto;
|
||||||
|
font-size: 12px;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const StepTitle = styled.div`
|
||||||
|
color: ${(props) =>
|
||||||
|
props.isCompleted || props.isActive ? 'rgb(0, 82, 204)' : '#738091'};
|
||||||
|
`;
|
||||||
|
const StepDescription = styled.div`
|
||||||
|
font-size: 12px;
|
||||||
|
margin-top: 10px;
|
||||||
|
color: ${(props) =>
|
||||||
|
props.isCompleted || props.isActive ? 'rgb(0, 82, 204)' : '#738091'};
|
||||||
|
`;
|
||||||
|
|
||||||
|
const StepIconWrap = styled.div`
|
||||||
|
display: flex;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const StepTextWrap = styled.div`
|
||||||
|
text-align: left;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const StepIconText = styled.div``;
|
||||||
1
packages/webapp/src/components/Stepper/index.ts
Normal file
1
packages/webapp/src/components/Stepper/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './Stepper';
|
||||||
7
packages/webapp/src/components/Stepper/types.ts
Normal file
7
packages/webapp/src/components/Stepper/types.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
|
||||||
|
|
||||||
|
export enum StepperStepState {
|
||||||
|
Progress = 'stepProgress',
|
||||||
|
Completed = 'stepCompleted',
|
||||||
|
Inactive = 'stepInactive',
|
||||||
|
}
|
||||||
@@ -1,6 +1,21 @@
|
|||||||
// @ts-nocheck
|
// @ts-nocheck
|
||||||
import { Classes } from '@blueprintjs/core';
|
import { Classes } from '@blueprintjs/core';
|
||||||
|
|
||||||
|
export const NS = 'bp4';
|
||||||
|
|
||||||
|
export const SECTION = `${NS}-section`;
|
||||||
|
export const SECTION_COLLAPSED = `${SECTION}-collapsed`;
|
||||||
|
export const SECTION_HEADER = `${SECTION}-header`;
|
||||||
|
export const SECTION_HEADER_LEFT = `${SECTION_HEADER}-left`;
|
||||||
|
export const SECTION_HEADER_TITLE = `${SECTION_HEADER}-title`;
|
||||||
|
export const SECTION_HEADER_SUB_TITLE = `${SECTION_HEADER}-sub-title`;
|
||||||
|
export const SECTION_HEADER_DIVIDER = `${SECTION_HEADER}-divider`;
|
||||||
|
export const SECTION_HEADER_TABS = `${SECTION_HEADER}-tabs`;
|
||||||
|
export const SECTION_HEADER_RIGHT = `${SECTION_HEADER}-right`;
|
||||||
|
export const SECTION_CARD = `${SECTION}-card`;
|
||||||
|
|
||||||
|
export const PADDED = `${NS}-padded`;
|
||||||
|
|
||||||
const CLASSES = {
|
const CLASSES = {
|
||||||
DASHBOARD_PAGE: 'dashboard__page',
|
DASHBOARD_PAGE: 'dashboard__page',
|
||||||
DASHBOARD_DATATABLE: 'dashboard__datatable',
|
DASHBOARD_DATATABLE: 'dashboard__datatable',
|
||||||
@@ -16,7 +31,7 @@ const CLASSES = {
|
|||||||
DASHBOARD_CONTENT_PREFERENCES: 'dashboard-content--preferences',
|
DASHBOARD_CONTENT_PREFERENCES: 'dashboard-content--preferences',
|
||||||
DASHBOARD_CONTENT_PANE: 'Pane2',
|
DASHBOARD_CONTENT_PANE: 'Pane2',
|
||||||
DASHBOARD_CENTERED_EMPTY_STATUS: 'dashboard__centered-empty-status',
|
DASHBOARD_CENTERED_EMPTY_STATUS: 'dashboard__centered-empty-status',
|
||||||
|
|
||||||
PAGE_FORM: 'page-form',
|
PAGE_FORM: 'page-form',
|
||||||
PAGE_FORM_HEADER: 'page-form__header',
|
PAGE_FORM_HEADER: 'page-form__header',
|
||||||
PAGE_FORM_HEADER_PRIMARY: 'page-form__primary-section',
|
PAGE_FORM_HEADER_PRIMARY: 'page-form__primary-section',
|
||||||
@@ -40,9 +55,9 @@ const CLASSES = {
|
|||||||
PAGE_FORM_ITEM: 'page-form--item',
|
PAGE_FORM_ITEM: 'page-form--item',
|
||||||
PAGE_FORM_MAKE_JOURNAL: 'page-form--make-journal-entries',
|
PAGE_FORM_MAKE_JOURNAL: 'page-form--make-journal-entries',
|
||||||
PAGE_FORM_EXPENSE: 'page-form--expense',
|
PAGE_FORM_EXPENSE: 'page-form--expense',
|
||||||
PAGE_FORM_CREDIT_NOTE:'page-form--credit-note',
|
PAGE_FORM_CREDIT_NOTE: 'page-form--credit-note',
|
||||||
PAGE_FORM_VENDOR_CREDIT_NOTE:'page-form--vendor-credit-note',
|
PAGE_FORM_VENDOR_CREDIT_NOTE: 'page-form--vendor-credit-note',
|
||||||
PAGE_FORM_WAREHOUSE_TRANSFER:'page-form--warehouse-transfer',
|
PAGE_FORM_WAREHOUSE_TRANSFER: 'page-form--warehouse-transfer',
|
||||||
|
|
||||||
FORM_GROUP_LIST_SELECT: 'form-group--select-list',
|
FORM_GROUP_LIST_SELECT: 'form-group--select-list',
|
||||||
|
|
||||||
@@ -66,31 +81,42 @@ const CLASSES = {
|
|||||||
PREFERENCES_TOPBAR: 'preferences-topbar',
|
PREFERENCES_TOPBAR: 'preferences-topbar',
|
||||||
|
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT: 'preferences-page__inside-content',
|
PREFERENCES_PAGE_INSIDE_CONTENT: 'preferences-page__inside-content',
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_GENERAL: 'preferences-page__inside-content--general',
|
PREFERENCES_PAGE_INSIDE_CONTENT_GENERAL:
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_USERS: 'preferences-page__inside-content--users',
|
'preferences-page__inside-content--general',
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_CURRENCIES: 'preferences-page__inside-content--currencies',
|
PREFERENCES_PAGE_INSIDE_CONTENT_USERS:
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_ACCOUNTANT: 'preferences-page__inside-content--accountant',
|
'preferences-page__inside-content--users',
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_SMS_INTEGRATION: 'preferences-page__inside-content--sms-integration',
|
PREFERENCES_PAGE_INSIDE_CONTENT_CURRENCIES:
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_ROLES_FORM: 'preferences-page__inside-content--roles-form',
|
'preferences-page__inside-content--currencies',
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_BRANCHES: 'preferences-page__inside-content--branches',
|
PREFERENCES_PAGE_INSIDE_CONTENT_ACCOUNTANT:
|
||||||
PREFERENCES_PAGE_INSIDE_CONTENT_WAREHOUSES: 'preferences-page__inside-content--warehouses',
|
'preferences-page__inside-content--accountant',
|
||||||
|
PREFERENCES_PAGE_INSIDE_CONTENT_SMS_INTEGRATION:
|
||||||
|
'preferences-page__inside-content--sms-integration',
|
||||||
|
PREFERENCES_PAGE_INSIDE_CONTENT_ROLES_FORM:
|
||||||
|
'preferences-page__inside-content--roles-form',
|
||||||
|
PREFERENCES_PAGE_INSIDE_CONTENT_BRANCHES:
|
||||||
|
'preferences-page__inside-content--branches',
|
||||||
|
PREFERENCES_PAGE_INSIDE_CONTENT_WAREHOUSES:
|
||||||
|
'preferences-page__inside-content--warehouses',
|
||||||
|
|
||||||
FINANCIAL_REPORT_INSIDER: 'dashboard__insider--financial-report',
|
FINANCIAL_REPORT_INSIDER: 'dashboard__insider--financial-report',
|
||||||
|
|
||||||
|
|
||||||
UNIVERSAL_SEARCH: 'universal-search',
|
UNIVERSAL_SEARCH: 'universal-search',
|
||||||
UNIVERSAL_SEARCH_OMNIBAR: 'universal-search__omnibar',
|
UNIVERSAL_SEARCH_OMNIBAR: 'universal-search__omnibar',
|
||||||
UNIVERSAL_SEARCH_OVERLAY: 'universal-search-overlay',
|
UNIVERSAL_SEARCH_OVERLAY: 'universal-search-overlay',
|
||||||
UNIVERSAL_SEARCH_INPUT: 'universal-search__input',
|
UNIVERSAL_SEARCH_INPUT: 'universal-search__input',
|
||||||
UNIVERSAL_SEARCH_INPUT_RIGHT_ELEMENTS: 'universal-search-input-right-elements',
|
UNIVERSAL_SEARCH_INPUT_RIGHT_ELEMENTS:
|
||||||
|
'universal-search-input-right-elements',
|
||||||
UNIVERSAL_SEARCH_TYPE_SELECT_OVERLAY: 'universal-search__type-select-overlay',
|
UNIVERSAL_SEARCH_TYPE_SELECT_OVERLAY: 'universal-search__type-select-overlay',
|
||||||
UNIVERSAL_SEARCH_TYPE_SELECT_BTN: 'universal-search__type-select-btn',
|
UNIVERSAL_SEARCH_TYPE_SELECT_BTN: 'universal-search__type-select-btn',
|
||||||
UNIVERSAL_SEARCH_FOOTER: 'universal-search__footer',
|
UNIVERSAL_SEARCH_FOOTER: 'universal-search__footer',
|
||||||
|
|
||||||
UNIVERSAL_SEARCH_ACTIONS: 'universal-search__actions',
|
UNIVERSAL_SEARCH_ACTIONS: 'universal-search__actions',
|
||||||
UNIVERSAL_SEARCH_ACTION_SELECT: 'universal-search__action universal-search__action--select',
|
UNIVERSAL_SEARCH_ACTION_SELECT:
|
||||||
UNIVERSAL_SEARCH_ACTION_CLOSE: 'universal-search__action universal-search__action--close',
|
'universal-search__action universal-search__action--select',
|
||||||
UNIVERSAL_SEARCH_ACTION_ARROWS: 'universal-search__action universal-search__action--arrows',
|
UNIVERSAL_SEARCH_ACTION_CLOSE:
|
||||||
|
'universal-search__action universal-search__action--close',
|
||||||
|
UNIVERSAL_SEARCH_ACTION_ARROWS:
|
||||||
|
'universal-search__action universal-search__action--arrows',
|
||||||
|
|
||||||
DIALOG_PDF_PREVIEW: 'dialog--pdf-preview-dialog',
|
DIALOG_PDF_PREVIEW: 'dialog--pdf-preview-dialog',
|
||||||
|
|
||||||
@@ -98,8 +124,19 @@ const CLASSES = {
|
|||||||
CARD: 'card',
|
CARD: 'card',
|
||||||
ALIGN_RIGHT: 'align-right',
|
ALIGN_RIGHT: 'align-right',
|
||||||
FONT_BOLD: 'font-bold',
|
FONT_BOLD: 'font-bold',
|
||||||
|
|
||||||
|
NS,
|
||||||
|
PADDED,
|
||||||
|
SECTION,
|
||||||
|
SECTION_COLLAPSED,
|
||||||
|
SECTION_HEADER,
|
||||||
|
SECTION_HEADER_LEFT,
|
||||||
|
SECTION_HEADER_TITLE,
|
||||||
|
SECTION_HEADER_SUB_TITLE,
|
||||||
|
SECTION_HEADER_DIVIDER,
|
||||||
|
SECTION_HEADER_TABS,
|
||||||
|
SECTION_HEADER_RIGHT,
|
||||||
|
SECTION_CARD,
|
||||||
};
|
};
|
||||||
|
|
||||||
export {
|
export { CLASSES };
|
||||||
CLASSES,
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ import {
|
|||||||
DashboardActionViewsList,
|
DashboardActionViewsList,
|
||||||
DashboardFilterButton,
|
DashboardFilterButton,
|
||||||
DashboardRowsHeightButton,
|
DashboardRowsHeightButton,
|
||||||
DashboardActionsBar
|
DashboardActionsBar,
|
||||||
} from '@/components';
|
} from '@/components';
|
||||||
|
|
||||||
import { AccountAction, AbilitySubject } from '@/constants/abilityOption';
|
import { AccountAction, AbilitySubject } from '@/constants/abilityOption';
|
||||||
@@ -37,6 +37,7 @@ import withSettings from '@/containers/Settings/withSettings';
|
|||||||
import withSettingsActions from '@/containers/Settings/withSettingsActions';
|
import withSettingsActions from '@/containers/Settings/withSettingsActions';
|
||||||
|
|
||||||
import { compose } from '@/utils';
|
import { compose } from '@/utils';
|
||||||
|
import { useHistory } from 'react-router-dom';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Accounts actions bar.
|
* Accounts actions bar.
|
||||||
@@ -67,6 +68,8 @@ function AccountsActionsBar({
|
|||||||
}) {
|
}) {
|
||||||
const { resourceViews, fields } = useAccountsChartContext();
|
const { resourceViews, fields } = useAccountsChartContext();
|
||||||
|
|
||||||
|
const history = useHistory();
|
||||||
|
|
||||||
const onClickNewAccount = () => {
|
const onClickNewAccount = () => {
|
||||||
openDialog(DialogsName.AccountForm, {});
|
openDialog(DialogsName.AccountForm, {});
|
||||||
};
|
};
|
||||||
@@ -111,6 +114,11 @@ function AccountsActionsBar({
|
|||||||
const handleTableRowSizeChange = (size) => {
|
const handleTableRowSizeChange = (size) => {
|
||||||
addSetting('accounts', 'tableSize', size);
|
addSetting('accounts', 'tableSize', size);
|
||||||
};
|
};
|
||||||
|
// handle the import button click.
|
||||||
|
const handleImportBtnClick = () => {
|
||||||
|
history.push('/accounts/import');
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<DashboardActionsBar>
|
<DashboardActionsBar>
|
||||||
<NavbarGroup>
|
<NavbarGroup>
|
||||||
@@ -183,6 +191,7 @@ function AccountsActionsBar({
|
|||||||
className={Classes.MINIMAL}
|
className={Classes.MINIMAL}
|
||||||
icon={<Icon icon="file-import-16" iconSize={16} />}
|
icon={<Icon icon="file-import-16" iconSize={16} />}
|
||||||
text={<T id={'import'} />}
|
text={<T id={'import'} />}
|
||||||
|
onClick={handleImportBtnClick}
|
||||||
/>
|
/>
|
||||||
<NavbarDivider />
|
<NavbarDivider />
|
||||||
<DashboardRowsHeightButton
|
<DashboardRowsHeightButton
|
||||||
|
|||||||
25
packages/webapp/src/containers/Accounts/AccountsImport.tsx
Normal file
25
packages/webapp/src/containers/Accounts/AccountsImport.tsx
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { DashboardInsider } from '@/components';
|
||||||
|
import { ImportView } from '../Import/ImportView';
|
||||||
|
import { useHistory } from 'react-router-dom';
|
||||||
|
|
||||||
|
export default function AccountsImport() {
|
||||||
|
const history = useHistory();
|
||||||
|
|
||||||
|
const handleCancelBtnClick = () => {
|
||||||
|
history.push('/accounts');
|
||||||
|
};
|
||||||
|
const handleImportSuccess = () => {
|
||||||
|
history.push('/accounts');
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<DashboardInsider name={'import-accounts'}>
|
||||||
|
<ImportView
|
||||||
|
resource={'accounts'}
|
||||||
|
onCancelClick={handleCancelBtnClick}
|
||||||
|
onImportSuccess={handleImportSuccess}
|
||||||
|
/>
|
||||||
|
</DashboardInsider>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -7,6 +7,7 @@ import {
|
|||||||
NavbarDivider,
|
NavbarDivider,
|
||||||
Alignment,
|
Alignment,
|
||||||
} from '@blueprintjs/core';
|
} from '@blueprintjs/core';
|
||||||
|
import { useHistory } from 'react-router-dom';
|
||||||
import {
|
import {
|
||||||
Icon,
|
Icon,
|
||||||
DashboardActionsBar,
|
DashboardActionsBar,
|
||||||
@@ -48,6 +49,8 @@ function AccountTransactionsActionsBar({
|
|||||||
const addMoneyInOptions = useMemo(() => getAddMoneyInOptions(), []);
|
const addMoneyInOptions = useMemo(() => getAddMoneyInOptions(), []);
|
||||||
const addMoneyOutOptions = useMemo(() => getAddMoneyOutOptions(), []);
|
const addMoneyOutOptions = useMemo(() => getAddMoneyOutOptions(), []);
|
||||||
|
|
||||||
|
const history = useHistory();
|
||||||
|
|
||||||
// Handle money in form
|
// Handle money in form
|
||||||
const handleMoneyInFormTransaction = (account) => {
|
const handleMoneyInFormTransaction = (account) => {
|
||||||
openDialog('money-in', {
|
openDialog('money-in', {
|
||||||
@@ -64,6 +67,11 @@ function AccountTransactionsActionsBar({
|
|||||||
account_name: account.name,
|
account_name: account.name,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
// Handle import button click.
|
||||||
|
const handleImportBtnClick = () => {
|
||||||
|
history.push(`/cashflow-accounts/${accountId}/import`);
|
||||||
|
};
|
||||||
|
|
||||||
// Refresh cashflow infinity transactions hook.
|
// Refresh cashflow infinity transactions hook.
|
||||||
const { refresh } = useRefreshCashflowTransactionsInfinity();
|
const { refresh } = useRefreshCashflowTransactionsInfinity();
|
||||||
|
|
||||||
@@ -106,6 +114,7 @@ function AccountTransactionsActionsBar({
|
|||||||
className={Classes.MINIMAL}
|
className={Classes.MINIMAL}
|
||||||
icon={<Icon icon="file-import-16" iconSize={16} />}
|
icon={<Icon icon="file-import-16" iconSize={16} />}
|
||||||
text={<T id={'import'} />}
|
text={<T id={'import'} />}
|
||||||
|
onClick={handleImportBtnClick}
|
||||||
/>
|
/>
|
||||||
<NavbarDivider />
|
<NavbarDivider />
|
||||||
<DashboardRowsHeightButton
|
<DashboardRowsHeightButton
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { DashboardInsider } from '@/components';
|
||||||
|
import { ImportView } from '@/containers/Import/ImportView';
|
||||||
|
import { useHistory, useParams } from 'react-router-dom';
|
||||||
|
|
||||||
|
export default function ImportUncategorizedTransactions() {
|
||||||
|
const history = useHistory();
|
||||||
|
const params = useParams();
|
||||||
|
|
||||||
|
const handleImportSuccess = () => {
|
||||||
|
history.push(
|
||||||
|
`/cashflow-accounts/${params.id}/transactions?filter=uncategorized`,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
const handleCnacelBtnClick = () => {
|
||||||
|
history.push(
|
||||||
|
`/cashflow-accounts/${params.id}/transactions?filter=uncategorized`,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<DashboardInsider name={'import-uncategorized-bank-transactions'}>
|
||||||
|
<ImportView
|
||||||
|
resource={'uncategorized_cashflow_transaction'}
|
||||||
|
params={{ accountId: params.id }}
|
||||||
|
onImportSuccess={handleImportSuccess}
|
||||||
|
onCancelClick={handleCnacelBtnClick}
|
||||||
|
sampleFileName={'sample_bank_transactions'}
|
||||||
|
/>
|
||||||
|
</DashboardInsider>
|
||||||
|
);
|
||||||
|
}
|
||||||
25
packages/webapp/src/containers/Customers/CustomersImport.tsx
Normal file
25
packages/webapp/src/containers/Customers/CustomersImport.tsx
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { DashboardInsider } from '@/components';
|
||||||
|
import { ImportView } from '../Import/ImportView';
|
||||||
|
import { useHistory } from 'react-router-dom';
|
||||||
|
|
||||||
|
export default function CustomersImport() {
|
||||||
|
const history = useHistory();
|
||||||
|
|
||||||
|
const handleImportSuccess = () => {
|
||||||
|
history.push('/customers');
|
||||||
|
};
|
||||||
|
const handleCancelBtnClick = () => {
|
||||||
|
history.push('/customers');
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<DashboardInsider name={'import-customers'}>
|
||||||
|
<ImportView
|
||||||
|
resource={'customers'}
|
||||||
|
onImportSuccess={handleImportSuccess}
|
||||||
|
onCancelClick={handleCancelBtnClick}
|
||||||
|
exampleTitle="Customers Example"
|
||||||
|
/>
|
||||||
|
</DashboardInsider>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -95,6 +95,11 @@ function CustomerActionsBar({
|
|||||||
addSetting('customers', 'tableSize', size);
|
addSetting('customers', 'tableSize', size);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Handle import button click.
|
||||||
|
const handleImportBtnClick = () => {
|
||||||
|
history.push('/customers/import');
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<DashboardActionsBar>
|
<DashboardActionsBar>
|
||||||
<NavbarGroup>
|
<NavbarGroup>
|
||||||
@@ -142,6 +147,7 @@ function CustomerActionsBar({
|
|||||||
<Button
|
<Button
|
||||||
className={Classes.MINIMAL}
|
className={Classes.MINIMAL}
|
||||||
icon={<Icon icon="file-import-16" iconSize={16} />}
|
icon={<Icon icon="file-import-16" iconSize={16} />}
|
||||||
|
onClick={handleImportBtnClick}
|
||||||
text={<T id={'import'} />}
|
text={<T id={'import'} />}
|
||||||
/>
|
/>
|
||||||
<Button
|
<Button
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
|
||||||
|
.root {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.title{
|
||||||
|
color: #5F6B7C;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.iconWrap{
|
||||||
|
color: #8F99A8;
|
||||||
|
}
|
||||||
|
.subtitle {
|
||||||
|
color: #8F99A8;
|
||||||
|
}
|
||||||
|
.dropzoneContent{
|
||||||
|
text-align: center;
|
||||||
|
margin: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.buttons-wrap {
|
||||||
|
margin-top: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzoneHint{
|
||||||
|
display: flex;
|
||||||
|
font-size: 12px;
|
||||||
|
margin-top: 6px;
|
||||||
|
color: #8F99A8;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
27
packages/webapp/src/containers/Import/ImportDropzone.tsx
Normal file
27
packages/webapp/src/containers/Import/ImportDropzone.tsx
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { Field } from 'formik';
|
||||||
|
import { Box, Group, Stack } from '@/components';
|
||||||
|
import styles from './ImportDropzone.module.css';
|
||||||
|
import { ImportDropzoneField } from './ImportDropzoneFile';
|
||||||
|
|
||||||
|
export function ImportDropzone() {
|
||||||
|
return (
|
||||||
|
<Stack spacing={0}>
|
||||||
|
<Field id={'file'} name={'file'} type="file">
|
||||||
|
{({ form }) => (
|
||||||
|
<ImportDropzoneField
|
||||||
|
value={form.file}
|
||||||
|
onChange={(file) => {
|
||||||
|
form.setFieldValue('file', file);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Field>
|
||||||
|
|
||||||
|
<Group className={styles.dropzoneHint}>
|
||||||
|
<Box>Supperted Formats: CSV, XLSX</Box>
|
||||||
|
<Box>Maximum size: 25MB</Box>
|
||||||
|
</Group>
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
}
|
||||||
80
packages/webapp/src/containers/Import/ImportDropzoneFile.tsx
Normal file
80
packages/webapp/src/containers/Import/ImportDropzoneFile.tsx
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { useRef } from 'react';
|
||||||
|
import { Button, Intent } from '@blueprintjs/core';
|
||||||
|
import { Box, Icon, Stack } from '@/components';
|
||||||
|
import { Dropzone, DropzoneProps } from '@/components/Dropzone';
|
||||||
|
import { MIME_TYPES } from '@/components/Dropzone/mine-types';
|
||||||
|
import styles from './ImportDropzone.module.css';
|
||||||
|
import { useUncontrolled } from '@/hooks/useUncontrolled';
|
||||||
|
|
||||||
|
interface ImportDropzoneFieldProps {
|
||||||
|
initialValue?: File;
|
||||||
|
value?: File;
|
||||||
|
onChange?: (file: File) => void;
|
||||||
|
dropzoneProps?: DropzoneProps;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ImportDropzoneField({
|
||||||
|
initialValue,
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
dropzoneProps,
|
||||||
|
}: ImportDropzoneFieldProps) {
|
||||||
|
const [localValue, handleChange] = useUncontrolled({
|
||||||
|
value,
|
||||||
|
initialValue,
|
||||||
|
finalValue: null,
|
||||||
|
onChange,
|
||||||
|
});
|
||||||
|
const openRef = useRef<() => void>(null);
|
||||||
|
|
||||||
|
const handleRemove = () => {
|
||||||
|
handleChange(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dropzone
|
||||||
|
onDrop={(files) => handleChange(files[0])}
|
||||||
|
onReject={(files) => console.log('rejected files', files)}
|
||||||
|
maxSize={5 * 1024 ** 2}
|
||||||
|
accept={[MIME_TYPES.csv, MIME_TYPES.xls, MIME_TYPES.xlsx]}
|
||||||
|
classNames={{ content: styles.dropzoneContent }}
|
||||||
|
activateOnClick={false}
|
||||||
|
openRef={openRef}
|
||||||
|
{...dropzoneProps}
|
||||||
|
>
|
||||||
|
<Stack spacing={12} align="center" className={styles.content}>
|
||||||
|
<Box className={styles.iconWrap}>
|
||||||
|
<Icon icon="download" iconSize={26} />
|
||||||
|
</Box>
|
||||||
|
{localValue ? (
|
||||||
|
<Stack spacing={6} justify="center" align="center">
|
||||||
|
<h4 className={styles.title}>{localValue.name}</h4>
|
||||||
|
<Button small minimal intent={Intent.DANGER} onClick={handleRemove}>
|
||||||
|
Remove
|
||||||
|
</Button>
|
||||||
|
</Stack>
|
||||||
|
) : (
|
||||||
|
<Stack spacing={4} align="center">
|
||||||
|
<h4 className={styles.title}>
|
||||||
|
Drag images here or click to select files
|
||||||
|
</h4>
|
||||||
|
<span className={styles.subtitle}>
|
||||||
|
Drag and Drop file here or Choose file
|
||||||
|
</span>
|
||||||
|
</Stack>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Button
|
||||||
|
intent="none"
|
||||||
|
onClick={() => openRef.current?.()}
|
||||||
|
style={{ pointerEvents: 'all' }}
|
||||||
|
minimal
|
||||||
|
outlined
|
||||||
|
>
|
||||||
|
{localValue ? 'Replace File' : 'Upload File'}
|
||||||
|
</Button>
|
||||||
|
</Stack>
|
||||||
|
</Dropzone>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
|
||||||
|
|
||||||
|
.root{
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
import styles from './ImportFileUploadStep.module.scss';
|
||||||
|
|
||||||
|
interface ImportFileContainerProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ImportFileContainer({ children }: ImportFileContainerProps) {
|
||||||
|
return <div className={styles.content}>{children}</div>;
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import clsx from 'classnames';
|
||||||
|
import { Group } from '@/components';
|
||||||
|
import { CLASSES } from '@/constants';
|
||||||
|
import { Button, Intent } from '@blueprintjs/core';
|
||||||
|
import { useFormikContext } from 'formik';
|
||||||
|
import styles from './ImportFileActions.module.scss';
|
||||||
|
import { useImportFileContext } from './ImportFileProvider';
|
||||||
|
|
||||||
|
export function ImportFileUploadFooterActions() {
|
||||||
|
const { isSubmitting } = useFormikContext();
|
||||||
|
const { onCancelClick } = useImportFileContext();
|
||||||
|
|
||||||
|
const handleCancelBtnClick = () => {
|
||||||
|
onCancelClick && onCancelClick();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={clsx(CLASSES.PAGE_FORM_FLOATING_ACTIONS, styles.root)}>
|
||||||
|
<Group spacing={10}>
|
||||||
|
<Button onClick={handleCancelBtnClick}>Cancel</Button>
|
||||||
|
<Button type="submit" intent={Intent.PRIMARY} loading={isSubmitting}>
|
||||||
|
Next
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
.table {
|
||||||
|
width: 100%;
|
||||||
|
margin-top: 1.4rem;
|
||||||
|
|
||||||
|
th.label,
|
||||||
|
td.label{
|
||||||
|
width: 32% !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
thead{
|
||||||
|
th{
|
||||||
|
border-top: 1px solid #d9d9da;
|
||||||
|
padding-top: 8px;
|
||||||
|
padding-bottom: 8px;
|
||||||
|
color: #738091;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tbody{
|
||||||
|
tr td {
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
|
||||||
|
tr td{
|
||||||
|
:global(.bp4-popover-target .bp4-button),
|
||||||
|
:global(.bp4-popover-wrapper){
|
||||||
|
max-width: 250px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.requiredSign{
|
||||||
|
color: rgb(250, 82, 82);
|
||||||
|
}
|
||||||
97
packages/webapp/src/containers/Import/ImportFileMapping.tsx
Normal file
97
packages/webapp/src/containers/Import/ImportFileMapping.tsx
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
import { useMemo } from 'react';
|
||||||
|
import clsx from 'classnames';
|
||||||
|
import { Button, Intent, Position } from '@blueprintjs/core';
|
||||||
|
import { useFormikContext } from 'formik';
|
||||||
|
import { FSelect, Group, Hint } from '@/components';
|
||||||
|
import { ImportFileMappingForm } from './ImportFileMappingForm';
|
||||||
|
import { EntityColumn, useImportFileContext } from './ImportFileProvider';
|
||||||
|
import { CLASSES } from '@/constants';
|
||||||
|
import { ImportFileContainer } from './ImportFileContainer';
|
||||||
|
import { ImportStepperStep } from './_types';
|
||||||
|
import { ImportFileMapBootProvider } from './ImportFileMappingBoot';
|
||||||
|
import styles from './ImportFileMapping.module.scss';
|
||||||
|
|
||||||
|
export function ImportFileMapping() {
|
||||||
|
const { importId } = useImportFileContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ImportFileMapBootProvider importId={importId}>
|
||||||
|
<ImportFileMappingForm>
|
||||||
|
<ImportFileContainer>
|
||||||
|
<p>
|
||||||
|
Review and map the column headers in your csv/xlsx file with the
|
||||||
|
Bigcapital fields.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<table className={clsx('bp4-html-table', styles.table)}>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th className={styles.label}>Bigcapital Fields</th>
|
||||||
|
<th className={styles.field}>Sheet Column Headers</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<ImportFileMappingFields />
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</ImportFileContainer>
|
||||||
|
|
||||||
|
<ImportFileMappingFloatingActions />
|
||||||
|
</ImportFileMappingForm>
|
||||||
|
</ImportFileMapBootProvider>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ImportFileMappingFields() {
|
||||||
|
const { entityColumns, sheetColumns } = useImportFileContext();
|
||||||
|
|
||||||
|
const items = useMemo(
|
||||||
|
() => sheetColumns.map((column) => ({ value: column, text: column })),
|
||||||
|
[sheetColumns],
|
||||||
|
);
|
||||||
|
const columnMapper = (column: EntityColumn, index: number) => (
|
||||||
|
<tr key={index}>
|
||||||
|
<td className={styles.label}>
|
||||||
|
{column.name}{' '}
|
||||||
|
{column.required && <span className={styles.requiredSign}>*</span>}
|
||||||
|
</td>
|
||||||
|
<td className={styles.field}>
|
||||||
|
<Group spacing={4}>
|
||||||
|
<FSelect
|
||||||
|
name={column.key}
|
||||||
|
items={items}
|
||||||
|
popoverProps={{ minimal: true }}
|
||||||
|
minimal={true}
|
||||||
|
fill={true}
|
||||||
|
/>
|
||||||
|
{column.hint && (
|
||||||
|
<Hint content={column.hint} position={Position.BOTTOM} />
|
||||||
|
)}
|
||||||
|
</Group>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
);
|
||||||
|
const columns = entityColumns.map(columnMapper);
|
||||||
|
|
||||||
|
return <>{columns}</>;
|
||||||
|
}
|
||||||
|
|
||||||
|
function ImportFileMappingFloatingActions() {
|
||||||
|
const { isSubmitting } = useFormikContext();
|
||||||
|
const { setStep } = useImportFileContext();
|
||||||
|
|
||||||
|
const handleCancelBtnClick = () => {
|
||||||
|
setStep(ImportStepperStep.Upload);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={clsx(CLASSES.PAGE_FORM_FLOATING_ACTIONS)}>
|
||||||
|
<Group spacing={10}>
|
||||||
|
<Button onClick={handleCancelBtnClick}>Back</Button>
|
||||||
|
<Button type="submit" intent={Intent.PRIMARY} loading={isSubmitting}>
|
||||||
|
Next
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,58 @@
|
|||||||
|
import { Spinner } from '@blueprintjs/core';
|
||||||
|
import React, { createContext, useContext } from 'react';
|
||||||
|
import { Box } from '@/components';
|
||||||
|
import { useImportFileMeta } from '@/hooks/query/import';
|
||||||
|
|
||||||
|
interface ImportFileMapBootContextValue {}
|
||||||
|
|
||||||
|
const ImportFileMapBootContext = createContext<ImportFileMapBootContextValue>(
|
||||||
|
{} as ImportFileMapBootContextValue,
|
||||||
|
);
|
||||||
|
|
||||||
|
export const useImportFileMapBootContext = () => {
|
||||||
|
const context = useContext<ImportFileMapBootContextValue>(
|
||||||
|
ImportFileMapBootContext,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!context) {
|
||||||
|
throw new Error(
|
||||||
|
'useImportFileMapBootContext must be used within an ImportFileMapBootProvider',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
};
|
||||||
|
|
||||||
|
interface ImportFileMapBootProps {
|
||||||
|
importId: string;
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ImportFileMapBootProvider = ({
|
||||||
|
importId,
|
||||||
|
children,
|
||||||
|
}: ImportFileMapBootProps) => {
|
||||||
|
const {
|
||||||
|
data: importFile,
|
||||||
|
isLoading: isImportFileLoading,
|
||||||
|
isFetching: isImportFileFetching,
|
||||||
|
} = useImportFileMeta(importId, {
|
||||||
|
enabled: Boolean(importId),
|
||||||
|
});
|
||||||
|
|
||||||
|
const value = {
|
||||||
|
importFile,
|
||||||
|
isImportFileLoading,
|
||||||
|
isImportFileFetching,
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<ImportFileMapBootContext.Provider value={value}>
|
||||||
|
{isImportFileLoading ? (
|
||||||
|
<Box style={{ padding: '2rem', textAlign: 'center' }}>
|
||||||
|
<Spinner size={26} />
|
||||||
|
</Box>
|
||||||
|
) : (
|
||||||
|
<>{children}</>
|
||||||
|
)}
|
||||||
|
</ImportFileMapBootContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
101
packages/webapp/src/containers/Import/ImportFileMappingForm.tsx
Normal file
101
packages/webapp/src/containers/Import/ImportFileMappingForm.tsx
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { Intent } from '@blueprintjs/core';
|
||||||
|
import { useImportFileMapping } from '@/hooks/query/import';
|
||||||
|
import { Form, Formik, FormikHelpers } from 'formik';
|
||||||
|
import { useImportFileContext } from './ImportFileProvider';
|
||||||
|
import { useMemo } from 'react';
|
||||||
|
import { isEmpty, lowerCase } from 'lodash';
|
||||||
|
import { AppToaster } from '@/components';
|
||||||
|
import { useImportFileMapBootContext } from './ImportFileMappingBoot';
|
||||||
|
import { transformToForm } from '@/utils';
|
||||||
|
|
||||||
|
interface ImportFileMappingFormProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
type ImportFileMappingFormValues = Record<string, string | null>;
|
||||||
|
|
||||||
|
export function ImportFileMappingForm({
|
||||||
|
children,
|
||||||
|
}: ImportFileMappingFormProps) {
|
||||||
|
const { mutateAsync: submitImportFileMapping } = useImportFileMapping();
|
||||||
|
const { importId, setStep } = useImportFileContext();
|
||||||
|
|
||||||
|
const initialValues = useImportFileMappingInitialValues();
|
||||||
|
|
||||||
|
const handleSubmit = (
|
||||||
|
values: ImportFileMappingFormValues,
|
||||||
|
{ setSubmitting }: FormikHelpers<ImportFileMappingFormValues>,
|
||||||
|
) => {
|
||||||
|
setSubmitting(true);
|
||||||
|
const _values = transformValueToReq(values);
|
||||||
|
|
||||||
|
submitImportFileMapping([importId, _values])
|
||||||
|
.then(() => {
|
||||||
|
setSubmitting(false);
|
||||||
|
setStep(2);
|
||||||
|
})
|
||||||
|
.catch(({ response: { data } }) => {
|
||||||
|
if (data.errors.find((e) => e.type === 'DUPLICATED_FROM_MAP_ATTR')) {
|
||||||
|
AppToaster.show({
|
||||||
|
message: 'Selected the same sheet columns to multiple fields.',
|
||||||
|
intent: Intent.DANGER,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
setSubmitting(false);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Formik initialValues={initialValues} onSubmit={handleSubmit}>
|
||||||
|
<Form>{children}</Form>
|
||||||
|
</Formik>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const transformValueToReq = (value: ImportFileMappingFormValues) => {
|
||||||
|
const mapping = Object.keys(value)
|
||||||
|
.filter((key) => !isEmpty(value[key]))
|
||||||
|
.map((key) => ({ from: value[key], to: key }));
|
||||||
|
return { mapping };
|
||||||
|
};
|
||||||
|
|
||||||
|
const transformResToFormValues = (value: { from: string; to: string }[]) => {
|
||||||
|
return value?.reduce((acc, map) => {
|
||||||
|
acc[map.to] = map.from;
|
||||||
|
return acc;
|
||||||
|
}, {});
|
||||||
|
};
|
||||||
|
|
||||||
|
const useImportFileMappingInitialValues = () => {
|
||||||
|
const { importFile } = useImportFileMapBootContext();
|
||||||
|
const { entityColumns, sheetColumns } = useImportFileContext();
|
||||||
|
|
||||||
|
const initialResValues = useMemo(
|
||||||
|
() => transformResToFormValues(importFile?.map || []),
|
||||||
|
[importFile?.map],
|
||||||
|
);
|
||||||
|
|
||||||
|
const initialValues = useMemo(
|
||||||
|
() =>
|
||||||
|
entityColumns.reduce((acc, { key, name }) => {
|
||||||
|
const _name = lowerCase(name);
|
||||||
|
const _matched = sheetColumns.find(
|
||||||
|
(column) => lowerCase(column) === _name,
|
||||||
|
);
|
||||||
|
// Match the default column name the same field name
|
||||||
|
// if matched one of sheet columns has the same field name.
|
||||||
|
acc[key] = _matched ? _matched : '';
|
||||||
|
return acc;
|
||||||
|
}, {}),
|
||||||
|
[entityColumns, sheetColumns],
|
||||||
|
);
|
||||||
|
|
||||||
|
return useMemo(
|
||||||
|
() => ({
|
||||||
|
...transformToForm(initialResValues, initialValues),
|
||||||
|
...initialValues,
|
||||||
|
}),
|
||||||
|
[initialValues, initialResValues],
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
|
||||||
|
.previewList {
|
||||||
|
list-style: none;
|
||||||
|
margin-top: 14px;
|
||||||
|
|
||||||
|
:global(li) {
|
||||||
|
border-top: 1px solid #d9d9da;
|
||||||
|
padding: 6px 0;
|
||||||
|
|
||||||
|
&:last-child{
|
||||||
|
padding-bottom: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.unmappedList {
|
||||||
|
padding-left: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.skippedTable {
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
thead{
|
||||||
|
th{
|
||||||
|
padding-top: 0;
|
||||||
|
padding-bottom: 8px;
|
||||||
|
color: #738091;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tbody{
|
||||||
|
tr td {
|
||||||
|
vertical-align: middle;
|
||||||
|
padding: 7px;
|
||||||
|
}
|
||||||
|
|
||||||
|
tr:hover td{
|
||||||
|
background: #F6F7F9;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
189
packages/webapp/src/containers/Import/ImportFilePreview.tsx
Normal file
189
packages/webapp/src/containers/Import/ImportFilePreview.tsx
Normal file
@@ -0,0 +1,189 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { Button, Callout, Intent, Text } from '@blueprintjs/core';
|
||||||
|
import clsx from 'classnames';
|
||||||
|
import {
|
||||||
|
ImportFilePreviewBootProvider,
|
||||||
|
useImportFilePreviewBootContext,
|
||||||
|
} from './ImportFilePreviewBoot';
|
||||||
|
import { useImportFileContext } from './ImportFileProvider';
|
||||||
|
import { useImportFileProcess } from '@/hooks/query/import';
|
||||||
|
import { AppToaster, Box, Group, Stack } from '@/components';
|
||||||
|
import { CLASSES } from '@/constants';
|
||||||
|
import { ImportStepperStep } from './_types';
|
||||||
|
import { ImportFileContainer } from './ImportFileContainer';
|
||||||
|
import { SectionCard, Section } from '@/components/Section';
|
||||||
|
import styles from './ImportFilePreview.module.scss';
|
||||||
|
|
||||||
|
export function ImportFilePreview() {
|
||||||
|
const { importId } = useImportFileContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ImportFilePreviewBootProvider importId={importId}>
|
||||||
|
<ImportFilePreviewContent />
|
||||||
|
</ImportFilePreviewBootProvider>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ImportFilePreviewContent() {
|
||||||
|
const { importPreview } = useImportFilePreviewBootContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box>
|
||||||
|
<ImportFileContainer>
|
||||||
|
<Stack spacing={20}>
|
||||||
|
<Callout
|
||||||
|
intent={
|
||||||
|
importPreview.createdCount <= 0 ? Intent.DANGER : Intent.NONE
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{importPreview.createdCount} of {importPreview.totalCount} Items in
|
||||||
|
your file are ready to be imported.
|
||||||
|
</Callout>
|
||||||
|
|
||||||
|
<ImportFilePreviewImported />
|
||||||
|
<ImportFilePreviewSkipped />
|
||||||
|
<ImportFilePreviewUnmapped />
|
||||||
|
</Stack>
|
||||||
|
</ImportFileContainer>
|
||||||
|
<ImportFilePreviewFloatingActions />
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ImportFilePreviewImported() {
|
||||||
|
const { importPreview } = useImportFilePreviewBootContext();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Section
|
||||||
|
collapseProps={{ defaultIsOpen: false }}
|
||||||
|
defaultIsOpen={true}
|
||||||
|
title={`(${importPreview.createdCount}) Items are ready to import`}
|
||||||
|
>
|
||||||
|
<SectionCard padded={true}>
|
||||||
|
<Text>
|
||||||
|
Items that are ready to be imported - {importPreview.createdCount}
|
||||||
|
</Text>
|
||||||
|
<ul className={styles.previewList}>
|
||||||
|
<li>
|
||||||
|
Items to be created: <span>({importPreview.createdCount})</span>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
Items to be skipped: <span>({importPreview.skippedCount})</span>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
Items have errors: <span>({importPreview.errorsCount})</span>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</SectionCard>
|
||||||
|
</Section>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ImportFilePreviewSkipped() {
|
||||||
|
const { importPreview } = useImportFilePreviewBootContext();
|
||||||
|
|
||||||
|
// Can't continue if there's no skipped items.
|
||||||
|
if (importPreview.skippedCount <= 0) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Section
|
||||||
|
collapseProps={{ defaultIsOpen: false }}
|
||||||
|
collapsible={true}
|
||||||
|
title={`(${importPreview.skippedCount}) Items are skipped`}
|
||||||
|
>
|
||||||
|
<SectionCard padded={true}>
|
||||||
|
<table className={clsx('bp4-html-table', styles.skippedTable)}>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th className={'number'}>#</th>
|
||||||
|
<th className={'name'}>Name</th>
|
||||||
|
<th>Error</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{importPreview?.errors.map((error, key) => (
|
||||||
|
<tr key={key}>
|
||||||
|
<td>{error.rowNumber}</td>
|
||||||
|
<td>{error.rowNumber}</td>
|
||||||
|
<td>
|
||||||
|
{error.errorMessage.map((message) => (
|
||||||
|
<div>{message}</div>
|
||||||
|
))}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
))}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</SectionCard>
|
||||||
|
</Section>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ImportFilePreviewUnmapped() {
|
||||||
|
const { importPreview } = useImportFilePreviewBootContext();
|
||||||
|
|
||||||
|
// Can't continue if there's no unmapped columns.
|
||||||
|
if (importPreview?.unmappedColumnsCount <= 0) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Section
|
||||||
|
collapseProps={{ defaultIsOpen: false }}
|
||||||
|
collapsible={true}
|
||||||
|
title={`(${importPreview?.unmappedColumnsCount}) Unmapped Columns`}
|
||||||
|
>
|
||||||
|
<SectionCard padded={true}>
|
||||||
|
<ul className={styles.unmappedList}>
|
||||||
|
{importPreview.unmappedColumns?.map((column, key) => (
|
||||||
|
<li key={key}>{column}</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</SectionCard>
|
||||||
|
</Section>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ImportFilePreviewFloatingActions() {
|
||||||
|
const { importId, setStep, onImportSuccess, onImportFailed } =
|
||||||
|
useImportFileContext();
|
||||||
|
const { importPreview } = useImportFilePreviewBootContext();
|
||||||
|
const { mutateAsync: importFile, isLoading: isImportFileLoading } =
|
||||||
|
useImportFileProcess();
|
||||||
|
|
||||||
|
const isValidToImport = importPreview?.createdCount > 0;
|
||||||
|
|
||||||
|
const handleSubmitBtn = () => {
|
||||||
|
importFile(importId)
|
||||||
|
.then(() => {
|
||||||
|
AppToaster.show({
|
||||||
|
intent: Intent.SUCCESS,
|
||||||
|
message: `The ${
|
||||||
|
importPreview.createdCount
|
||||||
|
} of ${10} has imported successfully.`,
|
||||||
|
});
|
||||||
|
onImportSuccess && onImportSuccess();
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
onImportFailed && onImportFailed();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
const handleCancelBtnClick = () => {
|
||||||
|
setStep(ImportStepperStep.Mapping);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={clsx(CLASSES.PAGE_FORM_FLOATING_ACTIONS)}>
|
||||||
|
<Group spacing={10}>
|
||||||
|
<Button onClick={handleCancelBtnClick}>Cancel</Button>
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
intent={Intent.PRIMARY}
|
||||||
|
loading={isImportFileLoading}
|
||||||
|
onClick={handleSubmitBtn}
|
||||||
|
disabled={!isValidToImport}
|
||||||
|
>
|
||||||
|
Import
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,59 @@
|
|||||||
|
import { Spinner } from '@blueprintjs/core';
|
||||||
|
import React, { createContext, useContext } from 'react';
|
||||||
|
import { Box } from '@/components';
|
||||||
|
import { useImportFilePreview } from '@/hooks/query/import';
|
||||||
|
|
||||||
|
interface ImportFilePreviewBootContextValue {}
|
||||||
|
|
||||||
|
const ImportFilePreviewBootContext =
|
||||||
|
createContext<ImportFilePreviewBootContextValue>(
|
||||||
|
{} as ImportFilePreviewBootContextValue,
|
||||||
|
);
|
||||||
|
|
||||||
|
export const useImportFilePreviewBootContext = () => {
|
||||||
|
const context = useContext<ImportFilePreviewBootContextValue>(
|
||||||
|
ImportFilePreviewBootContext,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!context) {
|
||||||
|
throw new Error(
|
||||||
|
'useImportFilePreviewBootContext must be used within an ImportFilePreviewBootProvider',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
};
|
||||||
|
|
||||||
|
interface ImportFilePreviewBootProps {
|
||||||
|
importId: string;
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ImportFilePreviewBootProvider = ({
|
||||||
|
importId,
|
||||||
|
children,
|
||||||
|
}: ImportFilePreviewBootProps) => {
|
||||||
|
const {
|
||||||
|
data: importPreview,
|
||||||
|
isLoading: isImportPreviewLoading,
|
||||||
|
isFetching: isImportPreviewFetching,
|
||||||
|
} = useImportFilePreview(importId, {
|
||||||
|
enabled: Boolean(importId),
|
||||||
|
});
|
||||||
|
|
||||||
|
const value = {
|
||||||
|
importPreview,
|
||||||
|
isImportPreviewLoading,
|
||||||
|
isImportPreviewFetching,
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<ImportFilePreviewBootContext.Provider value={value}>
|
||||||
|
{isImportPreviewLoading ? (
|
||||||
|
<Box style={{ padding: '2rem', textAlign: 'center' }}>
|
||||||
|
<Spinner size={26} />
|
||||||
|
</Box>
|
||||||
|
) : (
|
||||||
|
<>{children}</>
|
||||||
|
)}
|
||||||
|
</ImportFilePreviewBootContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
138
packages/webapp/src/containers/Import/ImportFileProvider.tsx
Normal file
138
packages/webapp/src/containers/Import/ImportFileProvider.tsx
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import React, {
|
||||||
|
Dispatch,
|
||||||
|
SetStateAction,
|
||||||
|
createContext,
|
||||||
|
useContext,
|
||||||
|
useState,
|
||||||
|
} from 'react';
|
||||||
|
|
||||||
|
export type EntityColumn = {
|
||||||
|
key: string;
|
||||||
|
name: string;
|
||||||
|
required?: boolean;
|
||||||
|
hint?: string;
|
||||||
|
};
|
||||||
|
export type SheetColumn = string;
|
||||||
|
export type SheetMap = { from: string; to: string };
|
||||||
|
|
||||||
|
interface ImportFileContextValue {
|
||||||
|
sheetColumns: SheetColumn[];
|
||||||
|
setSheetColumns: Dispatch<SetStateAction<SheetColumn[]>>;
|
||||||
|
|
||||||
|
entityColumns: EntityColumn[];
|
||||||
|
setEntityColumns: Dispatch<SetStateAction<EntityColumn[]>>;
|
||||||
|
|
||||||
|
sheetMapping: SheetMap[];
|
||||||
|
setSheetMapping: Dispatch<SetStateAction<SheetMap[]>>;
|
||||||
|
|
||||||
|
step: number;
|
||||||
|
setStep: Dispatch<SetStateAction<number>>;
|
||||||
|
|
||||||
|
importId: string;
|
||||||
|
setImportId: Dispatch<SetStateAction<string>>;
|
||||||
|
|
||||||
|
resource: string;
|
||||||
|
description?: string;
|
||||||
|
params: Record<string, any>;
|
||||||
|
onImportSuccess?: () => void;
|
||||||
|
onImportFailed?: () => void;
|
||||||
|
onCancelClick?: () => void;
|
||||||
|
sampleFileName?: string;
|
||||||
|
|
||||||
|
exampleDownload?: boolean;
|
||||||
|
exampleTitle?: string;
|
||||||
|
exampleDescription?: string;
|
||||||
|
}
|
||||||
|
interface ImportFileProviderProps {
|
||||||
|
resource: string;
|
||||||
|
description?: string;
|
||||||
|
params: Record<string, any>;
|
||||||
|
onImportSuccess?: () => void;
|
||||||
|
onImportFailed?: () => void;
|
||||||
|
onCancelClick?: () => void;
|
||||||
|
children: React.ReactNode;
|
||||||
|
sampleFileName?: string;
|
||||||
|
|
||||||
|
exampleDownload?: boolean;
|
||||||
|
exampleTitle?: string;
|
||||||
|
exampleDescription?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ExampleDescription =
|
||||||
|
'You can download the sample file to obtain detailed information about the data fields used during the import.';
|
||||||
|
const ExampleTitle = 'Table Example';
|
||||||
|
|
||||||
|
const ImportFileContext = createContext<ImportFileContextValue>(
|
||||||
|
{} as ImportFileContextValue,
|
||||||
|
);
|
||||||
|
|
||||||
|
export const useImportFileContext = () => {
|
||||||
|
const context = useContext<ImportFileContextValue>(ImportFileContext);
|
||||||
|
|
||||||
|
if (!context) {
|
||||||
|
throw new Error(
|
||||||
|
'useImportFileContext must be used within an ImportFileProvider',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ImportFileProvider = ({
|
||||||
|
resource,
|
||||||
|
children,
|
||||||
|
description,
|
||||||
|
params,
|
||||||
|
onImportFailed,
|
||||||
|
onImportSuccess,
|
||||||
|
onCancelClick,
|
||||||
|
sampleFileName,
|
||||||
|
|
||||||
|
exampleDownload = true,
|
||||||
|
exampleTitle = ExampleTitle,
|
||||||
|
exampleDescription = ExampleDescription,
|
||||||
|
}: ImportFileProviderProps) => {
|
||||||
|
const [sheetColumns, setSheetColumns] = useState<SheetColumn[]>([]);
|
||||||
|
const [entityColumns, setEntityColumns] = useState<SheetColumn[]>([]);
|
||||||
|
const [sheetMapping, setSheetMapping] = useState<SheetMap[]>([]);
|
||||||
|
const [importId, setImportId] = useState<string>('');
|
||||||
|
|
||||||
|
const [step, setStep] = useState<number>(0);
|
||||||
|
|
||||||
|
const value = {
|
||||||
|
sheetColumns,
|
||||||
|
setSheetColumns,
|
||||||
|
|
||||||
|
entityColumns,
|
||||||
|
setEntityColumns,
|
||||||
|
|
||||||
|
sheetMapping,
|
||||||
|
setSheetMapping,
|
||||||
|
|
||||||
|
step,
|
||||||
|
setStep,
|
||||||
|
|
||||||
|
importId,
|
||||||
|
setImportId,
|
||||||
|
|
||||||
|
resource,
|
||||||
|
description,
|
||||||
|
params,
|
||||||
|
|
||||||
|
onImportSuccess,
|
||||||
|
onImportFailed,
|
||||||
|
onCancelClick,
|
||||||
|
|
||||||
|
sampleFileName,
|
||||||
|
|
||||||
|
exampleDownload,
|
||||||
|
exampleTitle,
|
||||||
|
exampleDescription,
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ImportFileContext.Provider value={value}>
|
||||||
|
{children}
|
||||||
|
</ImportFileContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,86 @@
|
|||||||
|
// @ts-nocheck
|
||||||
|
import { AppToaster } from '@/components';
|
||||||
|
import { useImportFileUpload } from '@/hooks/query/import';
|
||||||
|
import { Intent } from '@blueprintjs/core';
|
||||||
|
import { Formik, Form, FormikHelpers } from 'formik';
|
||||||
|
import * as Yup from 'yup';
|
||||||
|
import { useImportFileContext } from './ImportFileProvider';
|
||||||
|
import { ImportStepperStep } from './_types';
|
||||||
|
|
||||||
|
const initialValues = {
|
||||||
|
file: null,
|
||||||
|
} as ImportFileUploadValues;
|
||||||
|
|
||||||
|
interface ImportFileUploadFormProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
const validationSchema = Yup.object().shape({
|
||||||
|
file: Yup.mixed().required('File is required'),
|
||||||
|
});
|
||||||
|
|
||||||
|
interface ImportFileUploadValues {
|
||||||
|
file: File | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ImportFileUploadForm({
|
||||||
|
children,
|
||||||
|
formikProps,
|
||||||
|
formProps,
|
||||||
|
}: ImportFileUploadFormProps) {
|
||||||
|
const { mutateAsync: uploadImportFile } = useImportFileUpload();
|
||||||
|
const {
|
||||||
|
resource,
|
||||||
|
params,
|
||||||
|
setStep,
|
||||||
|
setSheetColumns,
|
||||||
|
setEntityColumns,
|
||||||
|
setImportId,
|
||||||
|
} = useImportFileContext();
|
||||||
|
|
||||||
|
const handleSubmit = (
|
||||||
|
values: ImportFileUploadValues,
|
||||||
|
{ setSubmitting }: FormikHelpers<ImportFileUploadValues>,
|
||||||
|
) => {
|
||||||
|
if (!values.file) return;
|
||||||
|
|
||||||
|
setSubmitting(true);
|
||||||
|
const formData = new FormData();
|
||||||
|
formData.append('file', values.file);
|
||||||
|
formData.append('resource', resource);
|
||||||
|
formData.append('params', JSON.stringify(params));
|
||||||
|
|
||||||
|
uploadImportFile(formData)
|
||||||
|
.then(({ data }) => {
|
||||||
|
setImportId(data.import.import_id);
|
||||||
|
setSheetColumns(data.sheet_columns);
|
||||||
|
setEntityColumns(data.resource_columns);
|
||||||
|
setStep(ImportStepperStep.Mapping);
|
||||||
|
setSubmitting(false);
|
||||||
|
})
|
||||||
|
.catch(({ response: { data } }) => {
|
||||||
|
if (
|
||||||
|
data.errors.find(
|
||||||
|
(er) => er.type === 'IMPORTED_FILE_EXTENSION_INVALID',
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
AppToaster.show({
|
||||||
|
intent: Intent.DANGER,
|
||||||
|
message: 'The extenstion of uploaded file is not supported.',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
setSubmitting(false);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Formik
|
||||||
|
initialValues={initialValues}
|
||||||
|
onSubmit={handleSubmit}
|
||||||
|
validationSchema={validationSchema}
|
||||||
|
{...formikProps}
|
||||||
|
>
|
||||||
|
<Form {...formProps}>{children}</Form>
|
||||||
|
</Formik>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
.root {
|
||||||
|
margin-top: 2.2rem
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
flex: 1;
|
||||||
|
padding: 32px 20px;
|
||||||
|
padding-bottom: 80px;
|
||||||
|
min-width: 660px;
|
||||||
|
max-width: 760px;
|
||||||
|
width: 75%;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form {
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user