mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-03 21:43:21 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			364 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			364 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
"use strict";
 | 
						|
 | 
						|
const assert = require("node:assert/strict");
 | 
						|
 | 
						|
const {mock_esm, zrequire} = require("./lib/namespace.cjs");
 | 
						|
const {run_test} = require("./lib/test.cjs");
 | 
						|
 | 
						|
let next_id = 0;
 | 
						|
const messages = new Map();
 | 
						|
 | 
						|
function make_stream_message({stream_id, topic, sender_id}) {
 | 
						|
    next_id += 1;
 | 
						|
 | 
						|
    const message = {
 | 
						|
        type: "stream",
 | 
						|
        stream_id,
 | 
						|
        id: next_id,
 | 
						|
        topic,
 | 
						|
        sender_id,
 | 
						|
    };
 | 
						|
    messages.set(message.id, message);
 | 
						|
 | 
						|
    return message;
 | 
						|
}
 | 
						|
 | 
						|
mock_esm("../src/message_store", {
 | 
						|
    get: (message_id) => messages.get(message_id),
 | 
						|
});
 | 
						|
const people = zrequire("people");
 | 
						|
people.initialize_current_user(1);
 | 
						|
const rs = zrequire("recent_senders");
 | 
						|
zrequire("message_util.ts");
 | 
						|
 | 
						|
