obdfcascrape/ncas/cy.js
Martin Donnelly be5d3eae07 init
2019-05-05 20:13:56 +01:00

456 lines
12 KiB
JavaScript

const Scraper = require('../helpers/scraper');
const cheerio = require('cheerio');
const path = require('path');
const jsonfile = require('jsonfile');
const url = require('url');
const logger = require('log4js').getLogger('CY');
logger.level = process.env.LOGGER_LEVEL || 'warn';
// load env variables from file
class CYScrape extends Scraper {
constructor() {
super();
this.setID('CY');
this.on('done', () => {
this._done();
});
this.run = this._debounce(async () => {
await this.__run();
}, 5000);
if (process.env.NODE_ENV === 'production')
this._checkLock().then((l) => {
if(l)
this.run();
});
}
/**
*
* @param selector
* @returns {Promise<void>}
*/
async grabLink(selector) {
const clickableLinks = await this.page.$$(selector);
await this.page._client.send('Page.setDownloadBehavior', { 'behavior': 'allow', 'downloadPath': this.path });
if (clickableLinks.length > 0)
for (const item of clickableLinks) {
const href = await this.page.evaluate(el => el.href, item);
await this._randomWait(this.page, 3, 5);
await this.page.goto(href, { 'waitUntil': 'networkidle2' }).catch((err) => {
// log this error but Puppeteer isn't supposed to support this sort of download....
// mute the ERR_ABORTED error which happens everytime but alert for everything else.
if (!err.message.includes('net::ERR_ABORTED') )
logger.error('grabLink', err);
});
}
}
/**
*
* @param id
* @returns {Promise<void>}
*/
async downloadEmoney(id) {
const selector = ['#generic_article > div > div.row > div > div > ul > li:nth-child(1) > a', '#generic_article > div > div.row > div > div > ul > li:nth-child(2) > b > b > a'];
await this.grabLink(selector[id]);
}
/**
*
* @returns {Promise<void>}
*/
async downloadExcel() {
const selector = '#workshops > div > div.workshop-article-container > div > div > div > h3 > a';
await this.grabLink(selector);
}
/**
*
* @returns {Promise<void>}
*/
async handlePaymentInstitutions() {
await this._randomWait(this.page, 3, 5);
const filename = 'licensing-and-supervision-of-payment-institutions';
await this._makeScreenshotV2(this.page, `${this.path}/${filename}_main`, null);
await this._randomWait(this.page, 3, 5);
await this.downloadExcel();
await this._randomWait(this.page, 3, 5);
await this.page.goto(this.eMoneyUrl, { 'waitUntil': 'networkidle2' });
}
/**
*
* @returns {Promise<void>}
*/
async handleElectronicMoneyInstitutions() {
await this._randomWait(this.page, 3, 5);
const filename = 'licensing-and-supervision-of-electronic-money-institutions';
await this._makeScreenshotV2(this.page, `${this.path}/${filename}_main`, null);
await this._randomWait(this.page, 3, 5);
await this.downloadEmoney(0);
await this._randomWait(this.page, 3, 5);
await this.downloadEmoney(1);
await this._randomWait(this.page, 3, 5);
this.emit('startProcessingCreditServices');
}
/**
*
* @param body
* @returns {Promise<{}|Array>}
*/
async extractLocalCreditInstitutions(body) {
try{
const matchHeading = /LOCAL AUTHORISED CREDIT INSTITUTIONS/;
const sanity = /(\d+\.\s)(.+)/;
const $ = cheerio.load(body, {
'normalizeWhitespace': true
});
let nextItem;
$('p').each(function(i, elem) {
const lineText = $(this).text();
const isHeading = matchHeading.test(lineText);
if (isHeading)
nextItem = $(this).next();
});
if (typeof nextItem !== 'undefined' && nextItem !== null) {
const splitText = $(nextItem).text().split('\n');
const output = [];
splitText.forEach((item) => {
const newItem = this._cleanUp(item);
if ( newItem !== '')
output.push( sanity.exec(newItem)[2]);
});
return output;
}
return {};
}
catch( err) {
logger.error(err);
}
}
/**
*
* @param body
* @returns {Promise<void>}
*/
async extractForeignCreditInstitutions(body) {
try{
const matchHeading = /FOREIGN AUTHORISED CREDIT INSTITUTIONS AND BRANCHES OF FOREIGN CREDIT INSTITUTIONS FROM EU MEMBER STATES OPERATING/;
const sanity = /(\w+\.\s+)(.+)/;
const $ = cheerio.load(body, {
'normalizeWhitespace': true
});
const output = {};
let nextItem;
$('p').each(function(i, elem) {
const lineText = $(this).text();
const isHeading = matchHeading.test(lineText);
if (isHeading)
nextItem = $(this).next();
});
// Rolling this out for ease as it could be changed by hand
let nextElm;
let firstHeadOrig, firstHead;
if (typeof nextItem !== 'undefined' && nextItem !== null) {
firstHeadOrig = this._cleanUp($(nextItem).text());
firstHead = sanity.exec(firstHeadOrig)[2];
output[firstHead] = {};
nextElm = $(nextItem).next();
const secondHeadOrig = this._cleanUp($(nextElm).text());
const secondHead = sanity.exec(secondHeadOrig)[2];
nextElm = $(nextElm).next();
const li = $(nextElm).find('li');
const arrayA = [];
$(li).each(function (i, elem) {
const lineText = $(this).text();
arrayA.push(lineText);
});
output[firstHead][secondHead] = arrayA;
nextElm = $(nextElm).next();
}
if (typeof nextElm !== 'undefined' && nextElm !== null) {
const secondHeadOrig = this._cleanUp($(nextElm).text());
const secondHead = sanity.exec(secondHeadOrig)[2];
nextElm = $(nextElm).next();
const li = $(nextElm).find('li');
const arrayA = [];
$(li).each(function (i, elem) {
const lineText = $(this).text();
arrayA.push(lineText);
});
output[firstHead][secondHead] = arrayA;
nextElm = $(nextElm).next();
}
if (typeof nextElm !== 'undefined' && nextElm !== null) {
firstHeadOrig = this._cleanUp($(nextElm).text());
firstHead = sanity.exec(firstHeadOrig)[2];
output[firstHead] = {};
nextElm = $(nextElm).next();
const secondHeadOrig = this._cleanUp($(nextElm).text());
const secondHead = sanity.exec(secondHeadOrig)[2];
nextElm = $(nextElm).next();
const li = $(nextElm).find('li');
const arrayA = [];
$(li).each(function (i, elem) {
const lineText = $(this).text();
arrayA.push(lineText);
});
output[firstHead][secondHead] = arrayA;
nextElm = $(nextElm).next();
}
if (typeof nextElm !== 'undefined' && nextElm !== null) {
const secondHeadOrig = this._cleanUp($(nextElm).text());
const secondHead = sanity.exec(secondHeadOrig)[2];
nextElm = $(nextElm).next();
const li = $(nextElm).find('li');
const arrayA = [];
$(li).each(function (i, elem) {
const lineText = $(this).text();
arrayA.push(lineText);
});
output[firstHead][secondHead] = arrayA;
}
return output;
}
catch(err) {
logger.error(err);
}
}
/**
*
* @returns {Promise<{local: Promise<*|void>}>}
*/
async processCreditInstitute() {
logger.info('Credit institutes');
try{
await this._makeScreenshotV2(this.page, `${this.path}/creditInstitutes`, null);
const body = await this.page.content();
await this._dumpFile(`${this.path}/creditInstitutes.html`, body);
const $ = cheerio.load(body);
const content = $('.generic_page-intro');
const local = await this.extractLocalCreditInstitutions(content.html());
const creditInstitutes = await this.extractForeignCreditInstitutions(content.html());
await jsonfile.writeFile(`${this.path}/creditInstitutes.json`, { local, creditInstitutes });
this.emit('done');
return { local, creditInstitutes };
}
catch(err) {
logger.error(err);
}
}
/**
*
* @param filePath
* @returns {Promise<void>}
*/
async savePDF(filePath) {
logger.info('Saving the pdf:', filePath);
await this._randomWait(this.page, 5, 7);
await this.page.pdf({ 'path': filePath, 'format': 'A4' });
// this.emit('startProcessingCreditServices');
logger.debug('!! i SHOULD EMIT SOMETHING HERE !!');
}
/**
*
* @returns {Promise<void>}
*/
async processNewPage() {
// give the page a few seconds to settle
const checkPDF = /(.pdf)/g;
await this._randomWait(this.page, 3, 5);
const pageUrl = url.parse(await this.page.url());
if (pageUrl.href === 'chrome-error://chromewebdata/') {
logger.warn('Directed to: chrome-error://chromewebdata/');
this.emit('recover');
return;
}
let currentPath = pageUrl.pathname;
let pdfFile;
if (checkPDF.test(currentPath)) {
const splitPath = currentPath.split('/');
pdfFile = splitPath.pop();
currentPath = splitPath.join('/');
}
switch (currentPath) {
case '/en/licensing-supervision/payment-institutions/licensing-and-supervision-of-payment-institutions':
await this.handlePaymentInstitutions();
break;
case '/en/licensing-supervision/electronic-money-institutions/licensing-and-supervision-of-electronic-money-institutions':
await this.handleElectronicMoneyInstitutions();
break;
case '/images/media/redirectfile/Electronic%20Money%20Institutions':
logger.warn('We should only arrive here when in Non-headless mode');
await this.savePDF(pdfFile);
break;
case '/en/licensing-supervision/banks/register-of-credit-institutions-operating-in-cyprus':
await this.processCreditInstitute();
break;
default:
await this._uploadError();
throw new Error(`Unknown page: ${pageUrl.href}`);
break;
}
}
/**
*
* @returns {Promise<void>}
*/
async attachEvents() {
logger.info('Attaching events');
this.on('startProcessingCreditServices', async function() {
await this._goto(this.credit);
});
}
/**
*
* @returns {Promise<void>}
*/
async start() {
try {
super._start();
this.creditServices = {
'items': 0,
'links': [],
'step': 0,
'visited': false,
'done' : false,
'searchDone' : false
};
this.startPage = 'https://www.centralbank.cy/en/licensing-supervision/payment-institutions/licensing-and-supervision-of-payment-institutions';
this.eMoneyUrl = 'https://www.centralbank.cy/en/licensing-supervision/electronic-money-institutions/licensing-and-supervision-of-electronic-money-institutions';
this.credit = 'https://www.centralbank.cy/en/licensing-supervision/banks/register-of-credit-institutions-operating-in-cyprus';
this.path = path.resolve(`${__dirname }/../artefacts/CY/CBOC`);
await this._createDirectory(this.path);
await this._doNonRepudiation().catch((err) => {
logger.warn(err);
});
await this._initBrowser(true);
await this._createBrowserPage();
this.page.on('domcontentloaded', this._throttle(async () => {
this.processNewPage().catch((err) => {
logger.error('processNewPage fail', err);
});
}, 2500));
if (this.eventNames().length === 2)
await this.attachEvents();
await this.page.tracing.start({ 'path': `${this.path}/trace.json`, 'screenshots': true });
await this.page.setViewport({ 'width': 1200, 'height': 800 });
await this._goto(this.startPage);
await this._randomWait(this.page, 3, 5);
}
catch (e) {
throw new Error(e);
}
}
/**
*
* @returns {Promise<void>}
*/
async __run() {
logger.info('Scraping Cyprus...');
await this.start();
}
}
module.exports = CYScrape;