zulip/frontend_tests/node_tests/stream_events.js

408 lines
13 KiB
JavaScript

"use strict";
const {strict: assert} = require("assert");
const rewiremock = require("rewiremock/node");
const {set_global, zrequire} = require("../zjsunit/namespace");
const {make_stub} = require("../zjsunit/stub");
const {run_test} = require("../zjsunit/test");
const $ = require("../zjsunit/zjquery");
const noop = () => {};
const _settings_notifications = {
update_page: () => {},
};
set_global("settings_notifications", _settings_notifications);
const color_data = {__esModule: true};
rewiremock("../../static/js/color_data").with(color_data);
set_global("current_msg_list", {});
const message_util = set_global("message_util", {});
const stream_color = {__esModule: true};
rewiremock("../../static/js/stream_color").with(stream_color);
const stream_list = set_global("stream_list", {});
const stream_muting = {__esModule: true};
rewiremock("../../static/js/stream_muting").with(stream_muting);
let subs = set_global("subs", {});
set_global("recent_topics", {
complete_rerender: () => {},
});
set_global("message_list", {
all: {
all_messages() {
return ["msg"];
},
},
});
subs = set_global("subs", {update_settings_for_subscribed: noop});
set_global("overlays", {streams_open: () => true});
rewiremock.enable();
const peer_data = zrequire("peer_data");
const people = zrequire("people");
const stream_data = zrequire("stream_data");
const stream_events = zrequire("stream_events");
const {Filter} = zrequire("Filter", "js/filter");
const narrow_state = zrequire("narrow_state");
const message_view_header = zrequire("message_view_header");
const george = {
email: "george@zulip.com",
full_name: "George",
user_id: 103,
};
const me = {
email: "me@zulip.com",
full_name: "Me Myself",
user_id: 104,
};
people.add_active_user(george);
people.add_active_user(me);
people.initialize_current_user(me.user_id);
const frontend = {
subscribed: false,
color: "yellow",
name: "frontend",
stream_id: 101,
is_muted: true,
invite_only: false,
};
stream_data.add_sub(frontend);
const frontend_filter_terms = [{operator: "stream", operand: "frontend"}];
const frontend_filter = new Filter(frontend_filter_terms);
run_test("update_property", (override) => {
const stream_id = frontend.stream_id;
// Invoke error for non-existent stream/property
{
let errors = 0;
override(blueslip, "warn", () => {
errors += 1;
});
stream_events.update_property(99, "color", "blue");
assert.equal(errors, 1);
stream_events.update_property(stream_id, "not_real", 42);
assert.equal(errors, 2);
}
// Test update color
{
const stub = make_stub();
override(stream_color, "update_stream_color", stub.f);
stream_events.update_property(stream_id, "color", "blue");
assert.equal(stub.num_calls, 1);
const args = stub.get_args("sub", "val");
assert.equal(args.sub.stream_id, stream_id);
assert.equal(args.val, "blue");
}
// Test in home view
{
const stub = make_stub();
override(stream_muting, "update_is_muted", stub.f);
stream_events.update_property(stream_id, "in_home_view", false);
assert.equal(stub.num_calls, 1);
const args = stub.get_args("sub", "val");
assert.equal(args.sub.stream_id, stream_id);
assert.equal(args.val, true);
}
function checkbox_for(property) {
return $(`#${CSS.escape(property)}_${CSS.escape(stream_id)}`);
}
// Test desktop notifications
stream_events.update_property(stream_id, "desktop_notifications", true);
assert.equal(frontend.desktop_notifications, true);
let checkbox = checkbox_for("desktop_notifications");
assert.equal(checkbox.prop("checked"), true);
// Tests audible notifications
stream_events.update_property(stream_id, "audible_notifications", true);
assert.equal(frontend.audible_notifications, true);
checkbox = checkbox_for("audible_notifications");
assert.equal(checkbox.prop("checked"), true);
// Tests push notifications
stream_events.update_property(stream_id, "push_notifications", true);
assert.equal(frontend.push_notifications, true);
checkbox = checkbox_for("push_notifications");
assert.equal(checkbox.prop("checked"), true);
// Tests email notifications
stream_events.update_property(stream_id, "email_notifications", true);
assert.equal(frontend.email_notifications, true);
checkbox = checkbox_for("email_notifications");
assert.equal(checkbox.prop("checked"), true);
// Tests wildcard_mentions_notify notifications
stream_events.update_property(stream_id, "wildcard_mentions_notify", true);
assert.equal(frontend.wildcard_mentions_notify, true);
checkbox = checkbox_for("wildcard_mentions_notify");
assert.equal(checkbox.prop("checked"), true);
// Test name change
{
const stub = make_stub();
override(subs, "update_stream_name", stub.f);
stream_events.update_property(stream_id, "name", "the frontend");
assert.equal(stub.num_calls, 1);
const args = stub.get_args("sub", "val");
assert.equal(args.sub.stream_id, stream_id);
assert.equal(args.val, "the frontend");
}
// Test description change
{
const stub = make_stub();
override(subs, "update_stream_description", stub.f);
stream_events.update_property(stream_id, "description", "we write code", {
rendered_description: "we write code",
});
assert.equal(stub.num_calls, 1);
const args = stub.get_args("sub", "val");
assert.equal(args.sub.stream_id, stream_id);
assert.equal(args.val, "we write code");
}
// Test email address change
stream_events.update_property(stream_id, "email_address", "zooly@zulip.com");
assert.equal(frontend.email_address, "zooly@zulip.com");
// Test pin to top
{
override(stream_list, "refresh_pinned_or_unpinned_stream", noop);
stream_events.update_property(stream_id, "pin_to_top", true);
checkbox = checkbox_for("pin_to_top");
assert.equal(checkbox.prop("checked"), true);
}
// Test stream privacy change event
{
const stub = make_stub();
override(subs, "update_stream_privacy", stub.f);
stream_events.update_property(stream_id, "invite_only", true, {
history_public_to_subscribers: true,
});
assert.equal(stub.num_calls, 1);
const args = stub.get_args("sub", "val");
assert.equal(args.sub.stream_id, stream_id);
assert.deepEqual(args.val, {
invite_only: true,
history_public_to_subscribers: true,
});
}
// Test stream stream_post_policy change event
{
const stub = make_stub();
override(subs, "update_stream_post_policy", stub.f);
stream_events.update_property(
stream_id,
"stream_post_policy",
stream_data.stream_post_policy_values.admins.code,
);
assert.equal(stub.num_calls, 1);
const args = stub.get_args("sub", "val");
assert.equal(args.sub.stream_id, stream_id);
assert.equal(args.val, stream_data.stream_post_policy_values.admins.code);
}
// Test stream message_retention_days change event
{
const stub = make_stub();
override(subs, "update_message_retention_setting", stub.f);
stream_events.update_property(stream_id, "message_retention_days", 20);
assert.equal(stub.num_calls, 1);
const args = stub.get_args("sub", "val");
assert.equal(args.sub.stream_id, stream_id);
assert.equal(args.val, 20);
}
});
run_test("marked_unsubscribed (code coverage)", () => {
// We don't error for unsubscribed streams for some reason.
stream_events.mark_unsubscribed(undefined);
});
run_test("marked_(un)subscribed (early return)", () => {
// The early-return prevents us from exploding or needing
// to override functions with side effects
stream_events.mark_subscribed({subscribed: true});
stream_events.mark_unsubscribed({subscribed: false});
});
run_test("marked_subscribed (error)", (override) => {
// Test undefined error
let errors = 0;
override(blueslip, "error", () => {
errors += 1;
});
stream_events.mark_subscribed(undefined, [], "yellow");
assert.equal(errors, 1);
});
run_test("marked_subscribed (normal)", (override) => {
override(stream_data, "subscribe_myself", noop);
override(stream_data, "update_calculated_fields", noop);
override(stream_color, "update_stream_color", noop);
narrow_state.set_current_filter(frontend_filter);
let args;
let list_updated = false;
const stream_list_stub = make_stub();
const message_view_header_stub = make_stub();
const message_util_stub = make_stub();
override(stream_list, "add_sidebar_row", stream_list_stub.f);
override(message_util, "do_unread_count_updates", message_util_stub.f);
override(message_view_header, "render_title_area", message_view_header_stub.f);
override(current_msg_list, "update_trailing_bookend", () => {
list_updated = true;
});
stream_events.mark_subscribed(frontend, [], "blue");
args = message_util_stub.get_args("messages");
assert.deepEqual(args.messages, ["msg"]);
args = stream_list_stub.get_args("sub");
assert.equal(args.sub.stream_id, frontend.stream_id);
assert.equal(message_view_header_stub.num_calls, 1);
assert.equal(list_updated, true);
assert.equal(frontend.color, "blue");
narrow_state.reset_current_filter();
});
run_test("marked_subscribed (color)", (override) => {
override(stream_data, "subscribe_myself", noop);
override(stream_data, "update_calculated_fields", noop);
override(message_util, "do_unread_count_updates", noop);
override(stream_list, "add_sidebar_row", noop);
frontend.color = undefined;
override(color_data, "pick_color", () => "green");
let warnings = 0;
override(blueslip, "warn", () => {
warnings += 1;
});
// narrow state is undefined
{
const stub = make_stub();
override(subs, "set_color", stub.f);
stream_events.mark_subscribed(frontend, [], undefined);
assert.equal(stub.num_calls, 1);
const args = stub.get_args("id", "color");
assert.equal(args.id, frontend.stream_id);
assert.equal(args.color, "green");
assert.equal(warnings, 1);
}
});
run_test("marked_subscribed (emails)", (override) => {
override(stream_data, "update_calculated_fields", noop);
override(stream_color, "update_stream_color", noop);
// Test assigning subscriber emails
// narrow state is undefined
override(message_util, "do_unread_count_updates", noop);
override(stream_list, "add_sidebar_row", noop);
const subs_stub = make_stub();
override(subs, "update_settings_for_subscribed", subs_stub.f);
assert(!stream_data.is_subscribed(frontend.name));
const user_ids = [15, 20, 25, me.user_id];
stream_events.mark_subscribed(frontend, user_ids, "");
assert.deepEqual(new Set(peer_data.get_subscribers(frontend.stream_id)), new Set(user_ids));
assert(stream_data.is_subscribed(frontend.name));
const args = subs_stub.get_args("sub");
assert.deepEqual(frontend, args.sub);
});
run_test("mark_unsubscribed (update_settings_for_unsubscribed)", (override) => {
override(stream_data, "update_calculated_fields", noop);
// Test unsubscribe
frontend.subscribed = true;
const stub = make_stub();
override(subs, "update_settings_for_unsubscribed", stub.f);
override(stream_list, "remove_sidebar_row", noop);
override(stream_data, "unsubscribe_myself", noop);
stream_events.mark_unsubscribed(frontend);
const args = stub.get_args("sub");
assert.deepEqual(args.sub, frontend);
});
run_test("mark_unsubscribed (render_title_area)", (override) => {
override(stream_data, "update_calculated_fields", noop);
// Test update bookend and remove done event
narrow_state.set_current_filter(frontend_filter);
const message_view_header_stub = make_stub();
override(message_view_header, "render_title_area", message_view_header_stub.f);
override(stream_data, "unsubscribe_myself", noop);
override(subs, "update_settings_for_unsubscribed", noop);
override(current_msg_list, "update_trailing_bookend", noop);
override(stream_list, "remove_sidebar_row", noop);
stream_events.mark_unsubscribed(frontend);
assert.equal(message_view_header_stub.num_calls, 1);
narrow_state.reset_current_filter();
});
stream_data.clear_subscriptions();
const dev_help = {
subscribed: true,
color: "blue",
name: "dev help",
stream_id: 2,
is_muted: true,
invite_only: false,
};
stream_data.add_sub(dev_help);
run_test("remove_deactivated_user_from_all_streams", () => {
const subs_stub = make_stub();
subs.update_subscribers_ui = subs_stub.f;
dev_help.can_access_subscribers = true;
// assert starting state
assert(!stream_data.is_user_subscribed(dev_help.stream_id, george.user_id));
// verify that deactivating user should unsubscribe user from all streams
peer_data.add_subscriber(dev_help.stream_id, george.user_id);
assert(stream_data.is_user_subscribed(dev_help.stream_id, george.user_id));
stream_events.remove_deactivated_user_from_all_streams(george.user_id);
// verify that we issue a call to update subscriber count/list UI
assert.equal(subs_stub.num_calls, 1);
});
rewiremock.disable();