mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-04 05:53:43 +00:00 
			
		
		
		
	Previously, if you topic-edited locally echoed messages, it would
throw this exception:
TypeError: Cannot read property 'indexOf' of undefined
    at Object.h.is_status_message (https://chat.zulip.org/static/min/app.274582375298.js:1742:154)
           = static/js/markdown.js line 125 column 13
    at Object.h.apply_markdown (https://chat.zulip.org/static/min/app.274582375298.js:1741:291)
           = static/js/markdown.js line 93 column 29
    at Object.a.edit_locally (https://chat.zulip.org/static/min/app.274582375298.js:1750:195)
           = static/js/echo.js line 176 column 5
    at Object.d.save (https://chat.zulip.org/static/min/app.274582375298.js:1921:376)
           = static/js/message_edit.js line 132 column 13
The behavior with this patch is still not ideal, in that we don't ever
send the edit to the server (if the issue was lag, not an actual error
on sending), but this at least makes the on-error case correct.
		
	
		
			
				
	
	
		
			310 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			310 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
var echo = (function () {
 | 
						|
 | 
						|
var exports = {};
 | 
						|
 | 
						|
var waiting_for_id = {};
 | 
						|
var waiting_for_ack = {};
 | 
						|
 | 
						|
function resend_message(message, row) {
 | 
						|
    message.content = message.raw_content;
 | 
						|
    var retry_spinner = row.find('.refresh-failed-message');
 | 
						|
    retry_spinner.toggleClass('rotating', true);
 | 
						|
 | 
						|
    // Always re-set queue_id if we've gotten a new one
 | 
						|
    // since the time when the message object was initially created
 | 
						|
    message.queue_id = page_params.queue_id;
 | 
						|
 | 
						|
    var local_id = message.local_id;
 | 
						|
 | 
						|
    function on_success(data) {
 | 
						|
        var message_id = data.id;
 | 
						|
        var locally_echoed = true;
 | 
						|
 | 
						|
        retry_spinner.toggleClass('rotating', false);
 | 
						|
 | 
						|
        compose.send_message_success(local_id, message_id, locally_echoed);
 | 
						|
 | 
						|
        // Resend succeeded, so mark as no longer failed
 | 
						|
        message_store.get(message_id).failed_request = false;
 | 
						|
        ui.show_failed_message_success(message_id);
 | 
						|
    }
 | 
						|
 | 
						|
    function on_error(response) {
 | 
						|
        exports.message_send_error(local_id, response);
 | 
						|
        retry_spinner.toggleClass('rotating', false);
 | 
						|
        blueslip.log("Manual resend of message failed");
 | 
						|
    }
 | 
						|
 | 
						|
    sent_messages.start_resend(local_id);
 | 
						|
    transmit.send_message(message, on_success, on_error);
 | 
						|
}
 | 
						|
 | 
						|
function truncate_precision(float) {
 | 
						|
    return parseFloat(float.toFixed(3));
 | 
						|
}
 | 
						|
 | 
						|
var get_next_local_id = (function () {
 | 
						|
 | 
						|
    var already_used = {};
 | 
						|
 | 
						|
    return function () {
 | 
						|
        var local_id_increment = 0.01;
 | 
						|
        var latest = page_params.max_message_id;
 | 
						|
        if (typeof message_list.all !== 'undefined' && message_list.all.last() !== undefined) {
 | 
						|
            latest = message_list.all.last().id;
 | 
						|
        }
 | 
						|
        latest = Math.max(0, latest);
 | 
						|
        var next_local_id = truncate_precision(latest + local_id_increment);
 | 
						|
 | 
						|
        if (already_used[next_local_id]) {
 | 
						|
            // If our id is already used, it is probably an edge case like we had
 | 
						|
            // to abort a very recent message.
 | 
						|
            blueslip.warn("We don't reuse ids for local echo.");
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        if (next_local_id % 1 > local_id_increment * 5) {
 | 
						|
            blueslip.warn("Turning off local echo for this message to let host catch up");
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        if (next_local_id % 1 === 0) {
 | 
						|
            // The logic to stop at 0.05 should prevent us from ever wrapping around
 | 
						|
            // to the next integer.
 | 
						|
            blueslip.error("Programming error");
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        already_used[next_local_id] = true;
 | 
						|
 | 
						|
        return next_local_id;
 | 
						|
    };
 | 
						|
}());
 | 
						|
 | 
						|
function insert_local_message(message_request, local_id) {
 | 
						|
    // Shallow clone of message request object that is turned into something suitable
 | 
						|
    // for zulip.js:add_message
 | 
						|
    // Keep this in sync with changes to compose.create_message_object
 | 
						|
    var message = $.extend({}, message_request);
 | 
						|
 | 
						|
    // Locally delivered messages cannot be unread (since we sent them), nor
 | 
						|
    // can they alert the user.
 | 
						|
    message.unread = false;
 | 
						|
 | 
						|
    message.raw_content = message.content;
 | 
						|
 | 
						|
    // NOTE: This will parse synchronously. We're not using the async pipeline
 | 
						|
    markdown.apply_markdown(message);
 | 
						|
 | 
						|
    message.content_type = 'text/html';
 | 
						|
    message.sender_email = people.my_current_email();
 | 
						|
    message.sender_full_name = people.my_full_name();
 | 
						|
    message.avatar_url = page_params.avatar_url;
 | 
						|
    message.timestamp = new XDate().getTime() / 1000;
 | 
						|
    message.local_id = local_id;
 | 
						|
    message.locally_echoed = true;
 | 
						|
    message.id = message.local_id;
 | 
						|
    markdown.add_subject_links(message);
 | 
						|
 | 
						|
    waiting_for_id[message.local_id] = message;
 | 
						|
    waiting_for_ack[message.local_id] = message;
 | 
						|
 | 
						|
    if (message.type === 'stream') {
 | 
						|
        message.display_recipient = message.stream;
 | 
						|
    } else {
 | 
						|
        // Build a display recipient with the full names of each
 | 
						|
        // recipient.  Note that it's important that use
 | 
						|
        // util.extract_pm_recipients, which filters out any spurious
 | 
						|
        // ", " at the end of the recipient list
 | 
						|
        var emails = util.extract_pm_recipients(message_request.private_message_recipient);
 | 
						|
        message.display_recipient = _.map(emails, function (email) {
 | 
						|
            email = email.trim();
 | 
						|
            var person = people.get_by_email(email);
 | 
						|
            if (person === undefined) {
 | 
						|
                // For unknown users, we return a skeleton object.
 | 
						|
                return {email: email, full_name: email,
 | 
						|
                        unknown_local_echo_user: true};
 | 
						|
            }
 | 
						|
            // NORMAL PATH
 | 
						|
            return person;
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    // It is a little bit funny to go through the message_events
 | 
						|
    // codepath, but it's sort of the idea behind local echo that
 | 
						|
    // we are simulating server events before they actually arrive.
 | 
						|
    message_events.insert_new_messages([message], true);
 | 
						|
    return message.local_id;
 | 
						|
}
 | 
						|
 | 
						|
exports.try_deliver_locally = function try_deliver_locally(message_request) {
 | 
						|
    if (markdown.contains_backend_only_syntax(message_request.content)) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (narrow_state.active() && !narrow_state.filter().can_apply_locally()) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    var next_local_id = get_next_local_id();
 | 
						|
 | 
						|
    if (!next_local_id) {
 | 
						|
        // This can happen for legit reasons.
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    return insert_local_message(message_request, next_local_id);
 | 
						|
};
 | 
						|
 | 
						|
exports.edit_locally = function edit_locally(message, raw_content, new_topic) {
 | 
						|
    var message_content_edited = raw_content !== undefined && message.raw_content !== raw_content;
 | 
						|
 | 
						|
    if (new_topic !== undefined) {
 | 
						|
        topic_data.remove_message({
 | 
						|
            stream_id: message.stream_id,
 | 
						|
            topic_name: message.subject,
 | 
						|
        });
 | 
						|
 | 
						|
        message.subject = new_topic;
 | 
						|
 | 
						|
        topic_data.add_message({
 | 
						|
            stream_id: message.stream_id,
 | 
						|
            topic_name: message.subject,
 | 
						|
            message_id: message.id,
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    if (message_content_edited) {
 | 
						|
        message.raw_content = raw_content;
 | 
						|
        markdown.apply_markdown(message);
 | 
						|
    }
 | 
						|
 | 
						|
    // We don't handle unread counts since local messages must be sent by us
 | 
						|
 | 
						|
    home_msg_list.view.rerender_messages([message]);
 | 
						|
    if (current_msg_list === message_list.narrowed) {
 | 
						|
        message_list.narrowed.view.rerender_messages([message]);
 | 
						|
    }
 | 
						|
    stream_list.update_streams_sidebar();
 | 
						|
    pm_list.update_private_messages();
 | 
						|
};
 | 
						|
 | 
						|
exports.reify_message_id = function reify_message_id(local_id, server_id) {
 | 
						|
    var message = waiting_for_id[local_id];
 | 
						|
    delete waiting_for_id[local_id];
 | 
						|
 | 
						|
    // reify_message_id is called both on receiving a self-sent message
 | 
						|
    // from the server, and on receiving the response to the send request
 | 
						|
    // Reification is only needed the first time the server id is found
 | 
						|
    if (message === undefined) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    message.id = server_id;
 | 
						|
    message.locally_echoed = false;
 | 
						|
 | 
						|
    var opts = {old_id: local_id, new_id: server_id};
 | 
						|
 | 
						|
    message_store.reify_message_id(opts);
 | 
						|
    notifications.reify_message_id(opts);
 | 
						|
};
 | 
						|
 | 
						|
exports.process_from_server = function process_from_server(messages) {
 | 
						|
    var msgs_to_rerender = [];
 | 
						|
    var non_echo_messages = [];
 | 
						|
 | 
						|
    _.each(messages, function (message) {
 | 
						|
        // In case we get the sent message before we get the send ACK, reify here
 | 
						|
 | 
						|
        var client_message = waiting_for_ack[message.local_id];
 | 
						|
 | 
						|
        if (client_message === undefined) {
 | 
						|
            // For messages that weren't locally echoed, we go through
 | 
						|
            // the "main" codepath that doesn't have to id reconciliation.
 | 
						|
            // We simply return non-echo messages to our caller.
 | 
						|
            non_echo_messages.push(message);
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        exports.reify_message_id(message.local_id, message.id);
 | 
						|
 | 
						|
        if (client_message.content !== message.content) {
 | 
						|
            client_message.content = message.content;
 | 
						|
            sent_messages.mark_disparity(message.local_id);
 | 
						|
        }
 | 
						|
 | 
						|
        message_store.update_booleans(client_message, message.flags);
 | 
						|
 | 
						|
        // We don't try to highlight alert words locally, so we have to
 | 
						|
        // do it now.  (Note that we will indeed highlight alert words in
 | 
						|
        // messages that we sent to ourselves, since we might want to test
 | 
						|
        // that our alert words are set up correctly.)
 | 
						|
        alert_words.process_message(client_message);
 | 
						|
 | 
						|
        // Previously, the message had the "local echo" timestamp set
 | 
						|
        // by the browser; if there was some round-trip delay to the
 | 
						|
        // server, the actual server-side timestamp could be slightly
 | 
						|
        // different.  This corrects the frontend timestamp to match
 | 
						|
        // the backend.
 | 
						|
        client_message.timestamp = message.timestamp;
 | 
						|
 | 
						|
        msgs_to_rerender.push(client_message);
 | 
						|
        delete waiting_for_ack[client_message.id];
 | 
						|
    });
 | 
						|
 | 
						|
    if (msgs_to_rerender.length > 0) {
 | 
						|
        // In theory, we could just rerender messages where there were
 | 
						|
        // changes in either the rounded timestamp we display or the
 | 
						|
        // message content, but in practice, there's no harm to just
 | 
						|
        // doing it unconditionally.
 | 
						|
        home_msg_list.view.rerender_messages(msgs_to_rerender);
 | 
						|
        if (current_msg_list === message_list.narrowed) {
 | 
						|
            message_list.narrowed.view.rerender_messages(msgs_to_rerender);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return non_echo_messages;
 | 
						|
};
 | 
						|
 | 
						|
exports.message_send_error = function message_send_error(local_id, error_response) {
 | 
						|
    // Error sending message, show inline
 | 
						|
    message_store.get(local_id).failed_request = true;
 | 
						|
    ui.show_message_failed(local_id, error_response);
 | 
						|
};
 | 
						|
 | 
						|
function abort_message(message) {
 | 
						|
    // Remove in all lists in which it exists
 | 
						|
    _.each([message_list.all, home_msg_list, current_msg_list], function (msg_list) {
 | 
						|
        msg_list.remove_and_rerender([message]);
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
$(function () {
 | 
						|
    function on_failed_action(action, callback) {
 | 
						|
        $("#main_div").on("click", "." + action + "-failed-message", function (e) {
 | 
						|
            e.stopPropagation();
 | 
						|
            popovers.hide_all();
 | 
						|
            var row = $(this).closest(".message_row");
 | 
						|
            var message_id = rows.id(row);
 | 
						|
            // Message should be waiting for ack and only have a local id,
 | 
						|
            // otherwise send would not have failed
 | 
						|
            var message = waiting_for_ack[message_id];
 | 
						|
            if (message === undefined) {
 | 
						|
                blueslip.warn("Got resend or retry on failure request but did not find message in ack list " + message_id);
 | 
						|
                return;
 | 
						|
            }
 | 
						|
            callback(message, row);
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    on_failed_action('remove', abort_message);
 | 
						|
    on_failed_action('refresh', resend_message);
 | 
						|
});
 | 
						|
 | 
						|
return exports;
 | 
						|
 | 
						|
}());
 | 
						|
if (typeof module !== 'undefined') {
 | 
						|
    module.exports = echo;
 | 
						|
}
 |