You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-desktop/js/modules/loki_rss_api.js

181 lines
4.7 KiB
JavaScript

/* eslint-disable no-await-in-loop */
/* eslint-disable no-loop-func */
/* global log, window, textsecure, ConversationController */
const EventEmitter = require('events');
const nodeFetch = require('node-fetch');
const friendRequestStatusEnum = require('./loki_friend_request_status');
const RSS_FEED = 'https://loki.network/category/messenger-updates/feed/';
const CONVO_ID = 'rss://loki.network/category/messenger-updates/feed/';
const PER_MIN = 60 * 1000;
const PER_HR = 60 * PER_MIN;
const RSS_POLL_EVERY = 1 * PER_HR; // once an hour
/*
const dnsUtil = require('dns');
// how do we get our local version?
// how do we integrate with extensions.expired()
const VERSION_HOST = 'lastreleasedate.messenger.loki.network';
function getLastRelease(cb) {
// doesn't look to have a promise interface
dnsUtil.resolveTxt(VERSION_HOST, function handleResponse(err, records) {
if (err) {
console.error('getLastRelease error', err);
cb();
return;
}
if (records.length) {
cb();
}
// return first record...
cb(records[0]);
});
}
*/
function xml2json(xml) {
try {
let obj = {};
if (xml.children.length > 0) {
for (let i = 0; i < xml.children.length; i += 1) {
const item = xml.children.item(i);
const { nodeName } = item;
if (typeof obj[nodeName] === 'undefined') {
obj[nodeName] = xml2json(item);
} else {
if (typeof obj[nodeName].push === 'undefined') {
const old = obj[nodeName];
obj[nodeName] = [];
obj[nodeName].push(old);
}
obj[nodeName].push(xml2json(item));
}
}
} else {
obj = xml.textContent;
}
return obj;
} catch (e) {
log.error(e.message);
}
return {};
}
class LokiRssAPI extends EventEmitter {
constructor() {
super();
// properties
this.groupId = CONVO_ID;
this.feedTimer = null;
this.conversationSetup = false;
// initial set up
this.getFeed();
}
setupConversation() {
// only run once
if (this.conversationSetup) return;
// wait until conversations are loaded
if (ConversationController._initialFetchComplete) {
const conversation = ConversationController.getOrCreate(
this.groupId,
'group'
);
conversation.setFriendRequestStatus(friendRequestStatusEnum.friends);
conversation.setGroupNameAndAvatar(
'Loki.network News',
'images/loki/loki_icon.png'
);
conversation.updateTextInputState();
this.conversationSetup = true; // prevent running again
}
}
async getFeed() {
let response;
let success = true;
try {
response = await nodeFetch(RSS_FEED);
} catch (e) {
log.error('fetcherror', e);
success = false;
}
const responseXML = await response.text();
let feedDOM = {};
try {
feedDOM = await new window.DOMParser().parseFromString(
responseXML,
'text/xml'
);
} catch (e) {
log.error('xmlerror', e);
success = false;
}
if (!success) return;
const feedObj = xml2json(feedDOM);
let receivedAt = new Date().getTime();
// make sure conversation is set up properly
// (delay to after the network response intentionally)
this.setupConversation();
if (!feedObj || !feedObj.rss || !feedObj.rss.channel) {
log.error('rsserror', feedObj, feedDOM, responseXML);
return;
}
if (!feedObj.rss.channel.item) {
// no records
return;
}
feedObj.rss.channel.item.reverse().forEach(item => {
// log.debug('item', item)
const pubDate = new Date(item.pubDate);
// if we use group style, we can put the title in the source
const messageData = {
friendRequest: false,
source: this.groupId,
sourceDevice: 1,
timestamp: pubDate.getTime(),
serverTimestamp: pubDate.getTime(),
receivedAt,
isRss: true,
message: {
body: `<h2>${item.title} </h2>${item.description}`,
attachments: [],
group: {
id: this.groupId,
type: textsecure.protobuf.GroupContext.Type.DELIVER,
},
flags: 0,
expireTimer: 0,
profileKey: null,
timestamp: pubDate.getTime(),
received_at: receivedAt,
sent_at: pubDate.getTime(),
quote: null,
contact: [],
preview: [],
profile: null,
},
};
receivedAt += 1; // Ensure different arrival times
this.emit('rssMessage', {
message: messageData,
});
});
function callTimer() {
this.getFeed();
}
this.feedTimer = setTimeout(callTimer, RSS_POLL_EVERY);
}
}
module.exports = LokiRssAPI;