refactor(ticket): ticket endpoins now can receive transactions

This commit is contained in:
Carlos Jimenez Ruiz 2021-09-21 18:48:59 +02:00
parent ca00e3611c
commit cc90a08f03
24 changed files with 1074 additions and 732 deletions

View File

@ -1,15 +1,15 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
describe('SalesMonitor salesFilter()', () => { describe('SalesMonitor salesFilter()', () => {
it('should return the tickets matching the filter', async() => { it('should now return the tickets matching the filter', async() => {
const ctx = {req: {accessToken: {userId: 9}}, args: {}}; const ctx = {req: {accessToken: {userId: 9}}, args: {}};
const filter = {order: 'id DESC'}; const filter = {order: 'id DESC'};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
expect(result.length).toEqual(24); expect(result.length).toEqual(24);
}); });
it('should return the tickets matching the problems on true', async() => { it('should now return the tickets matching the problems on true', async() => {
const yesterday = new Date(); const yesterday = new Date();
yesterday.setHours(0, 0, 0, 0); yesterday.setHours(0, 0, 0, 0);
const today = new Date(); const today = new Date();
@ -21,12 +21,12 @@ describe('SalesMonitor salesFilter()', () => {
to: today to: today
}}; }};
const filter = {}; const filter = {};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
expect(result.length).toEqual(9); expect(result.length).toEqual(9);
}); });
it('should return the tickets matching the problems on false', async() => { it('should now return the tickets matching the problems on false', async() => {
const yesterday = new Date(); const yesterday = new Date();
yesterday.setDate(yesterday.getDate() - 1); yesterday.setDate(yesterday.getDate() - 1);
yesterday.setHours(0, 0, 0, 0); yesterday.setHours(0, 0, 0, 0);
@ -39,33 +39,33 @@ describe('SalesMonitor salesFilter()', () => {
to: today to: today
}}; }};
const filter = {}; const filter = {};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
expect(result.length).toEqual(0); expect(result.length).toEqual(0);
}); });
it('should return the tickets matching the problems on null', async() => { it('should now return the tickets matching the problems on null', async() => {
const ctx = {req: {accessToken: {userId: 9}}, args: {problems: null}}; const ctx = {req: {accessToken: {userId: 9}}, args: {problems: null}};
const filter = {}; const filter = {};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
expect(result.length).toEqual(24); expect(result.length).toEqual(24);
}); });
it('should return the tickets matching the orderId 11', async() => { it('should now return the tickets matching the orderId 11', async() => {
const ctx = {req: {accessToken: {userId: 9}}, args: {orderFk: 11}}; const ctx = {req: {accessToken: {userId: 9}}, args: {orderFk: 11}};
const filter = {}; const filter = {};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
const firstRow = result[0]; const firstRow = result[0];
expect(result.length).toEqual(1); expect(result.length).toEqual(1);
expect(firstRow.id).toEqual(11); expect(firstRow.id).toEqual(11);
}); });
it('should return the tickets with grouped state "Pending" and not "Ok" nor "BOARDING"', async() => { it('should now return the tickets with grouped state "Pending" and not "Ok" nor "BOARDING"', async() => {
const ctx = {req: {accessToken: {userId: 9}}, args: {pending: true}}; const ctx = {req: {accessToken: {userId: 9}}, args: {pending: true}};
const filter = {}; const filter = {};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
const length = result.length; const length = result.length;
const anyResult = result[Math.floor(Math.random() * Math.floor(length))]; const anyResult = result[Math.floor(Math.random() * Math.floor(length))];
@ -74,10 +74,10 @@ describe('SalesMonitor salesFilter()', () => {
expect(anyResult.state).toMatch(/(Libre|Arreglar)/); expect(anyResult.state).toMatch(/(Libre|Arreglar)/);
}); });
it('should return the tickets that are not pending', async() => { it('should now return the tickets that are not pending', async() => {
const ctx = {req: {accessToken: {userId: 9}}, args: {pending: false}}; const ctx = {req: {accessToken: {userId: 9}}, args: {pending: false}};
const filter = {}; const filter = {};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
const firstRow = result[0]; const firstRow = result[0];
const secondRow = result[1]; const secondRow = result[1];
const thirdRow = result[2]; const thirdRow = result[2];
@ -88,18 +88,18 @@ describe('SalesMonitor salesFilter()', () => {
expect(thirdRow.state).toEqual('Entregado'); expect(thirdRow.state).toEqual('Entregado');
}); });
it('should return the tickets from the worker team', async() => { it('should now return the tickets from the worker team', async() => {
const ctx = {req: {accessToken: {userId: 18}}, args: {myTeam: true}}; const ctx = {req: {accessToken: {userId: 18}}, args: {myTeam: true}};
const filter = {}; const filter = {};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
expect(result.length).toEqual(20); expect(result.length).toEqual(20);
}); });
it('should return the tickets that are not from the worker team', async() => { it('should now return the tickets that are not from the worker team', async() => {
const ctx = {req: {accessToken: {userId: 18}}, args: {myTeam: false}}; const ctx = {req: {accessToken: {userId: 18}}, args: {myTeam: false}};
const filter = {}; const filter = {};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
expect(result.length).toEqual(4); expect(result.length).toEqual(4);
}); });
@ -113,7 +113,7 @@ describe('SalesMonitor salesFilter()', () => {
const ctx = {req: {accessToken: {userId: 18}}, args: {}}; const ctx = {req: {accessToken: {userId: 18}}, args: {}};
const filter = {order: 'totalProblems DESC'}; const filter = {order: 'totalProblems DESC'};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
const firstTicket = result.shift(); const firstTicket = result.shift();
const secondTicket = result.shift(); const secondTicket = result.shift();
@ -131,7 +131,7 @@ describe('SalesMonitor salesFilter()', () => {
const ctx = {req: {accessToken: {userId: 18}}, args: {}}; const ctx = {req: {accessToken: {userId: 18}}, args: {}};
const filter = {order: 'totalProblems ASC'}; const filter = {order: 'totalProblems ASC'};
const result = await app.models.SalesMonitor.salesFilter(ctx, filter); const result = await models.SalesMonitor.salesFilter(ctx, filter);
const firstTicket = result.shift(); const firstTicket = result.shift();
const secondTicket = result.shift(); const secondTicket = result.shift();

View File

@ -18,19 +18,23 @@ module.exports = Self => {
} }
}); });
Self.isEditable = async(ctx, stateId) => { Self.isEditable = async(ctx, stateId, options) => {
const accessToken = ctx.req.accessToken; const accessToken = ctx.req.accessToken;
const models = Self.app.models; const models = Self.app.models;
const userId = accessToken.userId; const userId = accessToken.userId;
const myOptions = {};
let isProduction = await models.Account.hasRole(userId, 'production'); if (typeof options == 'object')
let isSalesPerson = await models.Account.hasRole(userId, 'salesPerson'); Object.assign(myOptions, options);
let isAdministrative = await models.Account.hasRole(userId, 'administrative');
let state = await models.State.findById(stateId);
let salesPersonAllowed = (isSalesPerson && (state.code == 'PICKER_DESIGNED' || state.code == 'PRINTED')); const isProduction = await models.Account.hasRole(userId, 'production', myOptions);
const isSalesPerson = await models.Account.hasRole(userId, 'salesPerson', myOptions);
const isAdministrative = await models.Account.hasRole(userId, 'administrative', myOptions);
const state = await models.State.findById(stateId, null, myOptions);
let isAllowed = isProduction || isAdministrative || salesPersonAllowed || state.alertLevel == 0; const salesPersonAllowed = (isSalesPerson && (state.code == 'PICKER_DESIGNED' || state.code == 'PRINTED'));
const isAllowed = isProduction || isAdministrative || salesPersonAllowed || state.alertLevel == 0;
return isAllowed; return isAllowed;
}; };
}; };

View File

@ -1,61 +1,127 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
describe('state isEditable()', () => { describe('state isEditable()', () => {
it('should return false if the state is not editable by a specific role', async() => { it('should return false if the state is not editable by a specific role', async() => {
const tx = await models.State.beginTransaction({});
try {
const options = {transaction: tx};
const salesPersonRole = 18; const salesPersonRole = 18;
const onDeliveryState = 13; const onDeliveryState = 13;
let ctx = {req: {accessToken: {userId: salesPersonRole}}}; const ctx = {req: {accessToken: {userId: salesPersonRole}}};
let result = await app.models.State.isEditable(ctx, onDeliveryState); const result = await models.State.isEditable(ctx, onDeliveryState, options);
expect(result).toBe(false); expect(result).toBe(false);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should return true if the state is editable by a specific role', async() => { it('should return true if the state is editable by a specific role', async() => {
const tx = await models.State.beginTransaction({});
try {
const options = {transaction: tx};
const salesPersonRole = 18; const salesPersonRole = 18;
const asignedState = 20; const asignedState = 20;
let ctx = {req: {accessToken: {userId: salesPersonRole}}}; const ctx = {req: {accessToken: {userId: salesPersonRole}}};
let result = await app.models.State.isEditable(ctx, asignedState); const result = await models.State.isEditable(ctx, asignedState, options);
expect(result).toBe(true); expect(result).toBe(true);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should return true again if the state is editable by a specific role', async() => { it('should return true again if the state is editable by a specific role', async() => {
const tx = await models.State.beginTransaction({});
try {
const options = {transaction: tx};
const employeeRole = 1; const employeeRole = 1;
const fixingState = 1; const fixingState = 1;
let ctx = {req: {accessToken: {userId: employeeRole}}}; const ctx = {req: {accessToken: {userId: employeeRole}}};
let result = await app.models.State.isEditable(ctx, fixingState); const result = await models.State.isEditable(ctx, fixingState, options);
expect(result).toBe(true); expect(result).toBe(true);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should return false if the state is not editable for the given role', async() => { it('should return false if the state is not editable for the given role', async() => {
const tx = await models.State.beginTransaction({});
try {
const options = {transaction: tx};
const employeeRole = 1; const employeeRole = 1;
const asignedState = 13; const asignedState = 13;
let ctx = {req: {accessToken: {userId: employeeRole}}}; const ctx = {req: {accessToken: {userId: employeeRole}}};
let result = await app.models.State.isEditable(ctx, asignedState); const result = await models.State.isEditable(ctx, asignedState, options);
expect(result).toBe(false); expect(result).toBe(false);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should return true if the state is editable for the given role', async() => { it('should return true if the state is editable for the given role', async() => {
const tx = await models.State.beginTransaction({});
try {
const options = {transaction: tx};
const productionRole = 49; const productionRole = 49;
const onDeliveryState = 13; const onDeliveryState = 13;
let ctx = {req: {accessToken: {userId: productionRole}}}; const ctx = {req: {accessToken: {userId: productionRole}}};
let result = await app.models.State.isEditable(ctx, onDeliveryState); const result = await models.State.isEditable(ctx, onDeliveryState, options);
expect(result).toBe(true); expect(result).toBe(true);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should return true if the ticket is editable, the role is salesPerson and the ticket state is printed', async() => { it('should return true if the ticket is editable, the role is salesPerson and the ticket state is printed', async() => {
const tx = await models.State.beginTransaction({});
try {
const options = {transaction: tx};
const salesPersonRole = 18; const salesPersonRole = 18;
const printedState = 4; const printedState = 4;
const okState = 3; const okState = 3;
const ctx = {req: {accessToken: {userId: salesPersonRole}}}; const ctx = {req: {accessToken: {userId: salesPersonRole}}};
let canEditCurrent = await app.models.State.isEditable(ctx, printedState); const canEditCurrent = await models.State.isEditable(ctx, printedState, options);
let canAsignNew = await app.models.State.isEditable(ctx, okState); const canAsignNew = await models.State.isEditable(ctx, okState, options);
let result = canEditCurrent && canAsignNew; const result = canEditCurrent && canAsignNew;
expect(result).toBe(true); expect(result).toBe(true);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
}); });

View File

@ -23,38 +23,64 @@ module.exports = Self => {
} }
}); });
Self.changeState = async(ctx, params) => { Self.changeState = async(ctx, params, options) => {
let userId = ctx.req.accessToken.userId; const models = Self.app.models;
let models = Self.app.models; const myOptions = {};
let tx;
if (typeof options == 'object')
Object.assign(myOptions, options);
if (!myOptions.transaction) {
tx = await Self.beginTransaction({});
myOptions.transaction = tx;
}
try {
const userId = ctx.req.accessToken.userId;
if (!params.stateFk && !params.code) if (!params.stateFk && !params.code)
throw new UserError('State cannot be blank'); throw new UserError('State cannot be blank');
if (params.code) { if (params.code) {
let state = await models.State.findOne({where: {code: params.code}, fields: ['id']}); const state = await models.State.findOne({
where: {code: params.code},
fields: ['id']
}, myOptions);
params.stateFk = state.id; params.stateFk = state.id;
} }
if (!params.workerFk) { if (!params.workerFk) {
let worker = await models.Worker.findOne({where: {userFk: userId}}); const worker = await models.Worker.findOne({
where: {userFk: userId}
}, myOptions);
params.workerFk = worker.id; params.workerFk = worker.id;
} }
let ticketState = await models.TicketState.findById( const ticketState = await models.TicketState.findById(params.ticketFk, {
params.ticketFk, fields: ['stateFk']
{fields: ['stateFk']} }, myOptions);
);
let oldStateAllowed; let oldStateAllowed;
if (ticketState) if (ticketState)
oldStateAllowed = await models.State.isEditable(ctx, ticketState.stateFk); oldStateAllowed = await models.State.isEditable(ctx, ticketState.stateFk, myOptions);
let newStateAllowed = await models.State.isEditable(ctx, params.stateFk); const newStateAllowed = await models.State.isEditable(ctx, params.stateFk, myOptions);
let isAllowed = (!ticketState || oldStateAllowed == true) && newStateAllowed == true; const isAllowed = (!ticketState || oldStateAllowed == true) && newStateAllowed == true;
if (!isAllowed) if (!isAllowed)
throw new UserError(`You don't have enough privileges`, 'ACCESS_DENIED'); throw new UserError(`You don't have enough privileges`, 'ACCESS_DENIED');
return models.TicketTracking.create(params); const ticketTracking = await models.TicketTracking.create(params, myOptions);
if (tx) await tx.commit();
return ticketTracking;
} catch (e) {
if (tx) await tx.rollback();
throw e;
}
}; };
}; };

View File

@ -1,15 +1,34 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
const LoopBackContext = require('loopback-context'); const LoopBackContext = require('loopback-context');
describe('ticket changeState()', () => { describe('ticket changeState()', () => {
const salesPersonId = 18; const salesPersonId = 18;
const employeeId = 1; const employeeId = 1;
const productionId = 49; const productionId = 49;
let activeCtx = { const activeCtx = {
accessToken: {userId: 9}, accessToken: {userId: 9},
}; };
let ctx = {req: activeCtx}; const ctx = {req: activeCtx};
let ticket; const now = new Date();
const sampleTicket = {
shipped: now,
landed: now,
nickname: 'Many Places',
packages: 0,
updated: now,
priority: 1,
zoneFk: 3,
zonePrice: 5,
zoneBonus: 1,
totalWithVat: 120,
totalWithoutVat: 100,
clientFk: 1106,
warehouseFk: 1,
addressFk: 126,
routeFk: 6,
companyFk: 442,
agencyModeFk: 7
};
beforeAll(async done => { beforeAll(async done => {
spyOn(LoopBackContext, 'getCurrentContext').and.returnValue({ spyOn(LoopBackContext, 'getCurrentContext').and.returnValue({
@ -19,92 +38,98 @@ describe('ticket changeState()', () => {
done(); done();
}); });
beforeEach(async done => {
try {
let originalTicket = await app.models.Ticket.findOne({where: {id: 16}});
originalTicket.id = null;
ticket = await app.models.Ticket.create(originalTicket);
} catch (error) {
console.error(error);
}
done();
});
afterEach(async done => {
try {
await app.models.Ticket.destroyById(ticket.id);
} catch (error) {
console.error(error);
}
done();
});
afterAll(async done => {
try {
await app.models.Ticket.destroyById(ticket.id);
} catch (error) {
console.error(error);
}
done();
});
it('should throw if the ticket is not editable and the user isnt production', async() => { it('should throw if the ticket is not editable and the user isnt production', async() => {
activeCtx.accessToken.userId = salesPersonId; const tx = await models.TicketTracking.beginTransaction({});
let params = {ticketFk: 2, stateFk: 3};
let error;
let errCode;
try { try {
await app.models.TicketTracking.changeState(ctx, params); const options = {transaction: tx};
activeCtx.accessToken.userId = salesPersonId;
const params = {ticketFk: 2, stateFk: 3};
await models.TicketTracking.changeState(ctx, params, options);
await tx.rollback();
} catch (e) { } catch (e) {
errCode = e.code; await tx.rollback();
error = e;
} }
expect(errCode).toBe('ACCESS_DENIED'); expect(error.code).toBe('ACCESS_DENIED');
}); });
it('should throw an error if a worker with employee role attemps to a forbidden state', async() => { it('should throw an error if a worker with employee role attemps to a forbidden state', async() => {
activeCtx.accessToken.userId = employeeId; const tx = await models.TicketTracking.beginTransaction({});
let params = {ticketFk: 11, stateFk: 13};
let error;
let errCode;
try { try {
await app.models.TicketTracking.changeState(ctx, params); const options = {transaction: tx};
activeCtx.accessToken.userId = employeeId;
const params = {ticketFk: 11, stateFk: 13};
await models.TicketTracking.changeState(ctx, params, options);
await tx.rollback();
} catch (e) { } catch (e) {
errCode = e.code; await tx.rollback();
error = e;
} }
expect(errCode).toBe('ACCESS_DENIED'); expect(error.code).toBe('ACCESS_DENIED');
}); });
it('should be able to create a ticket tracking line for a not editable ticket if the user has the production role', async() => { it('should be able to create a ticket tracking line for a not editable ticket if the user has the production role', async() => {
activeCtx.accessToken.userId = productionId; const tx = await models.TicketTracking.beginTransaction({});
let params = {ticketFk: ticket.id, stateFk: 3};
let ticketTracking = await app.models.TicketTracking.changeState(ctx, params); try {
const options = {transaction: tx};
const ticket = await models.Ticket.create(sampleTicket, options);
activeCtx.accessToken.userId = productionId;
const params = {ticketFk: ticket.id, stateFk: 3};
const ticketTracking = await models.TicketTracking.changeState(ctx, params, options);
expect(ticketTracking.__data.ticketFk).toBe(params.ticketFk); expect(ticketTracking.__data.ticketFk).toBe(params.ticketFk);
expect(ticketTracking.__data.stateFk).toBe(params.stateFk); expect(ticketTracking.__data.stateFk).toBe(params.stateFk);
expect(ticketTracking.__data.workerFk).toBe(49); expect(ticketTracking.__data.workerFk).toBe(49);
expect(ticketTracking.__data.id).toBeDefined(); expect(ticketTracking.__data.id).toBeDefined();
// restores await tx.rollback();
await app.models.TicketTracking.destroyById(ticketTracking.__data.id); } catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should update the ticket tracking line when the user is salesperson, uses the state assigned and a valid worker id', async() => { it('should update the ticket tracking line when the user is salesperson, uses the state assigned and a valid worker id', async() => {
let ctx = {req: {accessToken: {userId: 18}}}; const tx = await models.TicketTracking.beginTransaction({});
let assignedState = await app.models.State.findOne({where: {code: 'PICKER_DESIGNED'}});
let params = {ticketFk: ticket.id, stateFk: assignedState.id, workerFk: 1}; try {
let res = await app.models.TicketTracking.changeState(ctx, params); const options = {transaction: tx};
const ticket = await models.Ticket.create(sampleTicket, options);
const ctx = {req: {accessToken: {userId: 18}}};
const assignedState = await models.State.findOne({where: {code: 'PICKER_DESIGNED'}}, options);
const params = {ticketFk: ticket.id, stateFk: assignedState.id, workerFk: 1};
const res = await models.TicketTracking.changeState(ctx, params, options);
expect(res.__data.ticketFk).toBe(params.ticketFk); expect(res.__data.ticketFk).toBe(params.ticketFk);
expect(res.__data.stateFk).toBe(params.stateFk); expect(res.__data.stateFk).toBe(params.stateFk);
expect(res.__data.workerFk).toBe(params.workerFk); expect(res.__data.workerFk).toBe(params.workerFk);
expect(res.__data.workerFk).toBe(1); expect(res.__data.workerFk).toBe(1);
expect(res.__data.id).toBeDefined(); expect(res.__data.id).toBeDefined();
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
}); });

View File

@ -12,12 +12,12 @@ module.exports = Self => {
}, },
{ {
arg: 'shipped', arg: 'shipped',
type: 'Date', type: 'date',
description: `The shipment date filter` description: `The shipment date filter`
}, },
{ {
arg: 'landed', arg: 'landed',
type: 'Date', type: 'date',
description: `The landing date filter` description: `The landing date filter`
}, },
{ {
@ -142,7 +142,7 @@ module.exports = Self => {
if (tx) await tx.commit(); if (tx) await tx.commit();
return await ticket; return ticket;
} catch (e) { } catch (e) {
if (tx) await tx.rollback(); if (tx) await tx.rollback();
throw e; throw e;

View File

@ -11,7 +11,7 @@ module.exports = Self => {
http: {source: 'path'} http: {source: 'path'}
}], }],
returns: { returns: {
type: 'Number', type: 'number',
root: true root: true
}, },
http: { http: {
@ -20,12 +20,32 @@ module.exports = Self => {
} }
}); });
Self.recalculateComponents = async(ctx, id) => { Self.recalculateComponents = async(ctx, id, options) => {
const isEditable = await Self.isEditable(ctx, id); const myOptions = {};
let tx;
if (typeof options == 'object')
Object.assign(myOptions, options);
if (!myOptions.transaction) {
tx = await Self.beginTransaction({});
myOptions.transaction = tx;
}
try {
const isEditable = await Self.isEditable(ctx, id, myOptions);
if (!isEditable) if (!isEditable)
throw new UserError(`The current ticket can't be modified`); throw new UserError(`The current ticket can't be modified`);
return Self.rawSql('CALL vn.ticket_recalcComponents(?, NULL)', [id]); const recalculation = await Self.rawSql('CALL vn.ticket_recalcComponents(?, NULL)', [id], myOptions);
if (tx) await tx.commit();
return recalculation;
} catch (e) {
if (tx) await tx.rollback();
throw e;
}
}; };
}; };

View File

@ -6,7 +6,7 @@ module.exports = Self => {
accessType: 'WRITE', accessType: 'WRITE',
accepts: [{ accepts: [{
arg: 'id', arg: 'id',
type: 'Number', type: 'number',
required: true, required: true,
description: 'The ticket id', description: 'The ticket id',
http: {source: 'path'} http: {source: 'path'}

View File

@ -5,23 +5,23 @@ module.exports = Self => {
accessType: 'WRITE', accessType: 'WRITE',
accepts: [{ accepts: [{
arg: 'id', arg: 'id',
type: 'Number', type: 'number',
required: true, required: true,
description: 'The ticket id', description: 'The ticket id',
http: {source: 'path'} http: {source: 'path'}
}, },
{ {
arg: 'destination', arg: 'destination',
type: 'String', type: 'string',
required: true, required: true,
}, },
{ {
arg: 'message', arg: 'message',
type: 'String', type: 'string',
required: true, required: true,
}], }],
returns: { returns: {
type: 'Object', type: 'object',
root: true root: true
}, },
http: { http: {
@ -30,11 +30,23 @@ module.exports = Self => {
} }
}); });
Self.sendSms = async(ctx, id, destination, message) => { Self.sendSms = async(ctx, id, destination, message, options) => {
const myOptions = {};
let tx;
if (typeof options == 'object')
Object.assign(myOptions, options);
if (!myOptions.transaction) {
tx = await Self.beginTransaction({});
myOptions.transaction = tx;
}
const userId = ctx.req.accessToken.userId; const userId = ctx.req.accessToken.userId;
let sms = await Self.app.models.Sms.send(ctx, id, destination, message); try {
let logRecord = { const sms = await Self.app.models.Sms.send(ctx, id, destination, message);
const logRecord = {
originFk: id, originFk: id,
userFk: userId, userFk: userId,
action: 'insert', action: 'insert',
@ -48,10 +60,16 @@ module.exports = Self => {
} }
}; };
const ticketLog = await Self.app.models.TicketLog.create(logRecord); const ticketLog = await Self.app.models.TicketLog.create(logRecord, myOptions);
sms.logId = ticketLog.id; sms.logId = ticketLog.id;
if (tx) await tx.commit();
return sms; return sms;
} catch (e) {
if (tx) await tx.rollback();
throw e;
}
}; };
}; };

View File

@ -6,7 +6,7 @@ module.exports = Self => {
accessType: 'WRITE', accessType: 'WRITE',
accepts: [{ accepts: [{
arg: 'id', arg: 'id',
type: 'Number', type: 'number',
required: true, required: true,
description: 'The ticket id', description: 'The ticket id',
http: {source: 'path'} http: {source: 'path'}
@ -21,21 +21,33 @@ module.exports = Self => {
} }
}); });
Self.setDeleted = async(ctx, id) => { Self.setDeleted = async(ctx, id, options) => {
const models = Self.app.models; const models = Self.app.models;
const userId = ctx.req.accessToken.userId;
const isEditable = await Self.isEditable(ctx, id);
const $t = ctx.req.__; // $translate const $t = ctx.req.__; // $translate
const myOptions = {};
let tx;
if (typeof options == 'object')
Object.assign(myOptions, options);
if (!myOptions.transaction) {
tx = await Self.beginTransaction({});
myOptions.transaction = tx;
}
try {
const userId = ctx.req.accessToken.userId;
const isEditable = await Self.isEditable(ctx, id, myOptions);
if (!isEditable) if (!isEditable)
throw new UserError(`The sales of this ticket can't be modified`); throw new UserError(`The sales of this ticket can't be modified`);
// Check if has sales with shelving // Check if has sales with shelving
const isSalesAssistant = await models.Account.hasRole(userId, 'salesAssistant'); const isSalesAssistant = await models.Account.hasRole(userId, 'salesAssistant', myOptions);
const sales = await models.Sale.find({ const sales = await models.Sale.find({
include: {relation: 'itemShelvingSale'}, include: {relation: 'itemShelvingSale'},
where: {ticketFk: id} where: {ticketFk: id}
}); }, myOptions);
const hasItemShelvingSales = sales.some(sale => { const hasItemShelvingSales = sales.some(sale => {
return sale.itemShelvingSale(); return sale.itemShelvingSale();
}); });
@ -44,7 +56,7 @@ module.exports = Self => {
throw new UserError(`You cannot delete a ticket that part of it is being prepared`); throw new UserError(`You cannot delete a ticket that part of it is being prepared`);
// Check for existing claim // Check for existing claim
const claimOfATicket = await models.Claim.findOne({where: {ticketFk: id}}); const claimOfATicket = await models.Claim.findOne({where: {ticketFk: id}}, myOptions);
if (claimOfATicket) if (claimOfATicket)
throw new UserError('You must delete the claim id %d first', 'DELETE_CLAIM_FIRST', claimOfATicket.id); throw new UserError('You must delete the claim id %d first', 'DELETE_CLAIM_FIRST', claimOfATicket.id);
@ -52,7 +64,7 @@ module.exports = Self => {
const hasPurchaseRequests = await models.TicketRequest.count({ const hasPurchaseRequests = await models.TicketRequest.count({
ticketFk: id, ticketFk: id,
isOk: true isOk: true
}); }, myOptions);
if (hasPurchaseRequests) if (hasPurchaseRequests)
throw new UserError('You must delete all the buy requests first'); throw new UserError('You must delete all the buy requests first');
@ -63,7 +75,7 @@ module.exports = Self => {
for (let sale of sales) { for (let sale of sales) {
if (sale.itemShelvingSale()) { if (sale.itemShelvingSale()) {
const itemShelvingSale = sale.itemShelvingSale(); const itemShelvingSale = sale.itemShelvingSale();
const destroyedShelving = models.ItemShelvingSale.destroyById(itemShelvingSale.id); const destroyedShelving = models.ItemShelvingSale.destroyById(itemShelvingSale.id, myOptions);
promises.push(destroyedShelving); promises.push(destroyedShelving);
} }
} }
@ -71,15 +83,15 @@ module.exports = Self => {
} }
// Remove ticket greuges // Remove ticket greuges
const ticketGreuges = await models.Greuge.find({where: {ticketFk: id}}); const ticketGreuges = await models.Greuge.find({where: {ticketFk: id}}, myOptions);
const ownGreuges = ticketGreuges.every(greuge => { const ownGreuges = ticketGreuges.every(greuge => {
return greuge.ticketFk == id; return greuge.ticketFk == id;
}); });
if (ownGreuges) { if (ownGreuges) {
for (const greuge of ticketGreuges) { for (const greuge of ticketGreuges) {
const instance = await models.Greuge.findById(greuge.id); const instance = await models.Greuge.findById(greuge.id, null, myOptions);
await instance.destroy(); await instance.destroy(myOptions);
} }
} }
@ -100,7 +112,7 @@ module.exports = Self => {
}, { }, {
relation: 'stowaway' relation: 'stowaway'
}] }]
}); }, myOptions);
// Change state to "fixing" if contains an stowaway and remove the link between them // Change state to "fixing" if contains an stowaway and remove the link between them
let otherTicketId; let otherTicketId;
@ -110,11 +122,11 @@ module.exports = Self => {
otherTicketId = ticket.ship().id; otherTicketId = ticket.ship().id;
if (otherTicketId) { if (otherTicketId) {
await models.Ticket.deleteStowaway(ctx, otherTicketId); await models.Ticket.deleteStowaway(ctx, otherTicketId, myOptions);
await models.TicketTracking.changeState(ctx, { await models.TicketTracking.changeState(ctx, {
ticketFk: otherTicketId, ticketFk: otherTicketId,
code: 'FIXING' code: 'FIXING'
}); }, myOptions);
} }
// Send notification to salesPerson // Send notification to salesPerson
@ -128,6 +140,14 @@ module.exports = Self => {
await models.Chat.send(ctx, `@${salesPersonUser.name}`, message); await models.Chat.send(ctx, `@${salesPersonUser.name}`, message);
} }
return ticket.updateAttribute('isDeleted', true); const updatedTicket = await ticket.updateAttribute('isDeleted', true, myOptions);
if (tx) await tx.commit();
return updatedTicket;
} catch (e) {
if (tx) await tx.rollback();
throw e;
}
}; };
}; };

View File

@ -1,24 +1,43 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
describe('ticket recalculateComponents()', () => { describe('ticket recalculateComponents()', () => {
const ticketId = 11; const ticketId = 11;
it('should update the ticket components', async() => { it('should update the ticket components', async() => {
const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
const ctx = {req: {accessToken: {userId: 9}}}; const ctx = {req: {accessToken: {userId: 9}}};
const response = await app.models.Ticket.recalculateComponents(ctx, ticketId); const response = await models.Ticket.recalculateComponents(ctx, ticketId, options);
expect(response.affectedRows).toBeDefined(); expect(response.affectedRows).toBeDefined();
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should throw an error if the ticket is not editable', async() => { it('should throw an error if the ticket is not editable', async() => {
const tx = await models.Ticket.beginTransaction({});
let error;
try {
const options = {transaction: tx};
const ctx = {req: {accessToken: {userId: 9}}}; const ctx = {req: {accessToken: {userId: 9}}};
const immutableTicketId = 1; const immutableTicketId = 1;
await app.models.Ticket.recalculateComponents(ctx, immutableTicketId) await models.Ticket.recalculateComponents(ctx, immutableTicketId, options);
.catch(response => {
expect(response).toEqual(new Error(`The current ticket can't be modified`));
error = response;
});
expect(error).toBeDefined(); await tx.rollback();
} catch (e) {
await tx.rollback();
error = e;
}
expect(error).toEqual(new Error(`The current ticket can't be modified`));
}); });
}); });

View File

@ -1,29 +1,30 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
const soap = require('soap'); const soap = require('soap');
describe('ticket sendSms()', () => { describe('ticket sendSms()', () => {
let logId;
afterAll(async done => {
await app.models.TicketLog.destroyById(logId);
done();
});
it('should send a message and log it', async() => { it('should send a message and log it', async() => {
const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
spyOn(soap, 'createClientAsync').and.returnValue('a so fake client'); spyOn(soap, 'createClientAsync').and.returnValue('a so fake client');
let ctx = {req: {accessToken: {userId: 9}}}; const ctx = {req: {accessToken: {userId: 9}}};
let id = 11; const id = 11;
let destination = 222222222; const destination = 222222222;
let message = 'this is the message created in a test'; const message = 'this is the message created in a test';
let sms = await app.models.Ticket.sendSms(ctx, id, destination, message); const sms = await models.Ticket.sendSms(ctx, id, destination, message, options);
logId = sms.logId; const createdLog = await models.TicketLog.findById(sms.logId, null, options);
const json = JSON.parse(JSON.stringify(createdLog.newInstance));
let createdLog = await app.models.TicketLog.findById(logId);
let json = JSON.parse(JSON.stringify(createdLog.newInstance));
expect(json.message).toEqual(message); expect(json.message).toEqual(message);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
}); });

View File

@ -1,6 +1,5 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
const LoopBackContext = require('loopback-context'); const LoopBackContext = require('loopback-context');
const models = app.models;
describe('ticket setDeleted()', () => { describe('ticket setDeleted()', () => {
const userId = 1106; const userId = 1106;
@ -10,21 +9,32 @@ describe('ticket setDeleted()', () => {
}; };
it('should throw an error if the given ticket has a claim', async() => { it('should throw an error if the given ticket has a claim', async() => {
const tx = await models.Ticket.beginTransaction({});
let error;
try {
const options = {transaction: tx};
const ctx = {req: activeCtx}; const ctx = {req: activeCtx};
const ticketId = 16; const ticketId = 16;
let error;
try { await models.Ticket.setDeleted(ctx, ticketId, options);
await app.models.Ticket.setDeleted(ctx, ticketId);
await tx.rollback();
} catch (e) { } catch (e) {
await tx.rollback();
error = e; error = e;
} }
expect(error.translateArgs[0]).toEqual(2);
expect(error.message).toEqual('You must delete the claim id %d first'); expect(error.message).toEqual('You must delete the claim id %d first');
}); });
it('should delete the ticket, remove the stowaway link and change the stowaway ticket state to "FIXING" and get rid of the itemshelving', async() => { it('should delete the ticket, remove the stowaway link and change the stowaway ticket state to "FIXING" and get rid of the itemshelving', async() => {
const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
spyOn(LoopBackContext, 'getCurrentContext').and.returnValue({ spyOn(LoopBackContext, 'getCurrentContext').and.returnValue({
active: activeCtx active: activeCtx
}); });
@ -38,67 +48,71 @@ describe('ticket setDeleted()', () => {
} }
}; };
let sampleTicket = await models.Ticket.findById(12); const sampleTicket = await models.Ticket.findById(12);
let sampleStowaway = await models.Ticket.findById(13); const sampleStowaway = await models.Ticket.findById(13);
sampleTicket.id = undefined; sampleTicket.id = undefined;
let shipTicket = await models.Ticket.create(sampleTicket); const shipTicket = await models.Ticket.create(sampleTicket, options);
sampleStowaway.id = undefined; sampleStowaway.id = undefined;
let stowawayTicket = await models.Ticket.create(sampleStowaway); const stowawayTicket = await models.Ticket.create(sampleStowaway, options);
await models.Stowaway.rawSql(` await models.Stowaway.rawSql(`
INSERT INTO vn.stowaway(id, shipFk) INSERT INTO vn.stowaway(id, shipFk)
VALUES (?, ?)`, [stowawayTicket.id, shipTicket.id]); VALUES (?, ?)`, [stowawayTicket.id, shipTicket.id], options);
const boardingState = await models.State.findOne({ const boardingState = await models.State.findOne({
where: { where: {
code: 'BOARDING' code: 'BOARDING'
} }
}); }, options);
await models.TicketTracking.create({ await models.TicketTracking.create({
ticketFk: stowawayTicket.id, ticketFk: stowawayTicket.id,
stateFk: boardingState.id, stateFk: boardingState.id,
workerFk: ctx.req.accessToken.userId workerFk: ctx.req.accessToken.userId
}); }, options);
const okState = await models.State.findOne({ const okState = await models.State.findOne({
where: { where: {
code: 'OK' code: 'OK'
} }
}); }, options);
await models.TicketTracking.create({ await models.TicketTracking.create({
ticketFk: shipTicket.id, ticketFk: shipTicket.id,
stateFk: okState.id, stateFk: okState.id,
workerFk: ctx.req.accessToken.userId workerFk: ctx.req.accessToken.userId
}); }, options);
let stowawayTicketState = await models.TicketState.findOne({ let stowawayTicketState = await models.TicketState.findOne({
where: { where: {
ticketFk: stowawayTicket.id ticketFk: stowawayTicket.id
} }
}); }, options);
let stowaway = await models.Stowaway.findById(shipTicket.id); let stowaway = await models.Stowaway.findById(shipTicket.id, null, options);
expect(stowaway).toBeDefined(); expect(stowaway).toBeDefined();
expect(stowawayTicketState.code).toEqual('BOARDING'); expect(stowawayTicketState.code).toEqual('BOARDING');
await models.Ticket.setDeleted(ctx, shipTicket.id); await models.Ticket.setDeleted(ctx, shipTicket.id, options);
stowawayTicketState = await models.TicketState.findOne({ stowawayTicketState = await models.TicketState.findOne({
where: { where: {
ticketFk: stowawayTicket.id ticketFk: stowawayTicket.id
} }
}); }, options);
stowaway = await models.Stowaway.findById(shipTicket.id); stowaway = await models.Stowaway.findById(shipTicket.id, null, options);
expect(stowaway).toBeNull(); expect(stowaway).toBeNull();
expect(stowawayTicketState.code).toEqual('FIXING'); expect(stowawayTicketState.code).toEqual('FIXING');
// restores await tx.rollback();
await models.Ticket.destroyById(shipTicket.id); } catch (e) {
await models.Ticket.destroyById(stowawayTicket.id); await tx.rollback();
throw e;
}
}); });
}); });

View File

@ -1,28 +1,72 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
describe('ticket summary()', () => { describe('ticket summary()', () => {
it('should return a summary object containing data from 1 ticket', async() => { it('should return a summary object containing data from 1 ticket', async() => {
let result = await app.models.Ticket.summary(1); const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
const result = await models.Ticket.summary(1, options);
expect(result.id).toEqual(1); expect(result.id).toEqual(1);
expect(result.nickname).toEqual('Bat cave'); expect(result.nickname).toEqual('Bat cave');
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should return a summary object containing sales from 1 ticket', async() => { it('should return a summary object containing sales from 1 ticket', async() => {
let result = await app.models.Ticket.summary(1); const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
const result = await models.Ticket.summary(1, options);
expect(result.sales.length).toEqual(4); expect(result.sales.length).toEqual(4);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should return a summary object containing totalWithoutVat for 1 ticket', async() => { it('should return a summary object containing totalWithoutVat for 1 ticket', async() => {
let result = await app.models.Ticket.summary(1); const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
const result = await models.Ticket.summary(1, options);
expect(result.totalWithoutVat).toEqual(jasmine.any(Number)); expect(result.totalWithoutVat).toEqual(jasmine.any(Number));
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should return a summary object containing total for 1 ticket', async() => { it('should return a summary object containing total for 1 ticket', async() => {
let result = await app.models.Ticket.summary(1); const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
const result = await models.Ticket.summary(1, options);
expect(result.totalWithVat).toEqual(jasmine.any(Number)); expect(result.totalWithVat).toEqual(jasmine.any(Number));
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
}); });

View File

@ -1,4 +1,4 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
const LoopBackContext = require('loopback-context'); const LoopBackContext = require('loopback-context');
describe('sale transferSales()', () => { describe('sale transferSales()', () => {
@ -8,121 +8,138 @@ describe('sale transferSales()', () => {
}; };
const ctx = {req: activeCtx}; const ctx = {req: activeCtx};
let createdTicketsIds = [];
beforeAll(() => { beforeAll(() => {
spyOn(LoopBackContext, 'getCurrentContext').and.returnValue({ spyOn(LoopBackContext, 'getCurrentContext').and.returnValue({
active: activeCtx active: activeCtx
}); });
}); });
afterEach(async done => {
if (createdTicketsIds.length) {
try {
createdTicketsIds.forEach(async createdTicketId => {
await app.models.Ticket.destroyById(createdTicketId);
});
} catch (error) {
console.error(error);
}
}
done();
});
it('should throw an error as the ticket is not editable', async() => { it('should throw an error as the ticket is not editable', async() => {
const tx = await models.Ticket.beginTransaction({});
let error; let error;
try {
const options = {transaction: tx};
const currentTicketId = 1; const currentTicketId = 1;
const receiverTicketId = undefined; const receiverTicketId = undefined;
const sales = []; const sales = [];
await app.models.Ticket.transferSales(ctx, currentTicketId, receiverTicketId, sales) await models.Ticket.transferSales(ctx, currentTicketId, receiverTicketId, sales, options);
.catch(response => {
expect(response.message).toEqual(`The sales of this ticket can't be modified`);
error = response;
});
expect(error).toBeDefined(); await tx.rollback();
} catch (e) {
await tx.rollback();
error = e;
}
expect(error.message).toEqual(`The sales of this ticket can't be modified`);
}); });
it('should throw an error if the receiving ticket is not editable', async() => { it('should throw an error if the receiving ticket is not editable', async() => {
const tx = await models.Ticket.beginTransaction({});
let error; let error;
try {
const options = {transaction: tx};
const currentTicketId = 16; const currentTicketId = 16;
const receiverTicketId = 1; const receiverTicketId = 1;
const sales = []; const sales = [];
await app.models.Ticket.transferSales(ctx, currentTicketId, receiverTicketId, sales) await models.Ticket.transferSales(ctx, currentTicketId, receiverTicketId, sales, options);
.catch(response => {
expect(response.message).toEqual(`The sales of the receiver ticket can't be modified`);
error = response;
});
expect(error).toBeDefined(); await tx.rollback();
} catch (e) {
await tx.rollback();
error = e;
}
expect(error.message).toEqual(`The sales of the receiver ticket can't be modified`);
}); });
it('should transfer the sales from one ticket to a new one then send them back and delete the created ticket', async() => { it('should transfer the sales from one ticket to a new one then send them back and delete the created ticket', async() => {
const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
const formerTicketId = 11; const formerTicketId = 11;
let createdTicketId = undefined; let createdTicketId = undefined;
let formerTicketSales = await app.models.Ticket.getSales(formerTicketId); let formerTicketSales = await models.Ticket.getSales(formerTicketId, options);
expect(formerTicketSales.length).toEqual(2); expect(formerTicketSales.length).toEqual(2);
let createdTicket = await app.models.Ticket.transferSales( let createdTicket = await models.Ticket.transferSales(
ctx, formerTicketId, createdTicketId, formerTicketSales); ctx, formerTicketId, createdTicketId, formerTicketSales, options);
createdTicketId = createdTicket.id; createdTicketId = createdTicket.id;
createdTicketsIds.push(createdTicketId);
formerTicketSales = await app.models.Ticket.getSales(formerTicketId); formerTicketSales = await models.Ticket.getSales(formerTicketId, options);
createdTicketSales = await app.models.Ticket.getSales(createdTicketId); createdTicketSales = await models.Ticket.getSales(createdTicketId, options);
expect(formerTicketSales.length).toEqual(0); expect(formerTicketSales.length).toEqual(0);
expect(createdTicketSales.length).toEqual(2); expect(createdTicketSales.length).toEqual(2);
await app.models.Ticket.transferSales( await models.Ticket.transferSales(
ctx, createdTicketId, formerTicketId, createdTicketSales); ctx, createdTicketId, formerTicketId, createdTicketSales, options);
formerTicketSales = await app.models.Ticket.getSales(formerTicketId); formerTicketSales = await models.Ticket.getSales(formerTicketId, options);
createdTicketSales = await app.models.Ticket.getSales(createdTicketId); createdTicketSales = await models.Ticket.getSales(createdTicketId, options);
createdTicket = await app.models.Ticket.findById(createdTicketId); createdTicket = await models.Ticket.findById(createdTicketId, null, options);
expect(createdTicket.isDeleted).toBeTruthy(); expect(createdTicket.isDeleted).toBeTruthy();
expect(formerTicketSales.length).toEqual(2); expect(formerTicketSales.length).toEqual(2);
expect(createdTicketSales.length).toEqual(0); expect(createdTicketSales.length).toEqual(0);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
describe('sale transferPartialSales()', () => { describe('sale transferPartialSales()', () => {
it('should throw an error in the quantity to transfer exceeds the amount from the original sale', async() => { it('should throw an error in the quantity to transfer exceeds the amount from the original sale', async() => {
const tx = await models.Ticket.beginTransaction({});
let error; let error;
let currentTicket = await app.models.Ticket.findById(11); try {
let currentTicketSales = await app.models.Ticket.getSales(currentTicket.id); const options = {transaction: tx};
const currentTicket = await models.Ticket.findById(11, null, options);
const currentTicketSales = await models.Ticket.getSales(currentTicket.id, options);
const currentTicketId = currentTicket.id; const currentTicketId = currentTicket.id;
const receiverTicketId = undefined; const receiverTicketId = undefined;
currentTicketSales[0].quantity = 99; currentTicketSales[0].quantity = 99;
await app.models.Ticket.transferSales( await models.Ticket.transferSales(
ctx, currentTicketId, receiverTicketId, currentTicketSales) ctx, currentTicketId, receiverTicketId, currentTicketSales, options);
.catch(response => {
expect(response.message).toEqual(`Invalid quantity`);
error = response;
});
expect(error).toBeDefined(); await tx.rollback();
} catch (e) {
await tx.rollback();
error = e;
}
expect(error.message).toEqual(`Invalid quantity`);
}); });
it('should transfer two sales to a new ticket but one shall be partial', async() => { it('should transfer two sales to a new ticket but one shall be partial', async() => {
const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
const formerTicketId = 11; const formerTicketId = 11;
let createdTicketId = undefined; let createdTicketId = undefined;
let formerTicketSales = await app.models.Ticket.getSales(formerTicketId); let formerTicketSales = await models.Ticket.getSales(formerTicketId, options);
const partialSaleId = formerTicketSales[0].id;
const completeSaleId = formerTicketSales[1].id; const completeSaleId = formerTicketSales[1].id;
let partialSaleTotalQuantity = formerTicketSales[0].quantity; let partialSaleTotalQuantity = formerTicketSales[0].quantity;
@ -130,14 +147,13 @@ describe('sale transferSales()', () => {
formerTicketSales[0].quantity = 1; formerTicketSales[0].quantity = 1;
let createdTicket = await app.models.Ticket.transferSales( let createdTicket = await models.Ticket.transferSales(
ctx, formerTicketId, createdTicketId, formerTicketSales); ctx, formerTicketId, createdTicketId, formerTicketSales, options);
createdTicketId = createdTicket.id; createdTicketId = createdTicket.id;
createdTicketsIds.push(createdTicket.id);
formerTicketSales = await app.models.Ticket.getSales(formerTicketId); formerTicketSales = await models.Ticket.getSales(formerTicketId, options);
createdTicketSales = await app.models.Ticket.getSales(createdTicketId); createdTicketSales = await models.Ticket.getSales(createdTicketId, options);
const [createdPartialSale] = createdTicketSales.filter(sale => { const [createdPartialSale] = createdTicketSales.filter(sale => {
return sale.id != completeSaleId; return sale.id != completeSaleId;
@ -148,20 +164,11 @@ describe('sale transferSales()', () => {
expect(createdTicketSales.length).toEqual(2); expect(createdTicketSales.length).toEqual(2);
expect(createdPartialSale.quantity).toEqual(1); expect(createdPartialSale.quantity).toEqual(1);
let saleToRestore = await app.models.Sale.findById(partialSaleId); await tx.rollback();
await saleToRestore.updateAttribute('quantity', partialSaleTotalQuantity); } catch (e) {
await tx.rollback();
let saleToReturnToTicket = await app.models.Sale.findById(completeSaleId); throw e;
await saleToReturnToTicket.updateAttribute('ticketFk', formerTicketId); }
formerTicketSales = await app.models.Ticket.getSales(formerTicketId);
const [returningPartialSale] = formerTicketSales.filter(sale => {
return sale.id == partialSaleId;
});
expect(returningPartialSale.quantity).toEqual(partialSaleTotalQuantity);
expect(formerTicketSales.length).toEqual(2);
}); });
}); });
}); });

View File

@ -1,40 +1,15 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
describe('sale updateDiscount()', () => { describe('sale updateDiscount()', () => {
const originalSaleId = 8; const originalSaleId = 8;
let componentId;
let originalSale;
let salesPersonMana;
beforeAll(async done => {
try {
originalSale = await app.models.Sale.findById(originalSaleId);
let manaDiscount = await app.models.Component.findOne({where: {code: 'buyerDiscount'}});
componentId = manaDiscount.id;
let ticket = await app.models.Ticket.findById(originalSale.ticketFk);
let client = await app.models.Client.findById(ticket.clientFk);
salesPersonMana = await app.models.WorkerMana.findById(client.salesPersonFk);
} catch (error) {
console.error(error);
}
done();
});
afterAll(async done => {
try {
await originalSale.save();
await app.models.SaleComponent.updateAll({componentFk: componentId, saleFk: originalSaleId}, {value: 0});
await salesPersonMana.save();
} catch (error) {
console.error(error);
}
done();
});
it('should throw an error if no sales were selected', async() => { it('should throw an error if no sales were selected', async() => {
const tx = await models.Ticket.beginTransaction({});
let error;
try {
const options = {transaction: tx};
const ctx = { const ctx = {
req: { req: {
accessToken: {userId: 9}, accessToken: {userId: 9},
@ -42,21 +17,28 @@ describe('sale updateDiscount()', () => {
__: () => {} __: () => {}
} }
}; };
let error;
const ticketId = 11; const ticketId = 11;
const sales = []; const sales = [];
const newDiscount = 10; const newDiscount = 10;
try { await models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount, options);
await app.models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount);
} catch (err) { await tx.rollback();
error = err; } catch (e) {
await tx.rollback();
error = e;
} }
expect(error.message).toEqual('Please select at least one sale'); expect(error.message).toEqual('Please select at least one sale');
}); });
it('should throw an error if no sales belong to different tickets', async() => { it('should throw an error if no sales belong to different tickets', async() => {
const tx = await models.Ticket.beginTransaction({});
let error;
try {
const options = {transaction: tx};
const ctx = { const ctx = {
req: { req: {
accessToken: {userId: 9}, accessToken: {userId: 9},
@ -64,21 +46,28 @@ describe('sale updateDiscount()', () => {
__: () => {} __: () => {}
} }
}; };
let error;
const ticketId = 11; const ticketId = 11;
const sales = [1, 14]; const sales = [1, 14];
const newDiscount = 10; const newDiscount = 10;
try { await models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount, options);
await app.models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount);
} catch (err) { await tx.rollback();
error = err; } catch (e) {
await tx.rollback();
error = e;
} }
expect(error.message).toEqual('All sales must belong to the same ticket'); expect(error.message).toEqual('All sales must belong to the same ticket');
}); });
it('should throw an error if the ticket is invoiced already', async() => { it('should throw an error if the ticket is invoiced already', async() => {
const tx = await models.Ticket.beginTransaction({});
let error;
try {
const options = {transaction: tx};
const ctx = { const ctx = {
req: { req: {
accessToken: {userId: 9}, accessToken: {userId: 9},
@ -86,21 +75,27 @@ describe('sale updateDiscount()', () => {
__: () => {} __: () => {}
} }
}; };
let error;
const ticketId = 1; const ticketId = 1;
const sales = [1]; const sales = [1];
const newDiscount = 100; const newDiscount = 100;
try { await models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount, options);
await app.models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount);
} catch (err) { await tx.rollback();
error = err; } catch (e) {
await tx.rollback();
error = e;
} }
expect(error.message).toEqual(`The sales of this ticket can't be modified`); expect(error.message).toEqual(`The sales of this ticket can't be modified`);
}); });
it('should update the discount if the salesPerson has mana', async() => { it('should update the discount if the salesPerson has mana', async() => {
const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
const ctx = { const ctx = {
req: { req: {
accessToken: {userId: 18}, accessToken: {userId: 18},
@ -111,24 +106,35 @@ describe('sale updateDiscount()', () => {
const ticketId = 11; const ticketId = 11;
const sales = [originalSaleId]; const sales = [originalSaleId];
const newDiscount = 100; const newDiscount = 100;
let manaDiscount = await app.models.Component.findOne({where: {code: 'mana'}}); const manaDiscount = await models.Component.findOne({where: {code: 'mana'}}, options);
componentId = manaDiscount.id; const componentId = manaDiscount.id;
await app.models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount); await models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount, options);
let updatedSale = await app.models.Sale.findById(originalSaleId); const updatedSale = await models.Sale.findById(originalSaleId, null, options);
let createdSaleComponent = await app.models.SaleComponent.findOne({ const createdSaleComponent = await models.SaleComponent.findOne({
where: { where: {
componentFk: componentId, componentFk: componentId,
saleFk: originalSaleId saleFk: originalSaleId
} }
}); }, options);
expect(createdSaleComponent.componentFk).toEqual(componentId); expect(createdSaleComponent.componentFk).toEqual(componentId);
expect(updatedSale.discount).toEqual(100); expect(updatedSale.discount).toEqual(100);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
it('should update the discount and add company discount component if the worker does not have mana', async() => { it('should update the discount and add company discount component if the worker does not have mana', async() => {
const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
const ctx = { const ctx = {
req: { req: {
accessToken: {userId: 9}, accessToken: {userId: 9},
@ -140,17 +146,26 @@ describe('sale updateDiscount()', () => {
const sales = [originalSaleId]; const sales = [originalSaleId];
const newDiscount = 100; const newDiscount = 100;
await app.models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount); await models.Ticket.updateDiscount(ctx, ticketId, sales, newDiscount, options);
let updatedSale = await app.models.Sale.findById(originalSaleId); const updatedSale = await models.Sale.findById(originalSaleId, null, options);
let createdSaleComponent = await app.models.SaleComponent.findOne({ const manaDiscount = await models.Component.findOne({where: {code: 'buyerDiscount'}}, options);
const componentId = manaDiscount.id;
const createdSaleComponent = await models.SaleComponent.findOne({
where: { where: {
componentFk: componentId, componentFk: componentId,
saleFk: originalSaleId saleFk: originalSaleId
} }
}); }, options);
expect(createdSaleComponent.componentFk).toEqual(componentId); expect(createdSaleComponent.componentFk).toEqual(componentId);
expect(updatedSale.discount).toEqual(100); expect(updatedSale.discount).toEqual(100);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
}); });

View File

@ -1,4 +1,4 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
const LoopBackContext = require('loopback-context'); const LoopBackContext = require('loopback-context');
const userId = 9; const userId = 9;
@ -11,34 +11,44 @@ describe('ticket updateEditableTicket()', () => {
const validTicketId = 12; const validTicketId = 12;
const invalidTicketId = 1; const invalidTicketId = 1;
const data = {addressFk: 1}; const data = {addressFk: 1};
const originalData = {addressFk: 123};
afterAll(async done => {
await app.models.Ticket.updateEditableTicket(ctx, validTicketId, originalData);
done();
});
it('should now throw an error if the ticket is not editable', async() => { it('should now throw an error if the ticket is not editable', async() => {
let error; const tx = await models.Ticket.beginTransaction({});
await app.models.Ticket.updateEditableTicket(ctx, invalidTicketId, data).catch(e => { let error;
try {
const options = {transaction: tx};
await models.Ticket.updateEditableTicket(ctx, invalidTicketId, data, options);
await tx.rollback();
} catch (e) {
await tx.rollback();
error = e; error = e;
}).finally(() => { }
expect(error.message).toEqual('This ticket can not be modified'); expect(error.message).toEqual('This ticket can not be modified');
}); });
expect(error).toBeDefined();
});
it('should edit the ticket address', async() => { it('should edit the ticket address', async() => {
const tx = await models.Ticket.beginTransaction({});
try {
const options = {transaction: tx};
spyOn(LoopBackContext, 'getCurrentContext').and.returnValue({ spyOn(LoopBackContext, 'getCurrentContext').and.returnValue({
active: activeCtx active: activeCtx
}); });
await app.models.Ticket.updateEditableTicket(ctx, validTicketId, data); await models.Ticket.updateEditableTicket(ctx, validTicketId, data, options);
let updatedTicket = await app.models.Ticket.findById(validTicketId); const updatedTicket = await models.Ticket.findById(validTicketId, null, options);
expect(updatedTicket.addressFk).toEqual(1); expect(updatedTicket.addressFk).toEqual(1);
await tx.rollback();
} catch (e) {
await tx.rollback();
throw e;
}
}); });
}); });

View File

@ -1,19 +1,26 @@
const app = require('vn-loopback/server/server'); const models = require('vn-loopback/server/server').models;
describe('Ticket uploadFile()', () => { describe('Ticket uploadFile()', () => {
it(`should return an error for a user without enough privileges`, async() => { it(`should return an error for a user without enough privileges`, async() => {
let ticketId = 15; const tx = await models.Ticket.beginTransaction({});
let currentUserId = 1101;
let ticketTypeId = 14;
let ctx = {req: {accessToken: {userId: currentUserId}}, args: {dmsTypeId: ticketTypeId}};
let error; let error;
await app.models.Ticket.uploadFile(ctx, ticketId).catch(e => { try {
const options = {transaction: tx};
const ticketId = 15;
const currentUserId = 1101;
const ticketTypeId = 14;
const ctx = {req: {accessToken: {userId: currentUserId}}, args: {dmsTypeId: ticketTypeId}};
await models.Ticket.uploadFile(ctx, ticketId, options);
await tx.rollback();
} catch (e) {
await tx.rollback();
error = e; error = e;
}).finally(() => { }
expect(error.message).toEqual(`You don't have enough privileges`); expect(error.message).toEqual(`You don't have enough privileges`);
}); });
expect(error).toBeDefined();
});
}); });

View File

@ -19,10 +19,17 @@ module.exports = Self => {
} }
}); });
Self.summary = async ticketFk => { Self.summary = async(ticketFk, options) => {
let models = Self.app.models; const models = Self.app.models;
let summaryObj = await getTicketData(Self, ticketFk); const myOptions = {};
summaryObj.sales = await models.Ticket.getSales(ticketFk);
if (typeof options == 'object')
Object.assign(myOptions, options);
const summaryObj = await getTicketData(Self, ticketFk, myOptions);
summaryObj.sales = await models.Ticket.getSales(ticketFk, myOptions);
summaryObj.packagings = await models.TicketPackaging.find({ summaryObj.packagings = await models.TicketPackaging.find({
where: {ticketFk: ticketFk}, where: {ticketFk: ticketFk},
include: [{relation: 'packaging', include: [{relation: 'packaging',
@ -33,24 +40,25 @@ module.exports = Self => {
} }
} }
}] }]
}); }, myOptions);
summaryObj.requests = await getRequests(Self, ticketFk);
summaryObj.requests = await getRequests(Self, ticketFk, myOptions);
summaryObj.services = await models.TicketService.find({ summaryObj.services = await models.TicketService.find({
where: {ticketFk: ticketFk}, where: {ticketFk: ticketFk},
include: [{relation: 'taxClass'}] include: [{relation: 'taxClass'}]
}); }, myOptions);
return summaryObj; return summaryObj;
}; };
async function getTicketData(Self, ticketFk) { async function getTicketData(Self, ticketFk, options) {
let filter = { const filter = {
include: [ include: [
{relation: 'warehouse', scope: {fields: ['name']}}, {relation: 'warehouse', scope: {fields: ['name']}},
{relation: 'agencyMode', scope: {fields: ['name']}}, {relation: 'agencyMode', scope: {fields: ['name']}},
{relation: 'zone', scope: {fields: ['name']}}, {relation: 'zone', scope: {fields: ['name']}},
{ {relation: 'client',
relation: 'client',
scope: { scope: {
fields: ['salesPersonFk', 'name', 'phone', 'mobile'], fields: ['salesPersonFk', 'name', 'phone', 'mobile'],
include: { include: {
@ -99,11 +107,11 @@ module.exports = Self => {
where: {id: ticketFk} where: {id: ticketFk}
}; };
return await Self.findOne(filter); return Self.findOne(filter, options);
} }
async function getRequests(Self, ticketFk) { async function getRequests(Self, ticketFk, options) {
let filter = { const filter = {
where: { where: {
ticketFk: ticketFk ticketFk: ticketFk
}, },
@ -127,6 +135,6 @@ module.exports = Self => {
} }
] ]
}; };
return await Self.app.models.TicketRequest.find(filter); return Self.app.models.TicketRequest.find(filter, options);
} }
}; };

View File

@ -5,25 +5,25 @@ module.exports = Self => {
description: 'Transfer sales to a new or a given ticket', description: 'Transfer sales to a new or a given ticket',
accepts: [{ accepts: [{
arg: 'id', arg: 'id',
type: 'Number', type: 'number',
required: true, required: true,
description: 'Origin ticket id', description: 'Origin ticket id',
http: {source: 'path'} http: {source: 'path'}
}, },
{ {
arg: 'ticketId', arg: 'ticketId',
type: 'Number', type: 'number',
description: 'Destination ticket id', description: 'Destination ticket id',
required: false required: false
}, },
{ {
arg: 'sales', arg: 'sales',
type: ['Object'], type: ['object'],
description: 'The sales to transfer', description: 'The sales to transfer',
required: true required: true
}], }],
returns: { returns: {
type: 'Object', type: 'object',
root: true root: true
}, },
http: { http: {
@ -32,28 +32,35 @@ module.exports = Self => {
} }
}); });
Self.transferSales = async(ctx, id, ticketId, sales) => { Self.transferSales = async(ctx, id, ticketId, sales, options) => {
let userId = ctx.req.accessToken.userId; const userId = ctx.req.accessToken.userId;
const models = Self.app.models; const models = Self.app.models;
const myOptions = {};
let tx;
const isEditable = await models.Ticket.isEditable(ctx, id); if (typeof options == 'object')
Object.assign(myOptions, options);
if (!myOptions.transaction) {
tx = await Self.beginTransaction({});
myOptions.transaction = tx;
}
try {
const isEditable = await models.Ticket.isEditable(ctx, id, myOptions);
if (!isEditable) if (!isEditable)
throw new UserError(`The sales of this ticket can't be modified`); throw new UserError(`The sales of this ticket can't be modified`);
if (ticketId) { if (ticketId) {
const isReceiverEditable = await models.Ticket.isEditable(ctx, ticketId); const isReceiverEditable = await models.Ticket.isEditable(ctx, ticketId, myOptions);
if (!isReceiverEditable) if (!isReceiverEditable)
throw new UserError(`The sales of the receiver ticket can't be modified`); throw new UserError(`The sales of the receiver ticket can't be modified`);
} }
let tx = await Self.beginTransaction({}); const originalTicket = await models.Ticket.findById(id, null, myOptions);
try {
const options = {transaction: tx};
const originalTicket = await models.Ticket.findById(id, null, options);
const originalSales = await models.Sale.find({ const originalSales = await models.Sale.find({
where: {ticketFk: id} where: {ticketFk: id}
}, options); }, myOptions);
if (!ticketId) { if (!ticketId) {
const ticket = await models.Ticket.findById(id); const ticket = await models.Ticket.findById(id);
@ -61,7 +68,7 @@ module.exports = Self => {
if (!canCreateTicket) if (!canCreateTicket)
throw new UserError(`You can't create a ticket for a inactive client`); throw new UserError(`You can't create a ticket for a inactive client`);
ticketId = await cloneTicket(originalTicket, options); ticketId = await cloneTicket(originalTicket, myOptions);
} }
const map = new Map(); const map = new Map();
@ -80,10 +87,10 @@ module.exports = Self => {
if (sale.quantity == originalSale.quantity) { if (sale.quantity == originalSale.quantity) {
await models.Sale.updateAll({ await models.Sale.updateAll({
id: sale.id id: sale.id
}, {ticketFk: ticketId}, options); }, {ticketFk: ticketId}, myOptions);
} else if (sale.quantity != originalSale.quantity) { } else if (sale.quantity != originalSale.quantity) {
await transferPartialSale( await transferPartialSale(
ticketId, originalSale, sale, options); ticketId, originalSale, sale, myOptions);
} }
// Log to original ticket // Log to original ticket
@ -105,7 +112,7 @@ module.exports = Self => {
concept: sale.concept, concept: sale.concept,
ticket: ticketId ticket: ticketId
} }
}, options); }, myOptions);
// Log to destination ticket // Log to destination ticket
await models.TicketLog.create({ await models.TicketLog.create({
@ -126,22 +133,22 @@ module.exports = Self => {
concept: sale.concept, concept: sale.concept,
ticket: ticketId ticket: ticketId
} }
}, options); }, myOptions);
} }
const isTicketEmpty = await models.Ticket.isEmpty(id, options); const isTicketEmpty = await models.Ticket.isEmpty(id, myOptions);
if (isTicketEmpty) { if (isTicketEmpty) {
await originalTicket.updateAttributes({ await originalTicket.updateAttributes({
isDeleted: true isDeleted: true
}, options); }, myOptions);
} }
await tx.commit(); if (tx) await tx.commit();
return {id: ticketId}; return {id: ticketId};
} catch (error) { } catch (e) {
await tx.rollback(); if (tx) await tx.rollback();
throw error; throw e;
} }
}; };

View File

@ -35,14 +35,21 @@ module.exports = Self => {
} }
}); });
Self.updateDiscount = async(ctx, id, salesIds, newDiscount) => { Self.updateDiscount = async(ctx, id, salesIds, newDiscount, options) => {
const $t = ctx.req.__; // $translate const $t = ctx.req.__; // $translate
const models = Self.app.models; const models = Self.app.models;
const tx = await Self.beginTransaction({}); const myOptions = {};
let tx;
if (typeof options == 'object')
Object.assign(myOptions, options);
if (!myOptions.transaction) {
tx = await Self.beginTransaction({});
myOptions.transaction = tx;
}
try { try {
const options = {transaction: tx};
const filter = { const filter = {
where: { where: {
id: {inq: salesIds} id: {inq: salesIds}
@ -61,7 +68,7 @@ module.exports = Self => {
} }
}; };
const sales = await models.Sale.find(filter, options); const sales = await models.Sale.find(filter, myOptions);
if (sales.length === 0) if (sales.length === 0)
throw new UserError('Please select at least one sale'); throw new UserError('Please select at least one sale');
@ -71,15 +78,15 @@ module.exports = Self => {
throw new UserError('All sales must belong to the same ticket'); throw new UserError('All sales must belong to the same ticket');
const userId = ctx.req.accessToken.userId; const userId = ctx.req.accessToken.userId;
const isLocked = await models.Ticket.isLocked(id); const isLocked = await models.Ticket.isLocked(id, myOptions);
const roles = await models.Account.getRoles(userId); const roles = await models.Account.getRoles(userId, myOptions);
const hasAllowedRoles = roles.filter(role => const hasAllowedRoles = roles.filter(role =>
role == 'salesPerson' || role == 'claimManager' role == 'salesPerson' || role == 'claimManager'
); );
const state = await Self.app.models.TicketState.findOne({ const state = await Self.app.models.TicketState.findOne({
where: {ticketFk: id} where: {ticketFk: id}
}); }, myOptions);
const alertLevel = state ? state.alertLevel : null; const alertLevel = state ? state.alertLevel : null;
if (isLocked || (!hasAllowedRoles && alertLevel > 0)) if (isLocked || (!hasAllowedRoles && alertLevel > 0))
@ -89,11 +96,11 @@ module.exports = Self => {
where: { where: {
workerFk: userId workerFk: userId
}, },
fields: 'amount'}, options); fields: 'amount'}, myOptions);
const componentCode = usesMana ? 'mana' : 'buyerDiscount'; const componentCode = usesMana ? 'mana' : 'buyerDiscount';
const discountComponent = await models.Component.findOne({ const discountComponent = await models.Component.findOne({
where: {code: componentCode}}, options); where: {code: componentCode}}, myOptions);
const componentId = discountComponent.id; const componentId = discountComponent.id;
const promises = []; const promises = [];
@ -105,9 +112,9 @@ module.exports = Self => {
const newComponent = models.SaleComponent.upsert({ const newComponent = models.SaleComponent.upsert({
saleFk: sale.id, saleFk: sale.id,
value: value, value: value,
componentFk: componentId}, options); componentFk: componentId}, myOptions);
const updatedSale = sale.updateAttribute('discount', newDiscount, options); const updatedSale = sale.updateAttribute('discount', newDiscount, myOptions);
promises.push(newComponent, updatedSale); promises.push(newComponent, updatedSale);
changesMade += `\r\n-${sale.itemFk}: ${sale.concept} (${sale.quantity}) ${oldDiscount}% ➔ *${newDiscount}%*`; changesMade += `\r\n-${sale.itemFk}: ${sale.concept} (${sale.quantity}) ${oldDiscount}% ➔ *${newDiscount}%*`;
@ -116,7 +123,7 @@ module.exports = Self => {
await Promise.all(promises); await Promise.all(promises);
const query = `call vn.manaSpellersRequery(?)`; const query = `call vn.manaSpellersRequery(?)`;
await Self.rawSql(query, [userId], options); await Self.rawSql(query, [userId], myOptions);
const ticket = await models.Ticket.findById(id, { const ticket = await models.Ticket.findById(id, {
include: { include: {
@ -130,7 +137,7 @@ module.exports = Self => {
} }
} }
} }
}, options); }, myOptions);
const salesPerson = ticket.client().salesPersonUser(); const salesPerson = ticket.client().salesPersonUser();
if (salesPerson) { if (salesPerson) {
@ -144,10 +151,10 @@ module.exports = Self => {
await models.Chat.sendCheckingPresence(ctx, salesPerson.id, message); await models.Chat.sendCheckingPresence(ctx, salesPerson.id, message);
} }
await tx.commit(); if (tx) await tx.commit();
} catch (error) { } catch (e) {
await tx.rollback(); if (tx) await tx.rollback();
throw error; throw e;
} }
}; };
}; };

View File

@ -15,7 +15,7 @@ module.exports = Self => {
{ {
arg: 'data', arg: 'data',
description: 'Model instance data', description: 'Model instance data',
type: 'Object', type: 'object',
required: true, required: true,
http: {source: 'body'} http: {source: 'body'}
} }
@ -30,12 +30,30 @@ module.exports = Self => {
} }
}); });
Self.updateEditableTicket = async(ctx, id, data) => { Self.updateEditableTicket = async(ctx, id, data, options) => {
let ticketIsEditable = await Self.app.models.Ticket.isEditable(ctx, id); const myOptions = {};
let tx;
if (typeof options == 'object')
Object.assign(myOptions, options);
if (!myOptions.transaction) {
tx = await Self.beginTransaction({});
myOptions.transaction = tx;
}
try {
const ticketIsEditable = await Self.app.models.Ticket.isEditable(ctx, id, myOptions);
if (!ticketIsEditable) if (!ticketIsEditable)
throw new UserError('This ticket can not be modified'); throw new UserError('This ticket can not be modified');
let ticket = await Self.app.models.Ticket.findById(id); const ticket = await Self.app.models.Ticket.findById(id, null, myOptions);
await ticket.updateAttributes(data); await ticket.updateAttributes(data, myOptions);
if (tx) await tx.commit();
} catch (e) {
if (tx) await tx.rollback();
throw e;
}
}; };
}; };

View File

@ -4,40 +4,40 @@ module.exports = Self => {
accessType: 'WRITE', accessType: 'WRITE',
accepts: [{ accepts: [{
arg: 'id', arg: 'id',
type: 'Number', type: 'number',
description: 'The ticket id', description: 'The ticket id',
http: {source: 'path'} http: {source: 'path'}
}, { }, {
arg: 'warehouseId', arg: 'warehouseId',
type: 'Number', type: 'number',
description: 'The warehouse id', description: 'The warehouse id',
required: true required: true
}, { }, {
arg: 'companyId', arg: 'companyId',
type: 'Number', type: 'number',
description: 'The company id', description: 'The company id',
required: true required: true
}, { }, {
arg: 'dmsTypeId', arg: 'dmsTypeId',
type: 'Number', type: 'number',
description: 'The dms type id', description: 'The dms type id',
required: true required: true
}, { }, {
arg: 'reference', arg: 'reference',
type: 'String', type: 'string',
required: true required: true
}, { }, {
arg: 'description', arg: 'description',
type: 'String', type: 'string',
required: true required: true
}, { }, {
arg: 'hasFile', arg: 'hasFile',
type: 'Boolean', type: 'boolean',
description: 'True if has an attached file', description: 'True if has an attached file',
required: true required: true
}], }],
returns: { returns: {
type: 'Object', type: 'object',
root: true root: true
}, },
http: { http: {
@ -46,31 +46,38 @@ module.exports = Self => {
} }
}); });
Self.uploadFile = async(ctx, id) => { Self.uploadFile = async(ctx, id, options) => {
const models = Self.app.models; const models = Self.app.models;
const myOptions = {};
let tx;
if (typeof options == 'object')
Object.assign(myOptions, options);
if (!myOptions.transaction) {
tx = await Self.beginTransaction({});
myOptions.transaction = tx;
}
const promises = []; const promises = [];
const tx = await Self.beginTransaction({});
try { try {
const options = {transaction: tx}; const uploadedFiles = await models.Dms.uploadFile(ctx, myOptions);
const uploadedFiles = await models.Dms.uploadFile(ctx, options);
uploadedFiles.forEach(dms => { uploadedFiles.forEach(dms => {
const newTicketDms = models.TicketDms.create({ const newTicketDms = models.TicketDms.create({
ticketFk: id, ticketFk: id,
dmsFk: dms.id dmsFk: dms.id
}, options); }, myOptions);
promises.push(newTicketDms); promises.push(newTicketDms);
}); });
const resolvedPromises = await Promise.all(promises); const resolvedPromises = await Promise.all(promises);
await tx.commit(); if (tx) await tx.commit();
return resolvedPromises; return resolvedPromises;
} catch (err) { } catch (e) {
await tx.rollback(); if (tx) await tx.rollback();
throw err; throw e;
} }
}; };
}; };

View File

@ -35,7 +35,7 @@ module.exports = Self => {
}); });
Self.getLanded = async(ctx, shipped, addressFk, agencyModeFk, warehouseFk, options) => { Self.getLanded = async(ctx, shipped, addressFk, agencyModeFk, warehouseFk, options) => {
let myOptions = {}; const myOptions = {};
if (typeof options == 'object') if (typeof options == 'object')
Object.assign(myOptions, options); Object.assign(myOptions, options);
@ -57,8 +57,7 @@ module.exports = Self => {
agencyModeFk, agencyModeFk,
warehouseFk, warehouseFk,
showExpired showExpired
], ]
myOptions
)); ));
const rsIndex = stmts.push('SELECT * FROM tmp.zoneGetLanded') - 1; const rsIndex = stmts.push('SELECT * FROM tmp.zoneGetLanded') - 1;