2013-08-15 21:11:07 +02:00
|
|
|
var stream_data = (function () {
|
|
|
|
|
|
|
|
var exports = {};
|
|
|
|
|
2013-09-16 23:47:05 +02:00
|
|
|
|
|
|
|
// The stream_info variable maps stream names to stream properties objects
|
|
|
|
// Call clear_subscriptions() to initialize it.
|
|
|
|
var stream_info;
|
2014-02-05 20:35:16 +01:00
|
|
|
var subs_by_stream_id;
|
2016-10-28 18:54:39 +02:00
|
|
|
var recent_topics = new Dict({fold_case: true});
|
2013-09-16 23:47:05 +02:00
|
|
|
|
|
|
|
exports.clear_subscriptions = function () {
|
|
|
|
stream_info = new Dict({fold_case: true});
|
2014-02-05 20:35:16 +01:00
|
|
|
subs_by_stream_id = new Dict();
|
2013-09-16 23:47:05 +02:00
|
|
|
};
|
2016-06-04 22:40:25 +02:00
|
|
|
|
2013-09-16 23:47:05 +02:00
|
|
|
exports.clear_subscriptions();
|
2013-08-15 21:11:07 +02:00
|
|
|
|
2016-10-28 17:18:56 +02:00
|
|
|
exports.is_active = function (stream_name) {
|
2016-10-28 18:54:39 +02:00
|
|
|
return recent_topics.has(stream_name);
|
2016-10-28 17:18:56 +02:00
|
|
|
};
|
2013-08-15 21:11:07 +02:00
|
|
|
|
2016-10-30 17:33:23 +01:00
|
|
|
exports.rename_sub = function (stream_id, new_name) {
|
|
|
|
var sub = subs_by_stream_id.get(stream_id);
|
|
|
|
var old_name = sub.name;
|
|
|
|
sub.name = new_name;
|
|
|
|
stream_info.del(old_name);
|
|
|
|
stream_info.set(new_name, sub);
|
|
|
|
|
|
|
|
return sub;
|
|
|
|
};
|
|
|
|
|
2013-08-15 21:11:07 +02:00
|
|
|
exports.add_sub = function (stream_name, sub) {
|
2016-03-14 06:38:43 +01:00
|
|
|
if (!_.has(sub, 'subscribers')) {
|
|
|
|
sub.subscribers = Dict.from_array([], {fold_case: true});
|
|
|
|
}
|
|
|
|
|
2013-09-16 23:47:05 +02:00
|
|
|
stream_info.set(stream_name, sub);
|
2014-02-05 20:35:16 +01:00
|
|
|
subs_by_stream_id.set(sub.stream_id, sub);
|
2013-08-15 21:11:07 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
exports.get_sub = function (stream_name) {
|
2013-09-16 23:47:05 +02:00
|
|
|
return stream_info.get(stream_name);
|
2013-08-15 21:11:07 +02:00
|
|
|
};
|
|
|
|
|
2014-02-05 20:35:16 +01:00
|
|
|
exports.get_sub_by_id = function (stream_id) {
|
|
|
|
return subs_by_stream_id.get(stream_id);
|
|
|
|
};
|
|
|
|
|
2013-08-21 23:21:31 +02:00
|
|
|
exports.delete_sub = function (stream_name) {
|
2013-09-16 23:47:05 +02:00
|
|
|
stream_info.del(stream_name);
|
2013-08-21 23:21:31 +02:00
|
|
|
};
|
|
|
|
|
2013-10-25 16:59:52 +02:00
|
|
|
exports.subscribed_subs = function () {
|
2013-08-15 21:11:07 +02:00
|
|
|
return _.where(stream_info.values(), {subscribed: true});
|
2013-10-25 16:59:52 +02:00
|
|
|
};
|
2013-08-15 21:11:07 +02:00
|
|
|
|
2016-10-25 21:45:19 +02:00
|
|
|
exports.unsubscribed_subs = function () {
|
|
|
|
return _.where(stream_info.values(), {subscribed: false});
|
|
|
|
};
|
|
|
|
|
2013-08-15 21:11:07 +02:00
|
|
|
exports.subscribed_streams = function () {
|
2013-10-25 16:59:52 +02:00
|
|
|
return _.pluck(exports.subscribed_subs(), 'name');
|
2013-08-15 21:11:07 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
exports.get_colors = function () {
|
2013-10-25 16:59:52 +02:00
|
|
|
return _.pluck(exports.subscribed_subs(), 'color');
|
2013-08-15 21:11:07 +02:00
|
|
|
};
|
|
|
|
|
2016-03-14 06:38:43 +01:00
|
|
|
exports.update_subscribers_count = function (sub) {
|
|
|
|
var count = sub.subscribers.num_items();
|
|
|
|
sub.subscriber_count = count;
|
|
|
|
};
|
|
|
|
|
2013-08-15 21:11:07 +02:00
|
|
|
exports.all_subscribed_streams_are_in_home_view = function () {
|
2013-10-25 16:59:52 +02:00
|
|
|
return _.every(exports.subscribed_subs(), function (sub) {
|
2013-08-15 21:11:07 +02:00
|
|
|
return sub.in_home_view; }
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2014-01-15 20:59:31 +01:00
|
|
|
exports.home_view_stream_names = function () {
|
|
|
|
var home_view_subs = _.filter(exports.subscribed_subs(), function (sub) {
|
|
|
|
return sub.in_home_view;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
return _.map(home_view_subs, function (sub) {
|
|
|
|
return sub.name;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2013-08-15 21:11:07 +02:00
|
|
|
exports.canonicalized_name = function (stream_name) {
|
|
|
|
return stream_name.toString().toLowerCase();
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.get_color = function (stream_name) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
|
|
|
if (sub === undefined) {
|
|
|
|
return stream_color.default_color;
|
|
|
|
}
|
|
|
|
return sub.color;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.in_home_view = function (stream_name) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
|
|
|
return sub !== undefined && sub.in_home_view;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.is_subscribed = function (stream_name) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
|
|
|
return sub !== undefined && sub.subscribed;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.get_invite_only = function (stream_name) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
|
|
|
if (sub === undefined) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return sub.invite_only;
|
|
|
|
};
|
|
|
|
|
2013-08-19 19:25:44 +02:00
|
|
|
exports.get_name = function (stream_name) {
|
|
|
|
// This returns the actual name of a stream if we are subscribed to
|
|
|
|
// it (i.e "Denmark" vs. "denmark"), while falling thru to
|
|
|
|
// stream_name if we don't have a subscription. (Stream names
|
|
|
|
// are case-insensitive, but we try to display the actual name
|
|
|
|
// when we know it.)
|
|
|
|
var sub = exports.get_sub(stream_name);
|
|
|
|
if (sub === undefined) {
|
|
|
|
return stream_name;
|
|
|
|
}
|
|
|
|
return sub.name;
|
|
|
|
};
|
|
|
|
|
2013-09-16 22:26:53 +02:00
|
|
|
exports.set_subscribers = function (sub, emails) {
|
2013-09-16 23:40:57 +02:00
|
|
|
sub.subscribers = Dict.from_array(emails || [], {fold_case: true});
|
2013-09-16 22:26:53 +02:00
|
|
|
};
|
|
|
|
|
2013-09-07 02:48:44 +02:00
|
|
|
exports.add_subscriber = function (stream_name, user_email) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
2016-10-15 21:13:09 +02:00
|
|
|
if (typeof sub === 'undefined') {
|
|
|
|
blueslip.warn("We got an add_subscriber call for a non-existent stream.");
|
2013-09-07 03:10:08 +02:00
|
|
|
return;
|
2013-09-07 02:48:44 +02:00
|
|
|
}
|
|
|
|
sub.subscribers.set(user_email, true);
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.remove_subscriber = function (stream_name, user_email) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
2016-10-25 20:06:25 +02:00
|
|
|
if (typeof sub === 'undefined') {
|
|
|
|
blueslip.warn("We got a remove_subscriber call for a non-existent stream " + stream_name);
|
2013-09-07 02:48:44 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
sub.subscribers.del(user_email);
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.user_is_subscribed = function (stream_name, user_email) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
2013-09-07 04:22:18 +02:00
|
|
|
if (typeof sub === 'undefined' || !sub.subscribed) {
|
|
|
|
// If we don't know about the stream, or we ourselves are not
|
|
|
|
// subscribed, we can't keep track of the subscriber list in general,
|
|
|
|
// so we return undefined (treated as falsy if not explicitly handled).
|
2013-09-24 00:13:02 +02:00
|
|
|
blueslip.warn("We got a user_is_subscribed call for a non-existent or unsubscribed stream.");
|
2013-09-07 02:48:44 +02:00
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
return sub.subscribers.has(user_email);
|
|
|
|
};
|
|
|
|
|
2016-10-15 21:10:10 +02:00
|
|
|
exports.create_streams = function (streams) {
|
|
|
|
_.each(streams, function (stream) {
|
|
|
|
var attrs = _.defaults(stream, {
|
|
|
|
subscribed: false
|
|
|
|
});
|
|
|
|
exports.create_sub_from_server_data(stream.name, attrs);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2016-10-15 20:17:32 +02:00
|
|
|
exports.create_sub_from_server_data = function (stream_name, attrs) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
|
|
|
if (sub !== undefined) {
|
|
|
|
// We've already created this subscription, no need to continue.
|
|
|
|
return sub;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!attrs.stream_id) {
|
|
|
|
// fail fast (blueslip.fatal will throw an error on our behalf)
|
|
|
|
blueslip.fatal("We cannot create a sub without a stream_id");
|
|
|
|
return; // this line is never actually reached
|
|
|
|
}
|
|
|
|
|
|
|
|
// Our internal data structure for subscriptions is mostly plain dictionaries,
|
|
|
|
// so we just reuse the attrs that are passed in to us, but we encapsulate how
|
|
|
|
// we handle subscribers.
|
|
|
|
var subscriber_emails = attrs.subscribers;
|
|
|
|
var raw_attrs = _.omit(attrs, 'subscribers');
|
|
|
|
|
|
|
|
sub = _.defaults(raw_attrs, {
|
|
|
|
name: stream_name,
|
|
|
|
render_subscribers: !page_params.is_zephyr_mirror_realm || attrs.invite_only === true,
|
|
|
|
subscribed: true,
|
|
|
|
in_home_view: true,
|
|
|
|
invite_only: false,
|
|
|
|
desktop_notifications: page_params.stream_desktop_notifications_enabled,
|
|
|
|
audible_notifications: page_params.stream_sounds_enabled,
|
|
|
|
description: ''
|
|
|
|
});
|
|
|
|
|
|
|
|
exports.set_subscribers(sub, subscriber_emails);
|
|
|
|
|
|
|
|
if (!sub.color) {
|
|
|
|
var used_colors = exports.get_colors();
|
|
|
|
sub.color = stream_color.pick_color(used_colors);
|
|
|
|
}
|
|
|
|
|
|
|
|
exports.add_sub(stream_name, sub);
|
|
|
|
|
|
|
|
return sub;
|
|
|
|
};
|
|
|
|
|
2016-10-17 15:53:06 +02:00
|
|
|
exports.receives_desktop_notifications = function (stream_name) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
|
|
|
if (sub === undefined) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return sub.desktop_notifications;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.receives_audible_notifications = function (stream_name) {
|
|
|
|
var sub = exports.get_sub(stream_name);
|
|
|
|
if (sub === undefined) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return sub.audible_notifications;
|
|
|
|
};
|
|
|
|
|
2016-10-17 16:38:15 +02:00
|
|
|
exports.add_admin_options = function (sub) {
|
|
|
|
return _.extend(sub, {
|
|
|
|
'is_admin': page_params.is_admin,
|
|
|
|
'can_make_public': page_params.is_admin && sub.invite_only && sub.subscribed,
|
|
|
|
'can_make_private': page_params.is_admin && !sub.invite_only
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2016-10-28 17:31:18 +02:00
|
|
|
exports.process_message_for_recent_topics = function process_message_for_recent_topics(message, remove_message) {
|
|
|
|
var current_timestamp = 0;
|
|
|
|
var count = 0;
|
|
|
|
var stream = message.stream;
|
|
|
|
var canon_subject = exports.canonicalized_name(message.subject);
|
|
|
|
|
2016-10-28 18:54:39 +02:00
|
|
|
if (! recent_topics.has(stream)) {
|
|
|
|
recent_topics.set(stream, []);
|
2016-10-28 17:31:18 +02:00
|
|
|
} else {
|
2016-10-28 18:54:39 +02:00
|
|
|
recent_topics.set(stream, _.filter(recent_topics.get(stream), function (item) {
|
2016-10-28 17:31:18 +02:00
|
|
|
var is_duplicate = (item.canon_subject.toLowerCase() === canon_subject.toLowerCase());
|
|
|
|
if (is_duplicate) {
|
|
|
|
current_timestamp = item.timestamp;
|
|
|
|
count = item.count;
|
|
|
|
}
|
|
|
|
return !is_duplicate;
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
2016-10-28 18:54:39 +02:00
|
|
|
var recents = recent_topics.get(stream);
|
2016-10-28 17:31:18 +02:00
|
|
|
|
|
|
|
if (remove_message !== undefined) {
|
|
|
|
count = count - 1;
|
|
|
|
} else {
|
|
|
|
count = count + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count !== 0) {
|
|
|
|
recents.push({subject: message.subject,
|
|
|
|
canon_subject: canon_subject,
|
|
|
|
count: count,
|
|
|
|
timestamp: Math.max(message.timestamp, current_timestamp)});
|
|
|
|
}
|
|
|
|
|
|
|
|
recents.sort(function (a, b) {
|
|
|
|
return b.timestamp - a.timestamp;
|
|
|
|
});
|
|
|
|
|
2016-10-28 18:54:39 +02:00
|
|
|
recent_topics.set(stream, recents);
|
2016-10-28 17:31:18 +02:00
|
|
|
};
|
|
|
|
|
2016-10-25 21:45:19 +02:00
|
|
|
exports.get_streams_for_settings_page = function () {
|
2016-10-17 17:48:56 +02:00
|
|
|
// Build up our list of subscribed streams from the data we already have.
|
|
|
|
var subscribed_rows = exports.subscribed_subs();
|
2016-10-25 21:45:19 +02:00
|
|
|
var unsubscribed_rows = exports.unsubscribed_subs();
|
2016-10-17 17:48:56 +02:00
|
|
|
|
|
|
|
// Sort and combine all our streams.
|
|
|
|
function by_name(a,b) {
|
|
|
|
return util.strcmp(a.name, b.name);
|
|
|
|
}
|
|
|
|
subscribed_rows.sort(by_name);
|
|
|
|
unsubscribed_rows.sort(by_name);
|
|
|
|
var all_subs = subscribed_rows.concat(unsubscribed_rows);
|
|
|
|
|
2016-03-14 06:38:43 +01:00
|
|
|
// Add in admin options and stream counts.
|
2016-10-17 17:48:56 +02:00
|
|
|
var sub_rows = [];
|
|
|
|
_.each(all_subs, function (sub) {
|
|
|
|
sub = exports.add_admin_options(sub);
|
2016-03-14 06:38:43 +01:00
|
|
|
exports.update_subscribers_count(sub);
|
2016-10-17 17:48:56 +02:00
|
|
|
sub_rows.push(sub);
|
|
|
|
});
|
|
|
|
|
|
|
|
return sub_rows;
|
|
|
|
};
|
2016-10-17 16:38:15 +02:00
|
|
|
|
Clean up startup code for streams.
The startup code in subs.js used to intermingle data
stuff and UI stuff in a loop inside a called function,
which made the code hard to reason about.
Now there is a clear separation of concerns, with these methods
being called in succession:
stream_data.initialize_from_page_params();
stream_list.create_initial_sidebar_rows();
The first method was mostly extracted from subs.js, but I simplified
some things, like not needing to make a copy of the hashes
we were passed in, plus I now garbage collect email_dict. Also,
the code path that initialize_from_page_params() mostly replaces
used to call create_sub(), which fired a trigger, but now it
just does data stuff.
Once the data structure is built up, it's a very simple matter
to build the initial sidebar rows, and that's what the second
method does.
2016-10-17 19:34:58 +02:00
|
|
|
exports.initialize_from_page_params = function () {
|
|
|
|
function populate_subscriptions(subs, subscribed) {
|
|
|
|
subs.forEach(function (sub) {
|
|
|
|
var stream_name = sub.name;
|
|
|
|
sub.subscribed = subscribed;
|
|
|
|
|
|
|
|
// When we get subscriber lists from the back end,
|
|
|
|
// they are sent as user ids to save bandwidth,
|
|
|
|
// but the legacy JS code wants emails.
|
|
|
|
if (sub.subscribers) {
|
|
|
|
sub.subscribers = _.map(sub.subscribers, function (subscription) {
|
|
|
|
return page_params.email_dict[subscription];
|
|
|
|
});
|
|
|
|
}
|
|
|
|
exports.create_sub_from_server_data(stream_name, sub);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
populate_subscriptions(page_params.subbed_info, true);
|
|
|
|
populate_subscriptions(page_params.unsubbed_info, false);
|
2016-10-19 01:06:31 +02:00
|
|
|
populate_subscriptions(page_params.neversubbed_info, false);
|
Clean up startup code for streams.
The startup code in subs.js used to intermingle data
stuff and UI stuff in a loop inside a called function,
which made the code hard to reason about.
Now there is a clear separation of concerns, with these methods
being called in succession:
stream_data.initialize_from_page_params();
stream_list.create_initial_sidebar_rows();
The first method was mostly extracted from subs.js, but I simplified
some things, like not needing to make a copy of the hashes
we were passed in, plus I now garbage collect email_dict. Also,
the code path that initialize_from_page_params() mostly replaces
used to call create_sub(), which fired a trigger, but now it
just does data stuff.
Once the data structure is built up, it's a very simple matter
to build the initial sidebar rows, and that's what the second
method does.
2016-10-17 19:34:58 +02:00
|
|
|
|
|
|
|
// Garbage collect data structures that were only used for initialization.
|
|
|
|
delete page_params.subbed_info;
|
|
|
|
delete page_params.unsubbed_info;
|
2016-10-19 01:06:31 +02:00
|
|
|
delete page_params.neversubbed_info;
|
Clean up startup code for streams.
The startup code in subs.js used to intermingle data
stuff and UI stuff in a loop inside a called function,
which made the code hard to reason about.
Now there is a clear separation of concerns, with these methods
being called in succession:
stream_data.initialize_from_page_params();
stream_list.create_initial_sidebar_rows();
The first method was mostly extracted from subs.js, but I simplified
some things, like not needing to make a copy of the hashes
we were passed in, plus I now garbage collect email_dict. Also,
the code path that initialize_from_page_params() mostly replaces
used to call create_sub(), which fired a trigger, but now it
just does data stuff.
Once the data structure is built up, it's a very simple matter
to build the initial sidebar rows, and that's what the second
method does.
2016-10-17 19:34:58 +02:00
|
|
|
delete page_params.email_dict;
|
|
|
|
};
|
|
|
|
|
2016-10-28 18:47:57 +02:00
|
|
|
exports.get_recent_topics = function (stream_name) {
|
2016-10-28 18:54:39 +02:00
|
|
|
return recent_topics.get(stream_name);
|
2016-10-28 18:47:57 +02:00
|
|
|
};
|
|
|
|
|
2016-10-28 18:26:30 +02:00
|
|
|
exports.populate_stream_topics_for_tests = function (stream_map) {
|
|
|
|
// This is only used by tests.
|
2016-10-28 18:54:39 +02:00
|
|
|
recent_topics = new Dict.from(stream_map, {fold_case: true});
|
2016-10-28 18:26:30 +02:00
|
|
|
};
|
Clean up startup code for streams.
The startup code in subs.js used to intermingle data
stuff and UI stuff in a loop inside a called function,
which made the code hard to reason about.
Now there is a clear separation of concerns, with these methods
being called in succession:
stream_data.initialize_from_page_params();
stream_list.create_initial_sidebar_rows();
The first method was mostly extracted from subs.js, but I simplified
some things, like not needing to make a copy of the hashes
we were passed in, plus I now garbage collect email_dict. Also,
the code path that initialize_from_page_params() mostly replaces
used to call create_sub(), which fired a trigger, but now it
just does data stuff.
Once the data structure is built up, it's a very simple matter
to build the initial sidebar rows, and that's what the second
method does.
2016-10-17 19:34:58 +02:00
|
|
|
|
2013-08-15 21:11:07 +02:00
|
|
|
return exports;
|
|
|
|
|
|
|
|
}());
|
|
|
|
if (typeof module !== 'undefined') {
|
|
|
|
module.exports = stream_data;
|
|
|
|
}
|