module.exports = Self => { Self.remoteMethodCtx('refund', { description: 'Create refund tickets with sales and services if provided', accessType: 'WRITE', accepts: [ { arg: 'salesIds', type: ['number'] }, { arg: 'servicesIds', type: ['number'] }, { arg: 'withWarehouse', type: 'boolean', required: true } ], returns: { type: ['number'], root: true }, http: { path: `/refund`, verb: 'post' } }); Self.refund = async(ctx, salesIds, servicesIds, withWarehouse, options) => { const models = Self.app.models; const myOptions = {userId: ctx.req.accessToken.userId}; let tx; if (typeof options == 'object') Object.assign(myOptions, options); if (!myOptions.transaction) { tx = await Self.beginTransaction({}); myOptions.transaction = tx; } let refundTicket = null; try { const refundAgencyMode = await models.AgencyMode.findOne({ include: { relation: 'zones', scope: { limit: 1, field: ['id', 'name'] } }, where: {code: 'refund'} }, myOptions); const refoundZoneId = refundAgencyMode.zones()[0].id; if (salesIds) { const salesFilter = { where: {id: {inq: salesIds}}, include: { relation: 'components', scope: { fields: ['saleFk', 'componentFk', 'value'] } } }; const sales = await models.Sale.find(salesFilter, myOptions); const ticketsIds = [...new Set(sales.map(sale => sale.ticketFk))]; const now = Date.vnNew(); const [firstTicketId] = ticketsIds; // eslint-disable-next-line max-len refundTicket = await createTicketRefund(firstTicketId, now, refundAgencyMode, refoundZoneId, withWarehouse, myOptions); for (const sale of sales) { const createdSale = await models.Sale.create({ ticketFk: refundTicket.id, itemFk: sale.itemFk, quantity: - sale.quantity, concept: sale.concept, price: sale.price, discount: sale.discount, }, myOptions); const components = sale.components(); for (const component of components) component.saleFk = createdSale.id; await models.SaleComponent.create(components, myOptions); } } if (!refundTicket) { const servicesFilter = { where: {id: {inq: servicesIds}} }; const services = await models.TicketService.find(servicesFilter, myOptions); const ticketsIds = [...new Set(services.map(service => service.ticketFk))]; const now = Date.vnNew(); const [firstTicketId] = ticketsIds; // eslint-disable-next-line max-len refundTicket = await createTicketRefund(firstTicketId, now, refundAgencyMode, refoundZoneId, withWarehouse, myOptions); } if (servicesIds && servicesIds.length > 0) { const servicesFilter = { where: {id: {inq: servicesIds}} }; const services = await models.TicketService.find(servicesFilter, myOptions); for (const service of services) { await models.TicketService.create({ description: service.description, quantity: service.quantity, price: - service.price, taxClassFk: service.taxClassFk, ticketFk: refundTicket.id, ticketServiceTypeFk: service.ticketServiceTypeFk, }, myOptions); } } const query = `CALL vn.ticket_recalc(?, NULL)`; await Self.rawSql(query, [refundTicket.id], myOptions); if (tx) await tx.commit(); return refundTicket; } catch (e) { if (tx) await tx.rollback(); throw e; } }; async function createTicketRefund(ticketId, now, refundAgencyMode, refoundZoneId, withWarehouse, myOptions) { const models = Self.app.models; const filter = {include: {relation: 'address'}}; const ticket = await models.Ticket.findById(ticketId, filter, myOptions); const refundTicket = await models.Ticket.create({ clientFk: ticket.clientFk, shipped: now, addressFk: ticket.address().id, agencyModeFk: refundAgencyMode.id, nickname: ticket.address().nickname, warehouseFk: withWarehouse ? ticket.warehouseFk : null, companyFk: ticket.companyFk, landed: now, zoneFk: refoundZoneId }, myOptions); await models.TicketRefund.create({ refundTicketFk: refundTicket.id, originalTicketFk: ticket.id, }, myOptions); return refundTicket; } };