function test(label, f) {
 | 
						|
    run_test(label, ({override}) => {
 | 
						|
        messages.clear();
 | 
						|
        next_id = 0;
 | 
						|
        rs.clear_for_testing();
 | 
						|
        f({override});
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
test("IdTracker", () => {
 | 
						|
    const id_tracker = new rs.IdTracker();
 | 
						|
 | 
						|
    function test_add(id, expected_max_id) {
 | 
						|
        id_tracker.add(id);
 | 
						|
        assert.equal(id_tracker.max_id(), expected_max_id);
 | 
						|
    }
 | 
						|
 | 
						|
    test_add(5, 5);
 | 
						|
    test_add(7, 7);
 | 
						|
    test_add(3, 7);
 | 
						|
    test_add(10, 10);
 | 
						|
    test_add(12, 12);
 | 
						|
    test_add(11, 12);
 | 
						|
 | 
						|
    function test_remove(id, expected_max_id) {
 | 
						|
        id_tracker.remove(id);
 | 
						|
        assert.equal(id_tracker.max_id(), expected_max_id);
 | 
						|
    }
 | 
						|
 | 
						|
    test_remove(10, 12);
 | 
						|
    test_remove(999999, 12); // bogus id has no effect
 | 
						|
    test_remove(3, 12);
 | 
						|
    test_remove(12, 11);
 | 
						|
 | 
						|
    test_add(3, 11);
 | 
						|
    test_add(7, 11);
 | 
						|
    test_add(13, 13);
 | 
						|
 | 
						|
    test_remove(3, 13);
 | 
						|
    test_remove(13, 11);
 | 
						|
});
 | 
						|
 | 
						|
test("noop process_topic_edit", () => {
 | 
						|
    // Just get line coverage on defensive code.
 | 
						|
    const bogus_ids = [333, 444];
 | 
						|
    rs.process_topic_edit({message_ids: bogus_ids});
 | 
						|
});
 | 
						|
 | 
						|
test("update_topics_of_deleted_message_ids", () => {
 | 
						|
    // Just get line coverage on defensive code.
 | 
						|
    const stream_id = 555;
 | 
						|
    const topic = "whatever";
 | 
						|
    const sender_id = 999;
 | 
						|
 | 
						|
    const message = make_stream_message({
 | 
						|
        stream_id,
 | 
						|
        topic,
 | 
						|
        sender_id,
 | 
						|
    });
 | 
						|
    rs.update_topics_of_deleted_message_ids([message.id]);
 | 
						|
    assert.deepEqual(rs.get_topic_recent_senders(stream_id, topic), []);
 | 
						|
 | 
						|
    rs.process_stream_message(message);
 | 
						|
    assert.deepEqual(rs.get_topic_recent_senders(stream_id, topic), [sender_id]);
 | 
						|
});
 | 
						|
 | 
						|
test("process_stream_message", () => {
 | 
						|
    const stream1 = 1;
 | 
						|
    const stream2 = 2;
 | 
						|
    const stream3 = 3;
 | 
						|
 | 
						|
    const topic1 = "topic-1";
 | 
						|
    const topic2 = "topic-2";
 | 
						|
    const topic3 = "topic-3";
 | 
						|
    const topic4 = "topic-4";
 | 
						|
 | 
						|
    const sender1 = 1;
 | 
						|
    const sender2 = 2;
 | 
						|
    const sender3 = 3;
 | 
						|
    const stream4 = 4;
 | 
						|
    const stream5 = 5;
 | 
						|
 | 
						|
    // New stream
 | 
						|
    const message1 = make_stream_message({
 | 
						|
        stream_id: stream1,
 | 
						|
        topic: topic1,
 | 
						|
        sender_id: sender1,
 | 
						|
    });
 | 
						|
 | 
						|
    const message2 = make_stream_message({
 | 
						|
        stream_id: stream2,
 | 
						|
        topic: topic1,
 | 
						|
        sender_id: sender2,
 | 
						|
    });
 | 
						|
 | 
						|
    rs.process_stream_message(message1);
 | 
						|
    rs.process_stream_message(message2);
 | 
						|
 | 
						|
    // Users have posted in only one of the streams
 | 
						|
    assert.equal(
 | 
						|
        rs.compare_by_recency({user_id: sender1}, {user_id: sender2}, stream1, topic1) < 0,
 | 
						|
        true,
 | 
						|
    );
 | 
						|
    assert.equal(
 | 
						|
        rs.compare_by_recency({user_id: sender1}, {user_id: sender2}, stream2, topic1) > 0,
 | 
						|
        true,
 | 
						|
    );
 | 
						|
 | 
						|
    // Users haven't posted in this stream, return zero
 | 
						|
    assert.equal(
 | 
						|
        rs.compare_by_recency({user_id: sender1}, {user_id: sender2}, stream3, undefined) === 0,
 | 
						|
        true,
 | 
						|
    );
 | 
						|
 | 
						|
    // New topic
 | 
						|
    const message3 = make_stream_message({
 | 
						|
        stream_id: stream1,
 | 
						|
        topic: topic2,
 | 
						|
        sender_id: sender3,
 | 
						|
    });
 | 
						|
 | 
						|
    rs.process_stream_message(message3);
 | 
						|
    assert.equal(
 | 
						|
        rs.compare_by_recency({user_id: sender3}, {user_id: sender2}, stream1, topic2) < 0,
 | 
						|
        true,
 | 
						|
    );
 | 
						|
 | 
						|
    // New sender
 | 
						|
    const message4 = make_stream_message({
 | 
						|
        stream_id: stream1,
 | 
						|
        topic: topic1,
 | 
						|
        sender_id: sender2,
 | 
						|
    });
 | 
						|
 | 
						|
    rs.process_stream_message(message4);
 | 
						|
    assert.equal(
 | 
						|
        rs.compare_by_recency({user_id: sender1}, {user_id: sender2}, stream1, topic1) > 0,
 | 
						|
        true,
 | 
						|
    );
 | 
						|
 | 
						|
    // More recent message
 | 
						|
    const message5 = make_stream_message({
 | 
						|
        stream_id: stream1,
 | 
						|
        topic: topic1,
 | 
						|
        sender_id: sender1,
 | 
						|
    });
 | 
						|
 | 
						|
    rs.process_stream_message(message5);
 | 
						|
    assert.equal(
 | 
						|
        rs.compare_by_recency({user_id: sender1}, {user_id: sender2}, stream1, topic1) < 0,
 | 
						|
        true,
 | 
						|
    );
 | 
						|
 | 
						|
    // Messages sent by sender1 in stream1 > topic1
 | 
						|
    assert.equal(rs.get_topic_message_ids_for_sender(stream1, topic1, sender1).size, 2);
 | 
						|
    // Messages sent by sender1 in stream1 > topic2
 | 
						|
    assert.equal(rs.get_topic_message_ids_for_sender(stream1, topic2, sender1).size, 0);
 | 
						|
 | 
						|
    // Same stream, but different topics
 | 
						|
    const message6 = make_stream_message({
 | 
						|
        stream_id: stream3,
 | 
						|
        topic: topic1,
 | 
						|
        sender_id: sender1,
 | 
						|
    });
 | 
						|
    const message7 = make_stream_message({
 | 
						|
        stream_id: stream3,
 | 
						|
        topic: topic2,
 | 
						|
        sender_id: sender2,
 | 
						|
    });
 | 
						|
    const message8 = make_stream_message({
 | 
						|
        stream_id: stream3,
 | 
						|
        topic: topic3,
 | 
						|
        sender_id: sender3,
 | 
						|
    });
 | 
						|
 | 
						|
    rs.process_stream_message(message6);
 | 
						|
    rs.process_stream_message(message7);
 | 
						|
    rs.process_stream_message(message8);
 | 
						|
 | 
						|
    // topic3 has a message in it, but sender1 nor sender2 have participated, so sort by stream
 | 
						|
    assert.equal(
 | 
						|
        rs.compare_by_recency({user_id: sender1}, {user_id: sender2}, stream3, topic3) > 0,
 | 
						|
        true,
 | 
						|
    );
 | 
						|
    assert.equal(
 | 
						|
        rs.compare_by_recency({user_id: sender2}, {user_id: sender1}, stream3, topic3) < 0,
 | 
						|
        true,
 | 
						|
    );
 | 
						|
 | 
						|
    assert.equal(rs.compare_by_recency({}, {}, (next_id += 1), ""), 0);
 | 
						|
 | 
						|
    // new message in topic2
 | 
						|
    const message9 = make_stream_message({
 | 
						|
        stream_id: stream3,
 | 
						|
        topic: topic2,
 | 
						|
        sender_id: sender3,
 | 
						|
    });
 | 
						|
 | 
						|
    rs.process_stream_message(message9);
 | 
						|
 | 
						|
    // Test topic change
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream3, topic3).toString(), "3");
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream3, topic2).toString(), "3,2");
 | 
						|
 | 
						|
    // message7's topic was changed by user
 | 
						|
    message7.topic = topic3;
 | 
						|
 | 
						|
    rs.process_topic_edit({
 | 
						|
        message_ids: [message7.id],
 | 
						|
        old_stream_id: stream3,
 | 
						|
        new_stream_id: stream3,
 | 
						|
        old_topic: topic2,
 | 
						|
        new_topic: topic3,
 | 
						|
    });
 | 
						|
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream3, topic3).toString(), "3,2");
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream3, topic2).toString(), "3");
 | 
						|
 | 
						|
    // Test stream change
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream3, topic3).toString(), "3,2");
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream4, topic3).toString(), "");
 | 
						|
 | 
						|
    message7.stream_id = stream4;
 | 
						|
    message8.stream_id = stream4;
 | 
						|
    rs.process_topic_edit({
 | 
						|
        message_ids: [message7.id, message8.id],
 | 
						|
        old_stream_id: stream3,
 | 
						|
        new_stream_id: stream4,
 | 
						|
        old_topic: topic3,
 | 
						|
        new_topic: topic3,
 | 
						|
    });
 | 
						|
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream3, topic3).toString(), "");
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream4, topic3).toString(), "3,2");
 | 
						|
 | 
						|
    // Test stream & topic change
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream4, topic3).toString(), "3,2");
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream5, topic4).toString(), "");
 | 
						|
 | 
						|
    message7.stream_id = stream5;
 | 
						|
    message7.topic = topic4;
 | 
						|
 | 
						|
    message8.stream_id = stream5;
 | 
						|
    message8.topic = topic4;
 | 
						|
 | 
						|
    rs.process_topic_edit({
 | 
						|
        message_ids: [message7.id, message8.id],
 | 
						|
        old_stream_id: stream4,
 | 
						|
        new_stream_id: stream5,
 | 
						|
        old_topic: topic3,
 | 
						|
        new_topic: topic4,
 | 
						|
    });
 | 
						|
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream4, topic3).toString(), "");
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream5, topic4).toString(), "3,2");
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream1, topic1).toString(), "1,2");
 | 
						|
 | 
						|
    // delete message1 and message5 sent by sender1
 | 
						|
    rs.update_topics_of_deleted_message_ids([message1.id, message5.id]);
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream1, topic1).toString(), "2");
 | 
						|
 | 
						|
    // test that we can remove again, harmlessly
 | 
						|
    rs.update_topics_of_deleted_message_ids([message1.id, message5.id]);
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream1, topic1).toString(), "2");
 | 
						|
 | 
						|
    // remove some more senders
 | 
						|
    rs.update_topics_of_deleted_message_ids([message2.id, message3.id, message4.id, message5.id]);
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream1, topic1).toString(), "");
 | 
						|
 | 
						|
    rs.update_topics_of_deleted_message_ids([message6.id, message7.id, message8.id, message9.id]);
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream1, topic1).toString(), "");
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream2, topic2).toString(), "");
 | 
						|
    assert.equal(rs.get_topic_recent_senders(stream3, topic3).toString(), "");
 | 
						|
 | 
						|
    // deleting an old message which isn't locally stored.
 | 
						|
    // We are just testing that it doesn't raise an error;
 | 
						|
    // no changes should take place in this case.
 | 
						|
    rs.update_topics_of_deleted_message_ids([-1]);
 | 
						|
 | 
						|
    // Comparing on a non-existent topic doesn't crash.
 | 
						|
    assert.equal(
 | 
						|
        rs.compare_by_recency({user_id: sender2}, {user_id: sender1}, stream3, "bogus") < 0,
 | 
						|
        true,
 | 
						|
    );
 | 
						|
});
 | 
						|
 | 
						|
