import os.path import time import deltachat import pytest from deltachat.capi import lib as dclib TIMEOUT = 40 def get_user_crew(crewuser: deltachat.Account, id=11) -> deltachat.Chat: """Get the Team chat from the team member's point of view. :param crewuser: the account object of the team member :return: the chat object of the team chat """ for chat in crewuser.get_chats(): print(chat.id, chat.get_name()) user_crew = crewuser.get_chat_by_id(id) assert user_crew.get_name().startswith("Team") return user_crew @pytest.mark.timeout(TIMEOUT) def test_not_relay_groups(relaycrew, outsider, lp): bot = relaycrew.bot user = relaycrew.user def find_msg(ac, text): for chat in ac.get_chats(): for msg in chat.get_messages(): if msg.text == text: return msg text = "outsider -> bot 1:1 chat" lp.sec(text) outsider_botcontact = outsider.create_contact(bot.get_config("addr")) outsider_outside_chat = outsider.create_chat(outsider_botcontact) outsider_outside_chat.send_text(text) lp.sec("receiving message from outsider in 1:1 chat") bot_message_from_outsider = bot._evtracker.wait_next_incoming_message() bot_outside_chat = bot_message_from_outsider.chat assert bot_message_from_outsider.text == text assert not bot.relayplugin.is_relay_group(bot_outside_chat) lp.sec("leave relay group with user") relayed_msg = find_msg(user, text) if not relayed_msg: relayed_msg = user._evtracker.wait_next_incoming_message() relayed_msg.chat.remove_contact(user.get_config("addr")) leave_msg = bot._evtracker.wait_next_incoming_message() assert bot.relayplugin.is_relay_group(leave_msg.chat) text = "outsider -> bot group chat" lp.sec(text) outsider_bot_group = outsider.create_group_chat( "test with outsider", contacts=[outsider_botcontact] ) outsider_bot_group.send_text(text) lp.sec("receiving message from outsider in group chat") bot_message_from_outsider = bot._evtracker.wait_next_incoming_message() assert bot_message_from_outsider.text == text assert not bot.relayplugin.is_relay_group(bot_message_from_outsider.chat) text = "user -> bot 1:1 chat" lp.sec(text) user_botcontact = user.create_contact(bot.get_config("addr")) user_to_bot = user.create_chat(user_botcontact) user_to_bot.send_text(text) lp.sec("receiving message from user in 1:1 chat") # somehow the message doesn't trigger DC_EVENT_INCOMING_MSG # bot._evtracker.wait_next_incoming_message() bot_message_from_user = find_msg(bot, text) while not bot_message_from_user: bot_message_from_user = find_msg(bot, text) time.sleep(1) assert bot_message_from_user.text == text assert not bot.relayplugin.is_relay_group(bot_message_from_user.chat) text = "user -> bot group chat" lp.sec(text) user_group = user.create_group_chat("test with user", contacts=[user_botcontact]) user_group.send_text(text) lp.sec("receiving message from user in group chat") bot_message_from_user = bot._evtracker.wait_next_incoming_message() assert bot_message_from_user.text == text assert not bot.relayplugin.is_relay_group(bot_message_from_user.chat) @pytest.mark.timeout(TIMEOUT) def test_relay_group_forwarding(relaycrew, outsider): bot = relaycrew.bot user = relaycrew.user # create outside chat outsider_botcontact = outsider.create_contact(bot.get_config("addr")) outsider_outside_chat = outsider.create_chat(outsider_botcontact) outsider_outside_chat.send_text("test 1:1 message to bot") # get outside chat message_from_outsider = bot._evtracker.wait_next_incoming_message() bot_outside_chat = message_from_outsider.chat assert not bot.relayplugin.is_relay_group(bot_outside_chat) # get relay group user_forwarded_message_from_outsider = user._evtracker.wait_next_incoming_message() user_relay_group = user_forwarded_message_from_outsider.create_chat() user_relay_group.send_text( "Chatter in relay group" ) # send normal reply, not forwarded bot_chatter_in_relay_group = bot._evtracker.wait_next_incoming_message() bot_relay_group = bot_chatter_in_relay_group.chat # check if relay group has relay group properties assert bot_relay_group.get_name().startswith( "[%s] " % (bot.get_config("addr").split("@")[0],) ) assert ( bot_relay_group.get_messages()[0].get_sender_contact() == bot.get_self_contact() ) assert not bot_relay_group.is_protected() assert relaycrew.get_contacts() == bot_relay_group.get_contacts() assert bot.relayplugin.is_relay_group(bot_relay_group) # send direct reply, should be forwarded user_direct_reply = deltachat.Message.new_empty(user, view_type="text") user_direct_reply.set_text("This should be forwarded to the outsider") user_direct_reply.quote = user_forwarded_message_from_outsider sent_id = dclib.dc_send_msg( user._dc_context, user_relay_group.id, user_direct_reply._dc_msg ) assert sent_id == user_direct_reply.id # check that direct reply was forwarded to outsider outsider_direct_reply = outsider._evtracker.wait_next_incoming_message() assert outsider_direct_reply.text == "This should be forwarded to the outsider" assert outsider_direct_reply.chat == outsider_outside_chat assert outsider_direct_reply.get_sender_contact() == outsider_botcontact # check that normal reply was not forwarded to outsider assert bot_chatter_in_relay_group.text not in [ msg.text for msg in bot_outside_chat.get_messages() ] # reply with outsider outsider_outside_chat.send_text("Second message by outsider") # check that outsider's reply ends up in the same chat user_second_message_from_outsider = user._evtracker.wait_next_incoming_message() assert user_second_message_from_outsider.chat == user_relay_group # check that relay group explanation is not forwarded to outsider for chat in outsider.get_chats(): for msg in chat.get_messages(): assert "This is the relay group for" not in msg.text @pytest.mark.timeout(TIMEOUT) def test_offboarding(team_bot, relaycrew, outsider, team_user): # outsider sends message, creates relay group outsider_botcontact = outsider.create_contact(team_bot.get_config("addr")) outsider_outside_chat = outsider.create_chat(outsider_botcontact) outsider_outside_chat.send_text("test 1:1 message to bot") # get relay group user_relay_group = team_user._evtracker.wait_next_incoming_message().chat bot_relay_group = team_bot.get_chats()[-1] # outsider gets added to crew qr = relaycrew.get_join_qr() outsider.qr_join_chat(qr) outsider._evtracker.wait_securejoin_joiner_progress(1000) # user kicks outsider from crew user_crew = get_user_crew(team_user) user_crew.remove_contact(team_user.create_contact(outsider)) team_bot._evtracker.wait_next_incoming_message() # user leaves crew user_crew.remove_contact(team_user) # make sure they are also offboarded from relay group team_bot._evtracker.wait_next_incoming_message() team_user._evtracker.wait_next_incoming_message() team_user._evtracker.wait_next_incoming_message() team_user._evtracker.wait_next_incoming_message() for contact in bot_relay_group.get_contacts(): assert team_user.get_config("addr") != contact.addr # make sure there is no message in relay group that outsider was kicked for msg in user_relay_group.get_messages(): print(msg.text) assert outsider.get_config("addr") + " removed by " not in msg.text @pytest.mark.timeout(TIMEOUT) def test_default_outside_help(relaycrew, outsider): bot = relaycrew.bot user = relaycrew.user # create outside chat outsider_botcontact = outsider.create_contact(bot.get_config("addr")) outsider_outside_chat = outsider.create_chat(outsider_botcontact) outsider_outside_chat.send_text("/help") # get response outside_help_message = outsider._evtracker.wait_next_incoming_message() assert "I forward messages to the " in outside_help_message.text # assert no relay group was created assert len(bot.get_chats()) == 2 assert len(user.get_chats()) == 1 @pytest.mark.timeout(TIMEOUT) def test_empty_outside_help(relaycrew, outsider): bot = relaycrew.bot user = relaycrew.user # set outside_help_message empty for chat in user.get_chats(): print(chat.id, chat.get_name()) user_crew = user.get_chat_by_id(11) assert user_crew.get_name().startswith("Team") user_crew.send_text("/set_outside_help") # ensure /set_outside_help arrives before sending /help bot._evtracker.wait_next_incoming_message() # create outside chat outsider_botcontact = outsider.create_contact(bot.get_config("addr")) outsider_outside_chat = outsider.create_chat(outsider_botcontact) outsider_outside_chat.send_text("/help") # get forwarded /help message user._evtracker.wait_next_incoming_message() # "Removed help message for outsiders" user._evtracker.wait_next_incoming_message() # explanation message user_forwarded_message_from_outsider = user._evtracker.wait_next_incoming_message() assert user_forwarded_message_from_outsider.text == "/help" @pytest.mark.timeout(TIMEOUT) def test_changed_outside_help(relaycrew, outsider): bot = relaycrew.bot user = relaycrew.user # set outside_help_message empty for chat in user.get_chats(): print(chat.id, chat.get_name()) user_crew = user.get_chat_by_id(11) assert user_crew.get_name().startswith("Team") outside_help_text = "Hi friend :) send me messages to chat with the team" user_crew.send_text("/set_outside_help " + outside_help_text) # ensure /set_outside_help arrives before sending /help bot._evtracker.wait_next_incoming_message() # create outside chat outsider_botcontact = outsider.create_contact(bot.get_config("addr")) outsider_outside_chat = outsider.create_chat(outsider_botcontact) outsider_outside_chat.send_text("/help") # get response outside_help_message = outsider._evtracker.wait_next_incoming_message() assert outside_help_message.text == outside_help_text # assert no relay group was created assert len(bot.get_chats()) == 2 assert len(user.get_chats()) == 1 @pytest.mark.timeout(TIMEOUT) def test_change_avatar(relaycrew): bot = relaycrew.bot user = relaycrew.user for contact in user.get_contacts(): if contact.addr == bot.get_config("addr"): assert not contact.get_profile_image() botcontact = contact break else: pytest.fail("bot contact not found") example_png_path = "/usr/share/pixmaps/debian-logo.png" if not os.path.exists(example_png_path): pytest.skip(f"example image not available: {example_png_path}") # set avatar to example image user_crew = get_user_crew(user) msg = deltachat.Message.new_empty(user, "image") msg.set_text("/set_avatar") msg.set_file(example_png_path) sent_id = dclib.dc_send_msg(user._dc_context, user_crew.id, msg._dc_msg) assert sent_id == msg.id group_avatar_changed_msg = user._evtracker.wait_next_incoming_message() assert "Group image changed" in group_avatar_changed_msg.text assert user_crew.get_profile_image() confirmation_msg = user._evtracker.wait_next_incoming_message() assert confirmation_msg.text == "Avatar changed to this image." assert botcontact.get_profile_image() @pytest.mark.timeout(TIMEOUT * 2) def test_forward_sending_errors_to_relay_group(relaycrew): usercrew = relaycrew.user.get_chats()[-1] usercrew.send_text("/start_chat alice@example.org This_Message_will_fail test") while len(relaycrew.bot.get_chats()) < 3: time.sleep(0.1) out_chat = relaycrew.bot.get_chats()[-1] outgoing_message = out_chat.get_messages()[-1] print(outgoing_message) begin = int(time.time()) while not outgoing_message.is_out_failed() and int(time.time()) < begin + TIMEOUT: time.sleep(0.1) assert outgoing_message.is_out_failed() while len(relaycrew.user.get_chats()) < 2 and int(time.time()) < begin + TIMEOUT: time.sleep(0.1) for chat in relaycrew.user.get_chats(): if "This Message will fail" in chat.get_name(): relay_group = chat while len(relay_group.get_messages()) < 3 and int(time.time()) < begin + TIMEOUT: print(relay_group.get_messages()[-1].text) time.sleep(0.1) assert ( "Recipient address rejected: Domain example.org does not accept mail" not in relay_group.get_messages()[-1].text ) assert ( "Invalid unencrypted mail to " in relay_group.get_messages()[-1].text ) @pytest.mark.timeout(TIMEOUT) def test_public_invite(relaycrew, outsider): crew = get_user_crew(relaycrew.user) crew.send_text("/generate-invite") result = relaycrew.user._evtracker.wait_next_incoming_message() # assert result.filename # assert result.text.startswith("https://i.delta.chat") # qr = result.filename # invite = "OPENPGP4FPR:" + result.text[22::] chat = outsider.qr_setup_contact(result.text) outsider._evtracker.wait_securejoin_joiner_progress(1000) while not chat.is_protected(): print(chat.get_messages()[-1].text) time.sleep(1)