512 lines
13 KiB
JavaScript
512 lines
13 KiB
JavaScript
/**
|
|
* Created by marti on 30/01/2016.
|
|
*/
|
|
const http = require('http');
|
|
const request = require('request');
|
|
const cheerio = require('cheerio');
|
|
const util = require('util');
|
|
const cron = require('node-cron');
|
|
const dateFormat = require('dateformat');
|
|
const jsonfile = require('jsonfile');
|
|
const fs = require('fs');
|
|
const LocalStorage = require('node-localstorage').LocalStorage;
|
|
// var nano = require('nano')('http://martind2000:1V3D4m526i@localhost:5984');
|
|
const logger = require('log4js').getLogger();
|
|
const Sugar = require('sugar');
|
|
|
|
const calHandler = require('./lib/calHandler');
|
|
const swedishWord = require('./lib/swedishword');
|
|
const weather = require('./lib/weather');
|
|
const trains = require('./lib/trains');
|
|
const history = require('./lib/history');
|
|
const mdMailer = require('./lib/mailer');
|
|
// const mdFitbit = require('./lib/fitbit');
|
|
const todayFTSE = require('./lib/todayftse');
|
|
const quotes = require('./lib/quotes');
|
|
// var db_name = 'silvrgit';
|
|
// var dbCloudant = nano.use(db_name);
|
|
|
|
/*const memwatch = require('memwatch-next');
|
|
|
|
memwatch.on('leak', (info) => {
|
|
console.error('Memory leak detected:\n', info);
|
|
});*/
|
|
|
|
/*
|
|
|
|
We've moved to cloudant through IBM Bluemix for the database
|
|
|
|
https://25f854ee-1b51-49ff-acd9-5b0ff478d944-bluemix.cloudant.com/dashboard.html#usage
|
|
|
|
*/
|
|
localStorage = new LocalStorage('./scratch');
|
|
|
|
const credentials = {
|
|
'username': '25f854ee-1b51-49ff-acd9-5b0ff478d944-bluemix',
|
|
'password': '8e417af1b0462ca55726848846cc6b8696fc76defe9d1864cbc334be59549e0c',
|
|
'host': '25f854ee-1b51-49ff-acd9-5b0ff478d944-bluemix.cloudant.com',
|
|
'port': 443,
|
|
'url': 'https://25f854ee-1b51-49ff-acd9-5b0ff478d944-bluemix:8e417af1b0462ca55726848846cc6b8696fc76defe9d1864cbc334be59549e0c@25f854ee-1b51-49ff-acd9-5b0ff478d944-bluemix.cloudant.com',
|
|
'database': 'today'
|
|
};
|
|
|
|
const Cloudant = require('cloudant');
|
|
const cloudant = Cloudant({ 'account': credentials.username, 'password': credentials.password });
|
|
|
|
const dbCloudant = cloudant.db.use(credentials.database);
|
|
|
|
String.prototype.hashCode = function () {
|
|
if (Array.prototype.reduce)
|
|
return this.split('').reduce(function (a, b) {
|
|
a = ((a << 5) - a) + b.charCodeAt(0);
|
|
|
|
return a & a;
|
|
}, 0);
|
|
else {
|
|
let hash = 0, i, chr, len;
|
|
if (this.length === 0) return hash;
|
|
for (i = 0, len = this.length; i < len; i++) {
|
|
chr = this.charCodeAt(i);
|
|
hash = ((hash << 5) - hash) + chr;
|
|
hash |= 0; // Convert to 32bit integer
|
|
}
|
|
|
|
return hash;
|
|
}
|
|
};
|
|
|
|
const todayCacheSrc = {
|
|
'last': 0, 'data': {
|
|
'trains': { 'last': 0, 'data': [] },
|
|
'weather': {},
|
|
'history': [],
|
|
'today': '',
|
|
'tv': { 'entries': [] },
|
|
'cal': { 'today': [], 'tomorrow': [], 'week': [] },
|
|
'swedish': {},
|
|
'fitbit': {},
|
|
'ftse': {}
|
|
}, 'expire': ((60 * 1000) * 60)
|
|
};
|
|
|
|
let todayCache = {
|
|
'last': 0, 'data': {
|
|
'trains': { 'last': 0, 'data': [] },
|
|
'weather': {},
|
|
'history': [],
|
|
'today': '',
|
|
'tv': { 'entries': [] },
|
|
'cal': { 'today': [], 'tomorrow': [], 'week': [] },
|
|
'swedish': {},
|
|
'fitbit': {},
|
|
'ftse': {}
|
|
}, 'expire': ((60 * 1000) * 60)
|
|
};
|
|
const file = `${__dirname }/` + 'newdata.json';
|
|
const htmlfile = `${__dirname }/` + 'today.html';
|
|
let eventEmitter;
|
|
|
|
function runable() {
|
|
try {
|
|
const now = new Date().getTime();
|
|
console.log(todayCache.last);
|
|
console.log('last updated', ((now - todayCache.last) / 60000));
|
|
if (now - todayCache.last < 3600000)
|
|
return false;
|
|
else {
|
|
todayCache.last = now;
|
|
|
|
return true;
|
|
}
|
|
}
|
|
catch (e) {
|
|
logger.debug('Creating new today object.. Lets go!');
|
|
todayCache = Object.assign({}, todayCacheSrc);
|
|
todayCache.last = new Date().getTime();
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
function broadcastWeather() {
|
|
const wData = {
|
|
'temperature': todayCache.data.weather.data.currently.temperature,
|
|
'icon': todayCache.data.weather.data.currently.icon,
|
|
'summary': todayCache.data.weather.data.currently.summary
|
|
};
|
|
|
|
if (todayCache.data.weather.data.hasOwnProperty('alerts'))
|
|
wData.alerts = todayCache.data.weather.data.alerts;
|
|
|
|
// eventEmitter.emit('sendSocket', { 'id': 'weather', 'data': wData });
|
|
}
|
|
|
|
function loadData() {
|
|
console.log('Loading old data');
|
|
// localStorage.removeItem('today');
|
|
try {
|
|
const tempCache = localStorage.getItem('today');
|
|
if (tempCache !== null)
|
|
todayCache = JSON.parse(tempCache);
|
|
}
|
|
catch (e) {
|
|
console.error('Could not load previous data');
|
|
}
|
|
}
|
|
function saveData() {
|
|
todayCache.last = new Date().getTime();
|
|
logger.info('Saving...');
|
|
// jsonfile.writeFileSync(file, todayCache);
|
|
localStorage.setItem('today', JSON.stringify(todayCache));
|
|
}
|
|
function saveToDB(data) {
|
|
saveData();
|
|
|
|
logger.debug('Inserting into couch...');
|
|
// Logger.info(util.inspect(obj));
|
|
dbCloudant.insert(data, function (err, body, header) {
|
|
if (err) {
|
|
logger.error('Error inserting into couch');
|
|
logger.error(err);
|
|
|
|
return;
|
|
}
|
|
});
|
|
}
|
|
|
|
function nth(d) {
|
|
// If (d > 3 && d < 21) {return 'th';} // Thanks kennebec
|
|
// if (d % 10 === 1) {return 'st';} else if (d % 10 === 2) {return 'nd';} else if (d % 10 === 3) {return 'rd';} else {return 'th';}
|
|
const n = d;
|
|
|
|
return Math.floor(n / 10) === 1 ? 'th' : (n % 10 === 1 ? 'st' : (n % 10 === 2 ? 'nd' : (n % 10 === 3 ? 'rd' : 'th')));
|
|
}
|
|
|
|
function dayNumber() {
|
|
const now = new Date();
|
|
const start = new Date(now.getFullYear(), 0, 0);
|
|
const diff = now - start;
|
|
const oneDay = 1000 * 60 * 60 * 24;
|
|
|
|
return Math.floor(diff / oneDay);
|
|
}
|
|
|
|
function breakDay() {
|
|
const now = new Date();
|
|
|
|
return {
|
|
'year': now.getFullYear(),
|
|
'month': parseInt(now.getMonth()) + 1,
|
|
'day': now.getDate()
|
|
};
|
|
}
|
|
|
|
function reduceTrains(d) {
|
|
const titles = [], ta = [];
|
|
// console.log('reducetrains',d);
|
|
/* for (let items in d) {
|
|
if (typeof d[items].title !== 'undefined') {
|
|
const hash = d[items].title.hashCode();
|
|
if (titles.indexOf(hash) === -1) {
|
|
titles.push(hash);
|
|
ta.push(d[items]);
|
|
}
|
|
|
|
}
|
|
|
|
}*/
|
|
|
|
for (const item in d)
|
|
if (typeof item.title !== 'undefined') {
|
|
const hash = item.title.hashCode();
|
|
if (titles.indexOf(hash) === -1) {
|
|
titles.push(hash);
|
|
ta.push(item);
|
|
}
|
|
}
|
|
|
|
return ta;
|
|
}
|
|
|
|
/**
|
|
* @return {number}
|
|
*/
|
|
function DayDiff(CurrentDate) {
|
|
const TYear = CurrentDate.getFullYear();
|
|
const TDay = new Date(`January, 01, ${ parseInt(TYear) + 1}`);
|
|
TDay.getFullYear(TYear);
|
|
let DayCount = (TDay - CurrentDate) / (1000 * 60 * 60 * 24);
|
|
DayCount = Math.round(DayCount);
|
|
|
|
const d = new Date();
|
|
DayCount = d.daysSince('beginning of this year');
|
|
|
|
return (DayCount);
|
|
}
|
|
|
|
Array.prototype.indexOfOld = Array.prototype.indexOf;
|
|
Array.prototype.indexOf = function (e, fn) {
|
|
if (!fn)
|
|
return this.indexOfOld(e);
|
|
else {
|
|
if (typeof fn === 'string') {
|
|
const att = fn;
|
|
fn = function (e) {
|
|
return e[att];
|
|
};
|
|
}
|
|
|
|
return this.map(fn).indexOfOld(e);
|
|
}
|
|
};
|
|
|
|
module.exports = {
|
|
'setEmitter': function (newEmitter) {
|
|
console.log('Setting events', newEmitter);
|
|
eventEmitter = newEmitter;
|
|
},
|
|
'getClock': function (req, res) {
|
|
// Console.log(todayCache);
|
|
res.render('pages/clock', todayCache);
|
|
},
|
|
'getToday': function (req, res) {
|
|
logger.info(todayCache);
|
|
res.render('pages/today', todayCache);
|
|
},
|
|
'getData': function (req, res) {
|
|
res.setHeader('Content-Type', 'application/json');
|
|
res.end(JSON.stringify(todayCache));
|
|
},
|
|
'getTodayDate': function () {
|
|
let s;
|
|
const d = new Sugar.Date();
|
|
const nextYear = `${(parseInt(d.getFullYear().raw) + 1).toString() }-01-01`;
|
|
|
|
console.log(d.daysUntil('beginning of next year').raw);
|
|
const daysSinceStart = d.daysSince('beginning of this year').raw;
|
|
const daysRemaining = d.daysUntil('beginning of next year').raw;
|
|
|
|
todayCache.data.history = [];
|
|
s = `<strong>${ d.format('{Weekday} {Month} {dd}, {yyyy}').raw }</strong> - `;
|
|
|
|
/*
|
|
s = s + 'The ' + daysSinceStart + nth(daysSinceStart) + ' day of ' + dateFormat(
|
|
d,
|
|
'yyyy') + ', and there are ' + daysRemaining + ' days left until the end of the year.';
|
|
*/
|
|
|
|
s = `${s }The ${daysSinceStart + nth(daysSinceStart)} day of ${d.format('{yyyy}').raw}, and there are ${daysRemaining} days until the end of the year`;
|
|
|
|
logger.debug(s);
|
|
todayCache.data.today = s;
|
|
},
|
|
'refreshTrain': function () {
|
|
trains.updateTrains()
|
|
.then((d) => {
|
|
'use strict';
|
|
|
|
d = reduceTrains(d);
|
|
|
|
// eventEmitter.emit('sendSocket', { 'id': 'trains', 'data': d });
|
|
todayCache.data.trains.data = d;
|
|
todayCache.data.trains.last = new Date();
|
|
})
|
|
.catch((e) => {
|
|
'use strict';
|
|
logger.error(e);
|
|
});
|
|
},
|
|
'refreshWeather': function () {
|
|
weather.newDoGetWeather()
|
|
.then((d) => {
|
|
todayCache.data.weather = d;
|
|
logger.info('Updating weather');
|
|
broadcastWeather();
|
|
}).catch((e) => {
|
|
logger.error(e);
|
|
});
|
|
},
|
|
|
|
'refreshTrainAndWeather': function () {
|
|
this.refreshTrain();
|
|
this.refreshWeather();
|
|
},
|
|
'preLoadToday': async function () {
|
|
function compare(a, b) {
|
|
if (a.ts < b.ts)
|
|
return -1;
|
|
|
|
if (a.ts > b.ts)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
module.exports.getTodayDate();
|
|
todayCache.data.cal = { 'today': [], 'tomorrow': [], 'week': [] };
|
|
weather.newDoGetWeather()
|
|
.then((d) => {
|
|
todayCache.data.weather = d;
|
|
})
|
|
.catch((e) => {
|
|
logger.error(e);
|
|
});
|
|
trains.updateTrains()
|
|
.then((d) => {
|
|
'use strict';
|
|
console.log('Trains: ', d);
|
|
todayCache.data.trains.data = d;
|
|
todayCache.data.trains.last = new Date();
|
|
})
|
|
.catch((e) => {
|
|
'use strict';
|
|
console.error(e);
|
|
});
|
|
history.updateHistory()
|
|
.then((d) => {
|
|
'use strict';
|
|
console.log('History result: ', d);
|
|
todayCache.data.history = d;
|
|
})
|
|
.catch((e) => {
|
|
'use strict';
|
|
console.error(e);
|
|
});
|
|
|
|
calHandler.getSimpleCalV3(
|
|
'http://www.pogdesign.co.uk/cat/download_ics/60cfdff469d0490545d33d7e3b5c0bcc')
|
|
.then((d) => {
|
|
'use strict';
|
|
todayCache.data.tv = d;
|
|
})
|
|
.catch((e) => {
|
|
'use strict';
|
|
logger.error(e);
|
|
});
|
|
|
|
todayFTSE.getFTSE()
|
|
.then((d) => {
|
|
todayCache.data.ftse = d;
|
|
})
|
|
.catch((e) => {
|
|
logger.error(e);
|
|
});
|
|
|
|
quotes.GetQuotes()
|
|
.then((d) => {
|
|
todayCache.data.quotes = d;
|
|
})
|
|
.catch((e) => {
|
|
logger.error(e);
|
|
});
|
|
|
|
/* for (let t = 0; t < calHandler.calendars.length; t++)
|
|
calHandler.getAdvancedCalV3(calHandler.calendars[t])
|
|
.then((d) => {
|
|
'use strict';
|
|
todayCache.data.cal.today = todayCache.data.cal.today.concat(d.today);
|
|
todayCache.data.cal.tomorrow = todayCache.data.cal.tomorrow.concat(d.tomorrow);
|
|
todayCache.data.cal.week = todayCache.data.cal.week.concat(d.week);
|
|
})
|
|
.catch((e) => {
|
|
'use strict';
|
|
logger.error(e);
|
|
});*/
|
|
|
|
for (const item of calHandler.calendars)
|
|
await calHandler.getAdvancedCalV3(item)
|
|
.then((d) => {
|
|
todayCache.data.cal.today = todayCache.data.cal.today.concat(d.today);
|
|
todayCache.data.cal.tomorrow = todayCache.data.cal.tomorrow.concat(d.tomorrow);
|
|
todayCache.data.cal.week = todayCache.data.cal.week.concat(d.week);
|
|
})
|
|
.catch((e) => {
|
|
'use strict';
|
|
logger.error(e);
|
|
});
|
|
|
|
console.log('>> SORT!!');
|
|
todayCache.data.cal.today = todayCache.data.cal.today.sort(compare);
|
|
todayCache.data.cal.tomorrow = todayCache.data.cal.tomorrow.sort(compare);
|
|
todayCache.data.cal.week = todayCache.data.cal.week.sort(compare);
|
|
|
|
swedishWord.getSwedishWord()
|
|
.then((d) => {
|
|
'use strict';
|
|
console.log('Swedish result: ', d);
|
|
todayCache.data.swedish = d;
|
|
})
|
|
.catch((e) => {
|
|
'use strict';
|
|
console.error(e);
|
|
});
|
|
|
|
/* mdFitbit.getYesterdayFitbit()
|
|
.then((d) => {
|
|
todayCache.data.fitbit = d;
|
|
})
|
|
.catch((e) => {
|
|
'use strict';
|
|
console.error(e);
|
|
});*/
|
|
|
|
todayCache.date = breakDay();
|
|
},
|
|
'preLoadTodayTest': function () {
|
|
module.exports.getTodayDate();
|
|
todayCache.data.cal = { 'today': [], 'tomorrow': [], 'week': [] };
|
|
|
|
todayCache.date = breakDay();
|
|
},
|
|
'broadcast': function () {
|
|
console.log('BROADCAST');
|
|
broadcastWeather();
|
|
eventEmitter.emit('sendSocket', { 'id': 'trains', 'data': todayCache.data.trains.data });
|
|
}
|
|
};
|
|
|
|
setTimeout(function () {
|
|
loadData();
|
|
if (runable())
|
|
module.exports.preLoadToday();
|
|
|
|
// Module.exports.preLoadToday();
|
|
}, 5000);
|
|
|
|
setTimeout(function () {
|
|
logger.debug('Going to do the email...');
|
|
// mdMailer.sendEmailV1(todayCache, __dirname);
|
|
|
|
mdMailer.sendSMTP(todayCache, __dirname);
|
|
//
|
|
saveToDB(todayCache);
|
|
saveData();
|
|
}, 45000);
|
|
|
|
setInterval(function () {
|
|
|
|
// EventEmitter.emit('sendSocket',{id:'weather',data:todayCache.data.weather});
|
|
// broadcastWeather();
|
|
// eventEmitter.emit('sendSocket', {id: 'trains', data: todayCache.data.trains.data});
|
|
|
|
}, (60000));
|
|
|
|
cron.schedule('15 7 * * *', function () {
|
|
if (runable()) {
|
|
module.exports.preLoadToday();
|
|
|
|
saveToDB(todayCache);
|
|
saveData();
|
|
}
|
|
|
|
return -1;
|
|
});
|
|
|
|
cron.schedule('30 7 * * *', function () {
|
|
mdMailer.sendSMTP(todayCache, __dirname);
|
|
// saveToDB(todayCache);
|
|
// Console.log('tick');
|
|
return -1;
|
|
});
|
|
|