salix/modules/client/back/models/client.js

477 lines
17 KiB
JavaScript
Raw Normal View History

const got = require('got');
2020-11-02 13:30:33 +00:00
const UserError = require('vn-loopback/util/user-error');
const getFinalState = require('vn-loopback/util/hook').getFinalState;
const isMultiple = require('vn-loopback/util/hook').isMultiple;
const validateTin = require('vn-loopback/util/validateTin');
const validateIban = require('vn-loopback/util/validateIban');
const LoopBackContext = require('loopback-context');
module.exports = Self => {
// Methods
require('../methods/client/getCard')(Self);
require('../methods/client/createWithUser')(Self);
require('../methods/client/hasCustomerRole')(Self);
2020-07-01 12:45:48 +00:00
require('../methods/client/canCreateTicket')(Self);
require('../methods/client/isValidClient')(Self);
require('../methods/client/addressesPropagateRe')(Self);
require('../methods/client/getDebt')(Self);
2018-03-27 13:06:22 +00:00
require('../methods/client/getMana')(Self);
require('../methods/client/getAverageInvoiced')(Self);
require('../methods/client/summary')(Self);
2018-07-13 10:37:58 +00:00
require('../methods/client/updateFiscalData')(Self);
2018-09-28 13:00:12 +00:00
require('../methods/client/getTransactions')(Self);
require('../methods/client/confirmTransaction')(Self);
2019-04-05 10:24:39 +00:00
require('../methods/client/canBeInvoiced')(Self);
require('../methods/client/uploadFile')(Self);
2019-08-09 06:04:44 +00:00
require('../methods/client/lastActiveTickets')(Self);
2020-01-15 12:27:14 +00:00
require('../methods/client/sendSms')(Self);
2020-01-28 08:03:20 +00:00
require('../methods/client/createAddress')(Self);
require('../methods/client/updateAddress')(Self);
2020-06-12 07:18:19 +00:00
require('../methods/client/consumption')(Self);
2021-01-22 14:46:03 +00:00
require('../methods/client/createReceipt')(Self);
require('../methods/client/updatePortfolio')(Self);
require('../methods/client/checkDuplicated')(Self);
// Validations
Self.validatesPresenceOf('street', {
message: 'Street cannot be empty'
});
Self.validatesPresenceOf('city', {
message: 'City cannot be empty'
});
Self.validatesUniquenessOf('fi', {
2018-04-05 11:08:10 +00:00
message: 'TIN must be unique'
});
Self.validatesFormatOf('email', {
2018-04-05 11:20:27 +00:00
message: 'Invalid email',
allowNull: true,
allowBlank: true,
with: /^[\w|.|-]+@[\w|-]+(\.[\w|-]+)*(,[\w|.|-]+@[\w|-]+(\.[\w|-]+)*)*$/
});
Self.validatesLengthOf('postcode', {
allowNull: true,
allowBlank: true,
min: 3, max: 10
});
Self.validateAsync('socialName', socialNameIsUnique, {
message: 'The company name must be unique'
});
async function socialNameIsUnique(err, done) {
const filter = {
where: {
and: [
{socialName: this.socialName},
2022-04-25 08:30:44 +00:00
{isActive: true},
{id: {neq: this.id}}
]
}
};
const client = await Self.app.models.Client.findOne(filter);
if (client)
err();
done();
}
Self.validateAsync('iban', ibanNeedsValidation, {
message: 'The IBAN does not have the correct format'
});
2020-11-02 13:30:33 +00:00
async function ibanNeedsValidation(err, done) {
const filter = {
fields: ['code'],
where: {id: this.countryFk}
};
const country = await Self.app.models.Country.findOne(filter);
const code = country ? country.code.toLowerCase() : null;
if (code != 'es')
return done();
if (!validateIban(this.iban))
err();
done();
}
2018-03-21 11:57:23 +00:00
Self.validateAsync('fi', tinIsValid, {
2018-04-05 11:08:10 +00:00
message: 'Invalid TIN'
});
2019-03-13 07:35:25 +00:00
2018-03-21 11:57:23 +00:00
async function tinIsValid(err, done) {
if (!this.isTaxDataChecked)
return done();
const filter = {
2018-03-12 13:13:36 +00:00
fields: ['code'],
where: {id: this.countryFk}
};
const country = await Self.app.models.Country.findOne(filter);
const code = country ? country.code.toLowerCase() : null;
2018-03-12 13:13:36 +00:00
2019-03-13 07:35:25 +00:00
if (!this.fi || !validateTin(this.fi, code))
2018-03-12 13:13:36 +00:00
err();
done();
}
Self.validate('payMethod', hasSalesMan, {
2018-04-05 11:20:27 +00:00
message: 'Cannot change the payment method if no salesperson'
});
2018-10-16 11:06:58 +00:00
function hasSalesMan(err) {
if (this.payMethod && !this.salesPersonUser)
err();
}
Self.validate('isEqualizated', cannotHaveET, {
message: 'Cannot check Equalization Tax in this NIF/CIF'
});
function cannotHaveET(err) {
2019-09-19 06:58:42 +00:00
if (!this.fi) return;
const tin = this.fi.toUpperCase();
const cannotHaveET = /^[A-B]/.test(tin);
if (cannotHaveET && this.isEqualizated)
err();
}
Self.validateAsync('payMethodFk', hasIban, {
message: 'That payment method requires an IBAN'
});
function hasIban(err, done) {
Self.app.models.PayMethod.findById(this.payMethodFk, (_, instance) => {
const isMissingIban = instance && instance.isIbanRequiredForClients && !this.iban;
if (isMissingIban)
err();
done();
});
}
Self.validateAsync('bankEntityFk', hasBic, {
message: 'That payment method requires a BIC'
});
function hasBic(err, done) {
2018-11-22 14:17:35 +00:00
if (this.iban && !this.bankEntityFk)
err();
done();
}
Self.validateAsync('defaultAddressFk', isActive,
{message: 'Unable to default a disabled consignee'}
);
async function isActive(err, done) {
if (!this.defaultAddressFk)
return done();
const address = await Self.app.models.Address.findById(this.defaultAddressFk);
if (address && !address.isActive) err();
done();
}
Self.validateAsync('postCode', hasValidPostcode, {
2020-01-29 13:13:25 +00:00
message: `The postcode doesn't exist. Please enter a correct one`
});
async function hasValidPostcode(err, done) {
if (!this.postcode)
return done();
const models = Self.app.models;
const postcode = await models.Postcode.findById(this.postcode);
if (!postcode) err();
done();
}
function isAlpha(value) {
2019-10-01 13:12:02 +00:00
const regexp = new RegExp(/^[ñça-zA-Z0-9\s]*$/i);
return regexp.test(value);
}
Self.observe('before save', async ctx => {
const changes = ctx.data || ctx.instance;
const orgData = ctx.currentInstance;
const businessTypeFk = changes && changes.businessTypeFk || orgData && orgData.businessTypeFk;
const isTaxDataChecked = changes && changes.isTaxDataChecked || orgData && orgData.isTaxDataChecked;
let invalidBusinessType = false;
if (!ctx.isNewInstance) {
const isWorker = await Self.app.models.UserAccount.findById(orgData.id);
const changedFields = Object.keys(changes);
const hasChangedOtherFields = changedFields.some(key => key !== 'businessTypeFk');
if (!businessTypeFk && !isTaxDataChecked && !isWorker && !hasChangedOtherFields)
invalidBusinessType = true;
}
if (ctx.isNewInstance) {
if (!businessTypeFk && !isTaxDataChecked)
invalidBusinessType = true;
}
if (invalidBusinessType)
throw new UserError(`The type of business must be filled in basic data`);
});
Self.observe('before save', async ctx => {
const changes = ctx.data || ctx.instance;
const orgData = ctx.currentInstance;
const models = Self.app.models;
const loopBackContext = LoopBackContext.getCurrentContext();
const userId = loopBackContext.active.accessToken.userId;
const isAdministrative = await models.Account.hasRole(userId, 'administrative', ctx.options);
const isSalesAssistant = await models.Account.hasRole(userId, 'salesAssistant', ctx.options);
const hasChanges = orgData && changes;
const isTaxDataChecked = hasChanges && (changes.isTaxDataChecked || orgData.isTaxDataChecked);
const isTaxDataCheckedChanged = hasChanges && orgData.isTaxDataChecked != isTaxDataChecked;
const sageTaxType = hasChanges && (changes.sageTaxTypeFk || orgData.sageTaxTypeFk);
const sageTaxTypeChanged = hasChanges && orgData.sageTaxTypeFk != sageTaxType;
const sageTransactionType = hasChanges && (changes.sageTransactionTypeFk || orgData.sageTransactionTypeFk);
const sageTransactionTypeChanged = hasChanges && orgData.sageTransactionTypeFk != sageTransactionType;
if (isTaxDataCheckedChanged && !isAdministrative)
throw new UserError(`You don't have enough privileges`);
if ((sageTaxTypeChanged || sageTransactionTypeChanged) && !isSalesAssistant)
throw new UserError(`You don't have enough privileges`);
});
Self.observe('before save', async function(ctx) {
const changes = ctx.data || ctx.instance;
const orgData = ctx.currentInstance;
const finalState = getFinalState(ctx);
const payMethodWithIban = 4;
// Validate socialName format
const hasChanges = orgData && changes;
const socialName = changes && changes.socialName || orgData && orgData.socialName;
2020-09-30 09:19:52 +00:00
const isTaxDataChecked = hasChanges && (changes.isTaxDataChecked || orgData.isTaxDataChecked);
const socialNameChanged = hasChanges
&& orgData.socialName != socialName;
const isTaxDataCheckedChanged = hasChanges
&& orgData.isTaxDataChecked != isTaxDataChecked;
if ((socialNameChanged || isTaxDataCheckedChanged) && !isAlpha(socialName))
2022-01-05 15:23:58 +00:00
throw new UserError(`The social name has an invalid format`);
if (changes.salesPerson === null) {
changes.credit = 0;
changes.discount = 0;
changes.payMethodFk = 5; // Credit card
}
const payMethodFk = changes.payMethodFk || (orgData && orgData.payMethodFk);
const dueDay = changes.dueDay || (orgData && orgData.dueDay);
if (payMethodFk == payMethodWithIban && dueDay == 0)
changes.dueDay = 5;
if (isMultiple(ctx)) return;
if (!ctx.isNewInstance) {
const isTaxDataCheckedChanged = !orgData.isTaxDataChecked && changes.isTaxDataChecked;
if (isTaxDataCheckedChanged && !orgData.businessTypeFk)
throw new UserError(`Can't verify data unless the client has a business type`);
}
// Credit changes
if (changes.credit !== undefined)
await Self.changeCredit(ctx, finalState, changes);
2020-06-10 10:09:27 +00:00
});
2019-02-28 13:30:29 +00:00
Self.observe('after save', async ctx => {
if (ctx.isNewInstance) return;
const hookState = ctx.hookState;
const newInstance = hookState.newInstance;
const oldInstance = hookState.oldInstance;
const instance = ctx.instance;
2020-01-23 12:31:07 +00:00
const models = Self.app.models;
2019-02-28 13:30:29 +00:00
const payMethodChanged = oldInstance.payMethodFk != newInstance.payMethodFk;
const ibanChanged = oldInstance.iban != newInstance.iban;
const dueDayChanged = oldInstance.dueDay != newInstance.dueDay;
if (payMethodChanged || ibanChanged || dueDayChanged) {
2020-01-23 12:31:07 +00:00
const loopBackContext = LoopBackContext.getCurrentContext();
const httpCtx = {req: loopBackContext.active};
const httpRequest = httpCtx.req.http.req;
const $t = httpRequest.__;
const headers = httpRequest.headers;
const origin = headers.origin;
const authorization = headers.authorization;
2020-01-23 12:31:07 +00:00
const salesPersonId = instance.salesPersonFk;
if (salesPersonId) {
// Send email to client
if (instance.email) {
const worker = await models.EmailUser.findById(salesPersonId);
const params = {
authorization: authorization,
recipientId: instance.id,
recipient: instance.email,
replyTo: worker.email
};
await got.get(`${origin}/api/email/payment-update`, {
2022-01-11 12:52:01 +00:00
searchParams: params
});
}
2020-01-23 12:31:07 +00:00
const fullUrl = `${origin}/#!/client/${instance.id}/billing-data`;
const message = $t('Changed client paymethod', {
2020-01-23 12:31:07 +00:00
clientId: instance.id,
clientName: instance.name,
url: fullUrl
2019-02-28 13:30:29 +00:00
});
2020-01-23 12:31:07 +00:00
await models.Chat.sendCheckingPresence(httpCtx, salesPersonId, message);
2019-02-28 13:30:29 +00:00
}
}
2021-03-23 15:19:45 +00:00
const workerIdBefore = oldInstance.salesPersonFk;
const workerIdAfter = newInstance.salesPersonFk;
const assignmentChanged = workerIdBefore != workerIdAfter;
if (assignmentChanged)
2021-03-24 11:45:26 +00:00
await Self.notifyAssignment(instance, workerIdBefore, workerIdAfter);
2019-02-28 13:30:29 +00:00
});
2021-03-23 15:19:45 +00:00
// Send notification on client worker assignment
2021-03-24 11:45:26 +00:00
Self.notifyAssignment = async function notifyAssignment(client, previousWorkerId, currentWorkerId) {
2021-03-23 15:19:45 +00:00
const loopBackContext = LoopBackContext.getCurrentContext();
const httpCtx = {req: loopBackContext.active};
const httpRequest = httpCtx.req.http.req;
const $t = httpRequest.__;
const headers = httpRequest.headers;
const origin = headers.origin;
const models = Self.app.models;
2021-03-24 11:45:26 +00:00
let previousWorker = {name: $t('None')};
let currentWorker = {name: $t('None')};
2021-03-23 15:19:45 +00:00
if (previousWorkerId) {
const worker = await models.Worker.findById(previousWorkerId, {
include: {relation: 'user'}
});
2021-03-24 11:45:26 +00:00
previousWorker.user = worker && worker.user().name;
previousWorker.name = worker && worker.user().nickname;
2021-03-23 15:19:45 +00:00
}
if (currentWorkerId) {
const worker = await models.Worker.findById(currentWorkerId, {
include: {relation: 'user'}
});
2021-03-24 11:45:26 +00:00
currentWorker.user = worker && worker.user().name;
currentWorker.name = worker && worker.user().nickname;
2021-03-23 15:19:45 +00:00
}
const fullUrl = `${origin}/#!/client/${client.id}/basic-data`;
const message = $t('Client assignment has changed', {
clientId: client.id,
clientName: client.name,
url: fullUrl,
2021-03-24 11:45:26 +00:00
previousWorkerName: previousWorker.name,
currentWorkerName: currentWorker.name
2021-03-23 15:19:45 +00:00
});
2021-03-24 11:45:26 +00:00
if (previousWorkerId)
await models.Chat.send(httpCtx, `@${previousWorker.user}`, message);
2021-03-23 15:19:45 +00:00
2021-03-24 11:45:26 +00:00
if (currentWorkerId)
await models.Chat.send(httpCtx, `@${currentWorker.user}`, message);
};
2021-03-23 15:19:45 +00:00
// Credit change validations
Self.changeCredit = async function changeCredit(ctx, finalState, changes) {
const models = Self.app.models;
const userId = ctx.options.accessToken.userId;
const isManager = await models.Account.hasRole(userId, 'manager', ctx.options);
if (!isManager) {
const lastCredit = await models.ClientCredit.findOne({
where: {
clientFk: finalState.id
},
order: 'id DESC'
}, ctx.options);
const lastAmount = lastCredit && lastCredit.amount;
const lastWorkerId = lastCredit && lastCredit.workerFk;
const lastWorkerIsManager = await models.Account.hasRole(lastWorkerId, 'manager', ctx.options);
if (lastAmount == 0 && lastWorkerIsManager)
throw new UserError(`You can't change the credit set to zero from a manager`);
const creditLimits = await models.ClientCreditLimit.find({
fields: ['roleFk'],
where: {
maxAmount: {gte: changes.credit}
}
}, ctx.options);
const requiredRoles = [];
for (limit of creditLimits)
requiredRoles.push(limit.roleFk);
const userRequiredRoles = await models.RoleMapping.count({
roleId: {inq: requiredRoles},
principalType: 'USER',
principalId: userId
}, ctx.options);
if (userRequiredRoles <= 0)
throw new UserError(`You don't have enough privileges to set this credit amount`);
}
await models.ClientCredit.create({
amount: changes.credit,
clientFk: finalState.id,
workerFk: userId
}, ctx.options);
};
2021-03-23 15:19:45 +00:00
const app = require('vn-loopback/server/server');
app.on('started', function() {
let account = app.models.Account;
account.observe('before save', async ctx => {
if (ctx.isNewInstance) return;
ctx.hookState.oldInstance = JSON.parse(JSON.stringify(ctx.currentInstance));
});
account.observe('after save', async ctx => {
let changes = ctx.data || ctx.instance;
if (!ctx.isNewInstance && changes) {
let oldData = ctx.hookState.oldInstance;
let hasChanges = oldData.name != changes.name || oldData.active != changes.active;
if (!hasChanges) return;
let userId = ctx.options.accessToken.userId;
let logRecord = {
originFk: oldData.id,
userFk: userId,
action: 'update',
changedModel: 'Account',
oldInstance: {name: oldData.name, active: oldData.active},
newInstance: {name: changes.name, active: changes.active}
};
await Self.app.models.ClientLog.create(logRecord);
}
});
});
};