salix/back/methods/edi/updateData.js

240 lines
7.4 KiB
JavaScript
Raw Permalink Normal View History

/* eslint no-console: "off" */
const path = require('path');
const fs = require('fs-extra');
module.exports = Self => {
Self.remoteMethodCtx('updateData', {
description: 'Updates schema data from external provider',
accessType: 'WRITE',
returns: {
type: 'object',
root: true
},
http: {
path: `/updateData`,
2022-09-13 11:53:44 +00:00
verb: 'POST'
}
});
2023-06-01 06:32:06 +00:00
Self.updateData = async ctx => {
const models = Self.app.models;
2022-06-07 11:06:42 +00:00
// Get files checksum
2022-09-08 07:21:05 +00:00
const tx = await Self.beginTransaction({});
2022-09-08 07:21:05 +00:00
try {
2023-06-01 06:32:06 +00:00
const options = {transaction: tx, userId: ctx.req.accessToken.userId};
2022-09-08 07:21:05 +00:00
const files = await Self.rawSql('SELECT name, checksum, keyValue FROM edi.fileConfig', null, options);
2022-09-08 07:21:05 +00:00
const updatableFiles = [];
for (const file of files) {
const fileChecksum = await getChecksum(file);
if (file.checksum != fileChecksum) {
updatableFiles.push({
name: file.name,
checksum: fileChecksum
});
} else
console.debug(`File already updated, skipping...`);
}
2022-09-08 07:21:05 +00:00
if (updatableFiles.length === 0)
return false;
2022-09-08 07:21:05 +00:00
// Download files
const container = await models.TempContainer.container('edi');
const tempPath = path.join(container.client.root, container.name);
2022-09-08 07:21:05 +00:00
let remoteFile;
let tempDir;
let tempFile;
2022-09-08 07:21:05 +00:00
const fileNames = updatableFiles.map(file => file.name);
2022-09-08 07:21:05 +00:00
const tables = await Self.rawSql(`
2023-06-02 05:51:01 +00:00
SELECT fileName, toTable, file
FROM edi.tableConfig
WHERE file IN (?)`, [fileNames], options);
2022-06-03 11:22:32 +00:00
2022-09-08 07:21:05 +00:00
for (const table of tables) {
const fileName = table.file;
2022-09-08 07:21:05 +00:00
remoteFile = `codes/${fileName}.ZIP`;
tempDir = `${tempPath}/${fileName}`;
tempFile = `${tempPath}/${fileName}.zip`;
try {
await fs.readFile(tempFile);
} catch (error) {
if (error.code === 'ENOENT') {
console.debug(`Downloading file ${fileName}...`);
const downloadOutput = await downloadFile(remoteFile, tempFile);
if (downloadOutput.error)
continue;
}
2022-06-09 08:13:45 +00:00
}
2022-06-07 11:06:42 +00:00
2022-09-08 07:21:05 +00:00
await extractFile(fileName, tempFile, tempDir);
2022-06-07 11:06:42 +00:00
2022-09-08 07:21:05 +00:00
console.debug(`Updating table ${table.toTable}...`);
await dumpData(tempDir, table, options);
}
2022-06-07 11:06:42 +00:00
2022-09-08 07:21:05 +00:00
// Update files checksum
for (const file of updatableFiles) {
console.log(`Updating file ${file.name} checksum...`);
await Self.rawSql(`
2023-06-02 05:51:01 +00:00
UPDATE edi.fileConfig
SET checksum = ?
WHERE name = ?`,
2022-09-08 07:21:05 +00:00
[file.checksum, file.name], options);
}
2022-09-08 08:33:51 +00:00
await tx.commit();
2022-09-08 07:21:05 +00:00
// Clean files
try {
console.debug(`Cleaning files...`);
await fs.remove(tempPath);
} catch (error) {
if (error.code !== 'ENOENT')
throw e;
}
2022-09-08 09:42:16 +00:00
return true;
2022-06-09 08:13:45 +00:00
} catch (error) {
2022-09-08 07:21:05 +00:00
await tx.rollback();
throw error;
2022-06-09 08:13:45 +00:00
}
};
2022-06-07 11:06:42 +00:00
let ftpClient;
async function getFtpClient() {
if (!ftpClient) {
const [ftpConfig] = await Self.rawSql('SELECT host, user, password FROM edi.ftpConfig');
console.debug(`Openning FTP connection to ${ftpConfig.host}...\n`);
2022-06-07 11:06:42 +00:00
const FtpClient = require('ftps');
2022-06-07 11:06:42 +00:00
ftpClient = new FtpClient({
host: ftpConfig.host,
username: ftpConfig.user,
password: ftpConfig.password,
procotol: 'ftp',
additionalLftpCommands: 'set ssl:verify-certificate no'
2022-06-07 11:06:42 +00:00
});
}
2022-06-07 11:06:42 +00:00
return ftpClient;
}
2022-06-07 11:06:42 +00:00
async function getChecksum(file) {
const ftpClient = await getFtpClient();
console.debug(`Checking checksum for file ${file.name}...`);
ftpClient.cat(`codes/${file.name}.TXT`);
2022-06-07 11:06:42 +00:00
const response = await new Promise((resolve, reject) => {
ftpClient.exec((err, response) => {
2022-09-06 12:23:25 +00:00
if (err || response.error) {
2022-06-07 11:06:42 +00:00
console.debug(`Error downloading checksum file... ${response.error}`);
2023-09-18 05:10:12 +00:00
return reject(response.error || err);
2022-06-07 11:06:42 +00:00
}
resolve(response);
});
});
2022-06-07 11:06:42 +00:00
if (response && response.data) {
const fileContents = response.data;
const rows = fileContents.split('\n');
const row = rows[4];
const columns = row.split(/\s+/);
let fileChecksum;
if (file.keyValue)
fileChecksum = columns[1];
if (!file.keyValue)
fileChecksum = columns[0];
return fileChecksum;
}
}
2022-06-07 11:06:42 +00:00
async function downloadFile(remoteFile, tempFile) {
const ftpClient = await getFtpClient();
ftpClient.get(remoteFile, tempFile);
return new Promise((resolve, reject) => {
ftpClient.exec((err, response) => {
2022-09-07 08:26:32 +00:00
if (err || response.error) {
2022-06-07 11:06:42 +00:00
console.debug(`Error downloading file... ${response.error}`);
2022-09-07 08:26:32 +00:00
return reject(err);
2022-06-07 11:06:42 +00:00
}
resolve(response);
});
});
}
2022-09-07 08:26:32 +00:00
async function extractFile(fileName, tempFile, tempDir) {
2022-06-07 11:06:42 +00:00
const JSZip = require('jszip');
2022-06-09 08:13:45 +00:00
try {
await fs.mkdir(tempDir);
2022-09-07 08:26:32 +00:00
console.debug(`Extracting file ${fileName}...`);
2022-06-09 08:13:45 +00:00
} catch (error) {
if (error.code !== 'EEXIST')
throw e;
}
2022-06-07 11:06:42 +00:00
const fileStream = await fs.readFile(tempFile);
if (fileStream) {
const zip = new JSZip();
const zipContents = await zip.loadAsync(fileStream);
if (!zipContents) return;
const fileNames = Object.keys(zipContents.files);
for (const fileName of fileNames) {
const fileContent = await zip.file(fileName).async('nodebuffer');
const dest = path.join(tempDir, fileName);
2022-06-09 08:13:45 +00:00
await fs.writeFile(dest, fileContent);
2022-06-07 11:06:42 +00:00
}
}
}
2022-09-08 07:21:05 +00:00
async function dumpData(tempDir, table, options) {
2022-06-07 11:06:42 +00:00
const toTable = table.toTable;
const baseName = table.fileName;
2022-09-08 07:21:05 +00:00
console.log(`Emptying table ${toTable}...`);
const tableName = `edi.${toTable}`;
await Self.rawSql(`DELETE FROM ??`, [tableName]);
2022-06-03 10:57:39 +00:00
2022-09-08 07:21:05 +00:00
const dirFiles = await fs.readdir(tempDir);
const files = dirFiles.filter(file => file.startsWith(baseName));
2022-09-08 07:21:05 +00:00
for (const file of files) {
console.log(`Dumping data from file ${file}...`);
2022-09-08 07:21:05 +00:00
const templatePath = path.join(__dirname, `./sql/${toTable}.sql`);
const sqlTemplate = await fs.readFile(templatePath, 'utf8');
const filePath = path.join(tempDir, file);
2022-06-03 10:52:41 +00:00
2022-09-08 07:21:05 +00:00
await Self.rawSql(sqlTemplate, [filePath], options);
await Self.rawSql(`
2023-06-02 05:51:01 +00:00
UPDATE edi.tableConfig
SET updated = ?
WHERE fileName = ?
`, [Date.vnNew(), baseName], options);
}
2022-09-08 07:21:05 +00:00
2022-06-03 10:52:41 +00:00
console.log(`Updated table ${toTable}\n`);
}
};