/** * 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 = `${ d.format('{Weekday} {Month} {dd}, {yyyy}').raw } - `; /* 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; });