Merge branch 'dev' into 8077-sumDefaulter
gitea/salix/pipeline/pr-dev This commit looks good
Details
gitea/salix/pipeline/pr-dev This commit looks good
Details
This commit is contained in:
commit
ceaad1ef2f
|
@ -158,13 +158,13 @@ INSERT INTO `account`.`mailForward`(`account`, `forwardTo`)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
INSERT INTO `vn`.`currency`(`id`, `code`, `name`, `ratio`)
|
INSERT INTO `vn`.`currency`(`id`, `code`, `name`, `ratio`, `hasToDownloadRate`)
|
||||||
VALUES
|
VALUES
|
||||||
(1, 'EUR', 'Euro', 1),
|
(1, 'EUR', 'Euro', 1, FALSE),
|
||||||
(2, 'USD', 'Dollar USA', 1.4),
|
(2, 'USD', 'Dollar USA', 1.4, TRUE),
|
||||||
(3, 'GBP', 'Libra', 1),
|
(3, 'GBP', 'Libra', 1, TRUE),
|
||||||
(4, 'JPY', 'Yen Japones', 1),
|
(4, 'JPY', 'Yen Japones', 1, FALSE),
|
||||||
(5, 'CNY', 'Yuan Chino', 1.2);
|
(5, 'CNY', 'Yuan Chino', 1.2, TRUE);
|
||||||
|
|
||||||
INSERT INTO `vn`.`country`(`id`, `name`, `isUeeMember`, `code`, `currencyFk`, `ibanLength`, `continentFk`, `hasDailyInvoice`, `CEE`)
|
INSERT INTO `vn`.`country`(`id`, `name`, `isUeeMember`, `code`, `currencyFk`, `ibanLength`, `continentFk`, `hasDailyInvoice`, `CEE`)
|
||||||
VALUES
|
VALUES
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
ALTER TABLE `vn`.`entry`
|
||||||
|
ADD COLUMN `initialTemperature` decimal(10,2) DEFAULT NULL COMMENT 'Temperatura de como lo recibimos del proveedor ej. en colombia',
|
||||||
|
ADD COLUMN `finalTemperature` decimal(10,2) DEFAULT NULL COMMENT 'Temperatura final de como llega a nuestras instalaciones';
|
|
@ -0,0 +1,2 @@
|
||||||
|
ALTER TABLE `vn`.`currency`
|
||||||
|
ADD COLUMN `hasToDownloadRate` TINYINT(1) NOT NULL DEFAULT 0 comment 'Si se guarda el tipo de cambio diariamente en referenceRate';
|
|
@ -0,0 +1,3 @@
|
||||||
|
UPDATE `vn`.`currency`
|
||||||
|
SET `hasToDownloadRate` = TRUE
|
||||||
|
WHERE `code` IN ('USD', 'CNY', 'GBP');
|
|
@ -0,0 +1,2 @@
|
||||||
|
INSERT INTO salix.ACL (model,property,accessType,permission,principalType,principalId)
|
||||||
|
VALUES ('VnUser','adminUser','WRITE','ALLOW','ROLE','sysadmin');
|
|
@ -0,0 +1,2 @@
|
||||||
|
RENAME TABLE bi.f_tvc TO bi.f_tvc__;
|
||||||
|
ALTER TABLE bi.f_tvc__ COMMENT='@deprecated 2025-01-15';
|
|
@ -52,6 +52,14 @@ module.exports = function(Self) {
|
||||||
arg: 'customsAgentFk',
|
arg: 'customsAgentFk',
|
||||||
type: 'number'
|
type: 'number'
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
arg: 'longitude',
|
||||||
|
type: 'number'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
arg: 'latitude',
|
||||||
|
type: 'number'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
arg: 'isActive',
|
arg: 'isActive',
|
||||||
type: 'boolean'
|
type: 'boolean'
|
||||||
|
|
|
@ -119,6 +119,16 @@ module.exports = Self => {
|
||||||
arg: 'invoiceAmount',
|
arg: 'invoiceAmount',
|
||||||
type: 'number',
|
type: 'number',
|
||||||
description: `The invoice amount`
|
description: `The invoice amount`
|
||||||
|
},
|
||||||
|
{
|
||||||
|
arg: 'initialTemperature',
|
||||||
|
type: 'number',
|
||||||
|
description: 'Initial temperature value'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
arg: 'finalTemperature',
|
||||||
|
type: 'number',
|
||||||
|
description: 'Final temperature value'
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
returns: {
|
returns: {
|
||||||
|
@ -170,6 +180,10 @@ module.exports = Self => {
|
||||||
case 'invoiceInFk':
|
case 'invoiceInFk':
|
||||||
param = `e.${param}`;
|
param = `e.${param}`;
|
||||||
return {[param]: value};
|
return {[param]: value};
|
||||||
|
case 'initialTemperature':
|
||||||
|
return {'e.initialTemperature': {lte: value}};
|
||||||
|
case 'finalTemperature':
|
||||||
|
return {'e.finalTemperature': {gte: value}};
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
filter = mergeFilters(ctx.args.filter, {where});
|
filter = mergeFilters(ctx.args.filter, {where});
|
||||||
|
@ -204,6 +218,8 @@ module.exports = Self => {
|
||||||
e.gestDocFk,
|
e.gestDocFk,
|
||||||
e.invoiceInFk,
|
e.invoiceInFk,
|
||||||
e.invoiceAmount,
|
e.invoiceAmount,
|
||||||
|
e.initialTemperature,
|
||||||
|
e.finalTemperature,
|
||||||
t.landed,
|
t.landed,
|
||||||
s.name supplierName,
|
s.name supplierName,
|
||||||
s.nickname supplierAlias,
|
s.nickname supplierAlias,
|
||||||
|
|
|
@ -68,6 +68,12 @@
|
||||||
},
|
},
|
||||||
"invoiceAmount": {
|
"invoiceAmount": {
|
||||||
"type": "number"
|
"type": "number"
|
||||||
|
},
|
||||||
|
"initialTemperature": {
|
||||||
|
"type": "number"
|
||||||
|
},
|
||||||
|
"finalTemperature": {
|
||||||
|
"type": "number"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"relations": {
|
"relations": {
|
||||||
|
|
|
@ -13,66 +13,114 @@ module.exports = Self => {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
Self.exchangeRateUpdate = async() => {
|
Self.exchangeRateUpdate = async(options = {}) => {
|
||||||
const response = await axios.get('http://www.ecb.europa.eu/stats/eurofxref/eurofxref-hist-90d.xml');
|
|
||||||
const xmlData = response.data;
|
|
||||||
|
|
||||||
const doc = new DOMParser({errorHandler: {warning: () => {}}})?.parseFromString(xmlData, 'text/xml');
|
|
||||||
const cubes = doc?.getElementsByTagName('Cube');
|
|
||||||
if (!cubes || cubes.length === 0)
|
|
||||||
throw new UserError('No cubes found. Exiting the method.');
|
|
||||||
|
|
||||||
const models = Self.app.models;
|
const models = Self.app.models;
|
||||||
|
const myOptions = {};
|
||||||
|
let tx;
|
||||||
|
|
||||||
const maxDateRecord = await models.ReferenceRate.findOne({order: 'dated DESC'});
|
if (typeof options == 'object')
|
||||||
|
Object.assign(myOptions, options);
|
||||||
|
|
||||||
const maxDate = maxDateRecord?.dated ? new Date(maxDateRecord.dated) : null;
|
if (!myOptions.transaction) {
|
||||||
|
tx = await Self.beginTransaction({});
|
||||||
|
myOptions.transaction = tx;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await axios.get('http://www.ecb.europa.eu/stats/eurofxref/eurofxref-hist-90d.xml');
|
||||||
|
const xmlData = response.data;
|
||||||
|
|
||||||
|
const doc = new DOMParser({errorHandler: {warning: () => {}}})
|
||||||
|
.parseFromString(xmlData, 'text/xml');
|
||||||
|
const cubes = doc?.getElementsByTagName('Cube');
|
||||||
|
if (!cubes || cubes.length === 0)
|
||||||
|
throw new UserError('No cubes found. Exiting the method.');
|
||||||
|
|
||||||
|
const currencies = await models.Currency.find({where: {hasToDownloadRate: true}}, myOptions);
|
||||||
|
const maxDateRecord = await models.ReferenceRate.findOne({order: 'dated DESC'}, myOptions);
|
||||||
|
const maxDate = maxDateRecord?.dated ? new Date(maxDateRecord.dated) : null;
|
||||||
|
let lastProcessedDate = maxDate;
|
||||||
|
|
||||||
|
for (const cube of Array.from(cubes)) {
|
||||||
|
if (cube.nodeType === doc.ELEMENT_NODE && cube.attributes.getNamedItem('time')) {
|
||||||
|
const xmlDate = new Date(cube.getAttribute('time'));
|
||||||
|
const xmlDateWithoutTime = new Date(
|
||||||
|
xmlDate.getFullYear(),
|
||||||
|
xmlDate.getMonth(),
|
||||||
|
xmlDate.getDate()
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!maxDate || xmlDateWithoutTime > maxDate) {
|
||||||
|
if (lastProcessedDate && xmlDateWithoutTime > lastProcessedDate) {
|
||||||
|
for (const currency of currencies) {
|
||||||
|
await fillMissingDates(
|
||||||
|
models, currency, lastProcessedDate, xmlDateWithoutTime, myOptions
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (const cube of Array.from(cubes)) {
|
|
||||||
if (cube.nodeType === doc.ELEMENT_NODE && cube.attributes.getNamedItem('time')) {
|
|
||||||
const xmlDate = new Date(cube.getAttribute('time'));
|
|
||||||
const xmlDateWithoutTime = new Date(xmlDate.getFullYear(), xmlDate.getMonth(), xmlDate.getDate());
|
|
||||||
if (!maxDate || maxDate < xmlDateWithoutTime) {
|
|
||||||
for (const rateCube of Array.from(cube.childNodes)) {
|
for (const rateCube of Array.from(cube.childNodes)) {
|
||||||
if (rateCube.nodeType === doc.ELEMENT_NODE) {
|
if (rateCube.nodeType === doc.ELEMENT_NODE) {
|
||||||
const currencyCode = rateCube.getAttribute('currency');
|
const currencyCode = rateCube.getAttribute('currency');
|
||||||
const rate = rateCube.getAttribute('rate');
|
const rate = rateCube.getAttribute('rate');
|
||||||
if (['USD', 'CNY', 'GBP'].includes(currencyCode)) {
|
const currency = currencies.find(c => c.code === currencyCode);
|
||||||
const currency = await models.Currency.findOne({where: {code: currencyCode}});
|
if (currency) {
|
||||||
if (!currency) throw new UserError(`Currency not found for code: ${currencyCode}`);
|
|
||||||
const existingRate = await models.ReferenceRate.findOne({
|
const existingRate = await models.ReferenceRate.findOne({
|
||||||
where: {currencyFk: currency.id, dated: xmlDate}
|
where: {currencyFk: currency.id, dated: xmlDateWithoutTime}
|
||||||
});
|
}, myOptions);
|
||||||
|
|
||||||
if (existingRate) {
|
if (existingRate) {
|
||||||
if (existingRate.value !== rate)
|
if (existingRate.value !== rate)
|
||||||
await existingRate.updateAttributes({value: rate});
|
await existingRate.updateAttributes({value: rate}, myOptions);
|
||||||
} else {
|
} else {
|
||||||
await models.ReferenceRate.create({
|
await models.ReferenceRate.create({
|
||||||
currencyFk: currency.id,
|
currencyFk: currency.id,
|
||||||
dated: xmlDate,
|
dated: xmlDateWithoutTime,
|
||||||
value: rate
|
value: rate
|
||||||
});
|
}, myOptions);
|
||||||
}
|
|
||||||
const monday = 1;
|
|
||||||
if (xmlDateWithoutTime.getDay() === monday) {
|
|
||||||
const saturday = new Date(xmlDateWithoutTime);
|
|
||||||
saturday.setDate(xmlDateWithoutTime.getDate() - 2);
|
|
||||||
const sunday = new Date(xmlDateWithoutTime);
|
|
||||||
sunday.setDate(xmlDateWithoutTime.getDate() - 1);
|
|
||||||
|
|
||||||
for (const date of [saturday, sunday]) {
|
|
||||||
await models.ReferenceRate.upsertWithWhere(
|
|
||||||
{currencyFk: currency.id, dated: date},
|
|
||||||
{currencyFk: currency.id, dated: date, value: rate}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
lastProcessedDate = xmlDateWithoutTime;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (tx) await tx.commit();
|
||||||
|
} catch (error) {
|
||||||
|
if (tx) await tx.rollback();
|
||||||
|
throw error;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
async function getLastValidRate(models, currencyId, date, myOptions) {
|
||||||
|
return models.ReferenceRate.findOne({
|
||||||
|
where: {currencyFk: currencyId, dated: {lt: date}},
|
||||||
|
order: 'dated DESC'
|
||||||
|
}, myOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fillMissingDates(models, currency, startDate, endDate, myOptions) {
|
||||||
|
const cursor = new Date(startDate);
|
||||||
|
cursor.setDate(cursor.getDate() + 1);
|
||||||
|
while (cursor < endDate) {
|
||||||
|
const existingRate = await models.ReferenceRate.findOne({
|
||||||
|
where: {currencyFk: currency.id, dated: cursor}
|
||||||
|
}, myOptions);
|
||||||
|
|
||||||
|
if (!existingRate) {
|
||||||
|
const lastValid = await getLastValidRate(models, currency.id, cursor, myOptions);
|
||||||
|
if (lastValid) {
|
||||||
|
await models.ReferenceRate.create({
|
||||||
|
currencyFk: currency.id,
|
||||||
|
dated: new Date(cursor),
|
||||||
|
value: lastValid.value
|
||||||
|
}, myOptions);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cursor.setDate(cursor.getDate() + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,52 +1,190 @@
|
||||||
describe('exchangeRateUpdate functionality', function() {
|
describe('exchangeRateUpdate functionality', function() {
|
||||||
const axios = require('axios');
|
const axios = require('axios');
|
||||||
const models = require('vn-loopback/server/server').models;
|
const models = require('vn-loopback/server/server').models;
|
||||||
|
let tx; let options;
|
||||||
|
|
||||||
beforeEach(function() {
|
function formatYmd(d) {
|
||||||
spyOn(axios, 'get').and.returnValue(Promise.resolve({
|
const mm = (d.getMonth() + 1).toString().padStart(2, '0');
|
||||||
data: `<Cube>
|
const dd = d.getDate().toString().padStart(2, '0');
|
||||||
<Cube time='2024-04-12'>
|
return `${d.getFullYear()}-${mm}-${dd}`;
|
||||||
<Cube currency='USD' rate='1.1'/>
|
}
|
||||||
<Cube currency='CNY' rate='1.2'/>
|
|
||||||
</Cube>
|
afterEach(async() => {
|
||||||
</Cube>`
|
await tx.rollback();
|
||||||
}));
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should process XML data and update or create rates in the database', async function() {
|
beforeEach(async() => {
|
||||||
|
tx = await models.Sale.beginTransaction({});
|
||||||
|
options = {transaction: tx};
|
||||||
|
spyOn(axios, 'get').and.returnValue(Promise.resolve({data: ''}));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should process XML data and create rates', async function() {
|
||||||
|
const d1 = Date.vnNew();
|
||||||
|
const d4 = Date.vnNew();
|
||||||
|
d4.setDate(d4.getDate() + 1);
|
||||||
|
const xml = `<Cube>
|
||||||
|
<Cube time='${formatYmd(d1)}'>
|
||||||
|
<Cube currency='USD' rate='1.1'/>
|
||||||
|
<Cube currency='CNY' rate='1.2'/>
|
||||||
|
</Cube>
|
||||||
|
<Cube time='${formatYmd(d4)}'>
|
||||||
|
<Cube currency='USD' rate='1.3'/>
|
||||||
|
</Cube>
|
||||||
|
</Cube>`;
|
||||||
|
axios.get.and.returnValue(Promise.resolve({data: xml}));
|
||||||
spyOn(models.ReferenceRate, 'findOne').and.returnValue(Promise.resolve(null));
|
spyOn(models.ReferenceRate, 'findOne').and.returnValue(Promise.resolve(null));
|
||||||
spyOn(models.ReferenceRate, 'create').and.returnValue(Promise.resolve());
|
spyOn(models.ReferenceRate, 'create').and.returnValue(Promise.resolve());
|
||||||
|
await models.InvoiceIn.exchangeRateUpdate(options);
|
||||||
|
|
||||||
await models.InvoiceIn.exchangeRateUpdate();
|
expect(models.ReferenceRate.create).toHaveBeenCalledTimes(3);
|
||||||
|
|
||||||
expect(models.ReferenceRate.create).toHaveBeenCalledTimes(2);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should not create or update rates when no XML data is available', async function() {
|
it('should handle no data', async function() {
|
||||||
axios.get.and.returnValue(Promise.resolve({}));
|
axios.get.and.returnValue(Promise.resolve({}));
|
||||||
spyOn(models.ReferenceRate, 'create');
|
spyOn(models.ReferenceRate, 'create');
|
||||||
|
let e;
|
||||||
let thrownError = null;
|
|
||||||
try {
|
try {
|
||||||
await models.InvoiceIn.exchangeRateUpdate();
|
await models.InvoiceIn.exchangeRateUpdate(options);
|
||||||
} catch (error) {
|
} catch (err) {
|
||||||
thrownError = error;
|
e = err;
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(thrownError.message).toBe('No cubes found. Exiting the method.');
|
expect(e.message).toBe('No cubes found. Exiting the method.');
|
||||||
|
expect(models.ReferenceRate.create).not.toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should handle errors gracefully', async function() {
|
it('should handle errors', async function() {
|
||||||
axios.get.and.returnValue(Promise.reject(new Error('Network error')));
|
axios.get.and.returnValue(Promise.reject(new Error('Network error')));
|
||||||
let error;
|
let e;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await models.InvoiceIn.exchangeRateUpdate();
|
await models.InvoiceIn.exchangeRateUpdate(options);
|
||||||
} catch (e) {
|
} catch (err) {
|
||||||
error = e;
|
e = err;
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(error).toBeDefined();
|
expect(e).toBeDefined();
|
||||||
expect(error.message).toBe('Network error');
|
expect(e.message).toBe('Network error');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should update existing rate', async function() {
|
||||||
|
const existingRate = await models.ReferenceRate.findOne({
|
||||||
|
order: 'id DESC'
|
||||||
|
}, options);
|
||||||
|
|
||||||
|
if (!existingRate) return fail('No ReferenceRate records in DB');
|
||||||
|
|
||||||
|
const currency = await models.Currency.findById(existingRate.currencyFk, null, options);
|
||||||
|
|
||||||
|
const xml = `<Cube>
|
||||||
|
<Cube time='${formatYmd(existingRate.dated)}'>
|
||||||
|
<Cube currency='${currency.code}' rate='2.22'/>
|
||||||
|
</Cube>
|
||||||
|
</Cube>`;
|
||||||
|
|
||||||
|
axios.get.and.returnValue(Promise.resolve({data: xml}));
|
||||||
|
|
||||||
|
await models.InvoiceIn.exchangeRateUpdate(options);
|
||||||
|
|
||||||
|
const updatedRate = await models.ReferenceRate.findById(existingRate.id, null, options);
|
||||||
|
|
||||||
|
expect(updatedRate.value).toBeCloseTo('2.22');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not update if same rate', async function() {
|
||||||
|
const existingRate = await models.ReferenceRate.findOne({order: 'id DESC'}, options);
|
||||||
|
if (!existingRate) return fail('No existing ReferenceRate in DB');
|
||||||
|
|
||||||
|
const currency = await models.Currency.findById(existingRate.currencyFk, null, options);
|
||||||
|
|
||||||
|
const oldValue = existingRate.value;
|
||||||
|
const xml = `<Cube>
|
||||||
|
<Cube time='${formatYmd(existingRate.dated)}'>
|
||||||
|
<Cube currency='${currency.code}' rate='${oldValue}'/>
|
||||||
|
</Cube>
|
||||||
|
</Cube>`;
|
||||||
|
|
||||||
|
axios.get.and.returnValue(Promise.resolve({data: xml}));
|
||||||
|
|
||||||
|
await models.InvoiceIn.exchangeRateUpdate(options);
|
||||||
|
|
||||||
|
const updatedRate = await models.ReferenceRate.findById(existingRate.id, null, options);
|
||||||
|
|
||||||
|
expect(updatedRate.value).toBe(oldValue);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should backfill missing dates', async function() {
|
||||||
|
const lastRate = await models.ReferenceRate.findOne({order: 'dated DESC'}, options);
|
||||||
|
if (!lastRate) return fail('No existing ReferenceRate data in DB');
|
||||||
|
|
||||||
|
const currency = await models.Currency.findById(lastRate.currencyFk, null, options);
|
||||||
|
|
||||||
|
const d1 = new Date(lastRate.dated);
|
||||||
|
d1.setDate(d1.getDate() + 1);
|
||||||
|
const d4 = new Date(lastRate.dated);
|
||||||
|
d4.setDate(d4.getDate() + 4);
|
||||||
|
|
||||||
|
const xml = `<Cube>
|
||||||
|
<Cube time='${formatYmd(d1)}'>
|
||||||
|
<Cube currency='${currency.code}' rate='1.0'/>
|
||||||
|
</Cube>
|
||||||
|
<Cube time='${formatYmd(d4)}'>
|
||||||
|
<Cube currency='${currency.code}' rate='2.0'/>
|
||||||
|
</Cube>
|
||||||
|
</Cube>`;
|
||||||
|
|
||||||
|
axios.get.and.returnValue(Promise.resolve({data: xml}));
|
||||||
|
|
||||||
|
const beforeCount = await models.ReferenceRate.count({}, options);
|
||||||
|
await models.InvoiceIn.exchangeRateUpdate(options);
|
||||||
|
const afterCount = await models.ReferenceRate.count({}, options);
|
||||||
|
|
||||||
|
expect(afterCount - beforeCount).toBe(4);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should create entries for day1 and day2 from the feed, and not backfill day3', async function() {
|
||||||
|
const lastRate = await models.ReferenceRate.findOne({order: 'dated DESC'}, options);
|
||||||
|
if (!lastRate) return fail('No existing ReferenceRate data in DB');
|
||||||
|
|
||||||
|
const currency = await models.Currency.findById(lastRate.currencyFk, null, options);
|
||||||
|
if (!currency) return fail(`No currency for ID ${lastRate.currencyFk}`);
|
||||||
|
|
||||||
|
const day1 = new Date(lastRate.dated);
|
||||||
|
day1.setDate(day1.getDate() + 1);
|
||||||
|
|
||||||
|
const day2 = new Date(lastRate.dated);
|
||||||
|
day2.setDate(day2.getDate() + 2);
|
||||||
|
|
||||||
|
const day3 = new Date(lastRate.dated);
|
||||||
|
day3.setDate(day3.getDate() + 3);
|
||||||
|
|
||||||
|
const xml = `<Cube>
|
||||||
|
<Cube time='${formatYmd(day1)}'>
|
||||||
|
<Cube currency='${currency.code}' rate='1.1'/>
|
||||||
|
</Cube>
|
||||||
|
<Cube time='${formatYmd(day2)}'>
|
||||||
|
<Cube currency='${currency.code}' rate='2.2'/>
|
||||||
|
</Cube>
|
||||||
|
</Cube>`;
|
||||||
|
|
||||||
|
axios.get.and.returnValue(Promise.resolve({data: xml}));
|
||||||
|
|
||||||
|
await models.InvoiceIn.exchangeRateUpdate(options);
|
||||||
|
|
||||||
|
const day3Record = await models.ReferenceRate.findOne({
|
||||||
|
where: {currencyFk: currency.id, dated: day3}
|
||||||
|
}, options);
|
||||||
|
|
||||||
|
expect(day3Record).toBeNull();
|
||||||
|
|
||||||
|
const day1Record = await models.ReferenceRate.findOne({
|
||||||
|
where: {currencyFk: currency.id, dated: day1}
|
||||||
|
}, options);
|
||||||
|
const day2Record = await models.ReferenceRate.findOne({
|
||||||
|
where: {currencyFk: currency.id, dated: day2}
|
||||||
|
}, options);
|
||||||
|
|
||||||
|
expect(day1Record.value).toBeCloseTo('1.1');
|
||||||
|
expect(day2Record.value).toBeCloseTo('2.2');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -41,7 +41,9 @@ module.exports = Self => {
|
||||||
* b.stickers)/1000000) AS DECIMAL(10,2)) m3,
|
* b.stickers)/1000000) AS DECIMAL(10,2)) m3,
|
||||||
TRUNCATE(SUM(b.stickers)/(COUNT( b.id) / COUNT( DISTINCT b.id)),0) hb,
|
TRUNCATE(SUM(b.stickers)/(COUNT( b.id) / COUNT( DISTINCT b.id)),0) hb,
|
||||||
CAST(SUM(b.freightValue*b.quantity) AS DECIMAL(10,2)) freightValue,
|
CAST(SUM(b.freightValue*b.quantity) AS DECIMAL(10,2)) freightValue,
|
||||||
CAST(SUM(b.packageValue*b.quantity) AS DECIMAL(10,2)) packageValue
|
CAST(SUM(b.packageValue*b.quantity) AS DECIMAL(10,2)) packageValue,
|
||||||
|
e.initialTemperature,
|
||||||
|
e.finalTemperature
|
||||||
FROM vn.travel t
|
FROM vn.travel t
|
||||||
LEFT JOIN vn.entry e ON t.id = e.travelFk
|
LEFT JOIN vn.entry e ON t.id = e.travelFk
|
||||||
LEFT JOIN vn.buy b ON b.entryFk = e.id
|
LEFT JOIN vn.buy b ON b.entryFk = e.id
|
||||||
|
|
|
@ -20,6 +20,9 @@
|
||||||
},
|
},
|
||||||
"ratio": {
|
"ratio": {
|
||||||
"type": "number"
|
"type": "number"
|
||||||
|
},
|
||||||
|
"hasToDownloadRate": {
|
||||||
|
"type": "boolean"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"acls": [
|
"acls": [
|
||||||
|
|
Loading…
Reference in New Issue