refactor(ticket): ticket endpoins now can receive transactions
This commit is contained in:
parent
ca00e3611c
commit
cc90a08f03
|
@ -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();
|
||||||
|
|
|
@ -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;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -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'}
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -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`));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -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);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -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();
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Loading…
Reference in New Issue