define([ '/common/common-messaging.js', '/common/common-hash.js', '/common/common-util.js', ], function (Messaging, Hash, Util) { var getRandomTimeout = function (ctx) { var lag = ctx.store.realtime.getLag().lag || 0; return (Math.max(0, lag) + 300) * 20 * (0.5 + Math.random()); }; var handlers = {}; var removeHandlers = {}; // Store the friend request displayed to avoid duplicates var friendRequest = {}; handlers['FRIEND_REQUEST'] = function (ctx, box, data, cb) { // Check if the request is valid (send by the correct user) if (data.msg.author !== data.msg.content.curvePublic) { return void cb(true); } // Don't show duplicate friend request: if we already have a friend request // in memory from the same user, dismiss the new one if (friendRequest[data.msg.author]) { return void cb(true); } friendRequest[data.msg.author] = true; // If the user is already in our friend list, automatically accept the request if (Messaging.getFriend(ctx.store.proxy, data.msg.author) || ctx.store.proxy.friends_pending[data.msg.author]) { Messaging.acceptFriendRequest(ctx.store, data.msg.content, function (obj) { if (obj && obj.error) { return void cb(); } cb(true); }); return; } cb(); }; removeHandlers['FRIEND_REQUEST'] = function (ctx, box, data) { if (friendRequest[data.content.curvePublic]) { delete friendRequest[data.content.curvePublic]; } }; var friendRequestDeclined = {}; handlers['DECLINE_FRIEND_REQUEST'] = function (ctx, box, data, cb) { setTimeout(function () { // Our friend request was declined. if (!ctx.store.proxy.friends_pending[data.msg.author]) { return; } // Remove the pending message and display the "declined" state in the UI delete ctx.store.proxy.friends_pending[data.msg.author]; ctx.updateMetadata(); if (friendRequestDeclined[data.msg.author]) { return; } box.sendMessage({ type: 'FRIEND_REQUEST_DECLINED', content: { user: data.msg.author, name: data.msg.content.displayName } }, function () { if (friendRequestDeclined[data.msg.author]) { // TODO remove our message because another one was sent first? } friendRequestDeclined[data.msg.author] = true; }); }, getRandomTimeout(ctx)); cb(true); }; handlers['FRIEND_REQUEST_DECLINED'] = function (ctx, box, data, cb) { ctx.updateMetadata(); if (friendRequestDeclined[data.msg.content.user]) { return void cb(true); } friendRequestDeclined[data.msg.content.user] = true; cb(); }; removeHandlers['FRIEND_REQUEST_DECLINED'] = function (ctx, box, data) { if (friendRequestDeclined[data.content.user]) { delete friendRequestDeclined[data.content.user]; } }; var friendRequestAccepted = {}; handlers['ACCEPT_FRIEND_REQUEST'] = function (ctx, box, data, cb) { // Our friend request was accepted. setTimeout(function () { // Make sure we really sent it if (!ctx.store.proxy.friends_pending[data.msg.author]) { return; } // And add the friend Messaging.addToFriendList({ proxy: ctx.store.proxy, realtime: ctx.store.realtime, pinPads: ctx.pinPads }, data.msg.content, function (err) { if (err) { console.error(err); } delete ctx.store.proxy.friends_pending[data.msg.author]; if (ctx.store.messenger) { ctx.store.messenger.onFriendAdded(data.msg.content); } ctx.updateMetadata(); // If you have a profile page open, update it if (ctx.store.modules['profile']) { ctx.store.modules['profile'].update(); } if (friendRequestAccepted[data.msg.author]) { return; } // Display the "accepted" state in the UI box.sendMessage({ type: 'FRIEND_REQUEST_ACCEPTED', content: { user: data.msg.author, name: data.msg.content.displayName } }, function () { if (friendRequestAccepted[data.msg.author]) { // TODO remove our message because another one was sent first? } friendRequestAccepted[data.msg.author] = true; }); }); }, getRandomTimeout(ctx)); cb(true); }; handlers['FRIEND_REQUEST_ACCEPTED'] = function (ctx, box, data, cb) { ctx.updateMetadata(); if (friendRequestAccepted[data.msg.content.user]) { return void cb(true); } friendRequestAccepted[data.msg.content.user] = true; cb(); }; removeHandlers['FRIEND_REQUEST_ACCEPTED'] = function (ctx, box, data) { if (friendRequestAccepted[data.content.user]) { delete friendRequestAccepted[data.content.user]; } }; handlers['UNFRIEND'] = function (ctx, box, data, cb) { var curve = data.msg.content.curvePublic; var friend = Messaging.getFriend(ctx.store.proxy, curve); if (!friend) { return void cb(true); } delete ctx.store.proxy.friends[curve]; if (ctx.store.messenger) { ctx.store.messenger.onFriendRemoved(curve, friend.channel); } ctx.updateMetadata(); cb(true); }; handlers['UPDATE_DATA'] = function (ctx, box, data, cb) { var msg = data.msg; var curve = msg.author; var friend = ctx.store.proxy.friends && ctx.store.proxy.friends[curve]; if (!friend || typeof msg.content !== "object") { return void cb(true); } Object.keys(msg.content).forEach(function (key) { friend[key] = msg.content[key]; }); if (ctx.store.messenger) { ctx.store.messenger.onFriendUpdate(curve); } ctx.updateMetadata(); cb(true); }; // Hide duplicates when receiving a SHARE_PAD notification: // Keep only one notification per channel: the stronger and more recent one var channels = {}; handlers['SHARE_PAD'] = function (ctx, box, data, cb) { var msg = data.msg; var hash = data.hash; var content = msg.content; // content.name, content.title, content.href, content.password var channel = Hash.hrefToHexChannelId(content.href, content.password); var parsed = Hash.parsePadUrl(content.href); var mode = parsed.hashData && parsed.hashData.mode || 'n/a'; var old = channels[channel]; var toRemove; if (old) { // New hash is weaker, ignore if (old.mode === 'edit' && mode === 'view') { return void cb(true); } // New hash is not weaker, clear the old one toRemove = old.data; } // Update the data channels[channel] = { mode: mode, data: { type: box.type, hash: hash } }; cb(false, toRemove); }; removeHandlers['SHARE_PAD'] = function (ctx, box, data, hash) { var content = data.content; var channel = Hash.hrefToHexChannelId(content.href, content.password); var old = channels[channel]; if (old && old.data && old.data.hash === hash) { delete channels[channel]; } }; // Hide duplicates when receiving a SUPPORT_MESSAGE notification var supportMessage = false; handlers['SUPPORT_MESSAGE'] = function (ctx, box, data, cb) { if (supportMessage) { return void cb(true); } supportMessage = true; cb(); }; // Incoming edit rights request: add data before sending it to inner handlers['REQUEST_PAD_ACCESS'] = function (ctx, box, data, cb) { var msg = data.msg; var content = msg.content; if (msg.author !== content.user.curvePublic) { return void cb(true); } var channel = content.channel; var res = ctx.store.manager.findChannel(channel); if (!res.length) { return void cb(true); } var edPublic = ctx.store.proxy.edPublic; var title, href; if (!res.some(function (obj) { if (obj.data && Array.isArray(obj.data.owners) && obj.data.owners.indexOf(edPublic) !== -1 && obj.data.href) { href = obj.data.href; title = obj.data.filename || obj.data.title; return true; } })) { return void cb(true); } content.title = title; content.href = href; cb(false); }; handlers['GIVE_PAD_ACCESS'] = function (ctx, box, data, cb) { var msg = data.msg; var content = msg.content; if (msg.author !== content.user.curvePublic) { return void cb(true); } var channel = content.channel; var res = ctx.store.manager.findChannel(channel, true); var title; res.forEach(function (obj) { if (obj.data && !obj.data.href) { if (!title) { title = obj.data.filename || obj.data.title; } obj.userObject.setHref(channel, null, content.href); } }); content.title = title || content.title; cb(false); }; // Hide duplicates when receiving an ADD_OWNER notification: var addOwners = {}; handlers['ADD_OWNER'] = function (ctx, box, data, cb) { var msg = data.msg; var content = msg.content; if (msg.author !== content.user.curvePublic) { return void cb(true); } if (!content.teamChannel && !(content.href && content.title && content.channel)) { console.log('Remove invalid notification'); return void cb(true); } var channel = content.channel || content.teamChannel; if (addOwners[channel]) { return void cb(true); } addOwners[channel] = { type: box.type, hash: data.hash }; cb(false); }; removeHandlers['ADD_OWNER'] = function (ctx, box, data) { var channel = data.content.channel || data.content.teamChannel; if (addOwners[channel]) { delete addOwners[channel]; } }; handlers['RM_OWNER'] = function (ctx, box, data, cb) { var msg = data.msg; var content = msg.content; if (msg.author !== content.user.curvePublic) { return void cb(true); } if (!content.channel && !content.teamChannel) { console.log('Remove invalid notification'); return void cb(true); } var channel = content.channel || content.teamChannel; // If our ownership rights for a team have been removed, update the owner flag if (content.teamChannel) { var teams = ctx.store.proxy.teams || {}; Object.keys(teams).some(function (id) { if (teams[id].channel === channel) { teams[id].owner = false; return true; } }); } if (addOwners[channel] && content.pending) { return void cb(false, addOwners[channel]); } cb(false); }; var invitedTo = {}; handlers['INVITE_TO_TEAM'] = function (ctx, box, data, cb) { var msg = data.msg; var content = msg.content; if (msg.author !== content.user.curvePublic) { return void cb(true); } if (!content.team) { console.log('Remove invalid notification'); return void cb(true); } var invited = invitedTo[content.team.channel]; if (invited) { console.log('removing old invitation'); cb(false, invited); invitedTo[content.team.channel] = { type: box.type, hash: data.hash }; return; } var myTeams = Util.find(ctx, ['store', 'proxy', 'teams']) || {}; var alreadyMember = Object.keys(myTeams).some(function (k) { var team = myTeams[k]; return team.channel === content.team.channel; }); if (alreadyMember) { return void cb(true); } invitedTo[content.team.channel] = { type: box.type, hash: data.hash }; cb(false); }; removeHandlers['INVITE_TO_TEAM'] = function (ctx, box, data) { var channel = Util.find(data, ['content', 'team', 'channel']); delete invitedTo[channel]; }; handlers['KICKED_FROM_TEAM'] = function (ctx, box, data, cb) { var msg = data.msg; var content = msg.content; if (msg.author !== content.user.curvePublic) { return void cb(true); } if (!content.teamChannel) { console.log('Remove invalid notification'); return void cb(true); } if (invitedTo[content.teamChannel] && content.pending) { return void cb(true, invitedTo[content.teamChannel]); } cb(false); }; handlers['INVITE_TO_TEAM_ANSWER'] = function (ctx, box, data, cb) { var msg = data.msg; var content = msg.content; if (msg.author !== content.user.curvePublic) { return void cb(true); } if (!content.teamChannel) { console.log('Remove invalid notification'); return void cb(true); } var myTeams = Util.find(ctx, ['store', 'proxy', 'teams']) || {}; var teamId; var team; Object.keys(myTeams).some(function (k) { var _team = myTeams[k]; if (_team.channel === content.teamChannel) { teamId = k; team = _team; return true; } }); if (!teamId) { return void cb(true); } content.team = team; if (!content.answer) { // If they declined the invitation, remove them from the roster (as a pending member) try { var module = ctx.store.modules['team']; module.removeFromTeam(teamId, msg.author); } catch (e) { console.error(e); } } cb(false); }; return { add: function (ctx, box, data, cb) { /** * data = { msg: { type: 'STRING', author: 'curvePublicString', content: {} (depend on the "type") }, hash: 'string' } */ if (!data.msg) { return void cb(true); } var type = data.msg.type; if (handlers[type]) { try { handlers[type](ctx, box, data, cb); } catch (e) { console.error(e); cb(); } } else { cb(); } }, remove: function (ctx, box, data, h) { // We sometimes try to delete non-existant data (with "delete box.content[h]") // In this case, we don't have the data in memory so we don't need to call // any "remove" handler if (!data) { return; } var type = data.type; if (removeHandlers[type]) { try { removeHandlers[type](ctx, box, data, h); } catch (e) { console.error(e); } } } }; });