Skip to content

HTTPS clone URL

Subversion checkout URL

You can clone with
or
.
Download ZIP
Browse files

Disabling bugzilla, adding mentions.

  • Loading branch information...
commit ad42a3e1768737076d9d77f98fc4ae95d33e6d2b 1 parent 57a214e
@simonwex simonwex authored
View
2  app/feeds/daemon.js
@@ -5,7 +5,7 @@
* You can obtain one at http://mozilla.org/MPL/2.0/. */
require('../../lib/extensions/number');
-require('./bugzilla');
+// require('./bugzilla');
var
maxStoryAge = (1).hours(),
View
71 app/http/controllers/profile.js
@@ -14,7 +14,35 @@ Redis = require('../../../lib/redis');
*/
exports.index = {
get: function(req, res){
- res.render('profile/index', { user: req.user });
+ mysql.query(
+ "SELECT name FROM irc_channels",
+ function(err, rows){
+ if (err){
+ logger.error("Error retrieving IRC Channels in profile.js: " + err);
+ }
+
+ for(var i in rows){
+ rows[i].displayName = rows[i].name.substr(1);
+ }
+
+ mysql.query(
+ "SELECT title, url, verified FROM feeds WHERE user_id = ?",
+ [req.user.id],
+ function(err, feedRows){
+ mysql.query(
+ "SELECT token FROM watched_tokens WHERE user_id = ?",
+ [req.user.id],
+ function(err, tokens){
+ for(var i in tokens){
+ tokens[i] = tokens[i].token;
+ }
+ res.render('profile/index', { user: req.user, channels: rows, feeds: feedRows, watchedTokens: tokens});
+ }
+ );
+ }
+ );
+ }
+ );
},
put: function(req, res){
// Only allow the saving of nick and realName
@@ -32,6 +60,47 @@ exports.index = {
}
};
+exports.watchedTokens = {
+ put: function(req, res){
+ console.log(req.body);
+ var tokens = req.body.tokens;
+ var values = [];
+ var placeholders = [];
+ for (var i in tokens){
+ //Validate
+ if (tokens[i].match(/^\w[\w\-\_]+$/)){
+ values.push(req.user.id)
+ values.push(tokens[i]);
+ placeholders.push('(?, ?)');
+ }
+ else {
+ res.send('"ERROR"', {status: 500});
+ return;
+ }
+ }
+
+ mysql.query(
+ "DELETE FROM watched_tokens WHERE user_id = ?",
+ [req.user.id],
+ function(err, result){
+ if (err)
+ logger.error("Error deleting old tokens");
+
+ mysql.query(
+ "INSERT INTO watched_tokens (user_id, token) VALUES " + placeholders.join(','),
+ values,
+ function(err, result){
+ if (err)
+ logger.error("Error inserting new tokens: " + err);
+
+ res.send('"OK"');
+ }
+ );
+ }
+ );
+ }
+}
+
/*
* POST
*/
View
33 app/http/controllers/site.js
@@ -3,9 +3,10 @@
* You can obtain one at http://mozilla.org/MPL/2.0/. */
const
-createRedisClient = require('../../../lib/redis'),
-
+redis = require('../../../lib/redis'),
+uuid = require('node-uuid'),
logger = require('../../../lib/logger'),
+crypto = require('crypto'),
config = require('../../../lib/configuration');
/*
@@ -25,8 +26,7 @@ exports.authenticate = function(req, res) {
exports.signout = function(req, res){
if (req.user){
- var redis = createRedisClient();
- redis.publish('user.signout', req.user.id.toString(), function(err){
+ redis.pub.publish('user.signout', req.user.id.toString(), function(err){
if (err)
logger.error(err);
});
@@ -34,3 +34,28 @@ exports.signout = function(req, res){
req.logout();
res.redirect('/');
};
+
+
+exports.driver = function(req, res){
+ var storyId = null;
+
+ if ('story' in req.body){
+ storyId = req.user.email + ":" + uuid.v1()
+ var hash = crypto.createHash('md5').update(req.user.email).digest("hex");
+
+ var story = {
+ id: storyId,
+ people: req.user.email,
+ title: req.body.story.title,
+ href: req.body.story.url,
+ pubdate: new Date(),
+ image: {url: "http://www.gravatar.com/avatar/" + hash + "?s=30"}
+ };
+
+ redis.io.lpush("serializer:stories", JSON.stringify({'userIds': [req.user.id], 'story': story}), function (err, reply) {
+ if (err)
+ logger.error(err);
+ });
+ }
+ res.render('site/driver', {user: req.user, lastId: storyId});
+}
View
16 app/http/controllers/social.js
@@ -25,23 +25,22 @@ exports.manifest = function(req, res){
res.render('social/manifest.json.ejs', { baseUrl: config.get("public_url"), providerSuffix: config.get("social_provider")['name_suffix'], layout: false });
};
-exports.bugs = function(req, res){
- mysql.query('SELECT * FROM stories WHERE user_id = ? AND durable = ? ORDER BY seen_at, published_at', [req.user.id, true], function(err, rows){
+exports.mentions = function(req, res){
+ mysql.query('SELECT * FROM stories WHERE user_id = ? AND durable = ? ORDER BY seen_at, published_at limit 200', [req.user.id, true], function(err, rows){
if (err){
logger.error('Erorr getting stories');
}
- var bugs = [];
+ var mentions = [];
for (var i in rows){
var story = JSON.parse(rows[i].data);
story.seen_at = rows[i].seen_at;
- bugs.push(story);
+ mentions.push(story);
}
- res.render('social/bugs', {user: req.user, bugs: bugs, layout: false});
+ res.render('social/mentions', {user: req.user, mentions: mentions, layout: false});
});
};
-exports.markBugAsViewed = function(req, res){
- console.log(req.body);
+exports.markMentionAsViewed = function(req, res){
if (req.user){
mysql.query(
"UPDATE stories SET seen_at = NOW() WHERE user_id = ? and id = ?",
@@ -50,8 +49,7 @@ exports.markBugAsViewed = function(req, res){
if (err){
logger.error('Error marking story as read: ' + err);
}
- var redis = createRedisClient();
- redis.publish('notifications.bugzilla.read', req.body.id, function(err){
+ redis.publish('notifications.mention.read', req.user.id, function(err){
if (err)
logger.error(err);
});
View
144 app/http/public/javascripts/application.js
@@ -2,9 +2,6 @@
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
* You can obtain one at http://mozilla.org/MPL/2.0/. */
-var DATA = {
- profile: {}
-}
$(function(){
$("#browserid").click(function(){
@@ -26,12 +23,26 @@ $(function(){
$('body').addClass(controller);
$('ul.nav li.' + controller).addClass('active');
+ if (typeof($(".chzn-select").chosen) == 'function'){
+ $(".chzn-select").chosen();
+ }
+
// Page-specific stuff:
switch(document.location.pathname){
case '/profile':
+
+ $('#user_form').submit(function(){
+ saveUser();
+ return false;
+ });
+
var updateLastSaved = function(){
- $('#changes_saved').text("Changes last saved at " + $.format.date(new Date(), 'hh:mm:ssa'));
+
+ $('#changes_saved .timestamp').text($.format.date(new Date(), 'hh:mm:ssa'));
$('#changes_saved').show();
+ setTimeout(function(){
+ $('#changes_saved').fadeOut();
+ }, 2000);
};
var saveUser = function(){
@@ -52,21 +63,34 @@ $(function(){
data: $("#user_form").serialize(),
success: function(data){
$('#error_contacting_nickserv').hide();
- updateLastSaved();
- DATA.profile['realNameFromIRC'] = data['realName'];
+
+ var changeRealName = ($('#user_real_name').val().length == 0);
+
$('#user_real_name').attr('placeholder', 'Real Name');
- if ($('#source_from_irc').is(':checked')){
+ if (changeRealName){
$('#user_real_name').val(data['realName']);
}
- var ul = $('#networks ul');
+ var ul = $('#channels');
ul.empty();
for(var i in data.networks){
var network = data.networks[i];
+ var option = $("#autojoin_channels option[value=" + network + "]");
+
+ if (option.length > 0){
+ option.attr('selected', 'selected');
+ }
+ else{
+ // HACKHACK: This shouldn't be required, but it seems that the full list isn't properly returning from the IRC client...
+ $("#autojoin_channels").append($("<option selected=\"selected\" value=\"" + network + "\">" + network + "</option>"));
+ }
ul.append("<li>" + network + "</li>");
}
- $('#networks').show();
+
+ $('.chzn-select').trigger("liszt:updated");
+
+ $('#channels').show();
},
error: function(err){
$('#error_contacting_nickserv').show();
@@ -79,27 +103,15 @@ $(function(){
return false;
};
- $('#source_from_irc').change(function(){
- if ($('#source_from_irc').is(':checked')){
- if (DATA.profile.realNameFromIRC){
- $('#user_real_name').val(DATA.profile.realNameFromIRC);
- }
- else{
- postNick();
- }
- $('#user_real_name').attr('disabled', true);
- }
- else{
- $('#user_real_name').attr('disabled', false);
- }
- });
- $('#user_real_name').change(function(){
- saveUser();
- });
- $('#user_nick').change(postNick);
+ var nick = $('#user_nick');
+ nick.change(postNick);
+
+ if (nick.val().length > 0){
+ postNick();
+ }
+
+ $('#user_real_name').change(saveUser);
- break;
- case '/feeds':
$('.feeds #new_feed_form').click(function(){
createBlankFeedForm();
return false;
@@ -123,7 +135,6 @@ $(function(){
console.log("Error removing");
}
});
- console.log();
return false;
}
@@ -136,7 +147,7 @@ $(function(){
.removeClass('verified')
.removeClass('error')
.find('.delete').hide();
- console.log(form.serialize());
+
var reqData = form.serialize();
form.find('input[type=hidden]').val($(this).val());
@@ -146,7 +157,6 @@ $(function(){
url: '/feeds/feed',
data: reqData,
success: function(data){
- console.log(data);
if (data.url == ""){
form.remove();
}
@@ -171,18 +181,17 @@ $(function(){
$('.feeds form input[name=url]').change(validateAndSaveUrl);
- // $('input[value="Whatever"]');
- var template = $('#feed_template');
- template.removeAttr('id').remove();
+ var feedTemplate = $('#feed_template');
+ feedTemplate.removeAttr('id').remove();
function createBlankFeedForm(){
var blankInput = $('.feeds input[type=text][value=]');
- if (blankInput.length > 1){
+ if (blankInput.length > 0){
blankInput.focus();
return;
}
- var newFeed = template.clone();
+ var newFeed = feedTemplate.clone();
newFeed.find('input[name=url]').change(validateAndSaveUrl);
newFeed.find('.delete').click(deleteUrl);
newFeed.find('form').submit(function(){
@@ -190,10 +199,71 @@ $(function(){
});
$('ul.feeds').append(newFeed);
newFeed.show();
+ $(newFeed).focus();
}
createBlankFeedForm();
+ function validateAndSaveTokens(){
+ $('.mentionTokens li').removeClass('error');
+ var inputs = $('.mentionTokens li input[type=text]');
+ var tokens = [];
+ for (var i=0; i<inputs.length; i++){
+ if($(inputs[i]).val().match(/^\w[\w\-\_]+$/)){
+ tokens.push($(inputs[i]).val());
+ }
+ else{
+ $(inputs[i]).parent().parent().addClass('error');
+ return false;
+ }
+ }
+
+ tokens;
+ $.ajax({
+ url: '/profile/watchedTokens',
+ type: 'PUT',
+ data: {tokens: tokens},
+ success: updateLastSaved,
+ error: console.log
+ });
+
+ }
+
+ $('#mentions #new_token_input').click(function(){
+ createBlankTokenItem();
+ return false;
+ });
+
+
+ function deleteToken(){
+ $(this).parent().remove();
+ validateAndSaveTokens();
+ return false;
+ }
+
+ var tokenTemplate = $('#mention_token_template');
+ tokenTemplate.removeAttr('id').remove();
+
+ $('#mentions form').submit(function(){return false;})
+
+ function createBlankTokenItem(){
+ var blankInput = $('.mentionTokens input[type=text][value=]');
+ if (blankInput.length > 0){
+ blankInput.focus();
+ return;
+ }
+
+ var newTokenInput = tokenTemplate.clone();
+ newTokenInput.find('input').change(validateAndSaveTokens);
+ newTokenInput.find('.delete').click(deleteToken);
+ $('ul.mentionTokens').append(newTokenInput);
+ newTokenInput.show();
+ newTokenInput.focus();
+ }
+
+ $('ul.mentionTokens .delete').click(deleteToken);
+ $('ul.mentionTokens input[type=text]').change(validateAndSaveTokens);
+
break;
}
View
3  app/http/public/stylesheets/notifications.css
@@ -1,3 +1,6 @@
+h1 {
+ font-size: 2em;
+}
.notifications {
height: 400px;
overflow-y: auto;
View
36 app/http/public/stylesheets/style.css
@@ -44,33 +44,49 @@ body.profile li.search-choice span {
padding-right: 12px;
}
-body.feeds ul.feeds input[type=text]{
+body.profile ul.feeds input[type=text]{
width: 630px;
}
-body.feeds div.feeds {
+body.profile div.feeds {
display: inline;
margin-left: 25px;
}
-body.feeds ul.feeds {
- width: 805px;
+
+body.profile ul {
list-style-type: none;
+}
+
+body.profile ul.mentionTokens li{
+ margin-bottom: 1em;
+}
+
+body.profile ul.feeds {
+ width: 805px;
margin-bottom: 35px;
}
-body.feeds ul.feeds form{
+body.profile .feedcontainer ul form{
padding-left: 59px;
}
-body.feeds ul.feeds form.verified, body.feeds ul.feeds form.error {
+body.profile .feedcontainer ul form.verified, body.profile .feedcontainer ul form.error {
padding-left: 19px;
}
-body.feeds ul.feeds .btn-success, body.feeds ul.feeds .btn-error {
+body.profile .feedcontainer ul .btn-success, body.profile .feedcontainer ul .btn-error {
display: none;
}
-body.feeds ul.feeds .verified .btn-success, body.feeds ul.feeds .error .btn-danger {
+body.profile .feedcontainer ul .verified .btn-success, body.profile .feedcontainer ul .error .btn-danger {
display: inline;
}
-body.feeds .feedcontainer {
+body.profile .mentionTokens .errorMessage{
+ display: none;
+}
+
+body.profile .mentionTokens .error .errorMessage{
+ display: block;
+}
+
+body.profile .feedcontainer {
min-height: 200px;
padding-bottom: 100px;
background-repeat: no-repeat;
@@ -78,6 +94,6 @@ body.feeds .feedcontainer {
background-image: url('/images/rss-bot-montage.jpg');
}
-body.feeds form.well {
+body.profile form.well {
background-color: rgba(245, 245, 245, 0.8);
}
View
14 app/http/server.js
@@ -74,7 +74,10 @@ passport.use(new BrowserID({
var http = express.createServer();
+var redisConfig = config.get('redis');
var sessionStore = new RedisStore({
+ host: redisConfig.host,
+ port: redisConfig.port,
maxAge: (30).days
});
@@ -124,22 +127,23 @@ routes = {
feeds: require('./controllers/feeds')
};
-
-http.get('/', routes.site.index);
+http.get('/', routes.site.index);
http.get('/signout', routes.site.signout);
http.post('/auth/browserid', passport.authenticate('browserid', { failureRedirect: '/login' }), routes.site.authenticate);
+http.get( '/driver', application.authenticate, routes.site.driver);
+http.post('/driver', application.authenticate, routes.site.driver);
http.get('/social/worker.js', routes.social.worker);
http.get('/social/sidebar', routes.social.sidebar);
-http.get('/social/bugs', routes.social.bugs);
-http.post('/social/bug', routes.social.markBugAsViewed);
+http.get('/social/mentions', routes.social.mentions);
+http.post('/social/mention', routes.social.markMentionAsViewed);
http.get('/social/manifest.json', routes.social.manifest);
http.get('/profile', application.authenticate, routes.profile.index.get);
http.put('/profile', application.authenticate, routes.profile.index.put);
http.post('/profile/nick', application.authenticate, routes.profile.nick.post);
-
+http.put('/profile/watchedTokens', application.authenticate, routes.profile.watchedTokens.put);
http.get('/feeds', application.authenticate, routes.feeds.index.get);
http.post('/feeds/feed', application.authenticate, routes.feeds.feed.post);
View
70 app/http/socket.js
@@ -50,24 +50,15 @@ function broadcast(message){
* Story Stuff: This should really get rolled up into a model.
*/
function sendStoryToUser(userId, story){
- var message = {
- data: story
- };
-
- if (story.durable){
- // We're only doing bugzilla right now
- message['topic'] = 'notifications.bugzilla';
- }
- else {
- message['topic'] = 'feed.story';
- }
-
- sendMessageToUser(userId, JSON.stringify(message));
+ sendMessageToUser(userId, JSON.stringify({
+ topic: 'feed.story',
+ data: story
+ }));
}
function subscribeForStories(){
pubsubRedis.on("ready", function(){
- pubsubRedis.subscribe(["feeds.storyForUser", "notifications.bugzilla.read", "user.signout", 'contacts.userStatusUpdate', 'contacts.userOffline', 'contacts.reset']);
+ pubsubRedis.subscribe(["feeds.storyForUser", "notifications.mention.read", "notifications.mention.new", "user.signout", 'contacts.userStatusUpdate', 'contacts.userOffline', 'contacts.reset']);
pubsubRedis.on("message", function(channel, message){
switch(channel){
@@ -76,6 +67,14 @@ function subscribeForStories(){
sendStoryToUser(data.userId, data.story);
break;
+ case "notifications.mention.read":
+ var userId = parseInt(message);
+ sendMessageToUser(userId, '{"topic":"notifications.mention.read"}');
+ break;
+ case "notifications.mention.new":
+ var userId = parseInt(message);
+ sendMessageToUser(userId, '{"topic":"notifications.mention.new"}');
+ break;
case "user.signout":
var userId = parseInt(message);
@@ -101,9 +100,6 @@ function subscribeForStories(){
case "contacts.userOffline":
broadcast(message);
break;
- case "notifications.bugzilla.read":
- broadcast({topic: 'notifications.bugzilla.read', data: {id: message}});
- break;
default:
logger.error("Redis message received in socket.js on unexpected channel: " + channel);
}
@@ -138,6 +134,7 @@ function sendContactListToUser(userId){
if (rows && rows.length){
for (var i in rows){
+ logger.debug('User: ' + rows[i].nick);
sendMessageToUser(userId, JSON.stringify({
topic: 'contacts.userStatusUpdate',
data: {
@@ -154,8 +151,8 @@ function sendContactListToUser(userId){
);
}
-function sendRecentStories(user, durable){
- mysql.query("SELECT * FROM stories WHERE user_id = ? AND durable = ? AND seen_at IS NULL ORDER BY published_at DESC LIMIT 30", [user.id, durable], function(err, rows){
+function sendRecentStories(user){
+ mysql.query("SELECT * FROM stories WHERE user_id = ? AND durable = ? AND seen_at IS NULL ORDER BY published_at DESC LIMIT 30", [user.id, false], function(err, rows){
if (err){
logger.error("Error loading stories for user: " + user.email);
logger.error(err);
@@ -172,18 +169,37 @@ function sendRecentStories(user, durable){
});
}
+function sendNotificationCounts(user){
+ mysql.query(
+ "SELECT COUNT(*) as count FROM stories WHERE user_id = ? and durable = ? AND seen_at IS NULL",
+ [user.id, true],
+ function(err, rows){
+ if (err){
+ logger.error("Error counting notifications for user: " + user.email);
+ logger.error(err);
+ }
+
+ sendMessageToUser(user.id, JSON.stringify({
+ topic: 'notifications.count',
+ mentions: rows[0].count
+ }));
+ }
+ );
+}
+
function userConnected(user){
logger.debug("Loading stories for user. (id: " + user.id + ")");
- sendRecentStories(user, true);
- sendRecentStories(user, false);
+ sendRecentStories(user);
+ sendNotificationCounts(user);
var responseQueue = "irc-resp:" + uuid.v1();
var redis = createRedisClient();
redis.lpush("irc:user-connected", JSON.stringify([user.id, user.nick, responseQueue]));
- // We block for two minutes max.
- redis.brpop(responseQueue, 120, function(err, data){
+ // We block for twenty seconds max.
+ redis.brpop(responseQueue, 20, function(err, data){
+ redis.quit();
if (!data)
logger.error("Timeout exceeded waiting for IRC daemon to update user: " + user.id);
@@ -207,7 +223,7 @@ module.exports = {
socket.on('request', function(request){
if (!request.httpRequest.headers.cookie){
logger.info('Socket request rejected because it lacked cookie data.');
- return request.reject();
+ return request.reject(401);
}
// here we use parseCookie instead of the already parsed cookies because
@@ -217,17 +233,17 @@ module.exports = {
if (!cookies['express.sid']){
logger.info('Socket request rejected because it lacked an express.sid (Session)');
- return request.reject();
+ return request.reject(401);
}
var sessionID = cookies['express.sid'];
store.load(sessionID, function(err, session){
if (err || !session){
logger.error('Socket request rejected due to error loading session: ' + err);
- return request.reject();
+ return request.reject(500);
}
else if ( !session.passport || !session.passport.user){
logger.info('Socket request rejected. -- Passport user empty.');
- return request.reject();
+ return request.reject(401);
}
else{
var connection = request.accept();
View
1  app/http/views/layout.ejs
@@ -55,7 +55,6 @@
</li>
<% } else { %>
<li class="profile" ><a href="/profile" >Profile</a></li>
- <li class="feeds" ><a href="/feeds" >Feeds</a></li>
<li ><a href="/signout" >Sign Out</a></li>
<% } %>
</ul>
View
207 app/http/views/profile/index.ejs
@@ -1,61 +1,170 @@
-<h1>Your Profile <span id="changes_saved" class="label label-success" style="display:none"></span></h1>
-
-<div id="container" class="row">
- <div id="photo" class="span3">
- <div class="thumbnail">
- <img src="<%= user.getGravatarUrl(170) %>" width="170" height="170" />
- <div class="caption">
- <p>
- Image loaded from <a href="http://gravatar/">Gravatar</a>
- <p><a href="http://en.gravatar.com/emails/" target="_NEW" class="btn btn-primary">Manage</a>
- </p>
+<div class="page-header">
+ <h1>Your Profile</h1>
+</div>
+
+<div id="changes_saved" class="span10 alert alert-success" style="display:none">
+ <b>Profile Saved</b> at <span class="timestamp"></span>
+</div>
+
+<form id="user_form" class="form-horizontal">
+ <fieldset>
+ <section id="chat span10">
+ <div class="page-header">
+ <h1>
+ Networking
+ <small>Chat and availability</small>
+ </h1>
</div>
- </div>
- </div>
- <div id="details" class="span8" style="background-color: white">
- <p>
- MoTown uses IRC to update your availability and status. Currently we're
- only supporting this on <a href="irc://irc.mozilla.org">irc.mozilla.org</a>.
- </p>
- <form id="user_form" class="form-horizontal">
- <fieldset>
- <div class="control-group">
- <label class="control-label" for="user_nick">IRC Nick</label>
- <div class="controls">
- <input class="input-xlarge focused" id="user_nick" name="user[nick]" type="text" value="<%= user.nick %>" placeholder="Nick">
- <p class="help-block">We only support irc://irc.mozilla.org</p>
+ <p>
+ MoTown uses IRC to update your availability and chat to other Mozillians. Currently we're
+ only supporting this on <a href="irc://irc.mozilla.org">irc.mozilla.org</a>.
+ </p>
+
+ <div class="control-group">
+ <label class="control-label" for="user_nick">IRC Nickname</label>
+ <div class="controls">
+ <input class="input-xlarge focused" id="user_nick" name="user[nick]" type="text" value="<%= user.nick %>" placeholder="Nick">
+ <p class="help-block">We only support irc://irc.mozilla.org</p>
+ </div>
+ </div>
+ <!-- <div class="control-group">
+ <label class="control-label" for="irc_user">Automatically Connect to IRC For me:</label>
+ <div class="controls">
+ <label class="checkbox">
+ <input type="checkbox" id="irc_user" checked="checked" value="checked">
+ </label>
+ <p class="help-block"></p>
+ </div>
+ </div>
+
+ <div id="networks" class="control-group">
+ <label class="control-label">Auto Join Networks</label>
+ <div class="controls">
+ <select id="autojoin_channels" data-placeholder="Auto Join Networks" multiple="multiple" class="chzn-select">
+ <% for (var i in channels) { %>
+ <option value="<%= channels[i].name %>"><%= channels[i].displayName %></option>
+ <% } %>
+ </select>
+ </div>
+ </div>
+ -->
+ </section>
+ </fieldset>
+
+ <fieldset>
+ <section id="personal">
+ <div class="page-header">
+ <h1>
+ Personal Info
+ <small>your name, photo and the like</small>
+ </h1>
+ </div>
+ <div class="row">
+ <div id="photo" class="span2">
+ <div class="thumbnail">
+ <img src="<%= user.getGravatarUrl(160) %>" width="160" height="160" />
+ <div class="caption">
+ <p>
+ Image loaded from <a href="http://gravatar.com">Gravatar</a>
+ <p><a href="http://en.gravatar.com/emails/" target="_NEW" class="btn btn-primary">Manage</a>
+ </p>
</div>
</div>
- <div class="control-group">
- <label class="control-label" for="source_from_irc">Use IRC for Public Details:</label>
- <div class="controls">
- <label class="checkbox">
- <input type="checkbox" id="source_from_irc" value="checked">
- </label>
- </div>
- </div>
-
+ </div>
+ <div class="span9">
<div class="control-group">
<label class="control-label" for="name">Real Name</label>
<div class="controls">
<input class="input-xlarge" id="user_real_name" name="user[realName]" type="text" value="<%= user.realName %>" placeholder="Real Name">
- <p id="error_contacting_nickserv" style="display:none" class="help-block alert-error">
- Error contacting NickServ. Please set your Real Name manually.
- </p>
</div>
</div>
-
- <div id="networks" class="control-group" style="display: none;">
- <label class="control-label">Networks</label>
- <div class="controls">
- <ul>
- <li>#vancouver</li>
- <li>#b2g</li>
- </ul>
- <span class="help-inline">These are maintained by IRC</span>
+ </div>
+ </div>
+ </section>
+ </fieldset>
+</form>
+
+<section id="feeds">
+ <div class="page-header">
+ <h1>
+ Feeds
+ <small>RSS</small>
+ </h1>
+ <p>
+ MoTown uses RSS feeds to keep you updated. We've filled in a couple of examples for you:
+ </p>
+
+ </div>
+
+ <div class="feedcontainer listcontainer">
+ <ul class="feeds">
+ <% for (var i in feeds){
+ var feed = feeds[i];
+ %>
+ <li>
+ <form class="well form-inline <%= feed.verified ? 'verified' : 'error' %>">
+ <a class="btn btn-success" href="#"><i class="icon-ok-sign icon-white"></i></a>
+ <a class="btn btn-danger btn-error" href="#"><i class="icon-remove-sign icon-white"></i></a>
+ <input type="text" name="url" class="input-xxlarge" placeholder="URL" title="<%= feed.title %>" value="<%= feed.url%>">
+ <input type="hidden" name="previousUrl" value="<%= feed.url%>">
+ <a class="btn btn-danger delete" href="#"><i class="icon-trash icon-white"></i> Delete</a>
+ </form>
+ </li>
+ <% } %>
+ <li id="feed_template" style="display:none">
+ <form class="well form-inline">
+ <a class="btn btn-success btn-error" href="#"><i class="icon-ok-sign icon-white"></i></a>
+ <a class="btn btn-danger btn-error" href="#"><i class="icon-remove-sign icon-white"></i></a>
+ <input type="text" name="url" class="input-xxlarge" placeholder="URL">
+ <input type="hidden" name="previousUrl">
+ <a class="btn btn-danger delete" href="#"><i class="icon-trash icon-white"></i> Delete</a>
+ </form>
+ </li>
+ </ul>
+ <div class="well feeds">
+ <a href="#" id="new_feed_form" class="btn btn-primary"><i class="icon-forward icon-white"></i>New</a>
+ </div>
+ </div>
+
+</section>
+
+<section id="mentions">
+ <div class="page-header">
+ <h1>
+ Mentions
+ <small>When the net screams your name</small>
+ </h1>
+ <p>
+ MoTown checks each story and flags those with mentions of you in the UI.
+ </p>
+
+ </div>
+
+ <div class="listcontainer">
+ <form class="well form-inline <%= feed.verified ? 'verified' : 'error' %>">
+ <ul class="mentionTokens">
+ <% for (var i in watchedTokens){
+ var token = watchedTokens[i];
+ %>
+ <li class="control-group">
+ <div class="input-prepend">
+ <span class="add-on">@</span><input class="input-large" type="text" value="<%= token%>">
</div>
+
+ <a class="btn btn-danger delete" href="#"><i class="icon-trash icon-white"></i> Delete</a>
+ </li>
+ <% } %>
+ <li class="control-group" id="mention_token_template" style="display:none">
+ <div class="input-prepend">
+ <span class="add-on">@</span><input class="input-large" size="16" type="text">
</div>
- </fieldset>
- </form>
+
+ <a class="btn btn-danger delete" href="#"><i class="icon-trash icon-white"></i> Delete</a>
+ <p class="errorMessage help-block">Mentions may only contain alphanumeric characters, dashes and underscores and must begin with an alphanumeric character.</p>
+ </li>
+ </ul>
+ <a href="#" id="new_token_input" class="btn btn-primary"><i class="icon-forward icon-white"></i>New</a>
+ </form>
+
</div>
-</div>
+</section>
View
19 app/http/views/social/bugs.ejs → app/http/views/social/mentions.ejs
@@ -22,21 +22,21 @@
<![endif]-->
<script type="text/javascript">
$(function(){
- $('.bug a').click(function(e){
+ $('.mention a').click(function(e){
var li = $(this).parent();
- var id = li.attr('id').substr(4);
-
+ var id = li.attr('id').substr(8);
+
var self = $(this);
$.ajax({
type: 'POST',
- url: '/social/bug',
+ url: '/social/mention',
data: {'id': id},
success: function(data){
self.removeClass('new');
},
error: function(){
//TODO: Do something intelligent in the UI.
- console.log("Error marking bug as read.");
+ console.log("Error marking mention as read.");
}
});
@@ -49,11 +49,12 @@
<body>
<div id="notif">
+ <h1>Mentions:</h1>
<ul class="notifications">
- <% for (var i in bugs){ %>
- <% var readClass = bugs[i].seen_at ? '' : 'new' %>
- <li id="bug_<%= bugs[i].id%>" class="bug">
- <a target="_blank" href="<%= bugs[i].href %>" class="<%= readClass%>"><%= bugs[i].title%></a>
+ <% for (var i in mentions){ %>
+ <% var readClass = mentions[i].seen_at ? '' : 'new' %>
+ <li id="mention_<%= mentions[i].id%>" class="mention">
+ <a target="_blank" href="<%= mentions[i].href %>" class="<%= readClass%>"><%= mentions[i].title%></a>
</li>
<% } %>
</ul>
View
30 app/http/views/social/worker.js.ejs
@@ -15,6 +15,7 @@
var SPRITES = {
bugIcon: "",
+ mentionIcon: "%3D",
}
// Make sure we have the right WebSocket class.
@@ -26,6 +27,7 @@ var baseUrl = '<%= baseUrl%>';
var motown;
var reconnectInterval;
var notificationCount = {
+ mentions: 0,
bugzilla: 0
}
@@ -100,12 +102,12 @@ function initWebSocket(){
function setAmbientNotificationCount(){
apiPort.postMessage({
- topic: 'social.ambient-notification-update',
+ topic: 'social.ambient-notification',
data: {
- name: 'bugzilla',
- iconURL: SPRITES.bugIcon,
- counter: notificationCount.bugzilla,
- contentPanel: baseUrl + '/social/bugs'
+ name: 'mentions',
+ iconURL: SPRITES.mentionIcon,
+ counter: notificationCount.mentions,
+ contentPanel: baseUrl + '/social/mentions'
}
});
}
@@ -116,16 +118,20 @@ var handlers = {
'contacts.userOffline': rawSidebarMessage,
'contacts.reset': rawSidebarMessage,
'contacts.userStatusUpdate': rawSidebarMessage,
- 'notifications.bugzilla.read': function(message){
- notificationCount.bugzilla--;
+ 'notifications.mention.read': function(message){
+ if (notificationCount.mentions > 0)
+ notificationCount.mentions--;
- log(notificationCount.bugzilla, true);
setAmbientNotificationCount();
},
- 'notifications.bugzilla': function(message){
- notificationCount.bugzilla++;
+ 'notifications.mention.new': function(message){
+ notificationCount.mentions++;
setAmbientNotificationCount();
},
+ 'notifications.count': function(message){
+ notificationCount.mentions = message.mentions;
+ setAmbientNotificationCount();
+ },
'social.port-closing': function(data, port){
if (apiPort == port){
apiPort.close();
@@ -133,11 +139,11 @@ var handlers = {
}
},
'social.initialize': function(data, port){
- dump('social.initialize on port ' + JSON.stringify(port) + '\n');
+ log('social.initialize on port ' + JSON.stringify(port) + '\n');
apiPort = port;
apiPort.postMessage({
- topic: 'social.ambient-notification-area',
+ topic: 'social.user-profile',
data: {
userName: user.email,
displayName: user.realName,
View
110 app/serializer.js
@@ -8,13 +8,64 @@
// Module dependencies.
const
-redis = require('../lib/redis')(),
-logger = require('../lib/logger'),
-config = require('../lib/configuration'),
-mysql = require('mysql').createClient(config.get('mysql'));
+redis = require('../lib/redis').new(),
+pubRedis = require('../lib/redis').pub,
+logger = require('../lib/logger'),
+config = require('../lib/configuration'),
+mysql = require('mysql').createClient(config.get('mysql'));
+require('../lib/extensions/array');
+require('../lib/extensions/number');
+
+var watchedTokens = null; // {token: [<user.id>, ...], ...
+
+
+function loadWatchedTokens(callback){
+ mysql.query(
+ "SELECT token, user_id FROM watched_tokens ORDER BY token, user_id",
+ function(err, rows)
+ {
+ if (!err){
+ watchedTokens = {};
+ for (var i in rows){
+
+ var token = "@" + rows[i].token;
+ var userId = rows[i].user_id;
+
+ if (token in watchedTokens){
+ watchedTokens[token].push(userId);
+ }
+ else{
+ watchedTokens[token] = [userId];
+ }
+ }
+ if (typeof(callback) == 'function'){
+ callback();
+ }
+ }
+ else{
+ logger.error("Error loading watched_tokens in Feed Daemon: " + err);
+ }
+ }
+ );
+}
+
+// This method might not look super clear,
+// but it is checking each token found in a story against the
+// complete set of watched tokens. aka {token: [<user.id>, ...], ...}
+// so we return a possibly non-unique list of interested users.
+function checkForMentions(story){
+ var mentionedUsers = [];
+
+ var mentions = story.title.match(/\@\w+/g);
+
+ for (var i in mentions){
+ mentionedUsers = mentionedUsers.concat(watchedTokens[mentions[i]]);
+ }
+
+ return mentionedUsers;
+}
-redis.debug_mode = false;
// {
// "userIds": [ 13, 14 ],
@@ -36,11 +87,13 @@ redis.debug_mode = false;
// }
// }
-
function waitForStory(){
+ logger.debug('Waiting on serializer:stories');
// This is a blocking call only to the network layer (we keep ticking)
redis.brpop('serializer:stories', 0, function(err, data){
- // data == ['stories', '{story json as string}']
+
+ if (err)
+ logger.error("Error dequeueing story: " + err);
//TODO: Error handling including:
// - accepting an error from redis
@@ -53,14 +106,26 @@ function waitForStory(){
var story = JSON.parse(data);
var userIds = story.userIds;
+
story = story.story;
+ var mentionedUsers = [];
+
+ if (!story.durable){
+ mentionedUsers = checkForMentions(story);
+ }
+
+ userIds = Array.uniqueSort(userIds.concat(mentionedUsers));
+
for (var i in userIds){
var userId = userIds[i];
+ var durable = !!(story.durable || mentionedUsers.indexOf(userId) >= 0);
+ logger.debug("Durable? : " + durable);
+
// Persist the story to MySQL
mysql.query(
- 'REPLACE INTO stories SET id = ?, data = ?, user_id = ?, published_at = ?, durable = ?', [story.id, JSON.stringify(story), userId, story.pubdate, !!story.durable],
+ 'REPLACE INTO stories SET id = ?, data = ?, user_id = ?, published_at = ?, durable = ?', [story.id, JSON.stringify(story), userId, story.pubdate, !!durable],
function(err, data){
// TODO: Handle error wisely
if (err)
@@ -68,14 +133,21 @@ function waitForStory(){
// Publish the story data as a pub/sub event for the socket if it's a new record
if (data.affectedRows == 1){
- redis.publish('feeds.storyForUser', JSON.stringify({'userId': userId, 'story': story}), function(err){
- if (err)
- logger.error(err);
- });
+ if (durable){
+ pubRedis.publish('notifications.mention.new', userId.toString(), function(err){
+ if (err)
+ logger.error("Error publishing notifications.mention.new: " + err);
+ });
+ }
+ else {
+ pubRedis.publish('feeds.storyForUser', JSON.stringify({'userId': userId, 'story': story}), function(err){
+ if (err)
+ logger.error(err);
+ });
+ }
}
}
);
-
}
// Rinse and repeat
@@ -83,7 +155,13 @@ function waitForStory(){
});
}
+
+var reloadInterval = setInterval(loadWatchedTokens, (10).minutes());
+
+
redis.on("ready", function(){
- logger.info("Serializer ready.");
- waitForStory();
-});
+ loadWatchedTokens(function(){
+ logger.info("Serializer ready.");
+ waitForStory();
+ });
+});
Please sign in to comment.
Something went wrong with that request. Please try again.