// twister_io.js // 2013 Miguel Freitas // // low-level twister i/o. // implements requests of dht resources. multiple pending requests to the same resource are joined. // cache results (profile, avatar, etc) in memory. // avatars are cached in localstored (expiration = 24 hours) // main json rpc method. receives callbacks for success and error function twisterRpc(method, params, resultFunc, resultArg, errorFunc, errorArg) { // removing hardcoded username from javascript: please use url http://user:pwd@localhost:28332 instead //var foo = new $.JsonRpcClient({ ajaxUrl: '/', username: 'user', password: 'pwd'}); var foo = new $.JsonRpcClient({ajaxUrl: window.location.pathname.replace(/[^\/]*$/, '')}); foo.call(method, params, function(ret) {resultFunc(resultArg, ret);}, function(ret) {errorFunc(errorArg, ret);} // FIXME why only if "(ret != null)"? ); } // join multiple dhtgets to the same resources in this map var _dhtgetPendingMap = {}; // memory cache for profile and avatar var _profileMap = {}; var _avatarMap = {}; var _pubkeyMap = {}; // number of dhtgets in progress (requests to the daemon) var _dhtgetsInProgress = 0; // keep _maxDhtgets smaller than the number of daemon/browser sockets // most browsers limit to 6 per domain (see http://www.browserscope.org/?category=network) var _maxDhtgets = 5; // requests not yet sent to the daemon due to _maxDhtgets limit var _queuedDhtgets = []; // private function to define a key in _dhtgetPendingMap function _dhtgetLocator(peerAlias, resource, multi) { return peerAlias + ';' + resource + ';' + multi; } function _dhtgetAddPending(locator, cbFunc, cbReq) { if (!_dhtgetPendingMap[locator]) { _dhtgetPendingMap[locator] = []; } _dhtgetPendingMap[locator].push({cbFunc: cbFunc, cbReq: cbReq}); } function _dhtgetProcessPending(locator, multi, ret) { if (_dhtgetPendingMap[locator]) { for (var i = 0; i < _dhtgetPendingMap[locator].length; i++) { var cbFunc = _dhtgetPendingMap[locator][i].cbFunc; var cbReq = _dhtgetPendingMap[locator][i].cbReq; if (multi === 'url') { cbFunc(cbReq, ret); // here is decodeshorturl case } else if (multi === 's') { if (typeof ret[0] !== 'undefined') { cbFunc(cbReq, ret[0].p.v, ret); } else { cbFunc(cbReq); } } else { var multiret = []; for (var j = 0; j < ret.length; j++) { multiret.push(ret[j].p.v); } cbFunc(cbReq, multiret, ret); } } delete _dhtgetPendingMap[locator]; } else { console.warn('_dhtgetProcessPending(): unknown locator ' + locator); } } function _dhtgetAbortPending(locator) { if (_dhtgetPendingMap[locator]) { for (var i = 0; i < _dhtgetPendingMap[locator].length; i++) { var cbFunc = _dhtgetPendingMap[locator][i].cbFunc; var cbReq = _dhtgetPendingMap[locator][i].cbReq; cbFunc(cbReq); } delete _dhtgetPendingMap[locator]; } else { console.warn('_dhtgetAbortPending(): unknown locator ' + locator); } } // get data from dht resource // the value ["v"] is extracted from response and returned to callback // null is passed to callback in case of an error function dhtget(peerAlias, resource, multi, cbFunc, cbReq, timeoutArgs) { //console.log('dhtget ' + peerAlias + ' ' + resource + ' ' + multi); var locator = _dhtgetLocator(peerAlias, resource, multi); if (_dhtgetPendingMap[locator]) { _dhtgetAddPending(locator, cbFunc, cbReq); } else { _dhtgetAddPending(locator, cbFunc, cbReq); // limit the number of simultaneous dhtgets. // this should leave some sockets for other non-blocking daemon requests. if (_dhtgetsInProgress < _maxDhtgets) { _dhtgetInternal(peerAlias, resource, multi, timeoutArgs); } else { // just queue the locator. it will be unqueue when some dhtget completes. _queuedDhtgets.push(locator); } } } // decode shortened url // the expanded url is returned to callback // null is passed to callback in case of an error function decodeShortURI(locator, cbFunc, cbReq, timeoutArgs) { if (!locator) return; if (parseInt(twisterVersion) < 93500) { console.warn('can\'t fetch URI "' + req + '" — ' + polyglot.t('daemon_is_obsolete', {versionReq: '0.9.35'})); return; } if (_dhtgetPendingMap[locator]) { _dhtgetAddPending(locator, cbFunc, cbReq); } else { _dhtgetAddPending(locator, cbFunc, cbReq); // limit the number of simultaneous decodeshorturl's and dhtgets. // this should leave some sockets for other non-blocking daemon requests. if (_dhtgetsInProgress < _maxDhtgets) { _decodeshorturlInternal(locator, timeoutArgs); } else { // just queue the locator. it will be unqueue when some dhtget completes. _queuedDhtgets.push(locator); } } } function _dhtgetInternal(peerAlias, resource, multi, timeoutArgs) { var locator = _dhtgetLocator(peerAlias, resource, multi); _dhtgetsInProgress++; argsList = [peerAlias, resource, multi]; if (typeof timeoutArgs !== 'undefined') { argsList = argsList.concat(timeoutArgs); } twisterRpc('dhtget', argsList, function(req, ret) { _dhtgetsInProgress--; _dhtgetProcessPending(req.locator, req.multi, ret); _dhtgetDequeue(); }, {locator: locator, multi: multi}, function(req, ret) { console.warn('RPC "dhtget" error: ' + (ret && ret.message ? ret.message : ret)); _dhtgetsInProgress--; _dhtgetAbortPending(req); _dhtgetDequeue(); }, locator ); } function _decodeshorturlInternal(locator, timeoutArgs) { _dhtgetsInProgress++; argsList = [locator]; if (typeof timeoutArgs !== 'undefined') { argsList = argsList.concat(timeoutArgs); } twisterRpc('decodeshorturl', argsList, function(req, ret) { _dhtgetsInProgress--; _dhtgetProcessPending(req, 'url', ret); _dhtgetDequeue(); }, locator, function(req, ret) { console.warn('RPC "decodeshorturl" error: ' + (ret && ret.message ? ret.message : ret)); _dhtgetsInProgress--; _dhtgetAbortPending(req); _dhtgetDequeue(); }, locator ); } function _dhtgetDequeue() { if (_queuedDhtgets.length) { var locator = _queuedDhtgets.pop(); var locatorSplit = locator.split(';'); if (locatorSplit.length === 3) { _dhtgetInternal(locatorSplit[0], locatorSplit[1], locatorSplit[2]); } else { _decodeshorturlInternal( locator ) } } } // removes queued dhtgets (requests that have not been made to the daemon) // this is used by user search dropdown to discard old users we are not interested anymore function removeUserFromDhtgetQueue(peerAlias) { var resources = ['profile', 'avatar'] for (var i = 0; i < resources.length; i++) { var locator = _dhtgetLocator(peerAlias, resources[i], 's'); var locatorIndex = _queuedDhtgets.indexOf(locator); if (locatorIndex > -1) { _queuedDhtgets.splice(locatorIndex, 1); delete _dhtgetPendingMap[locator]; } } } function removeUsersFromDhtgetQueue(users) { for (var i = 0; i < users.length; i++) { removeUserFromDhtgetQueue(users[i]); } } // store value at the dht resource function dhtput(peerAlias, resource, multi, value, sig_user, seq, cbFunc, cbReq) { twisterRpc('dhtput', [peerAlias, resource, multi, value, sig_user, seq], function(req, ret) { if (req.cbFunc) req.cbFunc(req.cbReq, true); }, {cbFunc: cbFunc, cbReq: cbReq}, function(req, ret) { console.warn('RPC "dhtput" error: ' + (ret && ret.message ? ret.message : ret)); if (req.cbFunc) req.cbFunc(req.cbReq, false); }, {cbFunc: cbFunc, cbReq: cbReq} ); } // get something from profile and store it in elem.text or do callback function getProfileResource(peerAlias, resource, elem, cbFunc, cbReq) { var profile; if (_profileMap[peerAlias]) { profile = _profileMap[peerAlias]; } else { profile = _getResourceFromStorage('profile:' + peerAlias); } if (profile) { _profileMap[peerAlias] = profile; if (elem) elem.text(profile[resource]); if (cbFunc) cbFunc(cbReq, profile[resource]); } else { dhtget(peerAlias, 'profile', 's', function(req, profile) { if (profile) { _profileMap[req.peerAlias] = profile; _putResourceIntoStorage('profile:' + peerAlias, profile); if (req.elem) req.elem.text(profile[resource]); if (req.cbFunc) req.cbFunc(req.cbReq, profile[resource]); } else { if (req.cbFunc) req.cbFunc(req.cbReq); } }, {peerAlias: peerAlias, elem: elem, cbFunc: cbFunc, cbReq: cbReq} ); } } // get fullname and store it in elem.text function getFullname(peerAlias, elem) { elem.text(peerAlias); // fallback: set the peerAlias first in case the profile has no fullname getProfileResource(peerAlias, 'fullname', undefined, function(req, name) { if (name && (name = name.trim())) req.elem.text(name); if (typeof twisterFollowingO !== 'undefined' && // FIXME delete this check when you fix client init sequence ($.Options.isFollowingMe.val === 'everywhere' || req.elem.hasClass('profile-name'))) { // here we try to detect if peer follows us and then display it if (twisterFollowingO.knownFollowers.indexOf(req.peerAlias) > -1) { req.elem.addClass('isFollowing'); req.elem.attr('title', polyglot.t('follows you')); } else if (twisterFollowingO.notFollowers.indexOf(req.peerAlias) === -1) { if (twisterFollowingO.followingsFollowings[req.peerAlias] && twisterFollowingO.followingsFollowings[req.peerAlias].following) { if (twisterFollowingO.followingsFollowings[req.peerAlias].following.indexOf(defaultScreenName) > -1) { if (twisterFollowingO.knownFollowers.indexOf(req.peerAlias) === -1) { twisterFollowingO.knownFollowers.push(req.peerAlias); twisterFollowingO.save(); addPeerToFollowersList(getElem('.followers-modal .followers-list'), req.peerAlias, true); $('.module.mini-profile .open-followers') .attr('title', twisterFollowingO.knownFollowers.length.toString()); } req.elem.addClass('isFollowing'); req.elem.attr('title', polyglot.t('follows you')); } } else { loadFollowingFromDht(req.peerAlias, 1, [], 0, function (req, following, seqNum) { if (following.indexOf(defaultScreenName) > -1) { if (twisterFollowingO.knownFollowers.indexOf(req.peerAlias) === -1) { twisterFollowingO.knownFollowers.push(req.peerAlias); addPeerToFollowersList(getElem('.followers-modal .followers-list'), req.peerAlias, true); $('.module.mini-profile .open-followers') .attr('title', twisterFollowingO.knownFollowers.length.toString()); } req.elem.addClass('isFollowing'); req.elem.attr('title', polyglot.t('follows you')); } else if (twisterFollowingO.notFollowers.indexOf(req.peerAlias) === -1) twisterFollowingO.notFollowers.push(req.peerAlias); twisterFollowingO.save(); }, {elem: req.elem, peerAlias: req.peerAlias} ); } } } }, {elem: elem, peerAlias: peerAlias} ); } // get bio, format it as post message and store result to elem function getBioToElem(peerAlias, elem) { getProfileResource(peerAlias, 'bio', undefined, fillElemWithTxt, elem); } // get tox address and store it in elem.text function getTox(peerAlias, elem) { getProfileResource(peerAlias, 'tox', false, function(elem, val) { if (val) { elem.attr('href', 'tox:' + val); elem.next().attr('data', val).attr('title', 'Copy to clipboard'); elem.parent().css('display', 'inline-block').parent().show(); } }, elem ); } // get bitmessage address and store it in elem.text function getBitmessage(peerAlias, elem) { getProfileResource(peerAlias, 'bitmessage', false, function(elem, val) { if (val) { elem.attr('href', 'bitmsg:' + val + '?action=add&label=' + peerAlias); elem.next().attr('data', val).attr('title', 'Copy to clipboard'); elem.parent().css('display', 'inline-block').parent().show(); } }, elem ); } // get location and store it in elem.text function getLocation(peerAlias, elem) { getProfileResource(peerAlias, 'location', elem); } // get location and store it in elem.text function getWebpage(peerAlias, elem) { getProfileResource(peerAlias, 'url', elem, function(elem, val) { if (typeof(val) !== 'undefined') { if (val.indexOf('://') < 0) { val = 'http://' + val; } elem.attr('href', val); } }, elem ); } function getGroupChatName(groupAlias, elem) { twisterRpc('getgroupinfo', [groupAlias], function(elem, ret) { elem.text(ret.description); }, elem, function(req, ret) { console.warn('RPC "getgroupinfo" error: ' + (ret && ret.message ? ret.message : ret)); req.elem.text(req.groupAlias); }, {elem: elem, groupAlias: groupAlias} ); } // we must cache avatar results to disk to lower bandwidth on // other peers. dht server limits udp rate so requesting too much // data will only cause new requests to fail. function _getResourceFromStorage(locator) { var storage = $.localStorage; if (storage.isSet(locator)) { var storedResource = storage.get(locator); var curTime = new Date().getTime() / 1000; // avatar is downloaded once per day FIXME why once per day? what about profiles? // FIXME need to check what type of data is requested and what time is allowed for it if (storedResource.time + 86400 > curTime) { // 3600 * 24 return storedResource.data; } } return null; } function _putResourceIntoStorage(locator, data) { $.localStorage.set(locator, { time: Math.trunc(new Date().getTime() / 1000), data: data }); } function cleanupStorage() { var curTime = new Date().getTime() / 1000; var storage = $.localStorage, keys = storage.keys(), item = ''; var delAvatars = delProfiles = 0; for (var i = 0; i < keys.length; i++) { item = keys[i]; // FIXME need to decide what time for type of data is allowed if (item.substr(0, 7) === 'avatar:') { if (storage.get(item).time + 86400 < curTime) { // 3600 * 24 hours storage.remove(item); delAvatars++; //console.log('local storage item \'' + item + '\' was too old, deleted'); } } else if (item.substr(0, 8) === 'profile:') { if (storage.get(item).time + 86400 < curTime) { // 3600 * 24 hours storage.remove(item); delProfiles++; //console.log('local storage item \'' + item + '\' was too old, deleted'); } } } console.log('cleaning of storage is completed for ' + (new Date().getTime() / 1000 - curTime) + 's'); if (delAvatars) console.log(' ' + delAvatars + ' cached avatars was too old, deleted'); if (delProfiles) console.log(' ' + delProfiles + ' cached profiles was too old, deleted'); console.log(' ' + 'there was ' + i + ' items in total, now ' + (i - delAvatars - delProfiles)); } // get avatar and set it in img.attr("src") // TODO rename to getAvatarImgToELem(), move nin theme related stuff to nin's theme_option.js function getAvatar(peerAlias, img) { if (!img.length) return; if (peerAlias === 'nobody') { img.attr('src', ($.Options.theme.val === 'nin') ? 'theme_nin/img/tornado_avatar.png' : 'img/tornado_avatar.png'); return; } if (_avatarMap[peerAlias]) { //img.attr('src', 'data:image/jpg;base64,'+avatarMap[peerAlias]); img.attr('src', _avatarMap[peerAlias]); } else { var data = _getResourceFromStorage('avatar:' + peerAlias); if (data) { switch (data.substr(0, 4)) { case 'jpg/': data = '' + window.btoa(data.slice(4)); break; case 'png/': data = 'data:image/png;base64,' + window.btoa(data.slice(4)); break; case 'gif/': data = 'data:image/gif;base64,' + window.btoa(data.slice(4)); break; } _avatarMap[peerAlias] = data; img.attr('src', data); } else { dhtget(peerAlias, 'avatar', 's', function(req, imagedata) { if (imagedata && imagedata.length) { _avatarMap[req.peerAlias] = imagedata; if (imagedata !== 'img/genericPerson.png') { if (imagedata.substr(0, 27) === '') _putResourceIntoStorage('avatar:' + peerAlias, 'jpg/' + window.atob(imagedata.slice(27))); else { var s = imagedata.substr(0, 22); if (s === 'data:image/png;base64,' || s === 'data:image/gif;base64,') _putResourceIntoStorage('avatar:' + peerAlias, imagedata.substr(11, 3) + '/' + window.atob(imagedata.slice(22))); else _putResourceIntoStorage('avatar:' + peerAlias, imagedata); } } req.img.attr('src', imagedata); } }, {peerAlias: peerAlias, img: img} ); } } } function clearAvatarAndProfileCache(peerAlias) { var storage = $.localStorage; storage.remove('avatar:' + peerAlias); storage.remove('profile:' + peerAlias); if (_avatarMap[peerAlias]) { delete _avatarMap[peerAlias]; } if (_profileMap[peerAlias]) { delete _profileMap[peerAlias]; } } // get estimative for number of followers (use known peers of torrent tracker) function getFollowers(peerAlias, elem) { dhtget(peerAlias, 'tracker', 'm', function(elem, ret) { if (ret && ret.length && ret[0].followers) { elem.text(ret[0].followers) } }, elem ); } function getPostsCount(peerAlias, elem) { dhtget(peerAlias, 'status', 's', function(req, v) { var count = 0; if (v && v.userpost) { count = v.userpost.k + 1; } var oldCount = parseInt(req.elem.text()); if (!oldCount || count > oldCount) { req.elem.text(count); } if (peerAlias === defaultScreenName && count) { incLastPostId(v.userpost.k); } }, {peerAlias: peerAlias, elem: elem} ); } function getStatusTime(peerAlias, elem) { dhtget(peerAlias, 'status', 's', function (req, ret) { if (!ret || !ret.userpost) return; req.elem.text(timeGmtToText(ret.userpost.time)) .closest('.latest-activity') .attr('data-screen-name', req.peerAlias) .attr('data-id', ret.userpost.k) .attr('data-time', ret.userpost.time) ; }, {peerAlias: peerAlias, elem: elem} ); } function getPostMaxAvailability(peerAlias, k, cbFunc, cbReq) { twisterRpc('getpiecemaxseen', [peerAlias, k], function(req, ret) { req.cbFunc(req.cbReq, ret); }, {cbFunc: cbFunc, cbReq: cbReq}, function(req, ret) { console.warn('RPC "getpiecemaxseen" error: ' + (ret && ret.message ? ret.message : ret)); } ); } function checkPubkeyExists(peerAlias, cbFunc, cbReq) { // pubkey is checked in block chain db. // so only accepted registrations are reported (local wallet users are not) twisterRpc('dumppubkey', [peerAlias], function(req, ret) { req.cbFunc(req.cbReq, ret.length > 0); }, {cbFunc: cbFunc, cbReq: cbReq}, function(req, ret) { console.warn('RPC "dumppubkey" error: ' + (ret && ret.message ? ret.message : ret)); alert(polyglot.t('error_connecting_to_daemon')); } ); } // pubkey is obtained from block chain db. // so only accepted registrations are reported (local wallet users are not) // cbFunc is called as cbFunc(cbReq, pubkey) // if user doesn't exist then pubkey.length == 0 function dumpPubkey(peerAlias, cbFunc, cbReq) { if (_pubkeyMap[peerAlias]) { if (cbFunc) cbFunc(cbReq, _pubkeyMap[peerAlias]); } else { twisterRpc('dumppubkey', [peerAlias], function (req, ret) { if (ret.length > 0) { _pubkeyMap[peerAlias] = ret; } if (req.cbFunc) { req.cbFunc(req.cbReq, ret); } }, {cbFunc: cbFunc, cbReq: cbReq}, function (req, ret) { console.warn('RPC "dumppubkey" error: ' + (ret && ret.message ? ret.message : ret)); alert(polyglot.t('error_connecting_to_daemon')); } ); } } // privkey is obtained from wallet db // so privkey is returned even for unsent transactions function dumpPrivkey(peerAlias, cbFunc, cbReq) { twisterRpc('dumpprivkey', [peerAlias], function(req, ret) { req.cbFunc(req.cbReq, ret); }, {cbFunc: cbFunc, cbReq: cbReq}, function(req, ret) { req.cbFunc(req.cbReq, ''); console.warn('user unknown — RPC "dumppubkey" error: ' + (ret && ret.message ? ret.message : ret)); }, {cbFunc: cbFunc, cbReq: cbReq} ); }