test("process_pms", () => {
 | 
						|
    const sender1 = 1; // Current user id
 | 
						|
    const sender2 = 2;
 | 
						|
    const sender3 = 3;
 | 
						|
 | 
						|
    const user_ids_string = "2,3,4";
 | 
						|
    rs.process_private_message({
 | 
						|
        to_user_ids: user_ids_string,
 | 
						|
        sender_id: sender2,
 | 
						|
        id: 1,
 | 
						|
    });
 | 
						|
    rs.process_private_message({
 | 
						|
        to_user_ids: user_ids_string,
 | 
						|
        sender_id: sender3,
 | 
						|
        id: 2,
 | 
						|
    });
 | 
						|
    rs.process_private_message({
 | 
						|
        to_user_ids: user_ids_string,
 | 
						|
        sender_id: sender1,
 | 
						|
        id: 3,
 | 
						|
    });
 | 
						|
 | 
						|
    // Recent Conversations displays avatars in the opposite order to this since
 | 
						|
    // that was simpler to implement in HTML.
 | 
						|
    assert.deepEqual(rs.get_pm_recent_senders(user_ids_string), {
 | 
						|
        participants: [1, 3, 2],
 | 
						|
        non_participants: [4],
 | 
						|
    });
 | 
						|
    // Direct message doesn't exist.
 | 
						|
    assert.deepEqual(rs.get_pm_recent_senders("1000,2000"), {
 | 
						|
        participants: [],
 | 
						|
        non_participants: [],
 | 
						|
    });
 | 
						|
 | 
						|
    rs.process_private_message({
 | 
						|
        to_user_ids: "1",
 | 
						|
        sender_id: sender1,
 | 
						|
        id: 4,
 | 
						|
    });
 | 
						|
    assert.deepEqual(rs.get_pm_recent_senders("1"), {
 | 
						|
        participants: [1],
 | 
						|
        non_participants: [],
 | 
						|
    });
 | 
						|
});
 |