# frozen_string_literal: true require "rails_helper" RSpec.describe Chat::ChatController do fab!(:user) { Fabricate(:user) } fab!(:other_user) { Fabricate(:user) } fab!(:admin) { Fabricate(:admin) } fab!(:category) { Fabricate(:category) } fab!(:chat_channel) { Fabricate(:category_channel, chatable: category) } fab!(:dm_chat_channel) { Fabricate(:direct_message_channel, users: [user, other_user, admin]) } fab!(:tag) { Fabricate(:tag) } MESSAGE_COUNT = 70 MESSAGE_COUNT.times do |n| fab!("message_#{n}") do Fabricate( :chat_message, chat_channel: chat_channel, user: other_user, message: "message #{n}", ) end end before do SiteSetting.chat_enabled = true SiteSetting.chat_allowed_groups = Group::AUTO_GROUPS[:everyone] end def flag_message(message, flagger, flag_type: ReviewableScore.types[:off_topic]) Chat::ChatReviewQueue.new.flag_message(message, Guardian.new(flagger), flag_type)[:reviewable] end describe "#messages" do let(:page_size) { 30 } before do sign_in(user) Group.refresh_automatic_groups! end it "errors for user when they are not allowed to chat" do SiteSetting.chat_allowed_groups = Group::AUTO_GROUPS[:staff] get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.status).to eq(403) end it "errors when page size is over 50" do get "/chat/#{chat_channel.id}/messages.json", params: { page_size: 51 } expect(response.status).to eq(400) end it "errors when page size is nil" do get "/chat/#{chat_channel.id}/messages.json" expect(response.status).to eq(400) end it "returns the latest messages in created_at, id order" do get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } messages = response.parsed_body["chat_messages"] expect(messages.count).to eq(page_size) expect(messages.first["created_at"].to_time).to be < messages.last["created_at"].to_time end it "returns `can_flag=true` for public channels" do get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["meta"]["can_flag"]).to be true end it "returns `can_flag=true` for DM channels" do get "/chat/#{dm_chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["meta"]["can_flag"]).to be true end it "returns `can_moderate=true` based on whether the user can moderate the chatable" do 1.upto(4) do |n| user.update!(trust_level: n) get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["meta"]["can_moderate"]).to be false end get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["meta"]["can_moderate"]).to be false user.update!(admin: true) get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["meta"]["can_moderate"]).to be true user.update!(admin: false) SiteSetting.enable_category_group_moderation = true group = Fabricate(:group) group.add(user) category.update!(reviewable_by_group: group) get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["meta"]["can_moderate"]).to be true end it "serializes `user_flag_status` for user who has a pending flag" do chat_message = chat_channel.chat_messages.last reviewable = flag_message(chat_message, user) score = reviewable.reviewable_scores.last get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["chat_messages"].last["user_flag_status"]).to eq( score.status_for_database, ) end it "doesn't serialize `reviewable_ids` for non-staff" do reviewable = flag_message(chat_channel.chat_messages.last, admin) get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["chat_messages"].last["reviewable_id"]).to be_nil end it "serializes `reviewable_ids` correctly for staff" do sign_in(admin) reviewable = flag_message(chat_channel.chat_messages.last, admin) get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["chat_messages"].last["reviewable_id"]).to eq(reviewable.id) end it "correctly marks reactions as 'reacted' for the current_user" do heart_emoji = ":heart:" smile_emoji = ":smile" last_message = chat_channel.chat_messages.last last_message.reactions.create(user: user, emoji: heart_emoji) last_message.reactions.create(user: admin, emoji: smile_emoji) get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } reactions = response.parsed_body["chat_messages"].last["reactions"] expect(reactions[heart_emoji]["reacted"]).to be true expect(reactions[smile_emoji]["reacted"]).to be false end it "sends the last message bus id for the channel" do get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["meta"]["channel_message_bus_last_id"]).not_to eq(nil) end describe "scrolling to the past" do it "returns the correct messages in created_at, id order" do get "/chat/#{chat_channel.id}/messages.json", params: { message_id: message_40.id, direction: described_class::PAST, page_size: page_size, } messages = response.parsed_body["chat_messages"] expect(messages.count).to eq(page_size) expect(messages.first["created_at"].to_time).to eq_time(message_10.created_at) expect(messages.last["created_at"].to_time).to eq_time(message_39.created_at) end it "returns 'can_load...' properly when there are more past messages" do get "/chat/#{chat_channel.id}/messages.json", params: { message_id: message_40.id, direction: described_class::PAST, page_size: page_size, } expect(response.parsed_body["meta"]["can_load_more_past"]).to be true expect(response.parsed_body["meta"]["can_load_more_future"]).to be_nil end it "returns 'can_load...' properly when there are no past messages" do get "/chat/#{chat_channel.id}/messages.json", params: { message_id: message_3.id, direction: described_class::PAST, page_size: page_size, } expect(response.parsed_body["meta"]["can_load_more_past"]).to be false expect(response.parsed_body["meta"]["can_load_more_future"]).to be_nil end end describe "scrolling to the future" do it "returns the correct messages in created_at, id order when there are many after" do get "/chat/#{chat_channel.id}/messages.json", params: { message_id: message_10.id, direction: described_class::FUTURE, page_size: page_size, } messages = response.parsed_body["chat_messages"] expect(messages.count).to eq(page_size) expect(messages.first["created_at"].to_time).to eq_time(message_11.created_at) expect(messages.last["created_at"].to_time).to eq_time(message_40.created_at) end it "return 'can_load..' properly when there are future messages" do get "/chat/#{chat_channel.id}/messages.json", params: { message_id: message_10.id, direction: described_class::FUTURE, page_size: page_size, } expect(response.parsed_body["meta"]["can_load_more_past"]).to be_nil expect(response.parsed_body["meta"]["can_load_more_future"]).to be true end it "returns 'can_load..' properly when there are no future messages" do get "/chat/#{chat_channel.id}/messages.json", params: { message_id: message_60.id, direction: described_class::FUTURE, page_size: page_size, } expect(response.parsed_body["meta"]["can_load_more_past"]).to be_nil expect(response.parsed_body["meta"]["can_load_more_future"]).to be false end end describe "without direction (latest messages)" do it "signals there are no future messages" do get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["meta"]["can_load_more_future"]).to eq(false) end it "signals there are more messages in the past" do get "/chat/#{chat_channel.id}/messages.json", params: { page_size: page_size } expect(response.parsed_body["meta"]["can_load_more_past"]).to eq(true) end it "signals there are no more messages" do new_channel = Fabricate(:category_channel) Fabricate(:chat_message, chat_channel: new_channel, user: other_user, message: "message") chat_messages_qty = 1 get "/chat/#{new_channel.id}/messages.json", params: { page_size: chat_messages_qty + 1 } expect(response.parsed_body["meta"]["can_load_more_past"]).to eq(false) end end end describe "#enable_chat" do context "with category as chatable" do let!(:category) { Fabricate(:category) } let(:channel) { Fabricate(:category_channel, chatable: category) } it "ensures created channel can be seen" do Guardian.any_instance.expects(:can_join_chat_channel?).with(channel) sign_in(admin) post "/chat/enable.json", params: { chatable_type: "category", chatable_id: category.id } end # TODO: rewrite specs to ensure no exception is raised it "ensures existing channel can be seen" do Guardian.any_instance.expects(:can_join_chat_channel?) sign_in(admin) post "/chat/enable.json", params: { chatable_type: "category", chatable_id: category.id } end end end describe "#disable_chat" do context "with category as chatable" do it "ensures category can be seen" do category = Fabricate(:category) channel = Fabricate(:category_channel, chatable: category) message = Fabricate(:chat_message, chat_channel: channel) Guardian.any_instance.expects(:can_join_chat_channel?).with(channel) sign_in(admin) post "/chat/disable.json", params: { chatable_type: "category", chatable_id: category.id } end end end describe "#create_message" do let(:message) { "This is a message" } describe "for category" do fab!(:chat_channel) { Fabricate(:category_channel, chatable: category) } context "when current user is silenced" do before do UserChatChannelMembership.create(user: user, chat_channel: chat_channel, following: true) sign_in(user) UserSilencer.new(user).silence end it "raises invalid acces" do post "/chat/#{chat_channel.id}.json", params: { message: message } expect(response.status).to eq(403) end end it "errors for regular user when chat is staff-only" do sign_in(user) SiteSetting.chat_allowed_groups = Group::AUTO_GROUPS[:staff] post "/chat/#{chat_channel.id}.json", params: { message: message } expect(response.status).to eq(403) end it "errors when the user isn't following the channel" do sign_in(user) post "/chat/#{chat_channel.id}.json", params: { message: message } expect(response.status).to eq(403) end it "errors when the user is not staff and the channel is not open" do Fabricate(:user_chat_channel_membership, chat_channel: chat_channel, user: user) sign_in(user) chat_channel.update(status: :closed) post "/chat/#{chat_channel.id}.json", params: { message: message } expect(response.status).to eq(422) expect(response.parsed_body["errors"]).to include( I18n.t("chat.errors.channel_new_message_disallowed", status: chat_channel.status_name), ) end it "errors when the user is staff and the channel is not open or closed" do Fabricate(:user_chat_channel_membership, chat_channel: chat_channel, user: admin) sign_in(admin) chat_channel.update(status: :closed) post "/chat/#{chat_channel.id}.json", params: { message: message } expect(response.status).to eq(200) chat_channel.update(status: :read_only) post "/chat/#{chat_channel.id}.json", params: { message: message } expect(response.status).to eq(422) expect(response.parsed_body["errors"]).to include( I18n.t("chat.errors.channel_new_message_disallowed", status: chat_channel.status_name), ) end it "sends a message for regular user when staff-only is disabled and they are following channel" do sign_in(user) UserChatChannelMembership.create(user: user, chat_channel: chat_channel, following: true) expect { post "/chat/#{chat_channel.id}.json", params: { message: message } }.to change { ChatMessage.count }.by(1) expect(response.status).to eq(200) expect(ChatMessage.last.message).to eq(message) end end describe "for direct message" do fab!(:user1) { Fabricate(:user) } fab!(:user2) { Fabricate(:user) } fab!(:chatable) { Fabricate(:direct_message, users: [user1, user2]) } fab!(:direct_message_channel) { Fabricate(:direct_message_channel, chatable: chatable) } def create_memberships UserChatChannelMembership.create!( user: user1, chat_channel: direct_message_channel, following: true, desktop_notification_level: UserChatChannelMembership::NOTIFICATION_LEVELS[:always], mobile_notification_level: UserChatChannelMembership::NOTIFICATION_LEVELS[:always], ) UserChatChannelMembership.create!( user: user2, chat_channel: direct_message_channel, following: false, desktop_notification_level: UserChatChannelMembership::NOTIFICATION_LEVELS[:always], mobile_notification_level: UserChatChannelMembership::NOTIFICATION_LEVELS[:always], ) Group.refresh_automatic_groups! end it "forces users to follow the channel" do create_memberships expect(UserChatChannelMembership.find_by(user_id: user2.id).following).to be false ChatPublisher.expects(:publish_new_channel).once sign_in(user1) post "/chat/#{direct_message_channel.id}.json", params: { message: message } expect(UserChatChannelMembership.find_by(user_id: user2.id).following).to be true end it "errors when the user is not part of the direct message channel" do create_memberships DirectMessageUser.find_by(user: user1, direct_message: chatable).destroy! sign_in(user1) post "/chat/#{direct_message_channel.id}.json", params: { message: message } expect(response.status).to eq(403) UserChatChannelMembership.find_by(user_id: user2.id).update!(following: true) sign_in(user2) post "/chat/#{direct_message_channel.id}.json", params: { message: message } expect(response.status).to eq(200) end context "when current user is silenced" do before do create_memberships sign_in(user1) UserSilencer.new(user1).silence end it "raises invalid acces" do post "/chat/#{direct_message_channel.id}.json", params: { message: message } expect(response.status).to eq(403) end end context "if any of the direct message users is ignoring the acting user" do before do IgnoredUser.create!(user: user2, ignored_user: user1, expiring_at: 1.day.from_now) end it "does not force them to follow the channel or send a publish_new_channel message" do create_memberships expect(UserChatChannelMembership.find_by(user_id: user2.id).following).to be false ChatPublisher.expects(:publish_new_channel).never sign_in(user1) post "/chat/#{direct_message_channel.id}.json", params: { message: message } expect(UserChatChannelMembership.find_by(user_id: user2.id).following).to be false end end end end describe "#rebake" do fab!(:chat_message) { Fabricate(:chat_message, chat_channel: chat_channel, user: user) } context "as staff" do it "rebakes the post" do sign_in(Fabricate(:admin)) expect_enqueued_with( job: :process_chat_message, args: { chat_message_id: chat_message.id, }, ) do put "/chat/#{chat_channel.id}/#{chat_message.id}/rebake.json" expect(response.status).to eq(200) end end it "does not interfere with core's guardian can_rebake? for posts" do sign_in(Fabricate(:admin)) put "/chat/#{chat_channel.id}/#{chat_message.id}/rebake.json" expect(response.status).to eq(200) post = Fabricate(:post) put "/posts/#{post.id}/rebake.json" expect(response.status).to eq(200) end it "does not rebake the post when channel is read_only" do chat_message.chat_channel.update!(status: :read_only) sign_in(Fabricate(:admin)) put "/chat/#{chat_channel.id}/#{chat_message.id}/rebake.json" expect(response.status).to eq(403) end context "when cooked has changed" do it "marks the message as dirty" do sign_in(Fabricate(:admin)) chat_message.update!(message: "new content") expect_enqueued_with( job: :process_chat_message, args: { chat_message_id: chat_message.id, is_dirty: true, }, ) do put "/chat/#{chat_channel.id}/#{chat_message.id}/rebake.json" expect(response.status).to eq(200) end end end end context "when not staff" do it "forbids non staff to rebake" do sign_in(Fabricate(:user)) put "/chat/#{chat_channel.id}/#{chat_message.id}/rebake.json" expect(response.status).to eq(403) end context "as TL3 user" do it "forbids less then TL4 user tries to rebake" do sign_in(Fabricate(:user, trust_level: TrustLevel[3])) put "/chat/#{chat_channel.id}/#{chat_message.id}/rebake.json" expect(response.status).to eq(403) end end context "as TL4 user" do it "allows TL4 users to rebake" do sign_in(Fabricate(:user, trust_level: TrustLevel[4])) put "/chat/#{chat_channel.id}/#{chat_message.id}/rebake.json" expect(response.status).to eq(200) end it "does not rebake the post when channel is read_only" do chat_message.chat_channel.update!(status: :read_only) sign_in(Fabricate(:user, trust_level: TrustLevel[4])) put "/chat/#{chat_channel.id}/#{chat_message.id}/rebake.json" expect(response.status).to eq(403) end end end end describe "#edit_message" do fab!(:chat_message) { Fabricate(:chat_message, chat_channel: chat_channel, user: user) } context "when current user is silenced" do before do UserSilencer.new(user).silence sign_in(user) end it "raises an invalid request" do put "/chat/#{chat_channel.id}/edit/#{chat_message.id}.json", params: { new_message: "Hi" } expect(response.status).to eq(422) end end it "errors when a user tries to edit another user's message" do sign_in(Fabricate(:user)) put "/chat/#{chat_channel.id}/edit/#{chat_message.id}.json", params: { new_message: "edit!" } expect(response.status).to eq(422) end it "errors when staff tries to edit another user's message" do sign_in(admin) new_message = "Vrroooom cars go fast" put "/chat/#{chat_channel.id}/edit/#{chat_message.id}.json", params: { new_message: new_message, } expect(response.status).to eq(422) end it "allows a user to edit their own messages" do sign_in(user) new_message = "Wow markvanlan must be a good programmer" put "/chat/#{chat_channel.id}/edit/#{chat_message.id}.json", params: { new_message: new_message, } expect(response.status).to eq(200) expect(chat_message.reload.message).to eq(new_message) end end RSpec.shared_examples "chat_message_deletion" do it "doesn't allow a user to delete another user's message" do sign_in(other_user) delete "/chat/#{chat_channel.id}/#{ChatMessage.last.id}.json" expect(response.status).to eq(403) end it "doesn't allow a silenced user to delete their message" do sign_in(other_user) UserSilencer.new(other_user).silence delete "/chat/#{other_user_message.chat_channel.id}/#{other_user_message.id}.json" expect(response.status).to eq(403) end it "Allows admin to delete others' messages" do sign_in(admin) expect { delete "/chat/#{chat_channel.id}/#{ChatMessage.last.id}.json" }.to change { ChatMessage.count }.by(-1) expect(response.status).to eq(200) end it "does not allow message delete when chat channel is read_only" do sign_in(ChatMessage.last.user) chat_channel.update!(status: :read_only) expect { delete "/chat/#{chat_channel.id}/#{ChatMessage.last.id}.json" }.not_to change { ChatMessage.count } expect(response.status).to eq(403) sign_in(admin) delete "/chat/#{chat_channel.id}/#{ChatMessage.last.id}.json" expect(response.status).to eq(403) end it "only allows admin to delete when chat channel is closed" do sign_in(admin) chat_channel.update!(status: :read_only) expect { delete "/chat/#{chat_channel.id}/#{ChatMessage.last.id}.json" }.not_to change { ChatMessage.count } expect(response.status).to eq(403) chat_channel.update!(status: :closed) expect { delete "/chat/#{chat_channel.id}/#{ChatMessage.last.id}.json" }.to change { ChatMessage.count }.by(-1) expect(response.status).to eq(200) end end describe "#delete" do fab!(:second_user) { Fabricate(:user) } fab!(:second_user_message) do Fabricate(:chat_message, user: second_user, chat_channel: chat_channel) end before do ChatMessage.create(user: user, message: "this is a message", chat_channel: chat_channel) end describe "for category" do fab!(:chat_channel) { Fabricate(:category_channel, chatable: category) } it_behaves_like "chat_message_deletion" do let(:other_user) { second_user } let(:other_user_message) { second_user_message } end it "Allows users to delete their own messages" do sign_in(user) expect { delete "/chat/#{chat_channel.id}/#{ChatMessage.last.id}.json" }.to change { ChatMessage.count }.by(-1) expect(response.status).to eq(200) end end end RSpec.shared_examples "chat_message_restoration" do it "doesn't allow a user to restore another user's message" do sign_in(other_user) put "/chat/#{chat_channel.id}/restore/#{ChatMessage.unscoped.last.id}.json" expect(response.status).to eq(403) end it "allows a user to restore their own posts" do sign_in(user) deleted_message = ChatMessage.unscoped.last put "/chat/#{chat_channel.id}/restore/#{deleted_message.id}.json" expect(response.status).to eq(200) expect(deleted_message.reload.deleted_at).to be_nil end it "allows admin to restore others' posts" do sign_in(admin) deleted_message = ChatMessage.unscoped.last put "/chat/#{chat_channel.id}/restore/#{deleted_message.id}.json" expect(response.status).to eq(200) expect(deleted_message.reload.deleted_at).to be_nil end it "does not allow message restore when chat channel is read_only" do sign_in(ChatMessage.last.user) chat_channel.update!(status: :read_only) deleted_message = ChatMessage.unscoped.last put "/chat/#{chat_channel.id}/restore/#{deleted_message.id}.json" expect(response.status).to eq(403) expect(deleted_message.reload.deleted_at).not_to be_nil sign_in(admin) put "/chat/#{chat_channel.id}/restore/#{deleted_message.id}.json" expect(response.status).to eq(403) end it "only allows admin to restore when chat channel is closed" do sign_in(admin) chat_channel.update!(status: :read_only) deleted_message = ChatMessage.unscoped.last put "/chat/#{chat_channel.id}/restore/#{deleted_message.id}.json" expect(response.status).to eq(403) expect(deleted_message.reload.deleted_at).not_to be_nil chat_channel.update!(status: :closed) put "/chat/#{chat_channel.id}/restore/#{deleted_message.id}.json" expect(response.status).to eq(200) expect(deleted_message.reload.deleted_at).to be_nil end end describe "#restore" do fab!(:second_user) { Fabricate(:user) } before do message = ChatMessage.create(user: user, message: "this is a message", chat_channel: chat_channel) message.trash! end describe "for category" do fab!(:chat_channel) { Fabricate(:category_channel, chatable: category) } it_behaves_like "chat_message_restoration" do let(:other_user) { second_user } end end end describe "#update_user_last_read" do before { sign_in(user) } fab!(:message_1) { Fabricate(:chat_message, chat_channel: chat_channel, user: other_user) } fab!(:message_2) { Fabricate(:chat_message, chat_channel: chat_channel, user: other_user) } it "returns a 404 when the user is not a channel member" do put "/chat/#{chat_channel.id}/read/#{message_1.id}.json" expect(response.status).to eq(404) end it "returns a 404 when the user is not following the channel" do Fabricate( :user_chat_channel_membership, chat_channel: chat_channel, user: user, following: false, ) put "/chat/#{chat_channel.id}/read/#{message_1.id}.json" expect(response.status).to eq(404) end describe "when the user is a channel member" do fab!(:membership) do Fabricate(:user_chat_channel_membership, chat_channel: chat_channel, user: user) end context "when message_id param doesn't link to a message of the channel" do it "raises a not found" do put "/chat/#{chat_channel.id}/read/-999.json" expect(response.status).to eq(404) end end context "when message_id param is inferior to existing last read" do before { membership.update!(last_read_message_id: message_2.id) } it "raises an invalid request" do put "/chat/#{chat_channel.id}/read/#{message_1.id}.json" expect(response.status).to eq(400) expect(response.parsed_body["errors"][0]).to match(/message_id/) end end context "when message_id refers to deleted message" do before { message_1.trash!(Discourse.system_user) } it "works" do put "/chat/#{chat_channel.id}/read/#{message_1.id}.json" expect(response.status).to eq(200) end end it "updates timing records" do expect { put "/chat/#{chat_channel.id}/read/#{message_1.id}.json" }.not_to change { UserChatChannelMembership.count } membership.reload expect(membership.chat_channel_id).to eq(chat_channel.id) expect(membership.last_read_message_id).to eq(message_1.id) expect(membership.user_id).to eq(user.id) end def create_notification_and_mention_for(user, sender, msg) Notification .create!( notification_type: Notification.types[:chat_mention], user: user, high_priority: true, read: false, data: { message: "chat.mention_notification", chat_message_id: msg.id, chat_channel_id: msg.chat_channel_id, chat_channel_title: msg.chat_channel.title(user), chat_channel_slug: msg.chat_channel.slug, mentioned_by_username: sender.username, }.to_json, ) .tap do |notification| ChatMention.create!(user: user, chat_message: msg, notification: notification) end end it "marks all mention notifications as read for the channel" do notification = create_notification_and_mention_for(user, other_user, message_1) put "/chat/#{chat_channel.id}/read/#{message_2.id}.json" expect(response.status).to eq(200) expect(notification.reload.read).to eq(true) end it "doesn't mark notifications of messages that weren't read yet" do message_3 = Fabricate(:chat_message, chat_channel: chat_channel, user: other_user) notification = create_notification_and_mention_for(user, other_user, message_3) put "/chat/#{chat_channel.id}/read/#{message_2.id}.json" expect(response.status).to eq(200) expect(notification.reload.read).to eq(false) end end end describe "react" do fab!(:chat_channel) { Fabricate(:category_channel) } fab!(:chat_message) { Fabricate(:chat_message, chat_channel: chat_channel, user: user) } fab!(:user_membership) do Fabricate(:user_chat_channel_membership, chat_channel: chat_channel, user: user) end fab!(:private_chat_channel) do Fabricate(:category_channel, chatable: Fabricate(:private_category, group: Fabricate(:group))) end fab!(:private_chat_message) do Fabricate(:chat_message, chat_channel: private_chat_channel, user: admin) end fab!(:private_user_membership) do Fabricate(:user_chat_channel_membership, chat_channel: private_chat_channel, user: user) end fab!(:chat_channel_no_memberships) { Fabricate(:category_channel) } fab!(:chat_message_no_memberships) do Fabricate(:chat_message, chat_channel: chat_channel_no_memberships, user: user) end it "errors with invalid emoji" do sign_in(user) put "/chat/#{chat_channel.id}/react/#{chat_message.id}.json", params: { emoji: 12, react_action: "add", } expect(response.status).to eq(400) end it "errors with invalid action" do sign_in(user) put "/chat/#{chat_channel.id}/react/#{chat_message.id}.json", params: { emoji: ":heart:", react_action: "sdf", } expect(response.status).to eq(400) end it "creates a membership when reacting to channel without a membership record" do sign_in(user) expect { put "/chat/#{chat_channel_no_memberships.id}/react/#{chat_message_no_memberships.id}.json", params: { emoji: ":heart:", react_action: "add", } }.to change { UserChatChannelMembership.count }.by(1) expect(response.status).to eq(200) end it "errors when user tries to react to private channel they can't access" do sign_in(user) put "/chat/#{private_chat_channel.id}/react/#{private_chat_message.id}.json", params: { emoji: ":heart:", react_action: "add", } expect(response.status).to eq(403) end it "errors when the user tries to react to a read_only channel" do chat_channel.update(status: :read_only) sign_in(user) emoji = ":heart:" expect { put "/chat/#{chat_channel.id}/react/#{chat_message.id}.json", params: { emoji: emoji, react_action: "add", } }.not_to change { chat_message.reactions.where(user: user, emoji: emoji).count } expect(response.status).to eq(403) expect(response.parsed_body["errors"]).to include( I18n.t("chat.errors.channel_modify_message_disallowed", status: chat_channel.status_name), ) end it "errors when user is silenced" do UserSilencer.new(user).silence sign_in(user) put "/chat/#{chat_channel.id}/react/#{chat_message.id}.json", params: { emoji: ":heart:", react_action: "add", } expect(response.status).to eq(403) end it "errors when max unique reactions limit is reached" do Emoji .all .map(&:name) .take(29) .each { |emoji| chat_message.reactions.create(user: user, emoji: emoji) } sign_in(user) put "/chat/#{chat_channel.id}/react/#{chat_message.id}.json", params: { emoji: ":wink:", react_action: "add", } expect(response.status).to eq(200) put "/chat/#{chat_channel.id}/react/#{chat_message.id}.json", params: { emoji: ":wave:", react_action: "add", } expect(response.status).to eq(403) expect(response.parsed_body["errors"]).to include( I18n.t("chat.errors.max_reactions_limit_reached"), ) end it "does not error on new duplicate reactions" do another_user = Fabricate(:user) Emoji .all .map(&:name) .take(29) .each { |emoji| chat_message.reactions.create(user: another_user, emoji: emoji) } emoji = ":wink:" chat_message.reactions.create(user: another_user, emoji: emoji) sign_in(user) put "/chat/#{chat_channel.id}/react/#{chat_message.id}.json", params: { emoji: emoji, react_action: "add", } expect(response.status).to eq(200) end it "adds a reaction record correctly" do sign_in(user) emoji = ":heart:" expect { put "/chat/#{chat_channel.id}/react/#{chat_message.id}.json", params: { emoji: emoji, react_action: "add", } }.to change { chat_message.reactions.where(user: user, emoji: emoji).count }.by(1) expect(response.status).to eq(200) end it "removes a reaction record correctly" do sign_in(user) emoji = ":heart:" chat_message.reactions.create(user: user, emoji: emoji) expect { put "/chat/#{chat_channel.id}/react/#{chat_message.id}.json", params: { emoji: emoji, react_action: "remove", } }.to change { chat_message.reactions.where(user: user, emoji: emoji).count }.by(-1) expect(response.status).to eq(200) end end describe "invite_users" do fab!(:chat_channel) { Fabricate(:category_channel) } fab!(:chat_message) { Fabricate(:chat_message, chat_channel: chat_channel, user: admin) } fab!(:user2) { Fabricate(:user) } before do sign_in(admin) [user, user2].each { |u| u.user_option.update(chat_enabled: true) } end it "doesn't invite users who cannot chat" do SiteSetting.chat_allowed_groups = Group::AUTO_GROUPS[:admin] expect { put "/chat/#{chat_channel.id}/invite.json", params: { user_ids: [user.id] } }.not_to change { user.notifications.where(notification_type: Notification.types[:chat_invitation]).count } end it "creates an invitation notification for users who can chat" do expect { put "/chat/#{chat_channel.id}/invite.json", params: { user_ids: [user.id] } }.to change { user.notifications.where(notification_type: Notification.types[:chat_invitation]).count }.by(1) notification = user.notifications.where(notification_type: Notification.types[:chat_invitation]).last parsed_data = JSON.parse(notification[:data]) expect(parsed_data["chat_channel_title"]).to eq(chat_channel.title(user)) expect(parsed_data["chat_channel_slug"]).to eq(chat_channel.slug) end it "creates multiple invitations" do expect { put "/chat/#{chat_channel.id}/invite.json", params: { user_ids: [user.id, user2.id] } }.to change { Notification.where( notification_type: Notification.types[:chat_invitation], user_id: [user.id, user2.id], ).count }.by(2) end it "adds chat_message_id when param is present" do put "/chat/#{chat_channel.id}/invite.json", params: { user_ids: [user.id], chat_message_id: chat_message.id, } expect(JSON.parse(Notification.last.data)["chat_message_id"]).to eq(chat_message.id.to_s) end end describe "#dismiss_retention_reminder" do it "errors for anon" do post "/chat/dismiss-retention-reminder.json", params: { chatable_type: "Category" } expect(response.status).to eq(403) end it "errors when chatable_type isn't present" do sign_in(user) post "/chat/dismiss-retention-reminder.json", params: {} expect(response.status).to eq(400) end it "errors when chatable_type isn't a valid option" do sign_in(user) post "/chat/dismiss-retention-reminder.json", params: { chatable_type: "hi" } expect(response.status).to eq(400) end it "sets `dismissed_channel_retention_reminder` to true" do sign_in(user) expect { post "/chat/dismiss-retention-reminder.json", params: { chatable_type: "Category" } }.to change { user.user_option.reload.dismissed_channel_retention_reminder }.to (true) end it "sets `dismissed_dm_retention_reminder` to true" do sign_in(user) expect { post "/chat/dismiss-retention-reminder.json", params: { chatable_type: "DirectMessage" } }.to change { user.user_option.reload.dismissed_dm_retention_reminder }.to (true) end it "doesn't error if the fields are already true" do sign_in(user) user.user_option.update( dismissed_channel_retention_reminder: true, dismissed_dm_retention_reminder: true, ) post "/chat/dismiss-retention-reminder.json", params: { chatable_type: "Category" } expect(response.status).to eq(200) post "/chat/dismiss-retention-reminder.json", params: { chatable_type: "DirectMessage" } expect(response.status).to eq(200) end end describe "#quote_messages" do fab!(:channel) { Fabricate(:category_channel, chatable: category, name: "Cool Chat") } let(:user2) { Fabricate(:user) } let(:message1) do Fabricate( :chat_message, user: user, chat_channel: channel, message: "an extremely insightful response :)", ) end let(:message2) do Fabricate(:chat_message, user: user2, chat_channel: channel, message: "says you!") end let(:message3) { Fabricate(:chat_message, user: user, chat_channel: channel, message: "aw :(") } it "returns a 403 if the user can't chat" do SiteSetting.chat_allowed_groups = nil sign_in(user) post "/chat/#{channel.id}/quote.json", params: { message_ids: [message1.id, message2.id, message3.id], } expect(response.status).to eq(403) end it "returns a 403 if the user can't see the channel" do category.update!(read_restricted: true) group = Fabricate(:group) CategoryGroup.create(group: group, category: category, permission_type: CategoryGroup.permission_types[:create_post]) sign_in(user) post "/chat/#{channel.id}/quote.json", params: { message_ids: [message1.id, message2.id, message3.id], } expect(response.status).to eq(403) end it "returns a 404 for a not found channel" do channel.destroy sign_in(user) post "/chat/#{channel.id}/quote.json", params: { message_ids: [message1.id, message2.id, message3.id], } expect(response.status).to eq(404) end it "quotes the message ids provided" do sign_in(user) post "/chat/#{channel.id}/quote.json", params: { message_ids: [message1.id, message2.id, message3.id], } expect(response.status).to eq(200) markdown = response.parsed_body["markdown"] expect(markdown).to eq(<<~EXPECTED) [chat quote="#{user.username};#{message1.id};#{message1.created_at.iso8601}" channel="Cool Chat" channelId="#{channel.id}" multiQuote="true" chained="true"] an extremely insightful response :) [/chat] [chat quote="#{user2.username};#{message2.id};#{message2.created_at.iso8601}" chained="true"] says you! [/chat] [chat quote="#{user.username};#{message3.id};#{message3.created_at.iso8601}" chained="true"] aw :( [/chat] EXPECTED end end describe "#flag" do fab!(:admin_chat_message) { Fabricate(:chat_message, user: admin, chat_channel: chat_channel) } fab!(:user_chat_message) { Fabricate(:chat_message, user: user, chat_channel: chat_channel) } fab!(:admin_dm_message) { Fabricate(:chat_message, user: admin, chat_channel: dm_chat_channel) } before do sign_in(user) Group.refresh_automatic_groups! end it "creates reviewable" do expect { put "/chat/flag.json", params: { chat_message_id: admin_chat_message.id, flag_type_id: ReviewableScore.types[:off_topic], } }.to change { ReviewableChatMessage.where(target: admin_chat_message).count }.by(1) expect(response.status).to eq(200) end it "errors for silenced users" do UserSilencer.new(user).silence put "/chat/flag.json", params: { chat_message_id: admin_chat_message.id, flag_type_id: ReviewableScore.types[:off_topic], } expect(response.status).to eq(403) end it "doesn't allow flagging your own message" do put "/chat/flag.json", params: { chat_message_id: user_chat_message.id, flag_type_id: ReviewableScore.types[:off_topic], } expect(response.status).to eq(403) end it "doesn't allow flagging messages in a read_only channel" do user_chat_message.chat_channel.update(status: :read_only) put "/chat/flag.json", params: { chat_message_id: admin_chat_message.id, flag_type_id: ReviewableScore.types[:off_topic], } expect(response.status).to eq(403) end it "doesn't allow flagging staff if SiteSetting.allow_flagging_staff is false" do SiteSetting.allow_flagging_staff = false put "/chat/flag.json", params: { chat_message_id: admin_chat_message.id, flag_type_id: ReviewableScore.types[:off_topic], } expect(response.status).to eq(403) end it "returns a 429 when the user attempts to flag more than 4 messages in 1 minute" do RateLimiter.enable [message_1, message_2, message_3, message_4].each do |message| put "/chat/flag.json", params: { chat_message_id: message.id, flag_type_id: ReviewableScore.types[:off_topic], } expect(response.status).to eq(200) end put "/chat/flag.json", params: { chat_message_id: message_5.id, flag_type_id: ReviewableScore.types[:off_topic], } expect(response.status).to eq(429) end end describe "#set_draft" do fab!(:chat_channel) { Fabricate(:category_channel) } let(:dm_channel) { Fabricate(:direct_message_channel) } before { sign_in(user) } it "can create and destroy chat drafts" do expect { post "/chat/drafts.json", params: { chat_channel_id: chat_channel.id, data: "{}" } }.to change { ChatDraft.count }.by(1) expect { post "/chat/drafts.json", params: { chat_channel_id: chat_channel.id } }.to change { ChatDraft.count }.by(-1) end it "cannot create chat drafts for a category channel the user cannot access" do group = Fabricate(:group) private_category = Fabricate(:private_category, group: group) chat_channel.update!(chatable: private_category) post "/chat/drafts.json", params: { chat_channel_id: chat_channel.id, data: "{}" } expect(response.status).to eq(403) GroupUser.create!(user: user, group: group) expect { post "/chat/drafts.json", params: { chat_channel_id: chat_channel.id, data: "{}" } }.to change { ChatDraft.count }.by(1) end it "cannot create chat drafts for a direct message channel the user cannot access" do post "/chat/drafts.json", params: { chat_channel_id: dm_channel.id, data: "{}" } expect(response.status).to eq(403) DirectMessageUser.create(user: user, direct_message: dm_channel.chatable) expect { post "/chat/drafts.json", params: { chat_channel_id: dm_channel.id, data: "{}" } }.to change { ChatDraft.count }.by(1) end end describe "#message_link" do it "ensures message's channel can be seen" do channel = Fabricate(:category_channel, chatable: Fabricate(:category)) message = Fabricate(:chat_message, chat_channel: channel) Guardian.any_instance.expects(:can_join_chat_channel?).with(channel) sign_in(Fabricate(:user)) get "/chat/message/#{message.id}.json" end end describe "#lookup_message" do let!(:message) { Fabricate(:chat_message, chat_channel: channel) } let(:channel) { Fabricate(:direct_message_channel) } let(:chatable) { channel.chatable } fab!(:user) { Fabricate(:user) } before { sign_in(user) } it "ensures message's channel can be seen" do Guardian.any_instance.expects(:can_join_chat_channel?).with(channel) get "/chat/lookup/#{message.id}.json", { params: { chat_channel_id: channel.id } } end context "when the message doesn’t belong to the channel" do let!(:message) { Fabricate(:chat_message) } it "returns a 404" do get "/chat/lookup/#{message.id}.json", { params: { chat_channel_id: channel.id } } expect(response.status).to eq(404) end end context "when the chat channel is for a category" do let(:channel) { Fabricate(:category_channel) } it "ensures the user can access that category" do get "/chat/lookup/#{message.id}.json", { params: { chat_channel_id: channel.id } } expect(response.status).to eq(200) expect(response.parsed_body["chat_messages"][0]["id"]).to eq(message.id) group = Fabricate(:group) chatable.update!(read_restricted: true) Fabricate(:category_group, group: group, category: chatable) get "/chat/lookup/#{message.id}.json", { params: { chat_channel_id: channel.id } } expect(response.status).to eq(403) GroupUser.create!(user: user, group: group) get "/chat/lookup/#{message.id}.json", { params: { chat_channel_id: channel.id } } expect(response.status).to eq(200) expect(response.parsed_body["chat_messages"][0]["id"]).to eq(message.id) end end context "when the chat channel is for a direct message channel" do let(:channel) { Fabricate(:direct_message_channel) } it "ensures the user can access that direct message channel" do get "/chat/lookup/#{message.id}.json", { params: { chat_channel_id: channel.id } } expect(response.status).to eq(403) DirectMessageUser.create!(user: user, direct_message: chatable) get "/chat/lookup/#{message.id}.json", { params: { chat_channel_id: channel.id } } expect(response.status).to eq(200) expect(response.parsed_body["chat_messages"][0]["id"]).to eq(message.id) end end end describe "#move_messages_to_channel" do fab!(:message_to_move1) do Fabricate( :chat_message, chat_channel: chat_channel, message: "some cool message", created_at: 2.minutes.ago, ) end fab!(:message_to_move2) do Fabricate( :chat_message, chat_channel: chat_channel, message: "and another thing", created_at: 1.minute.ago, ) end fab!(:destination_channel) { Fabricate(:category_channel) } let(:message_ids) { [message_to_move1.id, message_to_move2.id] } let(:invalid_destination_channel) do Fabricate(:direct_message_channel, users: [admin, Fabricate(:user)]) end context "when the user is not admin" do it "returns an access denied error" do sign_in(user) put "/chat/#{chat_channel.id}/move_messages_to_channel.json", params: { destination_channel_id: destination_channel.id, message_ids: message_ids, } expect(response.status).to eq(403) end end context "when the user is admin" do before { sign_in(admin) } it "shows an error if the source channel is not found" do chat_channel.trash! put "/chat/#{chat_channel.id}/move_messages_to_channel.json", params: { destination_channel_id: destination_channel.id, message_ids: message_ids, } expect(response.status).to eq(404) end it "shows an error if the destination channel is not found" do destination_channel.trash! put "/chat/#{chat_channel.id}/move_messages_to_channel.json", params: { destination_channel_id: destination_channel.id, message_ids: message_ids, } expect(response.status).to eq(404) end it "successfully moves the messages to the new channel" do put "/chat/#{chat_channel.id}/move_messages_to_channel.json", params: { destination_channel_id: destination_channel.id, message_ids: message_ids, } expect(response.status).to eq(200) latest_destination_messages = destination_channel.chat_messages.last(2) expect(latest_destination_messages.first.message).to eq("some cool message") expect(latest_destination_messages.second.message).to eq("and another thing") expect(message_to_move1.reload.deleted_at).not_to eq(nil) expect(message_to_move2.reload.deleted_at).not_to eq(nil) end it "shows an error message when the destination channel is invalid" do put "/chat/#{chat_channel.id}/move_messages_to_channel.json", params: { destination_channel_id: invalid_destination_channel.id, message_ids: message_ids, } expect(response.status).to eq(422) expect(response.parsed_body["errors"]).to include( I18n.t("chat.errors.message_move_invalid_channel"), ) end it "shows an error when none of the messages can be found" do destroyed_message = Fabricate(:chat_message, chat_channel: chat_channel) destroyed_message.trash! put "/chat/#{chat_channel.id}/move_messages_to_channel.json", params: { destination_channel_id: destination_channel.id, message_ids: [destroyed_message], } expect(response.status).to eq(422) expect(response.parsed_body["errors"]).to include( I18n.t("chat.errors.message_move_no_messages_found"), ) end end end end