2022-11-02 21:41:30 +08:00
|
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
|
|
describe Chat do
|
|
|
|
|
before do
|
|
|
|
|
SiteSetting.clean_up_uploads = true
|
|
|
|
|
SiteSetting.clean_orphan_uploads_grace_period_hours = 1
|
|
|
|
|
Jobs::CleanUpUploads.new.reset_last_cleanup!
|
|
|
|
|
SiteSetting.chat_enabled = true
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "register_upload_unused" do
|
|
|
|
|
fab!(:chat_channel) { Fabricate(:chat_channel, chatable: Fabricate(:category)) }
|
2023-11-10 06:47:59 +08:00
|
|
|
|
fab!(:user)
|
2022-11-02 21:41:30 +08:00
|
|
|
|
fab!(:upload) { Fabricate(:upload, user: user, created_at: 1.month.ago) }
|
|
|
|
|
fab!(:unused_upload) { Fabricate(:upload, user: user, created_at: 1.month.ago) }
|
|
|
|
|
|
|
|
|
|
let!(:chat_message) do
|
2023-08-24 21:22:51 +08:00
|
|
|
|
Fabricate(
|
|
|
|
|
:chat_message,
|
2022-11-02 21:41:30 +08:00
|
|
|
|
chat_channel: chat_channel,
|
|
|
|
|
user: user,
|
2023-08-24 21:22:51 +08:00
|
|
|
|
message: "Hello world!",
|
|
|
|
|
uploads: [upload],
|
2022-11-02 21:41:30 +08:00
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
|
2023-01-24 11:28:21 +08:00
|
|
|
|
it "marks uploads with reference to ChatMessage via UploadReference in use" do
|
2022-11-02 21:41:30 +08:00
|
|
|
|
unused_upload
|
|
|
|
|
|
|
|
|
|
expect { Jobs::CleanUpUploads.new.execute({}) }.to change { Upload.count }.by(-1)
|
|
|
|
|
expect(Upload.exists?(id: upload.id)).to eq(true)
|
|
|
|
|
expect(Upload.exists?(id: unused_upload.id)).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "register_upload_in_use" do
|
|
|
|
|
fab!(:chat_channel) { Fabricate(:chat_channel, chatable: Fabricate(:category)) }
|
2023-11-10 06:47:59 +08:00
|
|
|
|
fab!(:user)
|
2022-11-02 21:41:30 +08:00
|
|
|
|
fab!(:message_upload) { Fabricate(:upload, user: user, created_at: 1.month.ago) }
|
|
|
|
|
fab!(:draft_upload) { Fabricate(:upload, user: user, created_at: 1.month.ago) }
|
|
|
|
|
fab!(:unused_upload) { Fabricate(:upload, user: user, created_at: 1.month.ago) }
|
|
|
|
|
|
|
|
|
|
let!(:chat_message) do
|
2023-08-24 21:22:51 +08:00
|
|
|
|
Fabricate(
|
|
|
|
|
:chat_message,
|
2022-11-02 21:41:30 +08:00
|
|
|
|
chat_channel: chat_channel,
|
|
|
|
|
user: user,
|
2023-08-24 21:22:51 +08:00
|
|
|
|
message: "Hello world! #{message_upload.sha1}",
|
2022-11-02 21:41:30 +08:00
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
let!(:draft_message) do
|
2023-03-17 21:24:38 +08:00
|
|
|
|
Chat::Draft.create!(
|
2022-11-02 21:41:30 +08:00
|
|
|
|
user: user,
|
|
|
|
|
chat_channel: chat_channel,
|
|
|
|
|
data:
|
|
|
|
|
"{\"value\":\"hello world \",\"uploads\":[\"#{draft_upload.sha1}\"],\"replyToMsg\":null}",
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
|
2023-01-24 11:28:21 +08:00
|
|
|
|
it "marks uploads with reference to ChatMessage via UploadReference in use" do
|
2022-11-02 21:41:30 +08:00
|
|
|
|
draft_upload
|
|
|
|
|
unused_upload
|
|
|
|
|
|
|
|
|
|
expect { Jobs::CleanUpUploads.new.execute({}) }.to change { Upload.count }.by(-1)
|
|
|
|
|
expect(Upload.exists?(id: message_upload.id)).to eq(true)
|
|
|
|
|
expect(Upload.exists?(id: draft_upload.id)).to eq(true)
|
|
|
|
|
expect(Upload.exists?(id: unused_upload.id)).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "user card serializer extension #can_chat_user" do
|
|
|
|
|
fab!(:target_user) { Fabricate(:user) }
|
|
|
|
|
let!(:user) { Fabricate(:user) }
|
|
|
|
|
let!(:guardian) { Guardian.new(user) }
|
|
|
|
|
let(:serializer) { UserCardSerializer.new(target_user, scope: guardian) }
|
2023-11-10 06:47:59 +08:00
|
|
|
|
fab!(:group)
|
2022-11-02 21:41:30 +08:00
|
|
|
|
|
|
|
|
|
context "when chat enabled" do
|
|
|
|
|
before { SiteSetting.chat_enabled = true }
|
|
|
|
|
|
|
|
|
|
it "returns true if the target user and the guardian user is in the Chat.allowed_group_ids" do
|
|
|
|
|
SiteSetting.chat_allowed_groups = group.id
|
2024-03-20 16:24:34 +08:00
|
|
|
|
SiteSetting.direct_message_enabled_groups = group.id
|
2022-11-02 21:41:30 +08:00
|
|
|
|
GroupUser.create(user: target_user, group: group)
|
|
|
|
|
GroupUser.create(user: user, group: group)
|
|
|
|
|
expect(serializer.can_chat_user).to eq(true)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "returns false if the target user but not the guardian user is in the Chat.allowed_group_ids" do
|
|
|
|
|
SiteSetting.chat_allowed_groups = group.id
|
|
|
|
|
GroupUser.create(user: target_user, group: group)
|
|
|
|
|
expect(serializer.can_chat_user).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "returns false if the guardian user but not the target user is in the Chat.allowed_group_ids" do
|
|
|
|
|
SiteSetting.chat_allowed_groups = group.id
|
|
|
|
|
GroupUser.create(user: user, group: group)
|
|
|
|
|
expect(serializer.can_chat_user).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
context "when guardian user is same as target user" do
|
|
|
|
|
let!(:guardian) { Guardian.new(target_user) }
|
|
|
|
|
|
|
|
|
|
it "returns false" do
|
|
|
|
|
expect(serializer.can_chat_user).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
context "when guardian user is anon" do
|
|
|
|
|
let!(:guardian) { Guardian.new }
|
|
|
|
|
|
|
|
|
|
it "returns false" do
|
|
|
|
|
expect(serializer.can_chat_user).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
end
|
2024-03-20 16:24:34 +08:00
|
|
|
|
|
|
|
|
|
context "when both users are in Chat.allowed_group_ids" do
|
|
|
|
|
before do
|
|
|
|
|
SiteSetting.chat_allowed_groups = group.id
|
|
|
|
|
SiteSetting.direct_message_enabled_groups = group.id
|
|
|
|
|
GroupUser.create(user: target_user, group: group)
|
|
|
|
|
GroupUser.create(user: user, group: group)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "returns true when both users are valid" do
|
|
|
|
|
expect(serializer.can_chat_user).to eq(true)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "returns false if current user has chat disabled" do
|
|
|
|
|
user.user_option.update!(chat_enabled: false)
|
|
|
|
|
expect(serializer.can_chat_user).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "returns false if target user has chat disabled" do
|
|
|
|
|
target_user.user_option.update!(chat_enabled: false)
|
|
|
|
|
expect(serializer.can_chat_user).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "returns false if user is not in dm allowed group" do
|
|
|
|
|
SiteSetting.direct_message_enabled_groups = 3
|
|
|
|
|
expect(serializer.can_chat_user).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
end
|
2022-11-02 21:41:30 +08:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
context "when chat not enabled" do
|
|
|
|
|
before { SiteSetting.chat_enabled = false }
|
|
|
|
|
|
|
|
|
|
it "returns false" do
|
|
|
|
|
expect(serializer.can_chat_user).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "chat oneboxes" do
|
|
|
|
|
fab!(:chat_channel) { Fabricate(:category_channel) }
|
2023-11-10 06:47:59 +08:00
|
|
|
|
fab!(:user)
|
2022-11-02 21:41:30 +08:00
|
|
|
|
|
2023-09-04 22:55:02 +08:00
|
|
|
|
fab!(:chat_message) do
|
2023-08-24 21:22:51 +08:00
|
|
|
|
Fabricate(:chat_message, chat_channel: chat_channel, user: user, message: "Hello world!")
|
2022-11-02 21:41:30 +08:00
|
|
|
|
end
|
|
|
|
|
|
2023-01-27 20:58:12 +08:00
|
|
|
|
let(:chat_url) { "#{Discourse.base_url}/chat/c/-/#{chat_channel.id}" }
|
2022-11-02 21:41:30 +08:00
|
|
|
|
|
|
|
|
|
context "when inline" do
|
|
|
|
|
it "renders channel" do
|
|
|
|
|
results = InlineOneboxer.new([chat_url], skip_cache: true).process
|
|
|
|
|
expect(results).to be_present
|
|
|
|
|
expect(results[0][:url]).to eq(chat_url)
|
|
|
|
|
expect(results[0][:title]).to eq("Chat ##{chat_channel.name}")
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "renders messages" do
|
2023-02-01 23:39:23 +08:00
|
|
|
|
results = InlineOneboxer.new(["#{chat_url}/#{chat_message.id}"], skip_cache: true).process
|
2022-11-02 21:41:30 +08:00
|
|
|
|
expect(results).to be_present
|
2023-02-01 23:39:23 +08:00
|
|
|
|
expect(results[0][:url]).to eq("#{chat_url}/#{chat_message.id}")
|
2022-11-02 21:41:30 +08:00
|
|
|
|
expect(results[0][:title]).to eq(
|
|
|
|
|
"Message ##{chat_message.id} by #{chat_message.user.username} – ##{chat_channel.name}",
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "auto-joining users to a channel" do
|
|
|
|
|
fab!(:chatters_group) { Fabricate(:group) }
|
|
|
|
|
fab!(:user) { Fabricate(:user, last_seen_at: 15.minutes.ago) }
|
|
|
|
|
let!(:channel) { Fabricate(:category_channel, auto_join_users: true, chatable: category) }
|
|
|
|
|
|
|
|
|
|
before { Jobs.run_immediately! }
|
|
|
|
|
|
|
|
|
|
def assert_user_following_state(user, channel, following:)
|
2023-03-17 21:24:38 +08:00
|
|
|
|
membership = Chat::UserChatChannelMembership.find_by(user: user, chat_channel: channel)
|
2022-11-02 21:41:30 +08:00
|
|
|
|
|
|
|
|
|
following ? (expect(membership.following).to eq(true)) : (expect(membership).to be_nil)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "when a user is added to a group with access to a channel through a category" do
|
|
|
|
|
let!(:category) { Fabricate(:private_category, group: chatters_group) }
|
|
|
|
|
|
|
|
|
|
it "joins the user to the channel if auto-join is enabled" do
|
|
|
|
|
chatters_group.add(user)
|
|
|
|
|
|
|
|
|
|
assert_user_following_state(user, channel, following: true)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "does nothing if auto-join is disabled" do
|
|
|
|
|
channel.update!(auto_join_users: false)
|
|
|
|
|
|
|
|
|
|
assert_user_following_state(user, channel, following: false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "when a user is created" do
|
2023-11-10 06:47:59 +08:00
|
|
|
|
fab!(:category)
|
2022-11-02 21:41:30 +08:00
|
|
|
|
let(:user) { Fabricate(:user, last_seen_at: nil, first_seen_at: nil) }
|
|
|
|
|
|
|
|
|
|
it "queues a job to auto-join the user the first time they log in" do
|
|
|
|
|
user.update_last_seen!
|
|
|
|
|
|
|
|
|
|
assert_user_following_state(user, channel, following: true)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "does nothing if it's not the first time we see the user" do
|
|
|
|
|
user.update!(first_seen_at: 2.minute.ago)
|
|
|
|
|
user.update_last_seen!
|
|
|
|
|
|
|
|
|
|
assert_user_following_state(user, channel, following: false)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "does nothing if auto-join is disabled" do
|
|
|
|
|
channel.update!(auto_join_users: false)
|
|
|
|
|
|
|
|
|
|
user.update_last_seen!
|
|
|
|
|
|
|
|
|
|
assert_user_following_state(user, channel, following: false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "when category permissions change" do
|
2023-11-10 06:47:59 +08:00
|
|
|
|
fab!(:category)
|
2022-11-02 21:41:30 +08:00
|
|
|
|
|
|
|
|
|
let(:chatters_group_permission) do
|
|
|
|
|
{ chatters_group.name => CategoryGroup.permission_types[:full] }
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "given permissions to a new group" do
|
|
|
|
|
it "adds the user to the channel" do
|
|
|
|
|
chatters_group.add(user)
|
|
|
|
|
|
|
|
|
|
category.update!(permissions: chatters_group_permission)
|
|
|
|
|
|
|
|
|
|
assert_user_following_state(user, channel, following: true)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "does nothing if there is no channel for the category" do
|
|
|
|
|
another_category = Fabricate(:category)
|
|
|
|
|
|
|
|
|
|
another_category.update!(permissions: chatters_group_permission)
|
|
|
|
|
|
|
|
|
|
assert_user_following_state(user, channel, following: false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2023-11-30 12:58:55 +08:00
|
|
|
|
describe "secure uploads compatibility" do
|
|
|
|
|
it "disables chat uploads if secure uploads changes from disabled to enabled" do
|
2022-11-02 21:41:30 +08:00
|
|
|
|
enable_secure_uploads
|
|
|
|
|
expect(SiteSetting.chat_allow_uploads).to eq(false)
|
|
|
|
|
last_history = UserHistory.last
|
|
|
|
|
expect(last_history.action).to eq(UserHistory.actions[:change_site_setting])
|
|
|
|
|
expect(last_history.previous_value).to eq("true")
|
|
|
|
|
expect(last_history.new_value).to eq("false")
|
|
|
|
|
expect(last_history.subject).to eq("chat_allow_uploads")
|
|
|
|
|
expect(last_history.context).to eq("Disabled because secure_uploads is enabled")
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "does not disable chat uploads if the allow_unsecure_chat_uploads global setting is set" do
|
|
|
|
|
global_setting :allow_unsecure_chat_uploads, true
|
|
|
|
|
expect { enable_secure_uploads }.not_to change { UserHistory.count }
|
|
|
|
|
expect(SiteSetting.chat_allow_uploads).to eq(true)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
describe "current_user_serializer#has_joinable_public_channels" do
|
|
|
|
|
before do
|
|
|
|
|
SiteSetting.chat_enabled = true
|
|
|
|
|
SiteSetting.chat_allowed_groups = Group::AUTO_GROUPS[:everyone]
|
|
|
|
|
end
|
|
|
|
|
|
2023-11-10 06:47:59 +08:00
|
|
|
|
fab!(:user)
|
2022-11-02 21:41:30 +08:00
|
|
|
|
let(:serializer) { CurrentUserSerializer.new(user, scope: Guardian.new(user)) }
|
|
|
|
|
|
|
|
|
|
context "when no channels exist" do
|
|
|
|
|
it "returns false" do
|
|
|
|
|
expect(serializer.has_joinable_public_channels).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
context "when no joinable channel exist" do
|
|
|
|
|
fab!(:channel) { Fabricate(:chat_channel) }
|
|
|
|
|
|
|
|
|
|
before do
|
|
|
|
|
Fabricate(:user_chat_channel_membership, user: user, chat_channel: channel, following: true)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "returns false" do
|
|
|
|
|
expect(serializer.has_joinable_public_channels).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
context "when no public channel exist" do
|
|
|
|
|
fab!(:private_category) { Fabricate(:private_category, group: Fabricate(:group)) }
|
|
|
|
|
fab!(:private_channel) { Fabricate(:chat_channel, chatable: private_category) }
|
|
|
|
|
|
|
|
|
|
it "returns false" do
|
|
|
|
|
expect(serializer.has_joinable_public_channels).to eq(false)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
context "when a joinable channel exists" do
|
|
|
|
|
fab!(:channel) { Fabricate(:chat_channel) }
|
|
|
|
|
|
|
|
|
|
it "returns true" do
|
|
|
|
|
expect(serializer.has_joinable_public_channels).to eq(true)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
2022-11-29 00:32:57 +08:00
|
|
|
|
|
|
|
|
|
describe "Deleting posts while deleting a user" do
|
2023-11-10 06:47:59 +08:00
|
|
|
|
fab!(:user)
|
2022-11-29 00:32:57 +08:00
|
|
|
|
|
|
|
|
|
it "queues a job to also delete chat messages" do
|
|
|
|
|
deletion_opts = { delete_posts: true }
|
|
|
|
|
|
|
|
|
|
expect { UserDestroyer.new(Discourse.system_user).destroy(user, deletion_opts) }.to change(
|
2023-03-17 21:24:38 +08:00
|
|
|
|
Jobs::Chat::DeleteUserMessages.jobs,
|
2022-11-29 00:32:57 +08:00
|
|
|
|
:size,
|
|
|
|
|
).by(1)
|
|
|
|
|
end
|
|
|
|
|
end
|
2024-09-05 19:05:19 +08:00
|
|
|
|
|
|
|
|
|
describe "when using topic tags changed trigger automation" do
|
|
|
|
|
describe "with the send message script" do
|
|
|
|
|
fab!(:automation_1) do
|
|
|
|
|
Fabricate(
|
|
|
|
|
:automation,
|
|
|
|
|
trigger: DiscourseAutomation::Triggers::TOPIC_TAGS_CHANGED,
|
|
|
|
|
script: :send_chat_message,
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
fab!(:tag_1) { Fabricate(:tag) }
|
|
|
|
|
fab!(:user_1) { Fabricate(:admin) }
|
|
|
|
|
fab!(:topic_1) { Fabricate(:topic) }
|
|
|
|
|
fab!(:channel_1) { Fabricate(:chat_channel) }
|
|
|
|
|
|
|
|
|
|
before do
|
|
|
|
|
SiteSetting.discourse_automation_enabled = true
|
|
|
|
|
SiteSetting.tagging_enabled = true
|
|
|
|
|
|
|
|
|
|
automation_1.upsert_field!(
|
|
|
|
|
"watching_tags",
|
|
|
|
|
"tags",
|
|
|
|
|
{ value: [tag_1.name] },
|
|
|
|
|
target: "trigger",
|
|
|
|
|
)
|
|
|
|
|
automation_1.upsert_field!(
|
|
|
|
|
"chat_channel_id",
|
|
|
|
|
"text",
|
|
|
|
|
{ value: channel_1.id },
|
|
|
|
|
target: "script",
|
|
|
|
|
)
|
|
|
|
|
automation_1.upsert_field!(
|
|
|
|
|
"message",
|
|
|
|
|
"message",
|
|
|
|
|
{ value: "[{{topic_title}}]({{topic_url}})" },
|
|
|
|
|
target: "script",
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "sends the message" do
|
|
|
|
|
DiscourseTagging.tag_topic_by_names(topic_1, Guardian.new(user_1), [tag_1.name])
|
|
|
|
|
|
|
|
|
|
expect(channel_1.chat_messages.last.message).to eq(
|
|
|
|
|
"[#{topic_1.title}](#{topic_1.relative_url})",
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
2024-09-05 23:17:18 +08:00
|
|
|
|
|
|
|
|
|
describe "when using post_edited_created trigger automation" do
|
|
|
|
|
describe "with the send message script" do
|
|
|
|
|
fab!(:automation_1) do
|
|
|
|
|
Fabricate(
|
|
|
|
|
:automation,
|
|
|
|
|
trigger: DiscourseAutomation::Triggers::POST_CREATED_EDITED,
|
|
|
|
|
script: :send_chat_message,
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
fab!(:user_1) { Fabricate(:admin) }
|
|
|
|
|
fab!(:channel_1) { Fabricate(:chat_channel) }
|
|
|
|
|
|
|
|
|
|
before do
|
|
|
|
|
SiteSetting.discourse_automation_enabled = true
|
|
|
|
|
|
|
|
|
|
automation_1.upsert_field!(
|
|
|
|
|
"chat_channel_id",
|
|
|
|
|
"text",
|
|
|
|
|
{ value: channel_1.id },
|
|
|
|
|
target: "script",
|
|
|
|
|
)
|
|
|
|
|
automation_1.upsert_field!(
|
|
|
|
|
"message",
|
|
|
|
|
"message",
|
|
|
|
|
{ value: "[{{topic_title}}]({{topic_url}})" },
|
|
|
|
|
target: "script",
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
it "sends the message" do
|
|
|
|
|
PostCreator.create(
|
|
|
|
|
user_1,
|
|
|
|
|
{ title: "hello world topic", raw: "my name is fred", archetype: Archetype.default },
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
topic_1 = Topic.last
|
|
|
|
|
|
|
|
|
|
expect(channel_1.chat_messages.last.message).to eq(
|
|
|
|
|
"[#{topic_1.title}](#{topic_1.relative_url})",
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
2022-11-02 21:41:30 +08:00
|
|
|
|
end
|