diff --git a/simplex-chat.cabal b/simplex-chat.cabal index 2f66f4c4f..f5c20ec9b 100644 --- a/simplex-chat.cabal +++ b/simplex-chat.cabal @@ -289,6 +289,11 @@ test-suite simplex-chat-test other-modules: ChatClient ChatTests + ChatTests.Direct + ChatTests.Files + ChatTests.Groups + ChatTests.Profiles + ChatTests.Utils MarkdownTests MobileTests ProtocolTests diff --git a/tests/ChatTests.hs b/tests/ChatTests.hs index 525a15025..ed81853ac 100644 --- a/tests/ChatTests.hs +++ b/tests/ChatTests.hs @@ -1,6018 +1,14 @@ -{-# LANGUAGE DuplicateRecordFields #-} -{-# LANGUAGE LambdaCase #-} -{-# LANGUAGE NamedFieldPuns #-} -{-# LANGUAGE OverloadedStrings #-} -{-# LANGUAGE PostfixOperators #-} -{-# LANGUAGE RankNTypes #-} -{-# LANGUAGE ScopedTypeVariables #-} - module ChatTests where -import ChatClient -import Control.Concurrent (threadDelay) -import Control.Concurrent.Async (concurrently_) -import Control.Concurrent.STM -import Control.Monad (forM_, unless, when) -import Data.Aeson (ToJSON) -import qualified Data.Aeson as J -import qualified Data.ByteString.Char8 as B -import qualified Data.ByteString.Lazy.Char8 as LB -import Data.Char (isDigit) -import Data.List (isPrefixOf, isSuffixOf) -import Data.Maybe (fromMaybe) -import Data.String -import qualified Data.Text as T -import Simplex.Chat.Call -import Simplex.Chat.Controller (ChatConfig (..), ChatController (..), InlineFilesConfig (..), defaultInlineFilesConfig) -import Simplex.Chat.Options (ChatOpts (..)) -import Simplex.Chat.Store (getUserContactProfiles) -import Simplex.Chat.Types -import Simplex.Messaging.Agent.Store.SQLite (withTransaction) -import qualified Simplex.Messaging.Crypto as C -import Simplex.Messaging.Encoding.String -import Simplex.Messaging.Util (unlessM) -import System.Directory (copyFile, createDirectoryIfMissing, doesDirectoryExist, doesFileExist) -import System.FilePath (()) +import ChatTests.Direct +import ChatTests.Files +import ChatTests.Groups +import ChatTests.Profiles import Test.Hspec -defaultPrefs :: Maybe Preferences -defaultPrefs = Just $ toChatPrefs defaultChatPrefs - -aliceProfile :: Profile -aliceProfile = Profile {displayName = "alice", fullName = "Alice", image = Nothing, preferences = defaultPrefs} - -bobProfile :: Profile -bobProfile = Profile {displayName = "bob", fullName = "Bob", image = Just (ImageData ""), preferences = defaultPrefs} - -cathProfile :: Profile -cathProfile = Profile {displayName = "cath", fullName = "Catherine", image = Nothing, preferences = defaultPrefs} - -danProfile :: Profile -danProfile = Profile {displayName = "dan", fullName = "Daniel", image = Nothing, preferences = defaultPrefs} - chatTests :: SpecWith FilePath chatTests = do - describe "direct messages" $ do - describe "add contact and send/receive message" testAddContact - it "deleting contact deletes profile" testDeleteContactDeletesProfile - it "direct message quoted replies" testDirectMessageQuotedReply - it "direct message update" testDirectMessageUpdate - it "direct message delete" testDirectMessageDelete - it "direct live message" testDirectLiveMessage - it "repeat AUTH errors disable contact" testRepeatAuthErrorsDisableContact - describe "chat groups" $ do - describe "add contacts, create group and send/receive messages" testGroup - it "add contacts, create group and send/receive messages, check messages" testGroupCheckMessages - it "create and join group with 4 members" testGroup2 - it "create and delete group" testGroupDelete - it "create group with the same displayName" testGroupSameName - it "invitee delete group when in status invited" testGroupDeleteWhenInvited - it "re-add member in status invited" testGroupReAddInvited - it "re-add member in status invited, change role" testGroupReAddInvitedChangeRole - it "delete contact before they accept group invitation, contact joins group" testGroupDeleteInvitedContact - it "member profile is kept when deleting group if other groups have this member" testDeleteGroupMemberProfileKept - it "remove contact from group and add again" testGroupRemoveAdd - it "list groups containing group invitations" testGroupList - it "group message quoted replies" testGroupMessageQuotedReply - it "group message update" testGroupMessageUpdate - it "group message delete" testGroupMessageDelete - it "group live message" testGroupLiveMessage - it "update group profile" testUpdateGroupProfile - it "update member role" testUpdateMemberRole - it "unused contacts are deleted after all their groups are deleted" testGroupDeleteUnusedContacts - it "group description is shown as the first message to new members" testGroupDescription - describe "async group connections" $ do - xit "create and join group when clients go offline" testGroupAsync - describe "user profiles" $ do - it "update user profile and notify contacts" testUpdateProfile - it "update user profile with image" testUpdateProfileImage - describe "sending and receiving files" $ do - describe "send and receive file" $ fileTestMatrix2 runTestFileTransfer - it "send and receive file inline (without accepting)" testInlineFileTransfer - it "accept inline file transfer, sender cancels during transfer" testAcceptInlineFileSndCancelDuringTransfer - it "send and receive small file inline (default config)" testSmallInlineFileTransfer - it "small file sent without acceptance is ignored in terminal by default" testSmallInlineFileIgnored - it "receive file inline with inline=on option" testReceiveInline - describe "send and receive a small file" $ fileTestMatrix2 runTestSmallFileTransfer - describe "sender cancelled file transfer before transfer" $ fileTestMatrix2 runTestFileSndCancelBeforeTransfer - it "sender cancelled file transfer during transfer" testFileSndCancelDuringTransfer - it "recipient cancelled file transfer" testFileRcvCancel - describe "send and receive file to group" $ fileTestMatrix3 runTestGroupFileTransfer - it "send and receive file inline to group (without accepting)" testInlineGroupFileTransfer - it "send and receive small file inline to group (default config)" testSmallInlineGroupFileTransfer - it "small file sent without acceptance is ignored in terminal by default" testSmallInlineGroupFileIgnored - describe "sender cancelled group file transfer before transfer" $ fileTestMatrix3 runTestGroupFileSndCancelBeforeTransfer - describe "messages with files" $ do - describe "send and receive message with file" $ fileTestMatrix2 runTestMessageWithFile - it "send and receive image" testSendImage - it "files folder: send and receive image" testFilesFoldersSendImage - it "files folder: sender deleted file during transfer" testFilesFoldersImageSndDelete - it "files folder: recipient deleted file during transfer" testFilesFoldersImageRcvDelete - it "send and receive image with text and quote" testSendImageWithTextAndQuote - describe "send and receive image to group" testGroupSendImage - it "send and receive image with text and quote to group" testGroupSendImageWithTextAndQuote - describe "user contact link" $ do - describe "create and connect via contact link" testUserContactLink - it "auto accept contact requests" testUserContactLinkAutoAccept - it "deduplicate contact requests" testDeduplicateContactRequests - it "deduplicate contact requests with profile change" testDeduplicateContactRequestsProfileChange - it "reject contact and delete contact link" testRejectContactAndDeleteUserContact - it "delete connection requests when contact link deleted" testDeleteConnectionRequests - it "auto-reply message" testAutoReplyMessage - it "auto-reply message in incognito" testAutoReplyMessageInIncognito - describe "incognito mode" $ do - it "connect incognito via invitation link" testConnectIncognitoInvitationLink - it "connect incognito via contact address" testConnectIncognitoContactAddress - it "accept contact request incognito" testAcceptContactRequestIncognito - it "join group incognito" testJoinGroupIncognito - it "can't invite contact to whom user connected incognito to a group" testCantInviteContactIncognito - it "can't see global preferences update" testCantSeeGlobalPrefsUpdateIncognito - it "deleting contact first, group second deletes incognito profile" testDeleteContactThenGroupDeletesIncognitoProfile - it "deleting group first, contact second deletes incognito profile" testDeleteGroupThenContactDeletesIncognitoProfile - describe "group links" $ do - it "create group link, join via group link" testGroupLink - it "delete group, re-join via same link" testGroupLinkDeleteGroupRejoin - it "sending message to contact created via group link marks it used" testGroupLinkContactUsed - it "create group link, join via group link - incognito membership" testGroupLinkIncognitoMembership - it "unused host contact is deleted after all groups with it are deleted" testGroupLinkUnusedHostContactDeleted - it "leaving groups with unused host contacts deletes incognito profiles" testGroupLinkIncognitoUnusedHostContactsDeleted - describe "contact aliases" $ do - it "set contact alias" testSetAlias - it "set connection alias" testSetConnectionAlias - describe "preferences" $ do - it "set contact preferences" testSetContactPrefs - it "feature offers" testFeatureOffers - it "update group preferences" testUpdateGroupPrefs - it "allow full deletion to contact" testAllowFullDeletionContact - it "allow full deletion to group" testAllowFullDeletionGroup - it "prohibit direct messages to group members" testProhibitDirectMessages - it "enable timed messages with contact" testEnableTimedMessagesContact - it "enable timed messages in group" testEnableTimedMessagesGroup - it "timed messages enabled globally, contact turns on" testTimedMessagesEnabledGlobally - describe "SMP servers" $ do - it "get and set SMP servers" testGetSetSMPServers - it "test SMP server connection" testTestSMPServerConnection - describe "async connection handshake" $ do - it "connect when initiating client goes offline" testAsyncInitiatingOffline - it "connect when accepting client goes offline" testAsyncAcceptingOffline - describe "connect, fully asynchronous (when clients are never simultaneously online)" $ do - it "v2" testFullAsync - -- it "v1" testFullAsyncV1 - -- it "v1 to v2" testFullAsyncV1toV2 - -- it "v2 to v1" testFullAsyncV2toV1 - describe "async sending and receiving files" $ do - it "send and receive file, sender restarts" testAsyncFileTransferSenderRestarts - it "send and receive file, receiver restarts" testAsyncFileTransferReceiverRestarts - xdescribe "send and receive file, fully asynchronous" $ do - it "v2" testAsyncFileTransfer - it "v1" testAsyncFileTransferV1 - xit "send and receive file to group, fully asynchronous" testAsyncGroupFileTransfer - describe "webrtc calls api" $ do - it "negotiate call" testNegotiateCall - describe "maintenance mode" $ do - it "start/stop/export/import chat" testMaintenanceMode - it "export/import chat with files" testMaintenanceModeWithFiles - it "encrypt/decrypt database" testDatabaseEncryption - describe "mute/unmute messages" $ do - it "mute/unmute contact" testMuteContact - it "mute/unmute group" testMuteGroup - describe "multiple users" $ do - it "create second user" testCreateSecondUser - it "multiple users subscribe and receive messages after restart" testUsersSubscribeAfterRestart - it "both users have contact link" testMultipleUserAddresses - it "create user with default servers" testCreateUserDefaultServers - it "create user with same servers" testCreateUserSameServers - it "delete user" testDeleteUser - it "users have different chat item TTL configuration, chat items expire" testUsersDifferentCIExpirationTTL - it "chat items expire after restart for all users according to per user configuration" testUsersRestartCIExpiration - it "chat items only expire for users who configured expiration" testEnableCIExpirationOnlyForOneUser - it "disabling chat item expiration doesn't disable it for other users" testDisableCIExpirationOnlyForOneUser - it "both users have configured timed messages with contacts, messages expire, restart" testUsersTimedMessages - describe "chat item expiration" $ do - it "set chat item TTL" testSetChatItemTTL - describe "queue rotation" $ do - it "switch contact to a different queue" testSwitchContact - it "switch group member to a different queue" testSwitchGroupMember - describe "connection verification code" $ do - it "verificationCode function converts ByteString to series of digits" $ \_ -> - verificationCode (C.sha256Hash "abcd") `shouldBe` "61889 38426 63934 09576 96390 79389 84124 85253 63658 69469 70853 37788 95900 68296 20156 25" - it "sameVerificationCode function should ignore spaces" $ \_ -> - sameVerificationCode "123 456 789" "12345 6789" `shouldBe` True - it "mark contact verified" testMarkContactVerified - it "mark group member verified" testMarkGroupMemberVerified - -versionTestMatrix2 :: (HasCallStack => TestCC -> TestCC -> IO ()) -> SpecWith FilePath -versionTestMatrix2 runTest = do - it "v2" $ testChat2 aliceProfile bobProfile runTest - it "v1" $ testChatCfg2 testCfgV1 aliceProfile bobProfile runTest - it "v1 to v2" $ runTestCfg2 testCfg testCfgV1 runTest - it "v2 to v1" $ runTestCfg2 testCfgV1 testCfg runTest - -versionTestMatrix3 :: (HasCallStack => TestCC -> TestCC -> TestCC -> IO ()) -> SpecWith FilePath -versionTestMatrix3 runTest = do - it "v2" $ testChat3 aliceProfile bobProfile cathProfile runTest - --- it "v1" $ testChatCfg3 testCfgV1 aliceProfile bobProfile cathProfile runTest --- it "v1 to v2" $ runTestCfg3 testCfg testCfgV1 testCfgV1 runTest --- it "v2+v1 to v2" $ runTestCfg3 testCfg testCfg testCfgV1 runTest --- it "v2 to v1" $ runTestCfg3 testCfgV1 testCfg testCfg runTest --- it "v2+v1 to v1" $ runTestCfg3 testCfgV1 testCfg testCfgV1 runTest - -inlineCfg :: Integer -> ChatConfig -inlineCfg n = testCfg {inlineFiles = defaultInlineFilesConfig {sendChunks = 0, offerChunks = n, receiveChunks = n}} - -fileTestMatrix2 :: (HasCallStack => TestCC -> TestCC -> IO ()) -> SpecWith FilePath -fileTestMatrix2 runTest = do - it "via connection" $ runTestCfg2 viaConn viaConn runTest - it "inline (accepting)" $ runTestCfg2 inline inline runTest - it "via connection (inline offered)" $ runTestCfg2 inline viaConn runTest - it "via connection (inline supported)" $ runTestCfg2 viaConn inline runTest - where - inline = inlineCfg 100 - viaConn = inlineCfg 0 - -fileTestMatrix3 :: (HasCallStack => TestCC -> TestCC -> TestCC -> IO ()) -> SpecWith FilePath -fileTestMatrix3 runTest = do - it "via connection" $ runTestCfg3 viaConn viaConn viaConn runTest - it "inline" $ runTestCfg3 inline inline inline runTest - it "via connection (inline offered)" $ runTestCfg3 inline viaConn viaConn runTest - it "via connection (inline supported)" $ runTestCfg3 viaConn inline inline runTest - where - inline = inlineCfg 100 - viaConn = inlineCfg 0 - -runTestCfg2 :: ChatConfig -> ChatConfig -> (HasCallStack => TestCC -> TestCC -> IO ()) -> FilePath -> IO () -runTestCfg2 aliceCfg bobCfg runTest tmp = - withNewTestChatCfg tmp aliceCfg "alice" aliceProfile $ \alice -> - withNewTestChatCfg tmp bobCfg "bob" bobProfile $ \bob -> - runTest alice bob - -runTestCfg3 :: ChatConfig -> ChatConfig -> ChatConfig -> (HasCallStack => TestCC -> TestCC -> TestCC -> IO ()) -> FilePath -> IO () -runTestCfg3 aliceCfg bobCfg cathCfg runTest tmp = - withNewTestChatCfg tmp aliceCfg "alice" aliceProfile $ \alice -> - withNewTestChatCfg tmp bobCfg "bob" bobProfile $ \bob -> - withNewTestChatCfg tmp cathCfg "cath" cathProfile $ \cath -> - runTest alice bob cath - -testAddContact :: HasCallStack => SpecWith FilePath -testAddContact = versionTestMatrix2 runTestAddContact - where - runTestAddContact alice bob = do - alice ##> "/_connect 1" - inv <- getInvitation alice - bob ##> ("/_connect 1 " <> inv) - bob <## "confirmation sent!" - concurrently_ - (bob <## "alice (Alice): contact is connected") - (alice <## "bob (Bob): contact is connected") - threadDelay 100000 - chatsEmpty alice bob - alice #> "@bob hello there 🙂" - bob <# "alice> hello there 🙂" - alice ##> "/_unread chat @2 on" - alice <## "ok" - alice ##> "/_unread chat @2 off" - alice <## "ok" - chatsOneMessage alice bob - bob #> "@alice hello there" - alice <# "bob> hello there" - bob #> "@alice how are you?" - alice <# "bob> how are you?" - chatsManyMessages alice bob - -- test adding the same contact one more time - local name will be different - alice ##> "/c" - inv' <- getInvitation alice - bob ##> ("/c " <> inv') - bob <## "confirmation sent!" - concurrently_ - (bob <## "alice_1 (Alice): contact is connected") - (alice <## "bob_1 (Bob): contact is connected") - alice #> "@bob_1 hello" - bob <# "alice_1> hello" - bob #> "@alice_1 hi" - alice <# "bob_1> hi" - alice @@@ [("@bob_1", "hi"), ("@bob", "how are you?")] - bob @@@ [("@alice_1", "hi"), ("@alice", "how are you?")] - -- test deleting contact - alice ##> "/d bob_1" - alice <## "bob_1: contact is deleted" - alice ##> "@bob_1 hey" - alice <## "no contact bob_1" - alice @@@ [("@bob", "how are you?")] - alice `hasContactProfiles` ["alice", "bob"] - bob @@@ [("@alice_1", "hi"), ("@alice", "how are you?")] - bob `hasContactProfiles` ["alice", "alice", "bob"] - -- test clearing chat - alice #$> ("/clear bob", id, "bob: all messages are removed locally ONLY") - alice #$> ("/_get chat @2 count=100", chat, []) - bob #$> ("/clear alice", id, "alice: all messages are removed locally ONLY") - bob #$> ("/_get chat @2 count=100", chat, []) - chatsEmpty alice bob = do - alice @@@ [("@bob", "Voice messages: enabled")] - alice #$> ("/_get chat @2 count=100", chat, chatFeatures) - bob @@@ [("@alice", "Voice messages: enabled")] - bob #$> ("/_get chat @2 count=100", chat, chatFeatures) - chatsOneMessage alice bob = do - alice @@@ [("@bob", "hello there 🙂")] - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hello there 🙂")]) - bob @@@ [("@alice", "hello there 🙂")] - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hello there 🙂")]) - chatsManyMessages alice bob = do - alice @@@ [("@bob", "how are you?")] - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hello there 🙂"), (0, "hello there"), (0, "how are you?")]) - bob @@@ [("@alice", "how are you?")] - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hello there 🙂"), (1, "hello there"), (1, "how are you?")]) - -- pagination - alice #$> ("/_get chat @2 after=" <> itemId 1 <> " count=100", chat, [(0, "hello there"), (0, "how are you?")]) - alice #$> ("/_get chat @2 before=" <> itemId 2 <> " count=100", chat, chatFeatures <> [(1, "hello there 🙂")]) - -- search - alice #$> ("/_get chat @2 count=100 search=ello ther", chat, [(1, "hello there 🙂"), (0, "hello there")]) - -- read messages - alice #$> ("/_read chat @2 from=1 to=100", id, "ok") - bob #$> ("/_read chat @2 from=1 to=100", id, "ok") - alice #$> ("/_read chat @2", id, "ok") - bob #$> ("/_read chat @2", id, "ok") - -testDeleteContactDeletesProfile :: HasCallStack => FilePath -> IO () -testDeleteContactDeletesProfile = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice <##> bob - -- alice deletes contact, profile is deleted - alice ##> "/d bob" - alice <## "bob: contact is deleted" - alice ##> "/_contacts 1" - (alice "/d alice" - bob <## "alice: contact is deleted" - bob ##> "/contacts" - (bob FilePath -> IO () -testDirectMessageQuotedReply = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice ##> "/_send @2 text hello! how are you?" - alice <# "@bob hello! how are you?" - bob <# "alice> hello! how are you?" - bob #> "@alice hi!" - alice <# "bob> hi!" - bob `send` "> @alice (hello) all good - you?" - bob <# "@alice > hello! how are you?" - bob <## " all good - you?" - alice <# "bob> > hello! how are you?" - alice <## " all good - you?" - bob #$> ("/_get chat @2 count=1", chat', [((1, "all good - you?"), Just (0, "hello! how are you?"))]) - alice #$> ("/_get chat @2 count=1", chat', [((0, "all good - you?"), Just (1, "hello! how are you?"))]) - bob `send` ">> @alice (all good) will tell more" - bob <# "@alice >> all good - you?" - bob <## " will tell more" - alice <# "bob> >> all good - you?" - alice <## " will tell more" - bob #$> ("/_get chat @2 count=1", chat', [((1, "will tell more"), Just (1, "all good - you?"))]) - alice #$> ("/_get chat @2 count=1", chat', [((0, "will tell more"), Just (0, "all good - you?"))]) - -testDirectMessageUpdate :: HasCallStack => FilePath -> IO () -testDirectMessageUpdate = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - - -- msg id 1 - alice #> "@bob hello 🙂" - bob <# "alice> hello 🙂" - - -- msg id 2 - bob `send` "> @alice (hello) hi alice" - bob <# "@alice > hello 🙂" - bob <## " hi alice" - alice <# "bob> > hello 🙂" - alice <## " hi alice" - - alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "hello 🙂"), Nothing), ((0, "hi alice"), Just (1, "hello 🙂"))]) - bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hello 🙂"), Nothing), ((1, "hi alice"), Just (0, "hello 🙂"))]) - - alice ##> ("/_update item @2 " <> itemId 1 <> " text hey 👋") - alice <# "@bob [edited] hey 👋" - bob <# "alice> [edited] hey 👋" - - alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "hey 👋"), Nothing), ((0, "hi alice"), Just (1, "hello 🙂"))]) - bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hey 👋"), Nothing), ((1, "hi alice"), Just (0, "hello 🙂"))]) - - -- msg id 3 - bob `send` "> @alice (hey) hey alice" - bob <# "@alice > hey 👋" - bob <## " hey alice" - alice <# "bob> > hey 👋" - alice <## " hey alice" - - alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "hey 👋"), Nothing), ((0, "hi alice"), Just (1, "hello 🙂")), ((0, "hey alice"), Just (1, "hey 👋"))]) - bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hey 👋"), Nothing), ((1, "hi alice"), Just (0, "hello 🙂")), ((1, "hey alice"), Just (0, "hey 👋"))]) - - alice ##> ("/_update item @2 " <> itemId 1 <> " text greetings 🤝") - alice <# "@bob [edited] greetings 🤝" - bob <# "alice> [edited] greetings 🤝" - - alice #$> ("/_update item @2 " <> itemId 2 <> " text updating bob's message", id, "cannot update this item") - - alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "greetings 🤝"), Nothing), ((0, "hi alice"), Just (1, "hello 🙂")), ((0, "hey alice"), Just (1, "hey 👋"))]) - bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "greetings 🤝"), Nothing), ((1, "hi alice"), Just (0, "hello 🙂")), ((1, "hey alice"), Just (0, "hey 👋"))]) - - bob ##> ("/_update item @2 " <> itemId 2 <> " text hey Alice") - bob <# "@alice [edited] > hello 🙂" - bob <## " hey Alice" - alice <# "bob> [edited] > hello 🙂" - alice <## " hey Alice" - - bob ##> ("/_update item @2 " <> itemId 3 <> " text greetings Alice") - bob <# "@alice [edited] > hey 👋" - bob <## " greetings Alice" - alice <# "bob> [edited] > hey 👋" - alice <## " greetings Alice" - - alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "greetings 🤝"), Nothing), ((0, "hey Alice"), Just (1, "hello 🙂")), ((0, "greetings Alice"), Just (1, "hey 👋"))]) - bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "greetings 🤝"), Nothing), ((1, "hey Alice"), Just (0, "hello 🙂")), ((1, "greetings Alice"), Just (0, "hey 👋"))]) - -testDirectMessageDelete :: HasCallStack => FilePath -> IO () -testDirectMessageDelete = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - - -- alice, bob: msg id 1 - alice #> "@bob hello 🙂" - bob <# "alice> hello 🙂" - - -- alice, bob: msg id 2 - bob `send` "> @alice (hello 🙂) hey alic" - bob <# "@alice > hello 🙂" - bob <## " hey alic" - alice <# "bob> > hello 🙂" - alice <## " hey alic" - - -- alice: deletes msg ids 1,2 - alice #$> ("/_delete item @2 " <> itemId 1 <> " internal", id, "message deleted") - alice #$> ("/_delete item @2 " <> itemId 2 <> " internal", id, "message deleted") - - alice @@@ [("@bob", "Voice messages: enabled")] - alice #$> ("/_get chat @2 count=100", chat, chatFeatures) - - -- alice: msg id 1 - bob ##> ("/_update item @2 " <> itemId 2 <> " text hey alice") - bob <# "@alice [edited] > hello 🙂" - bob <## " hey alice" - alice <# "bob> [edited] hey alice" - alice @@@ [("@bob", "hey alice")] - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hey alice")]) - - -- bob: marks deleted msg id 2 - bob #$> ("/_delete item @2 " <> itemId 2 <> " broadcast", id, "message marked deleted") - bob @@@ [("@alice", "hey alice [marked deleted]")] - alice <# "bob> [marked deleted] hey alice" - alice @@@ [("@bob", "hey alice [marked deleted]")] - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hey alice [marked deleted]")]) - - -- alice: deletes msg id 1 that was broadcast deleted by bob - alice #$> ("/_delete item @2 " <> itemId 1 <> " internal", id, "message deleted") - alice @@@ [("@bob", "Voice messages: enabled")] - alice #$> ("/_get chat @2 count=100", chat, chatFeatures) - - -- alice: msg id 1, bob: msg id 3 (quoting message alice deleted locally) - bob `send` "> @alice (hello 🙂) do you receive my messages?" - bob <# "@alice > hello 🙂" - bob <## " do you receive my messages?" - alice <# "bob> > hello 🙂" - alice <## " do you receive my messages?" - alice @@@ [("@bob", "do you receive my messages?")] - alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "do you receive my messages?"), Just (1, "hello 🙂"))]) - alice #$> ("/_delete item @2 " <> itemId 1 <> " broadcast", id, "cannot delete this item") - - -- alice: msg id 2, bob: msg id 4 - bob #> "@alice how are you?" - alice <# "bob> how are you?" - - -- alice: deletes msg id 2 - alice #$> ("/_delete item @2 " <> itemId 2 <> " internal", id, "message deleted") - - -- bob: marks deleted msg id 4 (that alice deleted locally) - bob #$> ("/_delete item @2 " <> itemId 4 <> " broadcast", id, "message marked deleted") - alice <## "bob> [deleted - original message not found]" - - alice @@@ [("@bob", "do you receive my messages?")] - alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "do you receive my messages?"), Just (1, "hello 🙂"))]) - bob @@@ [("@alice", "how are you? [marked deleted]")] - bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hello 🙂"), Nothing), ((1, "hey alice [marked deleted]"), Just (0, "hello 🙂")), ((1, "do you receive my messages?"), Just (0, "hello 🙂")), ((1, "how are you? [marked deleted]"), Nothing)]) - - -- bob: deletes msg ids 2,4 (that he has marked deleted) - bob #$> ("/_delete item @2 " <> itemId 2 <> " internal", id, "message deleted") - bob #$> ("/_delete item @2 " <> itemId 4 <> " internal", id, "message deleted") - bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hello 🙂"), Nothing), ((1, "do you receive my messages?"), Just (0, "hello 🙂"))]) - -testDirectLiveMessage :: HasCallStack => FilePath -> IO () -testDirectLiveMessage = - testChat2 aliceProfile bobProfile $ \alice bob -> do - connectUsers alice bob - -- non-empty live message is sent instantly - alice `send` "/live @bob hello" - bob <# "alice> [LIVE started] use /show [on/off/4] hello" - alice ##> ("/_update item @2 " <> itemId 1 <> " text hello there") - alice <# "@bob [LIVE] hello there" - bob <# "alice> [LIVE ended] hello there" - -- empty live message is also sent instantly - alice `send` "/live @bob" - bob <# "alice> [LIVE started] use /show [on/off/5]" - alice ##> ("/_update item @2 " <> itemId 2 <> " text hello 2") - alice <# "@bob [LIVE] hello 2" - bob <# "alice> [LIVE ended] hello 2" - -testRepeatAuthErrorsDisableContact :: HasCallStack => FilePath -> IO () -testRepeatAuthErrorsDisableContact = - testChat2 aliceProfile bobProfile $ \alice bob -> do - connectUsers alice bob - alice <##> bob - bob ##> "/d alice" - bob <## "alice: contact is deleted" - forM_ [1 .. authErrDisableCount] $ \_ -> sendAuth alice - alice <## "[bob] connection is disabled, to enable: /enable bob, to delete: /d bob" - alice ##> "@bob hey" - alice <## "bob: disabled, to enable: /enable bob, to delete: /d bob" - alice ##> "/enable bob" - alice <## "ok" - sendAuth alice - where - sendAuth alice = do - alice #> "@bob hey" - alice <## "[bob, contactId: 2, connId: 1] error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" - -testGroup :: HasCallStack => SpecWith FilePath -testGroup = versionTestMatrix3 runTestGroup - where - runTestGroup alice bob cath = testGroupShared alice bob cath False - -testGroupCheckMessages :: HasCallStack => FilePath -> IO () -testGroupCheckMessages = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> testGroupShared alice bob cath True - -testGroupShared :: HasCallStack => TestCC -> TestCC -> TestCC -> Bool -> IO () -testGroupShared alice bob cath checkMessages = do - connectUsers alice bob - connectUsers alice cath - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as admin" - bob <## "use /j team to accept" - ] - bob ##> "/j team" - concurrently_ - (alice <## "#team: bob joined the group") - (bob <## "#team: you joined the group") - when checkMessages $ threadDelay 1000000 -- for deterministic order of messages and "connected" events - alice ##> "/a team cath" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to cath", - do - cath <## "#team: alice invites you to join the group as admin" - cath <## "use /j team to accept" - ] - cath ##> "/j team" - concurrentlyN_ - [ alice <## "#team: cath joined the group", - do - cath <## "#team: you joined the group" - cath <## "#team: member bob (Bob) is connected", - do - bob <## "#team: alice added cath (Catherine) to the group (connecting...)" - bob <## "#team: new member cath is connected" - ] - when checkMessages $ threadDelay 1000000 -- for deterministic order of messages and "connected" events - alice #> "#team hello" - msgItem1 <- lastItemId alice - concurrently_ - (bob <# "#team alice> hello") - (cath <# "#team alice> hello") - when checkMessages $ threadDelay 1000000 -- server assigns timestamps with one second precision - bob #> "#team hi there" - concurrently_ - (alice <# "#team bob> hi there") - (cath <# "#team bob> hi there") - when checkMessages $ threadDelay 1000000 - cath #> "#team hey team" - concurrently_ - (alice <# "#team cath> hey team") - (bob <# "#team cath> hey team") - msgItem2 <- lastItemId alice - bob <##> cath - when checkMessages $ getReadChats msgItem1 msgItem2 - -- list groups - alice ##> "/gs" - alice <## "#team" - -- list group members - alice ##> "/ms team" - alice - <### [ "alice (Alice): owner, you, created group", - "bob (Bob): admin, invited, connected", - "cath (Catherine): admin, invited, connected" - ] - -- list contacts - alice ##> "/contacts" - alice <## "bob (Bob)" - alice <## "cath (Catherine)" - -- test observer role - -- to be enabled once the role is enabled in parser - -- alice ##> "/mr team bob observer" - -- concurrentlyN_ - -- [ alice <## "#team: you changed the role of bob from admin to observer", - -- bob <## "#team: alice changed your role from admin to observer", - -- cath <## "#team: alice changed the role of bob from admin to observer" - -- ] - -- bob ##> "#team hello" - -- bob <## "you don't have permission to send messages to this group" - -- bob ##> "/rm team cath" - -- bob <## "you have insufficient permissions for this action, the required role is admin" - -- cath #> "#team hello" - -- concurrentlyN_ - -- [ alice <# "#team cath> hello", - -- bob <# "#team cath> hello" - -- ] - -- alice ##> "/mr team bob admin" - -- concurrentlyN_ - -- [ alice <## "#team: you changed the role of bob from observer to admin", - -- bob <## "#team: alice changed your role from observer to admin", - -- cath <## "#team: alice changed the role of bob from observer to admin" - -- ] - -- remove member - bob ##> "/rm team cath" - concurrentlyN_ - [ bob <## "#team: you removed cath from the group", - alice <## "#team: bob removed cath from the group", - do - cath <## "#team: bob removed you from the group" - cath <## "use /d #team to delete the group" - ] - bob #> "#team hi" - concurrently_ - (alice <# "#team bob> hi") - (cath "#team hello" - concurrently_ - (bob <# "#team alice> hello") - (cath "#team hello" - cath <## "you are no longer a member of the group" - bob <##> cath - -- delete contact - alice ##> "/d bob" - alice <## "bob: contact is deleted" - alice ##> "@bob hey" - alice <## "no contact bob" - when checkMessages $ threadDelay 1000000 - alice #> "#team checking connection" - bob <# "#team alice> checking connection" - when checkMessages $ threadDelay 1000000 - bob #> "#team received" - alice <# "#team bob> received" - when checkMessages $ do - alice @@@ [("@cath", "sent invitation to join group team as admin"), ("#team", "received")] - bob @@@ [("@alice", "received invitation to join group team as admin"), ("@cath", "hey"), ("#team", "received")] - -- test clearing chat - alice #$> ("/clear #team", id, "#team: all messages are removed locally ONLY") - alice #$> ("/_get chat #1 count=100", chat, []) - bob #$> ("/clear #team", id, "#team: all messages are removed locally ONLY") - bob #$> ("/_get chat #1 count=100", chat, []) - cath #$> ("/clear #team", id, "#team: all messages are removed locally ONLY") - cath #$> ("/_get chat #1 count=100", chat, []) - where - getReadChats :: HasCallStack => String -> String -> IO () - getReadChats msgItem1 msgItem2 = do - alice @@@ [("#team", "hey team"), ("@cath", "sent invitation to join group team as admin"), ("@bob", "sent invitation to join group team as admin")] - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (0, "connected"), (1, "hello"), (0, "hi there"), (0, "hey team")]) - -- "before" and "after" define a chat item id across all chats, - -- so we take into account group event items as well as sent group invitations in direct chats - alice #$> ("/_get chat #1 after=" <> msgItem1 <> " count=100", chat, [(0, "hi there"), (0, "hey team")]) - alice #$> ("/_get chat #1 before=" <> msgItem2 <> " count=100", chat, [(0, "connected"), (0, "connected"), (1, "hello"), (0, "hi there")]) - alice #$> ("/_get chat #1 count=100 search=team", chat, [(0, "hey team")]) - bob @@@ [("@cath", "hey"), ("#team", "hey team"), ("@alice", "received invitation to join group team as admin")] - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "added cath (Catherine)"), (0, "connected"), (0, "hello"), (1, "hi there"), (0, "hey team")]) - cath @@@ [("@bob", "hey"), ("#team", "hey team"), ("@alice", "received invitation to join group team as admin")] - cath #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "connected"), (0, "hello"), (0, "hi there"), (1, "hey team")]) - alice #$> ("/_read chat #1 from=1 to=100", id, "ok") - bob #$> ("/_read chat #1 from=1 to=100", id, "ok") - cath #$> ("/_read chat #1 from=1 to=100", id, "ok") - alice #$> ("/_read chat #1", id, "ok") - bob #$> ("/_read chat #1", id, "ok") - cath #$> ("/_read chat #1", id, "ok") - alice #$> ("/_unread chat #1 on", id, "ok") - alice #$> ("/_unread chat #1 off", id, "ok") - -testGroup2 :: HasCallStack => FilePath -> IO () -testGroup2 = - testChat4 aliceProfile bobProfile cathProfile danProfile $ - \alice bob cath dan -> do - connectUsers alice bob - connectUsers alice cath - connectUsers bob dan - connectUsers alice dan - alice ##> "/g club" - alice <## "group #club is created" - alice <## "to add members use /a club or /create link #club" - alice ##> "/a club bob" - concurrentlyN_ - [ alice <## "invitation to join the group #club sent to bob", - do - bob <## "#club: alice invites you to join the group as admin" - bob <## "use /j club to accept" - ] - alice ##> "/a club cath" - concurrentlyN_ - [ alice <## "invitation to join the group #club sent to cath", - do - cath <## "#club: alice invites you to join the group as admin" - cath <## "use /j club to accept" - ] - bob ##> "/j club" - concurrently_ - (alice <## "#club: bob joined the group") - (bob <## "#club: you joined the group") - cath ##> "/j club" - concurrentlyN_ - [ alice <## "#club: cath joined the group", - do - cath <## "#club: you joined the group" - cath <## "#club: member bob (Bob) is connected", - do - bob <## "#club: alice added cath (Catherine) to the group (connecting...)" - bob <## "#club: new member cath is connected" - ] - bob ##> "/a club dan" - concurrentlyN_ - [ bob <## "invitation to join the group #club sent to dan", - do - dan <## "#club: bob invites you to join the group as admin" - dan <## "use /j club to accept" - ] - dan ##> "/j club" - concurrentlyN_ - [ bob <## "#club: dan joined the group", - do - dan <## "#club: you joined the group" - dan - <### [ "#club: member alice_1 (Alice) is connected", - "contact alice_1 is merged into alice", - "use @alice to send messages", - "#club: member cath (Catherine) is connected" - ], - do - alice <## "#club: bob added dan_1 (Daniel) to the group (connecting...)" - alice <## "#club: new member dan_1 is connected" - alice <## "contact dan_1 is merged into dan" - alice <## "use @dan to send messages", - do - cath <## "#club: bob added dan (Daniel) to the group (connecting...)" - cath <## "#club: new member dan is connected" - ] - alice #> "#club hello" - concurrentlyN_ - [ bob <# "#club alice> hello", - cath <# "#club alice> hello", - dan <# "#club alice> hello" - ] - bob #> "#club hi there" - concurrentlyN_ - [ alice <# "#club bob> hi there", - cath <# "#club bob> hi there", - dan <# "#club bob> hi there" - ] - cath #> "#club hey" - concurrentlyN_ - [ alice <# "#club cath> hey", - bob <# "#club cath> hey", - dan <# "#club cath> hey" - ] - dan #> "#club how is it going?" - concurrentlyN_ - [ alice <# "#club dan> how is it going?", - bob <# "#club dan> how is it going?", - cath <# "#club dan> how is it going?" - ] - bob <##> cath - dan <##> cath - dan <##> alice - -- show last messages - alice ##> "/t #club 8" - alice -- these strings are expected in any order because of sorting by time and rounding of time for sent - <##? [ "#club bob> connected", - "#club cath> connected", - "#club bob> added dan (Daniel)", - "#club dan> connected", - "#club hello", - "#club bob> hi there", - "#club cath> hey", - "#club dan> how is it going?" - ] - alice ##> "/t @dan 2" - alice - <##? [ "dan> hi", - "@dan hey" - ] - alice ##> "/t 21" - alice - <##? [ "@bob sent invitation to join group club as admin", - "@cath sent invitation to join group club as admin", - "#club bob> connected", - "#club cath> connected", - "#club bob> added dan (Daniel)", - "#club dan> connected", - "#club hello", - "#club bob> hi there", - "#club cath> hey", - "#club dan> how is it going?", - "dan> hi", - "@dan hey", - "dan> Disappearing messages: off", - "dan> Full deletion: off", - "dan> Voice messages: enabled", - "bob> Disappearing messages: off", - "bob> Full deletion: off", - "bob> Voice messages: enabled", - "cath> Disappearing messages: off", - "cath> Full deletion: off", - "cath> Voice messages: enabled" - ] - -- remove member - cath ##> "/rm club dan" - concurrentlyN_ - [ cath <## "#club: you removed dan from the group", - alice <## "#club: cath removed dan from the group", - bob <## "#club: cath removed dan from the group", - do - dan <## "#club: cath removed you from the group" - dan <## "use /d #club to delete the group" - ] - alice #> "#club hello" - concurrentlyN_ - [ bob <# "#club alice> hello", - cath <# "#club alice> hello", - (dan "#club hi there" - concurrentlyN_ - [ alice <# "#club bob> hi there", - cath <# "#club bob> hi there", - (dan "#club hey" - concurrentlyN_ - [ alice <# "#club cath> hey", - bob <# "#club cath> hey", - (dan "#club how is it going?" - dan <## "you are no longer a member of the group" - dan ##> "/d #club" - dan <## "#club: you deleted the group" - dan <##> cath - dan <##> alice - -- member leaves - bob ##> "/l club" - concurrentlyN_ - [ do - bob <## "#club: you left the group" - bob <## "use /d #club to delete the group", - alice <## "#club: bob left the group", - cath <## "#club: bob left the group" - ] - alice #> "#club hello" - concurrently_ - (cath <# "#club alice> hello") - (bob "#club hey" - concurrently_ - (alice <# "#club cath> hey") - (bob "#club how is it going?" - bob <## "you are no longer a member of the group" - bob ##> "/d #club" - bob <## "#club: you deleted the group" - bob <##> cath - bob <##> alice - -testGroupDelete :: HasCallStack => FilePath -> IO () -testGroupDelete = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - alice ##> "/d #team" - concurrentlyN_ - [ alice <## "#team: you deleted the group", - do - bob <## "#team: alice deleted the group" - bob <## "use /d #team to delete the local copy of the group", - do - cath <## "#team: alice deleted the group" - cath <## "use /d #team to delete the local copy of the group" - ] - alice ##> "#team hi" - alice <## "no group #team" - bob ##> "/d #team" - bob <## "#team: you deleted the group" - cath ##> "#team hi" - cath <## "you are no longer a member of the group" - cath ##> "/d #team" - cath <## "#team: you deleted the group" - alice <##> bob - alice <##> cath - -- unused group contacts are deleted - bob ##> "@cath hi" - bob <## "no contact cath" - (cath "@bob hi" - cath <## "no contact bob" - (bob FilePath -> IO () -testGroupSameName = - testChat2 aliceProfile bobProfile $ - \alice _ -> do - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/g team" - alice <## "group #team_1 (team) is created" - alice <## "to add members use /a team_1 or /create link #team_1" - -testGroupDeleteWhenInvited :: HasCallStack => FilePath -> IO () -testGroupDeleteWhenInvited = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as admin" - bob <## "use /j team to accept" - ] - bob ##> "/d #team" - bob <## "#team: you deleted the group" - -- alice doesn't receive notification that bob deleted group, - -- but she can re-add bob - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as admin" - bob <## "use /j team to accept" - ] - -testGroupReAddInvited :: HasCallStack => FilePath -> IO () -testGroupReAddInvited = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as admin" - bob <## "use /j team to accept" - ] - -- alice re-adds bob, he sees it as the same group - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as admin" - bob <## "use /j team to accept" - ] - -- if alice removes bob and then re-adds him, she uses a new connection request - -- and he sees it as a new group with a different local display name - alice ##> "/rm team bob" - alice <## "#team: you removed bob from the group" - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team_1 (team): alice invites you to join the group as admin" - bob <## "use /j team_1 to accept" - ] - -testGroupReAddInvitedChangeRole :: HasCallStack => FilePath -> IO () -testGroupReAddInvitedChangeRole = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as admin" - bob <## "use /j team to accept" - ] - -- alice re-adds bob, he sees it as the same group - alice ##> "/a team bob owner" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as owner" - bob <## "use /j team to accept" - ] - -- bob joins as owner - bob ##> "/j team" - concurrently_ - (alice <## "#team: bob joined the group") - (bob <## "#team: you joined the group") - bob ##> "/d #team" - concurrentlyN_ - [ bob <## "#team: you deleted the group", - do - alice <## "#team: bob deleted the group" - alice <## "use /d #team to delete the local copy of the group" - ] - bob ##> "#team hi" - bob <## "no group #team" - alice ##> "/d #team" - alice <## "#team: you deleted the group" - -testGroupDeleteInvitedContact :: HasCallStack => FilePath -> IO () -testGroupDeleteInvitedContact = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as admin" - bob <## "use /j team to accept" - ] - alice ##> "/d bob" - alice <## "bob: contact is deleted" - bob ##> "/j team" - concurrently_ - (alice <## "#team: bob joined the group") - (bob <## "#team: you joined the group") - alice #> "#team hello" - bob <# "#team alice> hello" - bob #> "#team hi there" - alice <# "#team bob> hi there" - alice ##> "@bob hey" - alice <## "no contact bob" - bob #> "@alice hey" - bob <## "[alice, contactId: 2, connId: 1] error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" - (alice FilePath -> IO () -testDeleteGroupMemberProfileKept = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - -- group 1 - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as admin" - bob <## "use /j team to accept" - ] - bob ##> "/j team" - concurrently_ - (alice <## "#team: bob joined the group") - (bob <## "#team: you joined the group") - alice #> "#team hello" - bob <# "#team alice> hello" - bob #> "#team hi there" - alice <# "#team bob> hi there" - -- group 2 - alice ##> "/g club" - alice <## "group #club is created" - alice <## "to add members use /a club or /create link #club" - alice ##> "/a club bob" - concurrentlyN_ - [ alice <## "invitation to join the group #club sent to bob", - do - bob <## "#club: alice invites you to join the group as admin" - bob <## "use /j club to accept" - ] - bob ##> "/j club" - concurrently_ - (alice <## "#club: bob joined the group") - (bob <## "#club: you joined the group") - alice #> "#club hello" - bob <# "#club alice> hello" - bob #> "#club hi there" - alice <# "#club bob> hi there" - -- delete contact - alice ##> "/d bob" - alice <## "bob: contact is deleted" - alice ##> "@bob hey" - alice <## "no contact bob" - bob #> "@alice hey" - bob <## "[alice, contactId: 2, connId: 1] error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" - (alice "/d #team" - concurrentlyN_ - [ alice <## "#team: you deleted the group", - do - bob <## "#team: alice deleted the group" - bob <## "use /d #team to delete the local copy of the group" - ] - alice ##> "#team hi" - alice <## "no group #team" - bob ##> "/d #team" - bob <## "#team: you deleted the group" - -- group 2 still works - alice #> "#club checking connection" - bob <# "#club alice> checking connection" - bob #> "#club received" - alice <# "#club bob> received" - -testGroupRemoveAdd :: HasCallStack => FilePath -> IO () -testGroupRemoveAdd = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - -- remove member - alice ##> "/rm team bob" - concurrentlyN_ - [ alice <## "#team: you removed bob from the group", - do - bob <## "#team: alice removed you from the group" - bob <## "use /d #team to delete the group", - cath <## "#team: alice removed bob from the group" - ] - alice ##> "/a team bob" - alice <## "invitation to join the group #team sent to bob" - bob <## "#team_1 (team): alice invites you to join the group as admin" - bob <## "use /j team_1 to accept" - bob ##> "/j team_1" - concurrentlyN_ - [ alice <## "#team: bob joined the group", - do - bob <## "#team_1: you joined the group" - bob <## "#team_1: member cath_1 (Catherine) is connected" - bob <## "contact cath_1 is merged into cath" - bob <## "use @cath to send messages", - do - cath <## "#team: alice added bob_1 (Bob) to the group (connecting...)" - cath <## "#team: new member bob_1 is connected" - cath <## "contact bob_1 is merged into bob" - cath <## "use @bob to send messages" - ] - alice #> "#team hi" - concurrently_ - (bob <# "#team_1 alice> hi") - (cath <# "#team alice> hi") - bob #> "#team_1 hey" - concurrently_ - (alice <# "#team bob> hey") - (cath <# "#team bob> hey") - cath #> "#team hello" - concurrently_ - (alice <# "#team cath> hello") - (bob <# "#team_1 cath> hello") - -testGroupList :: HasCallStack => FilePath -> IO () -testGroupList = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - createGroup2 "team" alice bob - alice ##> "/g tennis" - alice <## "group #tennis is created" - alice <## "to add members use /a tennis or /create link #tennis" - alice ##> "/a tennis bob" - concurrentlyN_ - [ alice <## "invitation to join the group #tennis sent to bob", - do - bob <## "#tennis: alice invites you to join the group as admin" - bob <## "use /j tennis to accept" - ] - -- alice sees both groups - alice ##> "/gs" - alice <### ["#team", "#tennis"] - -- bob sees #tennis as invitation - bob ##> "/gs" - bob - <### [ "#team", - "#tennis - you are invited (/j tennis to join, /d #tennis to delete invitation)" - ] - -- after deleting invitation bob sees only one group - bob ##> "/d #tennis" - bob <## "#tennis: you deleted the group" - bob ##> "/gs" - bob <## "#team" - -testGroupMessageQuotedReply :: HasCallStack => FilePath -> IO () -testGroupMessageQuotedReply = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - threadDelay 1000000 - alice #> "#team hello! how are you?" - concurrently_ - (bob <# "#team alice> hello! how are you?") - (cath <# "#team alice> hello! how are you?") - threadDelay 1000000 - bob `send` "> #team @alice (hello) hello, all good, you?" - bob <# "#team > alice hello! how are you?" - bob <## " hello, all good, you?" - concurrently_ - ( do - alice <# "#team bob> > alice hello! how are you?" - alice <## " hello, all good, you?" - ) - ( do - cath <# "#team bob> > alice hello! how are you?" - cath <## " hello, all good, you?" - ) - bob #$> ("/_get chat #1 count=2", chat', [((0, "hello! how are you?"), Nothing), ((1, "hello, all good, you?"), Just (0, "hello! how are you?"))]) - alice #$> ("/_get chat #1 count=2", chat', [((1, "hello! how are you?"), Nothing), ((0, "hello, all good, you?"), Just (1, "hello! how are you?"))]) - cath #$> ("/_get chat #1 count=2", chat', [((0, "hello! how are you?"), Nothing), ((0, "hello, all good, you?"), Just (0, "hello! how are you?"))]) - bob `send` "> #team bob (hello, all good) will tell more" - bob <# "#team > bob hello, all good, you?" - bob <## " will tell more" - concurrently_ - ( do - alice <# "#team bob> > bob hello, all good, you?" - alice <## " will tell more" - ) - ( do - cath <# "#team bob> > bob hello, all good, you?" - cath <## " will tell more" - ) - bob #$> ("/_get chat #1 count=1", chat', [((1, "will tell more"), Just (1, "hello, all good, you?"))]) - alice #$> ("/_get chat #1 count=1", chat', [((0, "will tell more"), Just (0, "hello, all good, you?"))]) - cath #$> ("/_get chat #1 count=1", chat', [((0, "will tell more"), Just (0, "hello, all good, you?"))]) - threadDelay 1000000 - cath `send` "> #team bob (hello) hi there!" - cath <# "#team > bob hello, all good, you?" - cath <## " hi there!" - concurrently_ - ( do - alice <# "#team cath> > bob hello, all good, you?" - alice <## " hi there!" - ) - ( do - bob <# "#team cath> > bob hello, all good, you?" - bob <## " hi there!" - ) - cath #$> ("/_get chat #1 count=1", chat', [((1, "hi there!"), Just (0, "hello, all good, you?"))]) - alice #$> ("/_get chat #1 count=1", chat', [((0, "hi there!"), Just (0, "hello, all good, you?"))]) - bob #$> ("/_get chat #1 count=1", chat', [((0, "hi there!"), Just (1, "hello, all good, you?"))]) - alice `send` "> #team (will tell) go on" - alice <# "#team > bob will tell more" - alice <## " go on" - concurrently_ - ( do - bob <# "#team alice> > bob will tell more" - bob <## " go on" - ) - ( do - cath <# "#team alice> > bob will tell more" - cath <## " go on" - ) - -testGroupMessageUpdate :: HasCallStack => FilePath -> IO () -testGroupMessageUpdate = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - threadDelay 1000000 - -- alice, bob: msg id 5, cath: msg id 4 (after group invitations & group events) - alice #> "#team hello!" - concurrently_ - (bob <# "#team alice> hello!") - (cath <# "#team alice> hello!") - - msgItemId1 <- lastItemId alice - alice ##> ("/_update item #1 " <> msgItemId1 <> " text hey 👋") - alice <# "#team [edited] hey 👋" - concurrently_ - (bob <# "#team alice> [edited] hey 👋") - (cath <# "#team alice> [edited] hey 👋") - - alice #$> ("/_get chat #1 count=1", chat', [((1, "hey 👋"), Nothing)]) - bob #$> ("/_get chat #1 count=1", chat', [((0, "hey 👋"), Nothing)]) - cath #$> ("/_get chat #1 count=1", chat', [((0, "hey 👋"), Nothing)]) - - threadDelay 1000000 - -- alice, bob: msg id 6, cath: msg id 5 - bob `send` "> #team @alice (hey) hi alice" - bob <# "#team > alice hey 👋" - bob <## " hi alice" - concurrently_ - ( do - alice <# "#team bob> > alice hey 👋" - alice <## " hi alice" - ) - ( do - cath <# "#team bob> > alice hey 👋" - cath <## " hi alice" - ) - - alice #$> ("/_get chat #1 count=2", chat', [((1, "hey 👋"), Nothing), ((0, "hi alice"), Just (1, "hey 👋"))]) - bob #$> ("/_get chat #1 count=2", chat', [((0, "hey 👋"), Nothing), ((1, "hi alice"), Just (0, "hey 👋"))]) - cath #$> ("/_get chat #1 count=2", chat', [((0, "hey 👋"), Nothing), ((0, "hi alice"), Just (0, "hey 👋"))]) - - alice ##> ("/_update item #1 " <> msgItemId1 <> " text greetings 🤝") - alice <# "#team [edited] greetings 🤝" - concurrently_ - (bob <# "#team alice> [edited] greetings 🤝") - (cath <# "#team alice> [edited] greetings 🤝") - - msgItemId2 <- lastItemId alice - alice #$> ("/_update item #1 " <> msgItemId2 <> " text updating bob's message", id, "cannot update this item") - - threadDelay 1000000 - cath `send` "> #team @alice (greetings) greetings!" - cath <# "#team > alice greetings 🤝" - cath <## " greetings!" - concurrently_ - ( do - alice <# "#team cath> > alice greetings 🤝" - alice <## " greetings!" - ) - ( do - bob <# "#team cath> > alice greetings 🤝" - bob <## " greetings!" - ) - - alice #$> ("/_get chat #1 count=3", chat', [((1, "greetings 🤝"), Nothing), ((0, "hi alice"), Just (1, "hey 👋")), ((0, "greetings!"), Just (1, "greetings 🤝"))]) - bob #$> ("/_get chat #1 count=3", chat', [((0, "greetings 🤝"), Nothing), ((1, "hi alice"), Just (0, "hey 👋")), ((0, "greetings!"), Just (0, "greetings 🤝"))]) - cath #$> ("/_get chat #1 count=3", chat', [((0, "greetings 🤝"), Nothing), ((0, "hi alice"), Just (0, "hey 👋")), ((1, "greetings!"), Just (0, "greetings 🤝"))]) - -testGroupMessageDelete :: HasCallStack => FilePath -> IO () -testGroupMessageDelete = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - threadDelay 1000000 - -- alice, bob: msg id 5, cath: msg id 4 (after group invitations & group events) - alice #> "#team hello!" - concurrently_ - (bob <# "#team alice> hello!") - (cath <# "#team alice> hello!") - - msgItemId1 <- lastItemId alice - alice #$> ("/_delete item #1 " <> msgItemId1 <> " internal", id, "message deleted") - - alice #$> ("/_get chat #1 count=1", chat, [(0, "connected")]) - bob #$> ("/_get chat #1 count=1", chat, [(0, "hello!")]) - cath #$> ("/_get chat #1 count=1", chat, [(0, "hello!")]) - - threadDelay 1000000 - -- alice: msg id 5, bob: msg id 6, cath: msg id 5 - bob `send` "> #team @alice (hello) hi alic" - bob <# "#team > alice hello!" - bob <## " hi alic" - concurrently_ - ( do - alice <# "#team bob> > alice hello!" - alice <## " hi alic" - ) - ( do - cath <# "#team bob> > alice hello!" - cath <## " hi alic" - ) - - alice #$> ("/_get chat #1 count=1", chat', [((0, "hi alic"), Just (1, "hello!"))]) - bob #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((1, "hi alic"), Just (0, "hello!"))]) - cath #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((0, "hi alic"), Just (0, "hello!"))]) - - msgItemId2 <- lastItemId alice - alice #$> ("/_delete item #1 " <> msgItemId2 <> " internal", id, "message deleted") - - alice #$> ("/_get chat #1 count=1", chat', [((0, "connected"), Nothing)]) - bob #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((1, "hi alic"), Just (0, "hello!"))]) - cath #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((0, "hi alic"), Just (0, "hello!"))]) - - -- alice: msg id 5 - msgItemId3 <- lastItemId bob - bob ##> ("/_update item #1 " <> msgItemId3 <> " text hi alice") - bob <# "#team [edited] > alice hello!" - bob <## " hi alice" - concurrently_ - (alice <# "#team bob> [edited] hi alice") - ( do - cath <# "#team bob> [edited] > alice hello!" - cath <## " hi alice" - ) - - alice #$> ("/_get chat #1 count=1", chat', [((0, "hi alice"), Nothing)]) - bob #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((1, "hi alice"), Just (0, "hello!"))]) - cath #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((0, "hi alice"), Just (0, "hello!"))]) - - threadDelay 1000000 - -- alice: msg id 6, bob: msg id 7, cath: msg id 6 - cath #> "#team how are you?" - concurrently_ - (alice <# "#team cath> how are you?") - (bob <# "#team cath> how are you?") - - msgItemId4 <- lastItemId cath - cath #$> ("/_delete item #1 " <> msgItemId4 <> " broadcast", id, "message marked deleted") - concurrently_ - (alice <# "#team cath> [marked deleted] how are you?") - (bob <# "#team cath> [marked deleted] how are you?") - - alice ##> "/last_item_id 1" - msgItemId6 <- getTermLine alice - alice #$> ("/_delete item #1 " <> msgItemId6 <> " broadcast", id, "cannot delete this item") - alice #$> ("/_delete item #1 " <> msgItemId6 <> " internal", id, "message deleted") - - alice #$> ("/_get chat #1 count=1", chat', [((0, "how are you? [marked deleted]"), Nothing)]) - bob #$> ("/_get chat #1 count=3", chat', [((0, "hello!"), Nothing), ((1, "hi alice"), Just (0, "hello!")), ((0, "how are you? [marked deleted]"), Nothing)]) - cath #$> ("/_get chat #1 count=3", chat', [((0, "hello!"), Nothing), ((0, "hi alice"), Just (0, "hello!")), ((1, "how are you? [marked deleted]"), Nothing)]) - -testGroupLiveMessage :: HasCallStack => FilePath -> IO () -testGroupLiveMessage = - testChat3 aliceProfile bobProfile cathProfile $ \alice bob cath -> do - createGroup3 "team" alice bob cath - threadDelay 500000 - -- non-empty live message is sent instantly - alice `send` "/live #team hello" - msgItemId1 <- lastItemId alice - bob <#. "#team alice> [LIVE started]" - cath <#. "#team alice> [LIVE started]" - alice ##> ("/_update item #1 " <> msgItemId1 <> " text hello there") - alice <# "#team [LIVE] hello there" - bob <# "#team alice> [LIVE ended] hello there" - cath <# "#team alice> [LIVE ended] hello there" - -- empty live message is also sent instantly - alice `send` "/live #team" - msgItemId2 <- lastItemId alice - bob <#. "#team alice> [LIVE started]" - cath <#. "#team alice> [LIVE started]" - alice ##> ("/_update item #1 " <> msgItemId2 <> " text hello 2") - alice <# "#team [LIVE] hello 2" - bob <# "#team alice> [LIVE ended] hello 2" - cath <# "#team alice> [LIVE ended] hello 2" - -testUpdateGroupProfile :: HasCallStack => FilePath -> IO () -testUpdateGroupProfile = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - threadDelay 1000000 - alice #> "#team hello!" - concurrently_ - (bob <# "#team alice> hello!") - (cath <# "#team alice> hello!") - bob ##> "/gp team my_team" - bob <## "you have insufficient permissions for this action, the required role is owner" - alice ##> "/gp team my_team" - alice <## "changed to #my_team" - concurrentlyN_ - [ do - bob <## "alice updated group #team:" - bob <## "changed to #my_team", - do - cath <## "alice updated group #team:" - cath <## "changed to #my_team" - ] - bob #> "#my_team hi" - concurrently_ - (alice <# "#my_team bob> hi") - (cath <# "#my_team bob> hi") - -testUpdateMemberRole :: HasCallStack => FilePath -> IO () -testUpdateMemberRole = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - connectUsers alice bob - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - addMember "team" alice bob GRAdmin - alice ##> "/mr team bob member" - alice <## "#team: you changed the role of bob from admin to member" - bob <## "#team: alice invites you to join the group as member" - bob <## "use /j team to accept" - bob ##> "/j team" - concurrently_ - (alice <## "#team: bob joined the group") - (bob <## "#team: you joined the group") - connectUsers bob cath - bob ##> "/a team cath" - bob <## "you have insufficient permissions for this action, the required role is admin" - alice ##> "/mr team bob admin" - concurrently_ - (alice <## "#team: you changed the role of bob from member to admin") - (bob <## "#team: alice changed your role from member to admin") - bob ##> "/a team cath owner" - bob <## "you have insufficient permissions for this action, the required role is owner" - addMember "team" bob cath GRMember - cath ##> "/j team" - concurrentlyN_ - [ bob <## "#team: cath joined the group", - do - cath <## "#team: you joined the group" - cath <## "#team: member alice (Alice) is connected", - do - alice <## "#team: bob added cath (Catherine) to the group (connecting...)" - alice <## "#team: new member cath is connected" - ] - alice ##> "/mr team alice admin" - concurrentlyN_ - [ alice <## "#team: you changed your role from owner to admin", - bob <## "#team: alice changed the role from owner to admin", - cath <## "#team: alice changed the role from owner to admin" - ] - alice ##> "/d #team" - alice <## "you have insufficient permissions for this action, the required role is owner" - -testGroupDeleteUnusedContacts :: HasCallStack => FilePath -> IO () -testGroupDeleteUnusedContacts = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - -- create group 1 - createGroup3 "team" alice bob cath - -- create group 2 - alice ##> "/g club" - alice <## "group #club is created" - alice <## "to add members use /a club or /create link #club" - alice ##> "/a club bob" - concurrentlyN_ - [ alice <## "invitation to join the group #club sent to bob", - do - bob <## "#club: alice invites you to join the group as admin" - bob <## "use /j club to accept" - ] - bob ##> "/j club" - concurrently_ - (alice <## "#club: bob joined the group") - (bob <## "#club: you joined the group") - alice ##> "/a club cath" - concurrentlyN_ - [ alice <## "invitation to join the group #club sent to cath", - do - cath <## "#club: alice invites you to join the group as admin" - cath <## "use /j club to accept" - ] - cath ##> "/j club" - concurrentlyN_ - [ alice <## "#club: cath joined the group", - do - cath <## "#club: you joined the group" - cath <## "#club: member bob_1 (Bob) is connected" - cath <## "contact bob_1 is merged into bob" - cath <## "use @bob to send messages", - do - bob <## "#club: alice added cath_1 (Catherine) to the group (connecting...)" - bob <## "#club: new member cath_1 is connected" - bob <## "contact cath_1 is merged into cath" - bob <## "use @cath to send messages" - ] - -- list contacts - bob ##> "/contacts" - bob <## "alice (Alice)" - bob <## "cath (Catherine)" - cath ##> "/contacts" - cath <## "alice (Alice)" - cath <## "bob (Bob)" - -- delete group 1, contacts and profiles are kept - deleteGroup alice bob cath "team" - bob ##> "/contacts" - bob <## "alice (Alice)" - bob <## "cath (Catherine)" - bob `hasContactProfiles` ["alice", "bob", "cath"] - cath ##> "/contacts" - cath <## "alice (Alice)" - cath <## "bob (Bob)" - cath `hasContactProfiles` ["alice", "bob", "cath"] - -- delete group 2, unused contacts and profiles are deleted - deleteGroup alice bob cath "club" - bob ##> "/contacts" - bob <## "alice (Alice)" - bob `hasContactProfiles` ["alice", "bob"] - cath ##> "/contacts" - cath <## "alice (Alice)" - cath `hasContactProfiles` ["alice", "cath"] - where - deleteGroup :: HasCallStack => TestCC -> TestCC -> TestCC -> String -> IO () - deleteGroup alice bob cath group = do - alice ##> ("/d #" <> group) - concurrentlyN_ - [ alice <## ("#" <> group <> ": you deleted the group"), - do - bob <## ("#" <> group <> ": alice deleted the group") - bob <## ("use /d #" <> group <> " to delete the local copy of the group"), - do - cath <## ("#" <> group <> ": alice deleted the group") - cath <## ("use /d #" <> group <> " to delete the local copy of the group") - ] - bob ##> ("/d #" <> group) - bob <## ("#" <> group <> ": you deleted the group") - cath ##> ("/d #" <> group) - cath <## ("#" <> group <> ": you deleted the group") - -testGroupDescription :: HasCallStack => FilePath -> IO () -testGroupDescription = testChat4 aliceProfile bobProfile cathProfile danProfile $ \alice bob cath dan -> do - connectUsers alice bob - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - addMember "team" alice bob GRAdmin - bob ##> "/j team" - concurrentlyN_ - [ alice <## "#team: bob joined the group", - bob <## "#team: you joined the group" - ] - alice ##> "/group_profile team" - alice <## "#team" - groupInfo alice - alice ##> "/group_descr team Welcome to the team!" - alice <## "description changed to:" - alice <## "Welcome to the team!" - bob <## "alice updated group #team:" - bob <## "description changed to:" - bob <## "Welcome to the team!" - alice ##> "/group_profile team" - alice <## "#team" - alice <## "description:" - alice <## "Welcome to the team!" - groupInfo alice - connectUsers alice cath - addMember "team" alice cath GRMember - cath ##> "/j team" - concurrentlyN_ - [ alice <## "#team: cath joined the group", - do - cath <## "#team: you joined the group" - cath <# "#team alice> Welcome to the team!" - cath <## "#team: member bob (Bob) is connected", - do - bob <## "#team: alice added cath (Catherine) to the group (connecting...)" - bob <## "#team: new member cath is connected" - ] - connectUsers bob dan - addMember "team" bob dan GRMember - dan ##> "/j team" - concurrentlyN_ - [ bob <## "#team: dan joined the group", - do - dan <## "#team: you joined the group" - dan <# "#team bob> Welcome to the team!" - dan - <### [ "#team: member alice (Alice) is connected", - "#team: member cath (Catherine) is connected" - ], - bobAddedDan alice, - bobAddedDan cath - ] - where - groupInfo :: HasCallStack => TestCC -> IO () - groupInfo alice = do - alice <## "group preferences:" - alice <## "Disappearing messages: off" - alice <## "Direct messages: on" - alice <## "Full deletion: off" - alice <## "Voice messages: on" - bobAddedDan :: HasCallStack => TestCC -> IO () - bobAddedDan cc = do - cc <## "#team: bob added dan (Daniel) to the group (connecting...)" - cc <## "#team: new member dan is connected" - -testGroupAsync :: HasCallStack => FilePath -> IO () -testGroupAsync tmp = do - print (0 :: Integer) - withNewTestChat tmp "alice" aliceProfile $ \alice -> do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - connectUsers alice bob - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/a team bob" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to bob", - do - bob <## "#team: alice invites you to join the group as admin" - bob <## "use /j team to accept" - ] - bob ##> "/j team" - concurrently_ - (alice <## "#team: bob joined the group") - (bob <## "#team: you joined the group") - alice #> "#team hello bob" - bob <# "#team alice> hello bob" - print (1 :: Integer) - withTestChat tmp "alice" $ \alice -> do - withNewTestChat tmp "cath" cathProfile $ \cath -> do - alice <## "1 contacts connected (use /cs for the list)" - alice <## "#team: connected to server(s)" - connectUsers alice cath - alice ##> "/a team cath" - concurrentlyN_ - [ alice <## "invitation to join the group #team sent to cath", - do - cath <## "#team: alice invites you to join the group as admin" - cath <## "use /j team to accept" - ] - cath ##> "/j team" - concurrentlyN_ - [ alice <## "#team: cath joined the group", - cath <## "#team: you joined the group" - ] - alice #> "#team hello cath" - cath <# "#team alice> hello cath" - print (2 :: Integer) - withTestChat tmp "bob" $ \bob -> do - withTestChat tmp "cath" $ \cath -> do - concurrentlyN_ - [ do - bob <## "1 contacts connected (use /cs for the list)" - bob <## "#team: connected to server(s)" - bob <## "#team: alice added cath (Catherine) to the group (connecting...)" - bob <# "#team alice> hello cath" - bob <## "#team: new member cath is connected", - do - cath <## "2 contacts connected (use /cs for the list)" - cath <## "#team: connected to server(s)" - cath <## "#team: member bob (Bob) is connected" - ] - threadDelay 500000 - print (3 :: Integer) - withTestChat tmp "bob" $ \bob -> do - withNewTestChat tmp "dan" danProfile $ \dan -> do - bob <## "2 contacts connected (use /cs for the list)" - bob <## "#team: connected to server(s)" - connectUsers bob dan - bob ##> "/a team dan" - concurrentlyN_ - [ bob <## "invitation to join the group #team sent to dan", - do - dan <## "#team: bob invites you to join the group as admin" - dan <## "use /j team to accept" - ] - dan ##> "/j team" - concurrentlyN_ - [ bob <## "#team: dan joined the group", - dan <## "#team: you joined the group" - ] - threadDelay 1000000 - threadDelay 1000000 - print (4 :: Integer) - withTestChat tmp "alice" $ \alice -> do - withTestChat tmp "cath" $ \cath -> do - withTestChat tmp "dan" $ \dan -> do - concurrentlyN_ - [ do - alice <## "2 contacts connected (use /cs for the list)" - alice <## "#team: connected to server(s)" - alice <## "#team: bob added dan (Daniel) to the group (connecting...)" - alice <## "#team: new member dan is connected", - do - cath <## "2 contacts connected (use /cs for the list)" - cath <## "#team: connected to server(s)" - cath <## "#team: bob added dan (Daniel) to the group (connecting...)" - cath <## "#team: new member dan is connected", - do - dan <## "3 contacts connected (use /cs for the list)" - dan <## "#team: connected to server(s)" - dan <## "#team: member alice (Alice) is connected" - dan <## "#team: member cath (Catherine) is connected" - ] - threadDelay 1000000 - print (5 :: Integer) - withTestChat tmp "alice" $ \alice -> do - withTestChat tmp "bob" $ \bob -> do - withTestChat tmp "cath" $ \cath -> do - withTestChat tmp "dan" $ \dan -> do - concurrentlyN_ - [ do - alice <## "3 contacts connected (use /cs for the list)" - alice <## "#team: connected to server(s)", - do - bob <## "3 contacts connected (use /cs for the list)" - bob <## "#team: connected to server(s)", - do - cath <## "3 contacts connected (use /cs for the list)" - cath <## "#team: connected to server(s)", - do - dan <## "3 contacts connected (use /cs for the list)" - dan <## "#team: connected to server(s)" - ] - alice #> "#team hello" - concurrentlyN_ - [ bob <# "#team alice> hello", - cath <# "#team alice> hello", - dan <# "#team alice> hello" - ] - bob #> "#team hi there" - concurrentlyN_ - [ alice <# "#team bob> hi there", - cath <# "#team bob> hi there", - dan <# "#team bob> hi there" - ] - cath #> "#team hey" - concurrentlyN_ - [ alice <# "#team cath> hey", - bob <# "#team cath> hey", - dan <# "#team cath> hey" - ] - dan #> "#team how is it going?" - concurrentlyN_ - [ alice <# "#team dan> how is it going?", - bob <# "#team dan> how is it going?", - cath <# "#team dan> how is it going?" - ] - bob <##> cath - dan <##> cath - dan <##> alice - -testUpdateProfile :: HasCallStack => FilePath -> IO () -testUpdateProfile = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - alice ##> "/p" - alice <## "user profile: alice (Alice)" - alice <## "use /p [] to change it" - alice <## "(the updated profile will be sent to all your contacts)" - alice ##> "/p alice" - concurrentlyN_ - [ alice <## "user full name removed (your contacts are notified)", - bob <## "contact alice removed full name", - cath <## "contact alice removed full name" - ] - alice ##> "/p alice Alice Jones" - concurrentlyN_ - [ alice <## "user full name changed to Alice Jones (your contacts are notified)", - bob <## "contact alice updated full name: Alice Jones", - cath <## "contact alice updated full name: Alice Jones" - ] - cath ##> "/p cate" - concurrentlyN_ - [ cath <## "user profile is changed to cate (your contacts are notified)", - do - alice <## "contact cath changed to cate" - alice <## "use @cate to send messages", - do - bob <## "contact cath changed to cate" - bob <## "use @cate to send messages" - ] - cath ##> "/p cat Cate" - concurrentlyN_ - [ cath <## "user profile is changed to cat (Cate) (your contacts are notified)", - do - alice <## "contact cate changed to cat (Cate)" - alice <## "use @cat to send messages", - do - bob <## "contact cate changed to cat (Cate)" - bob <## "use @cat to send messages" - ] - -testUpdateProfileImage :: HasCallStack => FilePath -> IO () -testUpdateProfileImage = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice ##> "/profile_image " - alice <## "profile image updated" - alice ##> "/profile_image" - alice <## "profile image removed" - alice ##> "/_profile 1 {\"displayName\": \"alice2\", \"fullName\": \"\"}" - alice <## "user profile is changed to alice2 (your contacts are notified)" - bob <## "contact alice changed to alice2" - bob <## "use @alice2 to send messages" - (bob TestCC -> TestCC -> IO () -runTestFileTransfer alice bob = do - connectUsers alice bob - startFileTransfer' alice bob "test.pdf" "266.0 KiB / 272376 bytes" - concurrentlyN_ - [ do - bob #> "@alice receiving here..." - bob <## "completed receiving file 1 (test.pdf) from alice", - alice - <### [ WithTime "bob> receiving here...", - "completed sending file 1 (test.pdf) to bob" - ] - ] - src <- B.readFile "./tests/fixtures/test.pdf" - dest <- B.readFile "./tests/tmp/test.pdf" - dest `shouldBe` src - -testInlineFileTransfer :: HasCallStack => FilePath -> IO () -testInlineFileTransfer = - testChatCfg2 cfg aliceProfile bobProfile $ \alice bob -> do - connectUsers alice bob - bob ##> "/_files_folder ./tests/tmp/" - bob <## "ok" - alice ##> "/_send @2 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/test.jpg\"}" - alice <# "@bob voice message (00:10)" - alice <# "/f @bob ./tests/fixtures/test.jpg" - -- below is not shown in "sent" mode - -- alice <## "use /fc 1 to cancel sending" - bob <# "alice> voice message (00:10)" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - -- below is not shown in "sent" mode - -- bob <## "use /fr 1 [/ | ] to receive it" - bob <## "started receiving file 1 (test.jpg) from alice" - concurrently_ - (alice <## "completed sending file 1 (test.jpg) to bob") - (bob <## "completed receiving file 1 (test.jpg) from alice") - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/test.jpg" - dest `shouldBe` src - where - cfg = testCfg {inlineFiles = defaultInlineFilesConfig {offerChunks = 100, sendChunks = 100, receiveChunks = 100}} - -testAcceptInlineFileSndCancelDuringTransfer :: HasCallStack => FilePath -> IO () -testAcceptInlineFileSndCancelDuringTransfer = - testChatCfg2 cfg aliceProfile bobProfile $ \alice bob -> do - connectUsers alice bob - bob ##> "/_files_folder ./tests/tmp/" - bob <## "ok" - alice #> "/f @bob ./tests/fixtures/test_1MB.pdf" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> sends file test_1MB.pdf (1017.7 KiB / 1042157 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1 inline=on" - bob <## "saving file 1 from alice to test_1MB.pdf" - alice <## "started sending file 1 (test_1MB.pdf) to bob" - bob <## "started receiving file 1 (test_1MB.pdf) from alice" - alice ##> "/fc 1" -- test that inline file cancel doesn't delete contact connection - concurrentlyN_ - [ do - alice <##. "cancelled sending file 1 (test_1MB.pdf)" - alice <## "completed sending file 1 (test_1MB.pdf) to bob", - do - bob <## "completed receiving file 1 (test_1MB.pdf) from alice" - bob <## "alice cancelled sending file 1 (test_1MB.pdf)" - ] - alice #> "@bob hi" - bob <# "alice> hi" - bob #> "@alice hey" - alice <# "bob> hey" - where - cfg = testCfg {inlineFiles = defaultInlineFilesConfig {offerChunks = 100, receiveChunks = 50}} - -testSmallInlineFileTransfer :: HasCallStack => FilePath -> IO () -testSmallInlineFileTransfer = - testChat2 aliceProfile bobProfile $ \alice bob -> do - connectUsers alice bob - bob ##> "/_files_folder ./tests/tmp/" - bob <## "ok" - alice ##> "/_send @2 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" - alice <# "@bob voice message (00:10)" - alice <# "/f @bob ./tests/fixtures/logo.jpg" - -- below is not shown in "sent" mode - -- alice <## "use /fc 1 to cancel sending" - bob <# "alice> voice message (00:10)" - bob <# "alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" - -- below is not shown in "sent" mode - -- bob <## "use /fr 1 [/ | ] to receive it" - bob <## "started receiving file 1 (logo.jpg) from alice" - concurrently_ - (alice <## "completed sending file 1 (logo.jpg) to bob") - (bob <## "completed receiving file 1 (logo.jpg) from alice") - src <- B.readFile "./tests/fixtures/logo.jpg" - dest <- B.readFile "./tests/tmp/logo.jpg" - dest `shouldBe` src - -testSmallInlineFileIgnored :: HasCallStack => FilePath -> IO () -testSmallInlineFileIgnored tmp = do - withNewTestChat tmp "alice" aliceProfile $ \alice -> - withNewTestChatOpts tmp testOpts {allowInstantFiles = False} "bob" bobProfile $ \bob -> do - connectUsers alice bob - bob ##> "/_files_folder ./tests/tmp/" - bob <## "ok" - alice ##> "/_send @2 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" - alice <# "@bob voice message (00:10)" - alice <# "/f @bob ./tests/fixtures/logo.jpg" - -- below is not shown in "sent" mode - -- alice <## "use /fc 1 to cancel sending" - bob <# "alice> voice message (00:10)" - bob <# "alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob <## "A small file sent without acceptance - you can enable receiving such files with -f option." - -- below is not shown in "sent" mode - -- bob <## "use /fr 1 [/ | ] to receive it" - alice <## "completed sending file 1 (logo.jpg) to bob" - bob ##> "/fr 1" - bob <## "file is already being received: logo.jpg" - -testReceiveInline :: HasCallStack => FilePath -> IO () -testReceiveInline = - testChatCfg2 cfg aliceProfile bobProfile $ \alice bob -> do - connectUsers alice bob - alice #> "/f @bob ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1 inline=on ./tests/tmp" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - alice <## "started sending file 1 (test.jpg) to bob" - alice <## "completed sending file 1 (test.jpg) to bob" - bob <## "started receiving file 1 (test.jpg) from alice" - bob <## "completed receiving file 1 (test.jpg) from alice" - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/test.jpg" - dest `shouldBe` src - where - cfg = testCfg {inlineFiles = defaultInlineFilesConfig {offerChunks = 10, receiveChunks = 5}} - -runTestSmallFileTransfer :: HasCallStack => TestCC -> TestCC -> IO () -runTestSmallFileTransfer alice bob = do - connectUsers alice bob - alice #> "/f @bob ./tests/fixtures/test.txt" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> sends file test.txt (11 bytes / 11 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1 ./tests/tmp" - bob <## "saving file 1 from alice to ./tests/tmp/test.txt" - concurrentlyN_ - [ do - bob <## "started receiving file 1 (test.txt) from alice" - bob <## "completed receiving file 1 (test.txt) from alice", - do - alice <## "started sending file 1 (test.txt) to bob" - alice <## "completed sending file 1 (test.txt) to bob" - ] - src <- B.readFile "./tests/fixtures/test.txt" - dest <- B.readFile "./tests/tmp/test.txt" - dest `shouldBe` src - -runTestFileSndCancelBeforeTransfer :: HasCallStack => TestCC -> TestCC -> IO () -runTestFileSndCancelBeforeTransfer alice bob = do - connectUsers alice bob - alice #> "/f @bob ./tests/fixtures/test.txt" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> sends file test.txt (11 bytes / 11 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - alice ##> "/fc 1" - concurrentlyN_ - [ alice <##. "cancelled sending file 1 (test.txt)", - bob <## "alice cancelled sending file 1 (test.txt)" - ] - alice ##> "/fs 1" - alice - <##.. [ "sending file 1 (test.txt): no file transfers", - "sending file 1 (test.txt) cancelled: bob" - ] - alice <## "file transfer cancelled" - bob ##> "/fs 1" - bob <## "receiving file 1 (test.txt) cancelled" - bob ##> "/fr 1 ./tests/tmp" - bob <## "file cancelled: test.txt" - -testFileSndCancelDuringTransfer :: HasCallStack => FilePath -> IO () -testFileSndCancelDuringTransfer = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - startFileTransfer' alice bob "test_1MB.pdf" "1017.7 KiB / 1042157 bytes" - alice ##> "/fc 1" - concurrentlyN_ - [ do - alice <## "cancelled sending file 1 (test_1MB.pdf) to bob" - alice ##> "/fs 1" - alice <## "sending file 1 (test_1MB.pdf) cancelled: bob" - alice <## "file transfer cancelled", - do - bob <## "alice cancelled sending file 1 (test_1MB.pdf)" - bob ##> "/fs 1" - bob <## "receiving file 1 (test_1MB.pdf) cancelled, received part path: ./tests/tmp/test_1MB.pdf" - ] - checkPartialTransfer "test_1MB.pdf" - -testFileRcvCancel :: HasCallStack => FilePath -> IO () -testFileRcvCancel = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - startFileTransfer alice bob - bob ##> "/fs 1" - getTermLine bob >>= (`shouldStartWith` "receiving file 1 (test.jpg) progress") - waitFileExists "./tests/tmp/test.jpg" - bob ##> "/fc 1" - concurrentlyN_ - [ do - bob <## "cancelled receiving file 1 (test.jpg) from alice" - bob ##> "/fs 1" - bob <## "receiving file 1 (test.jpg) cancelled, received part path: ./tests/tmp/test.jpg", - do - alice <## "bob cancelled receiving file 1 (test.jpg)" - alice ##> "/fs 1" - alice <## "sending file 1 (test.jpg) cancelled: bob" - ] - checkPartialTransfer "test.jpg" - -runTestGroupFileTransfer :: HasCallStack => TestCC -> TestCC -> TestCC -> IO () -runTestGroupFileTransfer alice bob cath = do - createGroup3 "team" alice bob cath - alice #> "/f #team ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - concurrentlyN_ - [ do - bob <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it", - do - cath <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - cath <## "use /fr 1 [/ | ] to receive it" - ] - alice ##> "/fs 1" - getTermLine alice >>= (`shouldStartWith` "sending file 1 (test.jpg): no file transfers") - bob ##> "/fr 1 ./tests/tmp/" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - concurrentlyN_ - [ do - alice <## "started sending file 1 (test.jpg) to bob" - alice <## "completed sending file 1 (test.jpg) to bob" - alice ##> "/fs 1" - alice <## "sending file 1 (test.jpg) complete: bob", - do - bob <## "started receiving file 1 (test.jpg) from alice" - bob <## "completed receiving file 1 (test.jpg) from alice" - ] - cath ##> "/fr 1 ./tests/tmp/" - cath <## "saving file 1 from alice to ./tests/tmp/test_1.jpg" - concurrentlyN_ - [ do - alice <## "started sending file 1 (test.jpg) to cath" - alice <## "completed sending file 1 (test.jpg) to cath" - alice ##> "/fs 1" - getTermLine alice >>= (`shouldStartWith` "sending file 1 (test.jpg) complete"), - do - cath <## "started receiving file 1 (test.jpg) from alice" - cath <## "completed receiving file 1 (test.jpg) from alice" - ] - src <- B.readFile "./tests/fixtures/test.jpg" - dest1 <- B.readFile "./tests/tmp/test.jpg" - dest2 <- B.readFile "./tests/tmp/test_1.jpg" - dest1 `shouldBe` src - dest2 `shouldBe` src - -testInlineGroupFileTransfer :: HasCallStack => FilePath -> IO () -testInlineGroupFileTransfer = - testChatCfg3 cfg aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - bob ##> "/_files_folder ./tests/tmp/bob/" - bob <## "ok" - cath ##> "/_files_folder ./tests/tmp/cath/" - cath <## "ok" - alice ##> "/_send #1 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" - alice <# "#team voice message (00:10)" - alice <# "/f #team ./tests/fixtures/logo.jpg" - -- below is not shown in "sent" mode - -- alice <## "use /fc 1 to cancel sending" - concurrentlyN_ - [ do - alice - <### [ "completed sending file 1 (logo.jpg) to bob", - "completed sending file 1 (logo.jpg) to cath" - ] - alice ##> "/fs 1" - alice <##. "sending file 1 (logo.jpg) complete", - do - bob <# "#team alice> voice message (00:10)" - bob <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" - bob <## "started receiving file 1 (logo.jpg) from alice" - bob <## "completed receiving file 1 (logo.jpg) from alice", - do - cath <# "#team alice> voice message (00:10)" - cath <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" - cath <## "started receiving file 1 (logo.jpg) from alice" - cath <## "completed receiving file 1 (logo.jpg) from alice" - ] - src <- B.readFile "./tests/fixtures/logo.jpg" - dest1 <- B.readFile "./tests/tmp/bob/logo.jpg" - dest2 <- B.readFile "./tests/tmp/cath/logo.jpg" - dest1 `shouldBe` src - dest2 `shouldBe` src - where - cfg = testCfg {inlineFiles = defaultInlineFilesConfig {offerChunks = 100, sendChunks = 100, totalSendChunks = 100, receiveChunks = 100}} - -testSmallInlineGroupFileTransfer :: HasCallStack => FilePath -> IO () -testSmallInlineGroupFileTransfer = - testChatCfg3 testCfg aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - bob ##> "/_files_folder ./tests/tmp/bob/" - bob <## "ok" - cath ##> "/_files_folder ./tests/tmp/cath/" - cath <## "ok" - alice ##> "/_send #1 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" - alice <# "#team voice message (00:10)" - alice <# "/f #team ./tests/fixtures/logo.jpg" - -- below is not shown in "sent" mode - -- alice <## "use /fc 1 to cancel sending" - concurrentlyN_ - [ do - alice - <### [ "completed sending file 1 (logo.jpg) to bob", - "completed sending file 1 (logo.jpg) to cath" - ] - alice ##> "/fs 1" - alice <##. "sending file 1 (logo.jpg) complete", - do - bob <# "#team alice> voice message (00:10)" - bob <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" - bob <## "started receiving file 1 (logo.jpg) from alice" - bob <## "completed receiving file 1 (logo.jpg) from alice", - do - cath <# "#team alice> voice message (00:10)" - cath <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" - cath <## "started receiving file 1 (logo.jpg) from alice" - cath <## "completed receiving file 1 (logo.jpg) from alice" - ] - src <- B.readFile "./tests/fixtures/logo.jpg" - dest1 <- B.readFile "./tests/tmp/bob/logo.jpg" - dest2 <- B.readFile "./tests/tmp/cath/logo.jpg" - dest1 `shouldBe` src - dest2 `shouldBe` src - -testSmallInlineGroupFileIgnored :: HasCallStack => FilePath -> IO () -testSmallInlineGroupFileIgnored tmp = do - withNewTestChat tmp "alice" aliceProfile $ \alice -> - withNewTestChatOpts tmp testOpts {allowInstantFiles = False} "bob" bobProfile $ \bob -> do - withNewTestChatOpts tmp testOpts {allowInstantFiles = False} "cath" cathProfile $ \cath -> do - createGroup3 "team" alice bob cath - bob ##> "/_files_folder ./tests/tmp/bob/" - bob <## "ok" - cath ##> "/_files_folder ./tests/tmp/cath/" - cath <## "ok" - alice ##> "/_send #1 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" - alice <# "#team voice message (00:10)" - alice <# "/f #team ./tests/fixtures/logo.jpg" - -- below is not shown in "sent" mode - -- alice <## "use /fc 1 to cancel sending" - concurrentlyN_ - [ do - alice - <### [ "completed sending file 1 (logo.jpg) to bob", - "completed sending file 1 (logo.jpg) to cath" - ] - alice ##> "/fs 1" - alice <##. "sending file 1 (logo.jpg) complete", - do - bob <# "#team alice> voice message (00:10)" - bob <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob <## "A small file sent without acceptance - you can enable receiving such files with -f option." - bob ##> "/fr 1" - bob <## "file is already being received: logo.jpg", - do - cath <# "#team alice> voice message (00:10)" - cath <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" - cath <## "use /fr 1 [/ | ] to receive it" - cath <## "A small file sent without acceptance - you can enable receiving such files with -f option." - cath ##> "/fr 1" - cath <## "file is already being received: logo.jpg" - ] - -runTestGroupFileSndCancelBeforeTransfer :: HasCallStack => TestCC -> TestCC -> TestCC -> IO () -runTestGroupFileSndCancelBeforeTransfer alice bob cath = do - createGroup3 "team" alice bob cath - alice #> "/f #team ./tests/fixtures/test.txt" - alice <## "use /fc 1 to cancel sending" - concurrentlyN_ - [ do - bob <# "#team alice> sends file test.txt (11 bytes / 11 bytes)" - bob <## "use /fr 1 [/ | ] to receive it", - do - cath <# "#team alice> sends file test.txt (11 bytes / 11 bytes)" - cath <## "use /fr 1 [/ | ] to receive it" - ] - alice ##> "/fc 1" - concurrentlyN_ - [ alice <## "cancelled sending file 1 (test.txt)", - bob <## "alice cancelled sending file 1 (test.txt)", - cath <## "alice cancelled sending file 1 (test.txt)" - ] - alice ##> "/fs 1" - alice <## "sending file 1 (test.txt): no file transfers" - alice <## "file transfer cancelled" - bob ##> "/fs 1" - bob <## "receiving file 1 (test.txt) cancelled" - bob ##> "/fr 1 ./tests/tmp" - bob <## "file cancelled: test.txt" - -runTestMessageWithFile :: HasCallStack => TestCC -> TestCC -> IO () -runTestMessageWithFile alice bob = do - connectUsers alice bob - alice ##> "/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"type\": \"text\", \"text\": \"hi, sending a file\"}}" - alice <# "@bob hi, sending a file" - alice <# "/f @bob ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> hi, sending a file" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1 ./tests/tmp" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - concurrently_ - (bob <## "started receiving file 1 (test.jpg) from alice") - (alice <## "started sending file 1 (test.jpg) to bob") - concurrently_ - (bob <## "completed receiving file 1 (test.jpg) from alice") - (alice <## "completed sending file 1 (test.jpg) to bob") - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/test.jpg" - dest `shouldBe` src - alice #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((1, "hi, sending a file"), Just "./tests/fixtures/test.jpg")]) - bob #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((0, "hi, sending a file"), Just "./tests/tmp/test.jpg")]) - -testSendImage :: HasCallStack => FilePath -> IO () -testSendImage = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice ##> "/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" - alice <# "/f @bob ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1 ./tests/tmp" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - concurrently_ - (bob <## "started receiving file 1 (test.jpg) from alice") - (alice <## "started sending file 1 (test.jpg) to bob") - concurrently_ - (bob <## "completed receiving file 1 (test.jpg) from alice") - (alice <## "completed sending file 1 (test.jpg) to bob") - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/test.jpg" - dest `shouldBe` src - alice #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((1, ""), Just "./tests/fixtures/test.jpg")]) - bob #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((0, ""), Just "./tests/tmp/test.jpg")]) - -- deleting contact without files folder set should not remove file - bob ##> "/d alice" - bob <## "alice: contact is deleted" - fileExists <- doesFileExist "./tests/tmp/test.jpg" - fileExists `shouldBe` True - -testFilesFoldersSendImage :: HasCallStack => FilePath -> IO () -testFilesFoldersSendImage = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice #$> ("/_files_folder ./tests/fixtures", id, "ok") - bob #$> ("/_files_folder ./tests/tmp/app_files", id, "ok") - alice ##> "/_send @2 json {\"filePath\": \"test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" - alice <# "/f @bob test.jpg" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1" - bob <## "saving file 1 from alice to test.jpg" - concurrently_ - (bob <## "started receiving file 1 (test.jpg) from alice") - (alice <## "started sending file 1 (test.jpg) to bob") - concurrently_ - (bob <## "completed receiving file 1 (test.jpg) from alice") - (alice <## "completed sending file 1 (test.jpg) to bob") - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/app_files/test.jpg" - dest `shouldBe` src - alice #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((1, ""), Just "test.jpg")]) - bob #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((0, ""), Just "test.jpg")]) - -- deleting contact with files folder set should remove file - checkActionDeletesFile "./tests/tmp/app_files/test.jpg" $ do - bob ##> "/d alice" - bob <## "alice: contact is deleted" - -testFilesFoldersImageSndDelete :: HasCallStack => FilePath -> IO () -testFilesFoldersImageSndDelete = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice #$> ("/_files_folder ./tests/tmp/alice_app_files", id, "ok") - copyFile "./tests/fixtures/test_1MB.pdf" "./tests/tmp/alice_app_files/test_1MB.pdf" - bob #$> ("/_files_folder ./tests/tmp/bob_app_files", id, "ok") - alice ##> "/_send @2 json {\"filePath\": \"test_1MB.pdf\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" - alice <# "/f @bob test_1MB.pdf" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> sends file test_1MB.pdf (1017.7 KiB / 1042157 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1" - bob <## "saving file 1 from alice to test_1MB.pdf" - concurrently_ - (bob <## "started receiving file 1 (test_1MB.pdf) from alice") - (alice <## "started sending file 1 (test_1MB.pdf) to bob") - -- deleting contact should cancel and remove file - checkActionDeletesFile "./tests/tmp/alice_app_files/test_1MB.pdf" $ do - alice ##> "/d bob" - alice <## "bob: contact is deleted" - bob ##> "/fs 1" - bob <##. "receiving file 1 (test_1MB.pdf) progress" - -- deleting contact should remove cancelled file - checkActionDeletesFile "./tests/tmp/bob_app_files/test_1MB.pdf" $ do - bob ##> "/d alice" - bob <## "alice: contact is deleted" - -testFilesFoldersImageRcvDelete :: HasCallStack => FilePath -> IO () -testFilesFoldersImageRcvDelete = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice #$> ("/_files_folder ./tests/fixtures", id, "ok") - bob #$> ("/_files_folder ./tests/tmp/app_files", id, "ok") - alice ##> "/_send @2 json {\"filePath\": \"test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" - alice <# "/f @bob test.jpg" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1" - bob <## "saving file 1 from alice to test.jpg" - concurrently_ - (bob <## "started receiving file 1 (test.jpg) from alice") - (alice <## "started sending file 1 (test.jpg) to bob") - -- deleting contact should cancel and remove file - waitFileExists "./tests/tmp/app_files/test.jpg" - checkActionDeletesFile "./tests/tmp/app_files/test.jpg" $ do - bob ##> "/d alice" - bob <## "alice: contact is deleted" - alice <## "bob cancelled receiving file 1 (test.jpg)" - alice ##> "/fs 1" - alice <## "sending file 1 (test.jpg) cancelled: bob" - -testSendImageWithTextAndQuote :: HasCallStack => FilePath -> IO () -testSendImageWithTextAndQuote = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - bob #> "@alice hi alice" - alice <# "bob> hi alice" - alice ##> ("/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"quotedItemId\": " <> itemId 1 <> ", \"msgContent\": {\"text\":\"hey bob\",\"type\":\"image\",\"image\":\"\"}}") - alice <# "@bob > hi alice" - alice <## " hey bob" - alice <# "/f @bob ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> > hi alice" - bob <## " hey bob" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1 ./tests/tmp" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - concurrently_ - (bob <## "started receiving file 1 (test.jpg) from alice") - (alice <## "started sending file 1 (test.jpg) to bob") - concurrently_ - (bob <## "completed receiving file 1 (test.jpg) from alice") - (alice <## "completed sending file 1 (test.jpg) to bob") - src <- B.readFile "./tests/fixtures/test.jpg" - B.readFile "./tests/tmp/test.jpg" `shouldReturn` src - alice #$> ("/_get chat @2 count=100", chat'', chatFeatures'' <> [((0, "hi alice"), Nothing, Nothing), ((1, "hey bob"), Just (0, "hi alice"), Just "./tests/fixtures/test.jpg")]) - alice @@@ [("@bob", "hey bob")] - bob #$> ("/_get chat @2 count=100", chat'', chatFeatures'' <> [((1, "hi alice"), Nothing, Nothing), ((0, "hey bob"), Just (1, "hi alice"), Just "./tests/tmp/test.jpg")]) - bob @@@ [("@alice", "hey bob")] - -- quoting (file + text) with file uses quoted text - bob ##> ("/_send @2 json {\"filePath\": \"./tests/fixtures/test.pdf\", \"quotedItemId\": " <> itemId 2 <> ", \"msgContent\": {\"text\":\"\",\"type\":\"file\"}}") - bob <# "@alice > hey bob" - bob <## " test.pdf" - bob <# "/f @alice ./tests/fixtures/test.pdf" - bob <## "use /fc 2 to cancel sending" - alice <# "bob> > hey bob" - alice <## " test.pdf" - alice <# "bob> sends file test.pdf (266.0 KiB / 272376 bytes)" - alice <## "use /fr 2 [/ | ] to receive it" - alice ##> "/fr 2 ./tests/tmp" - alice <## "saving file 2 from bob to ./tests/tmp/test.pdf" - concurrently_ - (alice <## "started receiving file 2 (test.pdf) from bob") - (bob <## "started sending file 2 (test.pdf) to alice") - concurrently_ - (alice <## "completed receiving file 2 (test.pdf) from bob") - (bob <## "completed sending file 2 (test.pdf) to alice") - txtSrc <- B.readFile "./tests/fixtures/test.pdf" - B.readFile "./tests/tmp/test.pdf" `shouldReturn` txtSrc - -- quoting (file without text) with file uses file name - alice ##> ("/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"quotedItemId\": " <> itemId 3 <> ", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}") - alice <# "@bob > test.pdf" - alice <## " test.jpg" - alice <# "/f @bob ./tests/fixtures/test.jpg" - alice <## "use /fc 3 to cancel sending" - bob <# "alice> > test.pdf" - bob <## " test.jpg" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 3 [/ | ] to receive it" - bob ##> "/fr 3 ./tests/tmp" - bob <## "saving file 3 from alice to ./tests/tmp/test_1.jpg" - concurrently_ - (bob <## "started receiving file 3 (test.jpg) from alice") - (alice <## "started sending file 3 (test.jpg) to bob") - concurrently_ - (bob <## "completed receiving file 3 (test.jpg) from alice") - (alice <## "completed sending file 3 (test.jpg) to bob") - B.readFile "./tests/tmp/test_1.jpg" `shouldReturn` src - -testGroupSendImage :: SpecWith FilePath -testGroupSendImage = versionTestMatrix3 runTestGroupSendImage - where - runTestGroupSendImage :: HasCallStack => TestCC -> TestCC -> TestCC -> IO () - runTestGroupSendImage alice bob cath = do - createGroup3 "team" alice bob cath - threadDelay 1000000 - alice ##> "/_send #1 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" - alice <# "/f #team ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - concurrentlyN_ - [ do - bob <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it", - do - cath <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - cath <## "use /fr 1 [/ | ] to receive it" - ] - bob ##> "/fr 1 ./tests/tmp/" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - concurrentlyN_ - [ do - alice <## "started sending file 1 (test.jpg) to bob" - alice <## "completed sending file 1 (test.jpg) to bob", - do - bob <## "started receiving file 1 (test.jpg) from alice" - bob <## "completed receiving file 1 (test.jpg) from alice" - ] - cath ##> "/fr 1 ./tests/tmp/" - cath <## "saving file 1 from alice to ./tests/tmp/test_1.jpg" - concurrentlyN_ - [ do - alice <## "started sending file 1 (test.jpg) to cath" - alice <## "completed sending file 1 (test.jpg) to cath", - do - cath <## "started receiving file 1 (test.jpg) from alice" - cath <## "completed receiving file 1 (test.jpg) from alice" - ] - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/test.jpg" - dest `shouldBe` src - dest2 <- B.readFile "./tests/tmp/test_1.jpg" - dest2 `shouldBe` src - alice #$> ("/_get chat #1 count=1", chatF, [((1, ""), Just "./tests/fixtures/test.jpg")]) - bob #$> ("/_get chat #1 count=1", chatF, [((0, ""), Just "./tests/tmp/test.jpg")]) - cath #$> ("/_get chat #1 count=1", chatF, [((0, ""), Just "./tests/tmp/test_1.jpg")]) - -testGroupSendImageWithTextAndQuote :: HasCallStack => FilePath -> IO () -testGroupSendImageWithTextAndQuote = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - threadDelay 1000000 - bob #> "#team hi team" - concurrently_ - (alice <# "#team bob> hi team") - (cath <# "#team bob> hi team") - threadDelay 1000000 - msgItemId <- lastItemId alice - alice ##> ("/_send #1 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"quotedItemId\": " <> msgItemId <> ", \"msgContent\": {\"text\":\"hey bob\",\"type\":\"image\",\"image\":\"\"}}") - alice <# "#team > bob hi team" - alice <## " hey bob" - alice <# "/f #team ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - concurrentlyN_ - [ do - bob <# "#team alice> > bob hi team" - bob <## " hey bob" - bob <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it", - do - cath <# "#team alice> > bob hi team" - cath <## " hey bob" - cath <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - cath <## "use /fr 1 [/ | ] to receive it" - ] - bob ##> "/fr 1 ./tests/tmp/" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - concurrentlyN_ - [ do - alice <## "started sending file 1 (test.jpg) to bob" - alice <## "completed sending file 1 (test.jpg) to bob", - do - bob <## "started receiving file 1 (test.jpg) from alice" - bob <## "completed receiving file 1 (test.jpg) from alice" - ] - cath ##> "/fr 1 ./tests/tmp/" - cath <## "saving file 1 from alice to ./tests/tmp/test_1.jpg" - concurrentlyN_ - [ do - alice <## "started sending file 1 (test.jpg) to cath" - alice <## "completed sending file 1 (test.jpg) to cath", - do - cath <## "started receiving file 1 (test.jpg) from alice" - cath <## "completed receiving file 1 (test.jpg) from alice" - ] - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/test.jpg" - dest `shouldBe` src - dest2 <- B.readFile "./tests/tmp/test_1.jpg" - dest2 `shouldBe` src - alice #$> ("/_get chat #1 count=2", chat'', [((0, "hi team"), Nothing, Nothing), ((1, "hey bob"), Just (0, "hi team"), Just "./tests/fixtures/test.jpg")]) - alice @@@ [("#team", "hey bob"), ("@bob", "sent invitation to join group team as admin"), ("@cath", "sent invitation to join group team as admin")] - bob #$> ("/_get chat #1 count=2", chat'', [((1, "hi team"), Nothing, Nothing), ((0, "hey bob"), Just (1, "hi team"), Just "./tests/tmp/test.jpg")]) - bob @@@ [("#team", "hey bob"), ("@alice", "received invitation to join group team as admin")] - cath #$> ("/_get chat #1 count=2", chat'', [((0, "hi team"), Nothing, Nothing), ((0, "hey bob"), Just (0, "hi team"), Just "./tests/tmp/test_1.jpg")]) - cath @@@ [("#team", "hey bob"), ("@alice", "received invitation to join group team as admin")] - -testUserContactLink :: SpecWith FilePath -testUserContactLink = versionTestMatrix3 $ \alice bob cath -> do - alice ##> "/ad" - cLink <- getContactLink alice True - bob ##> ("/c " <> cLink) - alice <#? bob - alice @@@ [("<@bob", "")] - alice ##> "/ac bob" - alice <## "bob (Bob): accepting contact request..." - concurrently_ - (bob <## "alice (Alice): contact is connected") - (alice <## "bob (Bob): contact is connected") - threadDelay 100000 - alice @@@ [("@bob", "Voice messages: enabled")] - alice <##> bob - - cath ##> ("/c " <> cLink) - alice <#? cath - alice @@@ [("<@cath", ""), ("@bob", "hey")] - alice ##> "/ac cath" - alice <## "cath (Catherine): accepting contact request..." - concurrently_ - (cath <## "alice (Alice): contact is connected") - (alice <## "cath (Catherine): contact is connected") - threadDelay 100000 - alice @@@ [("@cath", "Voice messages: enabled"), ("@bob", "hey")] - alice <##> cath - -testUserContactLinkAutoAccept :: HasCallStack => FilePath -> IO () -testUserContactLinkAutoAccept = - testChat4 aliceProfile bobProfile cathProfile danProfile $ - \alice bob cath dan -> do - alice ##> "/ad" - cLink <- getContactLink alice True - - bob ##> ("/c " <> cLink) - alice <#? bob - alice @@@ [("<@bob", "")] - alice ##> "/ac bob" - alice <## "bob (Bob): accepting contact request..." - concurrently_ - (bob <## "alice (Alice): contact is connected") - (alice <## "bob (Bob): contact is connected") - threadDelay 100000 - alice @@@ [("@bob", "Voice messages: enabled")] - alice <##> bob - - alice ##> "/auto_accept on" - alice <## "auto_accept on" - - cath ##> ("/c " <> cLink) - cath <## "connection request sent!" - alice <## "cath (Catherine): accepting contact request..." - concurrently_ - (cath <## "alice (Alice): contact is connected") - (alice <## "cath (Catherine): contact is connected") - threadDelay 100000 - alice @@@ [("@cath", "Voice messages: enabled"), ("@bob", "hey")] - alice <##> cath - - alice ##> "/auto_accept off" - alice <## "auto_accept off" - - dan ##> ("/c " <> cLink) - alice <#? dan - alice @@@ [("<@dan", ""), ("@cath", "hey"), ("@bob", "hey")] - alice ##> "/ac dan" - alice <## "dan (Daniel): accepting contact request..." - concurrently_ - (dan <## "alice (Alice): contact is connected") - (alice <## "dan (Daniel): contact is connected") - threadDelay 100000 - alice @@@ [("@dan", "Voice messages: enabled"), ("@cath", "hey"), ("@bob", "hey")] - alice <##> dan - -testDeduplicateContactRequests :: HasCallStack => FilePath -> IO () -testDeduplicateContactRequests = testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - alice ##> "/ad" - cLink <- getContactLink alice True - - bob ##> ("/c " <> cLink) - alice <#? bob - alice @@@ [("<@bob", "")] - bob @@@! [(":1", "", Just ConnJoined)] - - bob ##> ("/c " <> cLink) - alice <#? bob - bob ##> ("/c " <> cLink) - alice <#? bob - alice @@@ [("<@bob", "")] - bob @@@! [(":3", "", Just ConnJoined), (":2", "", Just ConnJoined), (":1", "", Just ConnJoined)] - - alice ##> "/ac bob" - alice <## "bob (Bob): accepting contact request..." - concurrently_ - (bob <## "alice (Alice): contact is connected") - (alice <## "bob (Bob): contact is connected") - - bob ##> ("/c " <> cLink) - bob <## "alice (Alice): contact already exists" - alice @@@ [("@bob", "Voice messages: enabled")] - bob @@@ [("@alice", "Voice messages: enabled"), (":2", ""), (":1", "")] - bob ##> "/_delete :1" - bob <## "connection :1 deleted" - bob ##> "/_delete :2" - bob <## "connection :2 deleted" - - alice <##> bob - alice @@@ [("@bob", "hey")] - bob @@@ [("@alice", "hey")] - - bob ##> ("/c " <> cLink) - bob <## "alice (Alice): contact already exists" - - alice <##> bob - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hi"), (0, "hey"), (1, "hi"), (0, "hey")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hi"), (1, "hey"), (0, "hi"), (1, "hey")]) - - cath ##> ("/c " <> cLink) - alice <#? cath - alice @@@ [("<@cath", ""), ("@bob", "hey")] - alice ##> "/ac cath" - alice <## "cath (Catherine): accepting contact request..." - concurrently_ - (cath <## "alice (Alice): contact is connected") - (alice <## "cath (Catherine): contact is connected") - threadDelay 100000 - alice @@@ [("@cath", "Voice messages: enabled"), ("@bob", "hey")] - alice <##> cath - -testDeduplicateContactRequestsProfileChange :: HasCallStack => FilePath -> IO () -testDeduplicateContactRequestsProfileChange = testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - alice ##> "/ad" - cLink <- getContactLink alice True - - bob ##> ("/c " <> cLink) - alice <#? bob - alice @@@ [("<@bob", "")] - - bob ##> "/p bob" - bob <## "user full name removed (your contacts are notified)" - bob ##> ("/c " <> cLink) - bob <## "connection request sent!" - alice <## "bob wants to connect to you!" - alice <## "to accept: /ac bob" - alice <## "to reject: /rc bob (the sender will NOT be notified)" - alice @@@ [("<@bob", "")] - - bob ##> "/p bob Bob Ross" - bob <## "user full name changed to Bob Ross (your contacts are notified)" - bob ##> ("/c " <> cLink) - alice <#? bob - alice @@@ [("<@bob", "")] - - bob ##> "/p robert Robert" - bob <## "user profile is changed to robert (Robert) (your contacts are notified)" - bob ##> ("/c " <> cLink) - alice <#? bob - alice @@@ [("<@robert", "")] - - alice ##> "/ac bob" - alice <## "no contact request from bob" - alice ##> "/ac robert" - alice <## "robert (Robert): accepting contact request..." - concurrently_ - (bob <## "alice (Alice): contact is connected") - (alice <## "robert (Robert): contact is connected") - - bob ##> ("/c " <> cLink) - bob <## "alice (Alice): contact already exists" - alice @@@ [("@robert", "Voice messages: enabled")] - bob @@@ [("@alice", "Voice messages: enabled"), (":3", ""), (":2", ""), (":1", "")] - bob ##> "/_delete :1" - bob <## "connection :1 deleted" - bob ##> "/_delete :2" - bob <## "connection :2 deleted" - bob ##> "/_delete :3" - bob <## "connection :3 deleted" - - alice <##> bob - alice @@@ [("@robert", "hey")] - bob @@@ [("@alice", "hey")] - - bob ##> ("/c " <> cLink) - bob <## "alice (Alice): contact already exists" - - alice <##> bob - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hi"), (0, "hey"), (1, "hi"), (0, "hey")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hi"), (1, "hey"), (0, "hi"), (1, "hey")]) - - cath ##> ("/c " <> cLink) - alice <#? cath - alice @@@ [("<@cath", ""), ("@robert", "hey")] - alice ##> "/ac cath" - alice <## "cath (Catherine): accepting contact request..." - concurrently_ - (cath <## "alice (Alice): contact is connected") - (alice <## "cath (Catherine): contact is connected") - threadDelay 100000 - alice @@@ [("@cath", "Voice messages: enabled"), ("@robert", "hey")] - alice <##> cath - -testRejectContactAndDeleteUserContact :: HasCallStack => FilePath -> IO () -testRejectContactAndDeleteUserContact = testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - alice ##> "/_address 1" - cLink <- getContactLink alice True - bob ##> ("/c " <> cLink) - alice <#? bob - alice ##> "/rc bob" - alice <## "bob: contact request rejected" - (bob "/_show_address 1" - cLink' <- getContactLink alice False - alice <## "auto_accept off" - cLink' `shouldBe` cLink - - alice ##> "/_delete_address 1" - alice <## "Your chat address is deleted - accepted contacts will remain connected." - alice <## "To create a new chat address use /ad" - - cath ##> ("/c " <> cLink) - cath <## "error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" - -testDeleteConnectionRequests :: HasCallStack => FilePath -> IO () -testDeleteConnectionRequests = testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - alice ##> "/ad" - cLink <- getContactLink alice True - bob ##> ("/c " <> cLink) - alice <#? bob - cath ##> ("/c " <> cLink) - alice <#? cath - - alice ##> "/da" - alice <## "Your chat address is deleted - accepted contacts will remain connected." - alice <## "To create a new chat address use /ad" - - alice ##> "/ad" - cLink' <- getContactLink alice True - bob ##> ("/c " <> cLink') - -- same names are used here, as they were released at /da - alice <#? bob - cath ##> ("/c " <> cLink') - alice <#? cath - -testAutoReplyMessage :: HasCallStack => FilePath -> IO () -testAutoReplyMessage = testChat2 aliceProfile bobProfile $ - \alice bob -> do - alice ##> "/ad" - cLink <- getContactLink alice True - alice ##> "/_auto_accept 1 on incognito=off text hello!" - alice <## "auto_accept on" - alice <## "auto reply:" - alice <## "hello!" - - bob ##> ("/c " <> cLink) - bob <## "connection request sent!" - alice <## "bob (Bob): accepting contact request..." - concurrentlyN_ - [ do - bob <## "alice (Alice): contact is connected" - bob <# "alice> hello!", - do - alice <## "bob (Bob): contact is connected" - alice <# "@bob hello!" - ] - -testAutoReplyMessageInIncognito :: HasCallStack => FilePath -> IO () -testAutoReplyMessageInIncognito = testChat2 aliceProfile bobProfile $ - \alice bob -> do - alice ##> "/ad" - cLink <- getContactLink alice True - alice ##> "/auto_accept on incognito=on text hello!" - alice <## "auto_accept on, incognito" - alice <## "auto reply:" - alice <## "hello!" - - bob ##> ("/c " <> cLink) - bob <## "connection request sent!" - alice <## "bob (Bob): accepting contact request..." - aliceIncognito <- getTermLine alice - concurrentlyN_ - [ do - bob <## (aliceIncognito <> ": contact is connected") - bob <# (aliceIncognito <> "> hello!"), - do - alice <## ("bob (Bob): contact is connected, your incognito profile for this contact is " <> aliceIncognito) - alice - <### [ "use /i bob to print out this incognito profile again", - WithTime "i @bob hello!" - ] - ] - -testConnectIncognitoInvitationLink :: HasCallStack => FilePath -> IO () -testConnectIncognitoInvitationLink = testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - alice #$> ("/incognito on", id, "ok") - bob #$> ("/incognito on", id, "ok") - alice ##> "/c" - inv <- getInvitation alice - bob ##> ("/c " <> inv) - bob <## "confirmation sent!" - bobIncognito <- getTermLine bob - aliceIncognito <- getTermLine alice - concurrentlyN_ - [ do - bob <## (aliceIncognito <> ": contact is connected, your incognito profile for this contact is " <> bobIncognito) - bob <## ("use /i " <> aliceIncognito <> " to print out this incognito profile again"), - do - alice <## (bobIncognito <> ": contact is connected, your incognito profile for this contact is " <> aliceIncognito) - alice <## ("use /i " <> bobIncognito <> " to print out this incognito profile again") - ] - -- after turning incognito mode off conversation is incognito - alice #$> ("/incognito off", id, "ok") - bob #$> ("/incognito off", id, "ok") - alice ?#> ("@" <> bobIncognito <> " psst, I'm incognito") - bob ?<# (aliceIncognito <> "> psst, I'm incognito") - bob ?#> ("@" <> aliceIncognito <> " me too") - alice ?<# (bobIncognito <> "> me too") - -- new contact is connected non incognito - connectUsers alice cath - alice <##> cath - -- bob is not notified on profile change - alice ##> "/p alice" - concurrentlyN_ - [ alice <## "user full name removed (your contacts are notified)", - cath <## "contact alice removed full name" - ] - alice ?#> ("@" <> bobIncognito <> " do you see that I've changed profile?") - bob ?<# (aliceIncognito <> "> do you see that I've changed profile?") - bob ?#> ("@" <> aliceIncognito <> " no") - alice ?<# (bobIncognito <> "> no") - alice ##> "/_set prefs @2 {}" - alice <## ("your preferences for " <> bobIncognito <> " did not change") - (bob "/_set prefs @2 {\"fullDelete\": {\"allow\": \"always\"}}" - alice <## ("you updated preferences for " <> bobIncognito <> ":") - alice <## "Full deletion: enabled for contact (you allow: always, contact allows: no)" - bob <## (aliceIncognito <> " updated preferences for you:") - bob <## "Full deletion: enabled for you (you allow: no, contact allows: always)" - bob ##> "/_set prefs @2 {}" - bob <## ("your preferences for " <> aliceIncognito <> " did not change") - (alice "/_set prefs @2 {\"fullDelete\": {\"allow\": \"no\"}}" - alice <## ("you updated preferences for " <> bobIncognito <> ":") - alice <## "Full deletion: off (you allow: no, contact allows: no)" - bob <## (aliceIncognito <> " updated preferences for you:") - bob <## "Full deletion: off (you allow: no, contact allows: no)" - -- list contacts - alice ##> "/contacts" - alice - <### [ ConsoleString $ "i " <> bobIncognito, - "cath (Catherine)" - ] - alice `hasContactProfiles` ["alice", T.pack aliceIncognito, T.pack bobIncognito, "cath"] - bob ##> "/contacts" - bob <## ("i " <> aliceIncognito) - bob `hasContactProfiles` ["bob", T.pack aliceIncognito, T.pack bobIncognito] - -- alice deletes contact, incognito profile is deleted - alice ##> ("/d " <> bobIncognito) - alice <## (bobIncognito <> ": contact is deleted") - alice ##> "/contacts" - alice <## "cath (Catherine)" - alice `hasContactProfiles` ["alice", "cath"] - -- bob deletes contact, incognito profile is deleted - bob ##> ("/d " <> aliceIncognito) - bob <## (aliceIncognito <> ": contact is deleted") - bob ##> "/contacts" - (bob FilePath -> IO () -testConnectIncognitoContactAddress = testChat2 aliceProfile bobProfile $ - \alice bob -> do - alice ##> "/ad" - cLink <- getContactLink alice True - bob #$> ("/incognito on", id, "ok") - bob ##> ("/c " <> cLink) - bobIncognito <- getTermLine bob - bob <## "connection request sent incognito!" - alice <## (bobIncognito <> " wants to connect to you!") - alice <## ("to accept: /ac " <> bobIncognito) - alice <## ("to reject: /rc " <> bobIncognito <> " (the sender will NOT be notified)") - alice ##> ("/ac " <> bobIncognito) - alice <## (bobIncognito <> ": accepting contact request...") - _ <- getTermLine bob - concurrentlyN_ - [ do - bob <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) - bob <## "use /i alice to print out this incognito profile again", - alice <## (bobIncognito <> ": contact is connected") - ] - -- after turning incognito mode off conversation is incognito - alice #$> ("/incognito off", id, "ok") - bob #$> ("/incognito off", id, "ok") - alice #> ("@" <> bobIncognito <> " who are you?") - bob ?<# "alice> who are you?" - bob ?#> "@alice I'm Batman" - alice <# (bobIncognito <> "> I'm Batman") - -- list contacts - bob ##> "/contacts" - bob <## "i alice (Alice)" - bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognito] - -- delete contact, incognito profile is deleted - bob ##> "/d alice" - bob <## "alice: contact is deleted" - bob ##> "/contacts" - (bob FilePath -> IO () -testAcceptContactRequestIncognito = testChat2 aliceProfile bobProfile $ - \alice bob -> do - alice ##> "/ad" - cLink <- getContactLink alice True - bob ##> ("/c " <> cLink) - alice <#? bob - alice #$> ("/incognito on", id, "ok") - alice ##> "/ac bob" - alice <## "bob (Bob): accepting contact request..." - aliceIncognito <- getTermLine alice - concurrentlyN_ - [ bob <## (aliceIncognito <> ": contact is connected"), - do - alice <## ("bob (Bob): contact is connected, your incognito profile for this contact is " <> aliceIncognito) - alice <## "use /i bob to print out this incognito profile again" - ] - -- after turning incognito mode off conversation is incognito - alice #$> ("/incognito off", id, "ok") - bob #$> ("/incognito off", id, "ok") - alice ?#> "@bob my profile is totally inconspicuous" - bob <# (aliceIncognito <> "> my profile is totally inconspicuous") - bob #> ("@" <> aliceIncognito <> " I know!") - alice ?<# "bob> I know!" - -- list contacts - alice ##> "/contacts" - alice <## "i bob (Bob)" - alice `hasContactProfiles` ["alice", "bob", T.pack aliceIncognito] - -- delete contact, incognito profile is deleted - alice ##> "/d bob" - alice <## "bob: contact is deleted" - alice ##> "/contacts" - (alice FilePath -> IO () -testJoinGroupIncognito = testChat4 aliceProfile bobProfile cathProfile danProfile $ - \alice bob cath dan -> do - -- non incognito connections - connectUsers alice bob - connectUsers alice dan - connectUsers bob cath - connectUsers bob dan - connectUsers cath dan - -- cath connected incognito to alice - alice ##> "/c" - inv <- getInvitation alice - cath #$> ("/incognito on", id, "ok") - cath ##> ("/c " <> inv) - cath <## "confirmation sent!" - cathIncognito <- getTermLine cath - concurrentlyN_ - [ do - cath <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> cathIncognito) - cath <## "use /i alice to print out this incognito profile again", - alice <## (cathIncognito <> ": contact is connected") - ] - -- alice creates group - alice ##> "/g secret_club" - alice <## "group #secret_club is created" - alice <## "to add members use /a secret_club or /create link #secret_club" - -- alice invites bob - alice ##> "/a secret_club bob" - concurrentlyN_ - [ alice <## "invitation to join the group #secret_club sent to bob", - do - bob <## "#secret_club: alice invites you to join the group as admin" - bob <## "use /j secret_club to accept" - ] - bob ##> "/j secret_club" - concurrently_ - (alice <## "#secret_club: bob joined the group") - (bob <## "#secret_club: you joined the group") - -- alice invites cath - alice ##> ("/a secret_club " <> cathIncognito) - concurrentlyN_ - [ alice <## ("invitation to join the group #secret_club sent to " <> cathIncognito), - do - cath <## "#secret_club: alice invites you to join the group as admin" - cath <## ("use /j secret_club to join incognito as " <> cathIncognito) - ] - -- cath uses the same incognito profile when joining group, disabling incognito mode doesn't affect it - cath #$> ("/incognito off", id, "ok") - cath ##> "/j secret_club" - -- cath and bob don't merge contacts - concurrentlyN_ - [ alice <## ("#secret_club: " <> cathIncognito <> " joined the group"), - do - cath <## ("#secret_club: you joined the group incognito as " <> cathIncognito) - cath <## "#secret_club: member bob_1 (Bob) is connected", - do - bob <## ("#secret_club: alice added " <> cathIncognito <> " to the group (connecting...)") - bob <## ("#secret_club: new member " <> cathIncognito <> " is connected") - ] - -- cath cannot invite to the group because her membership is incognito - cath ##> "/a secret_club dan" - cath <## "you've connected to this group using an incognito profile - prohibited to invite contacts" - -- alice invites dan - alice ##> "/a secret_club dan" - concurrentlyN_ - [ alice <## "invitation to join the group #secret_club sent to dan", - do - dan <## "#secret_club: alice invites you to join the group as admin" - dan <## "use /j secret_club to accept" - ] - dan ##> "/j secret_club" - -- cath and dan don't merge contacts - concurrentlyN_ - [ alice <## "#secret_club: dan joined the group", - do - dan <## "#secret_club: you joined the group" - dan - <### [ ConsoleString $ "#secret_club: member " <> cathIncognito <> " is connected", - "#secret_club: member bob_1 (Bob) is connected", - "contact bob_1 is merged into bob", - "use @bob to send messages" - ], - do - bob <## "#secret_club: alice added dan_1 (Daniel) to the group (connecting...)" - bob <## "#secret_club: new member dan_1 is connected" - bob <## "contact dan_1 is merged into dan" - bob <## "use @dan to send messages", - do - cath <## "#secret_club: alice added dan_1 (Daniel) to the group (connecting...)" - cath <## "#secret_club: new member dan_1 is connected" - ] - -- send messages - group is incognito for cath - alice #> "#secret_club hello" - concurrentlyN_ - [ bob <# "#secret_club alice> hello", - cath ?<# "#secret_club alice> hello", - dan <# "#secret_club alice> hello" - ] - bob #> "#secret_club hi there" - concurrentlyN_ - [ alice <# "#secret_club bob> hi there", - cath ?<# "#secret_club bob_1> hi there", - dan <# "#secret_club bob> hi there" - ] - cath ?#> "#secret_club hey" - concurrentlyN_ - [ alice <# ("#secret_club " <> cathIncognito <> "> hey"), - bob <# ("#secret_club " <> cathIncognito <> "> hey"), - dan <# ("#secret_club " <> cathIncognito <> "> hey") - ] - dan #> "#secret_club how is it going?" - concurrentlyN_ - [ alice <# "#secret_club dan> how is it going?", - bob <# "#secret_club dan> how is it going?", - cath ?<# "#secret_club dan_1> how is it going?" - ] - -- cath and bob can send messages via new direct connection, cath is incognito - bob #> ("@" <> cathIncognito <> " hi, I'm bob") - cath ?<# "bob_1> hi, I'm bob" - cath ?#> "@bob_1 hey, I'm incognito" - bob <# (cathIncognito <> "> hey, I'm incognito") - -- cath and dan can send messages via new direct connection, cath is incognito - dan #> ("@" <> cathIncognito <> " hi, I'm dan") - cath ?<# "dan_1> hi, I'm dan" - cath ?#> "@dan_1 hey, I'm incognito" - dan <# (cathIncognito <> "> hey, I'm incognito") - -- non incognito connections are separate - bob <##> cath - dan <##> cath - -- list groups - cath ##> "/gs" - cath <## "i #secret_club" - -- list group members - alice ##> "/ms secret_club" - alice - <### [ "alice (Alice): owner, you, created group", - "bob (Bob): admin, invited, connected", - ConsoleString $ cathIncognito <> ": admin, invited, connected", - "dan (Daniel): admin, invited, connected" - ] - bob ##> "/ms secret_club" - bob - <### [ "alice (Alice): owner, host, connected", - "bob (Bob): admin, you, connected", - ConsoleString $ cathIncognito <> ": admin, connected", - "dan (Daniel): admin, connected" - ] - cath ##> "/ms secret_club" - cath - <### [ "alice (Alice): owner, host, connected", - "bob_1 (Bob): admin, connected", - ConsoleString $ "i " <> cathIncognito <> ": admin, you, connected", - "dan_1 (Daniel): admin, connected" - ] - dan ##> "/ms secret_club" - dan - <### [ "alice (Alice): owner, host, connected", - "bob (Bob): admin, connected", - ConsoleString $ cathIncognito <> ": admin, connected", - "dan (Daniel): admin, you, connected" - ] - -- remove member - bob ##> ("/rm secret_club " <> cathIncognito) - concurrentlyN_ - [ bob <## ("#secret_club: you removed " <> cathIncognito <> " from the group"), - alice <## ("#secret_club: bob removed " <> cathIncognito <> " from the group"), - dan <## ("#secret_club: bob removed " <> cathIncognito <> " from the group"), - do - cath <## "#secret_club: bob_1 removed you from the group" - cath <## "use /d #secret_club to delete the group" - ] - bob #> "#secret_club hi" - concurrentlyN_ - [ alice <# "#secret_club bob> hi", - dan <# "#secret_club bob> hi", - (cath "#secret_club hello" - concurrentlyN_ - [ bob <# "#secret_club alice> hello", - dan <# "#secret_club alice> hello", - (cath "#secret_club hello" - cath <## "you are no longer a member of the group" - -- cath can still message members directly - bob #> ("@" <> cathIncognito <> " I removed you from group") - cath ?<# "bob_1> I removed you from group" - cath ?#> "@bob_1 ok" - bob <# (cathIncognito <> "> ok") - -testCantInviteContactIncognito :: HasCallStack => FilePath -> IO () -testCantInviteContactIncognito = testChat2 aliceProfile bobProfile $ - \alice bob -> do - -- alice connected incognito to bob - alice #$> ("/incognito on", id, "ok") - alice ##> "/c" - inv <- getInvitation alice - bob ##> ("/c " <> inv) - bob <## "confirmation sent!" - aliceIncognito <- getTermLine alice - concurrentlyN_ - [ bob <## (aliceIncognito <> ": contact is connected"), - do - alice <## ("bob (Bob): contact is connected, your incognito profile for this contact is " <> aliceIncognito) - alice <## "use /i bob to print out this incognito profile again" - ] - -- alice creates group non incognito - alice #$> ("/incognito off", id, "ok") - alice ##> "/g club" - alice <## "group #club is created" - alice <## "to add members use /a club or /create link #club" - alice ##> "/a club bob" - alice <## "you're using your main profile for this group - prohibited to invite contacts to whom you are connected incognito" - -- bob doesn't receive invitation - (bob FilePath -> IO () -testCantSeeGlobalPrefsUpdateIncognito = testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - alice #$> ("/incognito on", id, "ok") - alice ##> "/c" - invIncognito <- getInvitation alice - alice #$> ("/incognito off", id, "ok") - alice ##> "/c" - inv <- getInvitation alice - bob ##> ("/c " <> invIncognito) - bob <## "confirmation sent!" - aliceIncognito <- getTermLine alice - cath ##> ("/c " <> inv) - cath <## "confirmation sent!" - concurrentlyN_ - [ bob <## (aliceIncognito <> ": contact is connected"), - do - alice <## ("bob (Bob): contact is connected, your incognito profile for this contact is " <> aliceIncognito) - alice <## "use /i bob to print out this incognito profile again", - do - cath <## "alice (Alice): contact is connected" - ] - alice <## "cath (Catherine): contact is connected" - alice ##> "/_profile 1 {\"displayName\": \"alice\", \"fullName\": \"\", \"preferences\": {\"fullDelete\": {\"allow\": \"always\"}}}" - alice <## "user full name removed (your contacts are notified)" - alice <## "updated preferences:" - alice <## "Full deletion allowed: always" - (alice "/_set prefs @2 {\"fullDelete\": {\"allow\": \"always\"}}" - bob <## ("you updated preferences for " <> aliceIncognito <> ":") - bob <## "Full deletion: enabled for contact (you allow: always, contact allows: no)" - alice <## "bob updated preferences for you:" - alice <## "Full deletion: enabled for you (you allow: no, contact allows: always)" - alice ##> "/_set prefs @2 {\"fullDelete\": {\"allow\": \"yes\"}}" - alice <## "you updated preferences for bob:" - alice <## "Full deletion: enabled (you allow: yes, contact allows: always)" - bob <## (aliceIncognito <> " updated preferences for you:") - bob <## "Full deletion: enabled (you allow: always, contact allows: yes)" - (cath "/_set prefs @3 {\"fullDelete\": {\"allow\": \"always\"}}" - alice <## "your preferences for cath did not change" - alice ##> "/_set prefs @3 {\"fullDelete\": {\"allow\": \"yes\"}}" - alice <## "you updated preferences for cath:" - alice <## "Full deletion: off (you allow: yes, contact allows: no)" - cath <## "alice updated preferences for you:" - cath <## "Full deletion: off (you allow: default (no), contact allows: yes)" - -testDeleteContactThenGroupDeletesIncognitoProfile :: HasCallStack => FilePath -> IO () -testDeleteContactThenGroupDeletesIncognitoProfile = testChat2 aliceProfile bobProfile $ - \alice bob -> do - -- bob connects incognito to alice - alice ##> "/c" - inv <- getInvitation alice - bob #$> ("/incognito on", id, "ok") - bob ##> ("/c " <> inv) - bob <## "confirmation sent!" - bobIncognito <- getTermLine bob - concurrentlyN_ - [ alice <## (bobIncognito <> ": contact is connected"), - do - bob <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) - bob <## "use /i alice to print out this incognito profile again" - ] - -- bob joins group using incognito profile - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> ("/a team " <> bobIncognito) - concurrentlyN_ - [ alice <## ("invitation to join the group #team sent to " <> bobIncognito), - do - bob <## "#team: alice invites you to join the group as admin" - bob <## ("use /j team to join incognito as " <> bobIncognito) - ] - bob ##> "/j team" - concurrently_ - (alice <## ("#team: " <> bobIncognito <> " joined the group")) - (bob <## ("#team: you joined the group incognito as " <> bobIncognito)) - bob ##> "/contacts" - bob <## "i alice (Alice)" - bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognito] - -- delete contact - bob ##> "/d alice" - bob <## "alice: contact is deleted" - bob ##> "/contacts" - (bob "/l team" - concurrentlyN_ - [ do - bob <## "#team: you left the group" - bob <## "use /d #team to delete the group", - alice <## ("#team: " <> bobIncognito <> " left the group") - ] - bob ##> "/d #team" - bob <## "#team: you deleted the group" - bob `hasContactProfiles` ["bob"] - -testDeleteGroupThenContactDeletesIncognitoProfile :: HasCallStack => FilePath -> IO () -testDeleteGroupThenContactDeletesIncognitoProfile = testChat2 aliceProfile bobProfile $ - \alice bob -> do - -- bob connects incognito to alice - alice ##> "/c" - inv <- getInvitation alice - bob #$> ("/incognito on", id, "ok") - bob ##> ("/c " <> inv) - bob <## "confirmation sent!" - bobIncognito <- getTermLine bob - concurrentlyN_ - [ alice <## (bobIncognito <> ": contact is connected"), - do - bob <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) - bob <## "use /i alice to print out this incognito profile again" - ] - -- bob joins group using incognito profile - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> ("/a team " <> bobIncognito) - concurrentlyN_ - [ alice <## ("invitation to join the group #team sent to " <> bobIncognito), - do - bob <## "#team: alice invites you to join the group as admin" - bob <## ("use /j team to join incognito as " <> bobIncognito) - ] - bob ##> "/j team" - concurrently_ - (alice <## ("#team: " <> bobIncognito <> " joined the group")) - (bob <## ("#team: you joined the group incognito as " <> bobIncognito)) - bob ##> "/contacts" - bob <## "i alice (Alice)" - bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognito] - -- delete group - bob ##> "/l team" - concurrentlyN_ - [ do - bob <## "#team: you left the group" - bob <## "use /d #team to delete the group", - alice <## ("#team: " <> bobIncognito <> " left the group") - ] - bob ##> "/d #team" - bob <## "#team: you deleted the group" - bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognito] - -- delete contact - bob ##> "/d alice" - bob <## "alice: contact is deleted" - bob ##> "/contacts" - (bob FilePath -> IO () -testSetAlias = testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice #$> ("/_set alias @2 my friend bob", id, "contact bob alias updated: my friend bob") - alice ##> "/contacts" - alice <## "bob (Bob) (alias: my friend bob)" - alice #$> ("/_set alias @2", id, "contact bob alias removed") - alice ##> "/contacts" - alice <## "bob (Bob)" - -testSetConnectionAlias :: HasCallStack => FilePath -> IO () -testSetConnectionAlias = testChat2 aliceProfile bobProfile $ - \alice bob -> do - alice ##> "/c" - inv <- getInvitation alice - alice @@@ [(":1", "")] - alice ##> "/_set alias :1 friend" - alice <## "connection 1 alias updated: friend" - bob ##> ("/c " <> inv) - bob <## "confirmation sent!" - concurrently_ - (alice <## "bob (Bob): contact is connected") - (bob <## "alice (Alice): contact is connected") - threadDelay 100000 - alice @@@ [("@bob", "Voice messages: enabled")] - alice ##> "/contacts" - alice <## "bob (Bob) (alias: friend)" - -testSetContactPrefs :: HasCallStack => FilePath -> IO () -testSetContactPrefs = testChat2 aliceProfile bobProfile $ - \alice bob -> do - alice #$> ("/_files_folder ./tests/tmp/alice", id, "ok") - bob #$> ("/_files_folder ./tests/tmp/bob", id, "ok") - createDirectoryIfMissing True "./tests/tmp/alice" - createDirectoryIfMissing True "./tests/tmp/bob" - copyFile "./tests/fixtures/test.txt" "./tests/tmp/alice/test.txt" - copyFile "./tests/fixtures/test.txt" "./tests/tmp/bob/test.txt" - bob ##> "/_profile 1 {\"displayName\": \"bob\", \"fullName\": \"Bob\", \"preferences\": {\"voice\": {\"allow\": \"no\"}}}" - bob <## "profile image removed" - bob <## "updated preferences:" - bob <## "Voice messages allowed: no" - (bob "/_set prefs @2 {}" - alice <## "your preferences for bob did not change" - (bob ("/_get chat @2 count=100", chat, startFeatures) - bob #$> ("/_get chat @2 count=100", chat, startFeatures) - let sendVoice = "/_send @2 json {\"filePath\": \"test.txt\", \"msgContent\": {\"type\": \"voice\", \"text\": \"\", \"duration\": 10}}" - voiceNotAllowed = "bad chat command: feature not allowed Voice messages" - alice ##> sendVoice - alice <## voiceNotAllowed - bob ##> sendVoice - bob <## voiceNotAllowed - -- alice ##> "/_set prefs @2 {\"voice\": {\"allow\": \"always\"}}" - alice ##> "/set voice @bob always" - alice <## "you updated preferences for bob:" - alice <## "Voice messages: enabled for contact (you allow: always, contact allows: no)" - alice #$> ("/_get chat @2 count=100", chat, startFeatures <> [(1, "Voice messages: enabled for contact")]) - bob <## "alice updated preferences for you:" - bob <## "Voice messages: enabled for you (you allow: default (no), contact allows: always)" - bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you")]) - alice ##> sendVoice - alice <## voiceNotAllowed - bob ##> sendVoice - bob <# "@alice voice message (00:10)" - bob <# "/f @alice test.txt" - bob <## "completed sending file 1 (test.txt) to alice" - alice <# "bob> voice message (00:10)" - alice <# "bob> sends file test.txt (11 bytes / 11 bytes)" - alice <## "started receiving file 1 (test.txt) from bob" - alice <## "completed receiving file 1 (test.txt) from bob" - (bob "/_profile 1 {\"displayName\": \"alice\", \"fullName\": \"Alice\", \"preferences\": {\"voice\": {\"allow\": \"no\"}}}" - alice ##> "/set voice no" - alice <## "updated preferences:" - alice <## "Voice messages allowed: no" - (alice "/_set prefs @2 {\"voice\": {\"allow\": \"yes\"}}" - alice <## "you updated preferences for bob:" - alice <## "Voice messages: off (you allow: yes, contact allows: no)" - alice #$> ("/_get chat @2 count=100", chat, startFeatures <> [(1, "Voice messages: enabled for contact"), (0, "voice message (00:10)"), (1, "Voice messages: off")]) - bob <## "alice updated preferences for you:" - bob <## "Voice messages: off (you allow: default (no), contact allows: yes)" - bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you"), (1, "voice message (00:10)"), (0, "Voice messages: off")]) - (bob "/_profile 1 {\"displayName\": \"bob\", \"fullName\": \"\", \"preferences\": {\"voice\": {\"allow\": \"yes\"}}}" - bob <## "user full name removed (your contacts are notified)" - bob <## "updated preferences:" - bob <## "Voice messages allowed: yes" - bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you"), (1, "voice message (00:10)"), (0, "Voice messages: off"), (1, "Voice messages: enabled")]) - (bob ("/_get chat @2 count=100", chat, startFeatures <> [(1, "Voice messages: enabled for contact"), (0, "voice message (00:10)"), (1, "Voice messages: off"), (0, "Voice messages: enabled")]) - (alice "/_set prefs @2 {}" - bob <## "your preferences for alice did not change" - -- no change - bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you"), (1, "voice message (00:10)"), (0, "Voice messages: off"), (1, "Voice messages: enabled")]) - (bob "/_set prefs @2 {\"voice\": {\"allow\": \"no\"}}" - alice <## "you updated preferences for bob:" - alice <## "Voice messages: off (you allow: no, contact allows: yes)" - alice #$> ("/_get chat @2 count=100", chat, startFeatures <> [(1, "Voice messages: enabled for contact"), (0, "voice message (00:10)"), (1, "Voice messages: off"), (0, "Voice messages: enabled"), (1, "Voice messages: off")]) - bob <## "alice updated preferences for you:" - bob <## "Voice messages: off (you allow: default (yes), contact allows: no)" - bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you"), (1, "voice message (00:10)"), (0, "Voice messages: off"), (1, "Voice messages: enabled"), (0, "Voice messages: off")]) - -testFeatureOffers :: HasCallStack => FilePath -> IO () -testFeatureOffers = testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice ##> "/set delete @bob yes" - alice <## "you updated preferences for bob:" - alice <## "Full deletion: off (you allow: yes, contact allows: no)" - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Full deletion")]) - bob <## "alice updated preferences for you:" - bob <## "Full deletion: off (you allow: default (no), contact allows: yes)" - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Full deletion")]) - alice ##> "/set delete @bob no" - alice <## "you updated preferences for bob:" - alice <## "Full deletion: off (you allow: no, contact allows: no)" - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Full deletion"), (1, "you cancelled Full deletion")]) - bob <## "alice updated preferences for you:" - bob <## "Full deletion: off (you allow: default (no), contact allows: no)" - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Full deletion"), (0, "cancelled Full deletion")]) - -testUpdateGroupPrefs :: HasCallStack => FilePath -> IO () -testUpdateGroupPrefs = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - createGroup2 "team" alice bob - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected")]) - threadDelay 500000 - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected")]) - alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"fullDelete\": {\"enable\": \"on\"}, \"directMessages\": {\"enable\": \"on\"}}}" - alice <## "updated group preferences:" - alice <## "Full deletion: on" - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on")]) - bob <## "alice updated group #team:" - bob <## "updated group preferences:" - bob <## "Full deletion: on" - threadDelay 500000 - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Full deletion: on")]) - alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"fullDelete\": {\"enable\": \"off\"}, \"voice\": {\"enable\": \"off\"}, \"directMessages\": {\"enable\": \"on\"}}}" - alice <## "updated group preferences:" - alice <## "Full deletion: off" - alice <## "Voice messages: off" - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on"), (1, "Full deletion: off"), (1, "Voice messages: off")]) - bob <## "alice updated group #team:" - bob <## "updated group preferences:" - bob <## "Full deletion: off" - bob <## "Voice messages: off" - threadDelay 500000 - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Full deletion: on"), (0, "Full deletion: off"), (0, "Voice messages: off")]) - -- alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"fullDelete\": {\"enable\": \"off\"}, \"voice\": {\"enable\": \"on\"}}}" - alice ##> "/set voice #team on" - alice <## "updated group preferences:" - alice <## "Voice messages: on" - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on"), (1, "Full deletion: off"), (1, "Voice messages: off"), (1, "Voice messages: on")]) - bob <## "alice updated group #team:" - bob <## "updated group preferences:" - bob <## "Voice messages: on" - threadDelay 500000 - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Full deletion: on"), (0, "Full deletion: off"), (0, "Voice messages: off"), (0, "Voice messages: on")]) - threadDelay 500000 - alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"fullDelete\": {\"enable\": \"off\"}, \"voice\": {\"enable\": \"on\"}, \"directMessages\": {\"enable\": \"on\"}}}" - -- no update - threadDelay 500000 - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on"), (1, "Full deletion: off"), (1, "Voice messages: off"), (1, "Voice messages: on")]) - alice #> "#team hey" - bob <# "#team alice> hey" - threadDelay 1000000 - bob #> "#team hi" - alice <# "#team bob> hi" - threadDelay 500000 - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on"), (1, "Full deletion: off"), (1, "Voice messages: off"), (1, "Voice messages: on"), (1, "hey"), (0, "hi")]) - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Full deletion: on"), (0, "Full deletion: off"), (0, "Voice messages: off"), (0, "Voice messages: on"), (0, "hey"), (1, "hi")]) - -testAllowFullDeletionContact :: HasCallStack => FilePath -> IO () -testAllowFullDeletionContact = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice <##> bob - alice ##> "/set delete @bob always" - alice <## "you updated preferences for bob:" - alice <## "Full deletion: enabled for contact (you allow: always, contact allows: no)" - bob <## "alice updated preferences for you:" - bob <## "Full deletion: enabled for you (you allow: default (no), contact allows: always)" - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hi"), (0, "hey"), (1, "Full deletion: enabled for contact")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hi"), (1, "hey"), (0, "Full deletion: enabled for you")]) - bob #$> ("/_delete item @2 " <> itemId 2 <> " broadcast", id, "message deleted") - alice <# "bob> [deleted] hey" - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hi"), (1, "Full deletion: enabled for contact")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hi"), (0, "Full deletion: enabled for you")]) - -testAllowFullDeletionGroup :: HasCallStack => FilePath -> IO () -testAllowFullDeletionGroup = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - createGroup2 "team" alice bob - threadDelay 1000000 - alice #> "#team hi" - bob <# "#team alice> hi" - threadDelay 1000000 - bob #> "#team hey" - bob ##> "/last_item_id #team" - msgItemId <- getTermLine bob - alice <# "#team bob> hey" - alice ##> "/set delete #team on" - alice <## "updated group preferences:" - alice <## "Full deletion: on" - bob <## "alice updated group #team:" - bob <## "updated group preferences:" - bob <## "Full deletion: on" - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "hi"), (0, "hey"), (1, "Full deletion: on")]) - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "hi"), (1, "hey"), (0, "Full deletion: on")]) - bob #$> ("/_delete item #1 " <> msgItemId <> " broadcast", id, "message deleted") - alice <# "#team bob> [deleted] hey" - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "hi"), (1, "Full deletion: on")]) - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "hi"), (0, "Full deletion: on")]) - -testProhibitDirectMessages :: HasCallStack => FilePath -> IO () -testProhibitDirectMessages = - testChat4 aliceProfile bobProfile cathProfile danProfile $ \alice bob cath dan -> do - createGroup3 "team" alice bob cath - threadDelay 1000000 - alice ##> "/set direct #team off" - alice <## "updated group preferences:" - alice <## "Direct messages: off" - directProhibited bob - directProhibited cath - threadDelay 1000000 - -- still can send direct messages to direct contacts - alice #> "@bob hello again" - bob <# "alice> hello again" - alice #> "@cath hello again" - cath <# "alice> hello again" - bob ##> "@cath hello again" - bob <## "direct messages to indirect contact cath are prohibited" - (cath "/j #team" - concurrentlyN_ - [ cath <## "#team: dan joined the group", - do - dan <## "#team: you joined the group" - dan - <### [ "#team: member alice (Alice) is connected", - "#team: member bob (Bob) is connected" - ], - do - alice <## "#team: cath added dan (Daniel) to the group (connecting...)" - alice <## "#team: new member dan is connected", - do - bob <## "#team: cath added dan (Daniel) to the group (connecting...)" - bob <## "#team: new member dan is connected" - ] - alice ##> "@dan hi" - alice <## "direct messages to indirect contact dan are prohibited" - bob ##> "@dan hi" - bob <## "direct messages to indirect contact dan are prohibited" - (dan "@alice hi" - dan <## "direct messages to indirect contact alice are prohibited" - dan ##> "@bob hi" - dan <## "direct messages to indirect contact bob are prohibited" - dan #> "@cath hi" - cath <# "dan> hi" - cath #> "@dan hi" - dan <# "cath> hi" - where - directProhibited :: HasCallStack => TestCC -> IO () - directProhibited cc = do - cc <## "alice updated group #team:" - cc <## "updated group preferences:" - cc <## "Direct messages: off" - -testEnableTimedMessagesContact :: HasCallStack => FilePath -> IO () -testEnableTimedMessagesContact = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice ##> "/_set prefs @2 {\"timedMessages\": {\"allow\": \"yes\", \"ttl\": 1}}" - alice <## "you updated preferences for bob:" - alice <## "Disappearing messages: off (you allow: yes (1 sec), contact allows: no)" - bob <## "alice updated preferences for you:" - bob <## "Disappearing messages: off (you allow: no, contact allows: yes (1 sec))" - bob ##> "/set disappear @alice yes" - bob <## "you updated preferences for alice:" - bob <## "Disappearing messages: enabled (you allow: yes (1 sec), contact allows: yes (1 sec))" - alice <## "bob updated preferences for you:" - alice <## "Disappearing messages: enabled (you allow: yes (1 sec), contact allows: yes (1 sec))" - alice <##> bob - threadDelay 500000 - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Disappearing messages (1 sec)"), (0, "Disappearing messages: enabled (1 sec)"), (1, "hi"), (0, "hey")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Disappearing messages (1 sec)"), (1, "Disappearing messages: enabled (1 sec)"), (0, "hi"), (1, "hey")]) - threadDelay 1000000 - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Disappearing messages (1 sec)"), (0, "Disappearing messages: enabled (1 sec)")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Disappearing messages (1 sec)"), (1, "Disappearing messages: enabled (1 sec)")]) - -- turn off, messages are not disappearing - bob ##> "/set disappear @alice no" - bob <## "you updated preferences for alice:" - bob <## "Disappearing messages: off (you allow: no, contact allows: yes (1 sec))" - alice <## "bob updated preferences for you:" - alice <## "Disappearing messages: off (you allow: yes (1 sec), contact allows: no)" - alice <##> bob - threadDelay 1500000 - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Disappearing messages (1 sec)"), (0, "Disappearing messages: enabled (1 sec)"), (0, "Disappearing messages: off"), (1, "hi"), (0, "hey")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Disappearing messages (1 sec)"), (1, "Disappearing messages: enabled (1 sec)"), (1, "Disappearing messages: off"), (0, "hi"), (1, "hey")]) - -- test api - bob ##> "/set disappear @alice yes 30s" - bob <## "you updated preferences for alice:" - bob <## "Disappearing messages: enabled (you allow: yes (30 sec), contact allows: yes (1 sec))" - alice <## "bob updated preferences for you:" - alice <## "Disappearing messages: enabled (you allow: yes (30 sec), contact allows: yes (30 sec))" - bob ##> "/set disappear @alice week" -- "yes" is optional - bob <## "you updated preferences for alice:" - bob <## "Disappearing messages: enabled (you allow: yes (1 week), contact allows: yes (1 sec))" - alice <## "bob updated preferences for you:" - alice <## "Disappearing messages: enabled (you allow: yes (1 week), contact allows: yes (1 week))" - -testEnableTimedMessagesGroup :: HasCallStack => FilePath -> IO () -testEnableTimedMessagesGroup = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - createGroup2 "team" alice bob - threadDelay 1000000 - alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"timedMessages\": {\"enable\": \"on\", \"ttl\": 1}, \"directMessages\": {\"enable\": \"on\"}}}" - alice <## "updated group preferences:" - alice <## "Disappearing messages: on (1 sec)" - bob <## "alice updated group #team:" - bob <## "updated group preferences:" - bob <## "Disappearing messages: on (1 sec)" - threadDelay 1000000 - alice #> "#team hi" - bob <# "#team alice> hi" - threadDelay 500000 - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Disappearing messages: on (1 sec)"), (1, "hi")]) - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Disappearing messages: on (1 sec)"), (0, "hi")]) - threadDelay 1000000 - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Disappearing messages: on (1 sec)")]) - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Disappearing messages: on (1 sec)")]) - -- turn off, messages are not disappearing - alice ##> "/set disappear #team off" - alice <## "updated group preferences:" - alice <## "Disappearing messages: off" - bob <## "alice updated group #team:" - bob <## "updated group preferences:" - bob <## "Disappearing messages: off" - threadDelay 1000000 - alice #> "#team hey" - bob <# "#team alice> hey" - threadDelay 1500000 - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Disappearing messages: on (1 sec)"), (1, "Disappearing messages: off"), (1, "hey")]) - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Disappearing messages: on (1 sec)"), (0, "Disappearing messages: off"), (0, "hey")]) - -- test api - alice ##> "/set disappear #team on 30s" - alice <## "updated group preferences:" - alice <## "Disappearing messages: on (30 sec)" - bob <## "alice updated group #team:" - bob <## "updated group preferences:" - bob <## "Disappearing messages: on (30 sec)" - alice ##> "/set disappear #team week" -- "on" is optional - alice <## "updated group preferences:" - alice <## "Disappearing messages: on (1 week)" - bob <## "alice updated group #team:" - bob <## "updated group preferences:" - bob <## "Disappearing messages: on (1 week)" - -testTimedMessagesEnabledGlobally :: HasCallStack => FilePath -> IO () -testTimedMessagesEnabledGlobally = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - alice ##> "/set disappear yes" - alice <## "updated preferences:" - alice <## "Disappearing messages allowed: yes" - connectUsers alice bob - bob ##> "/_set prefs @2 {\"timedMessages\": {\"allow\": \"yes\", \"ttl\": 1}}" - bob <## "you updated preferences for alice:" - bob <## "Disappearing messages: enabled (you allow: yes (1 sec), contact allows: yes)" - alice <## "bob updated preferences for you:" - alice <## "Disappearing messages: enabled (you allow: yes (1 sec), contact allows: yes (1 sec))" - alice <##> bob - threadDelay 500000 - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "Disappearing messages: enabled (1 sec)"), (1, "hi"), (0, "hey")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "Disappearing messages: enabled (1 sec)"), (0, "hi"), (1, "hey")]) - threadDelay 1000000 - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "Disappearing messages: enabled (1 sec)")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "Disappearing messages: enabled (1 sec)")]) - -testGetSetSMPServers :: HasCallStack => FilePath -> IO () -testGetSetSMPServers = - testChat2 aliceProfile bobProfile $ - \alice _ -> do - alice #$> ("/_smp 1", id, "smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001") - alice #$> ("/smp smp://1234-w==@smp1.example.im", id, "ok") - alice #$> ("/smp", id, "smp://1234-w==@smp1.example.im") - alice #$> ("/smp smp://1234-w==:password@smp1.example.im", id, "ok") - alice #$> ("/smp", id, "smp://1234-w==:password@smp1.example.im") - alice #$> ("/smp smp://2345-w==@smp2.example.im;smp://3456-w==@smp3.example.im:5224", id, "ok") - alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") - alice #$> ("/smp default", id, "ok") - alice #$> ("/smp", id, "smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001") - -testTestSMPServerConnection :: HasCallStack => FilePath -> IO () -testTestSMPServerConnection = - testChat2 aliceProfile bobProfile $ - \alice _ -> do - alice ##> "/smp test 1 smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=@localhost:7001" - alice <## "SMP server test passed" - -- to test with password: - -- alice <## "SMP server test failed at CreateQueue, error: SMP AUTH" - -- alice <## "Server requires authorization to create queues, check password" - alice ##> "/smp test 1 smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001" - alice <## "SMP server test passed" - alice ##> "/smp test 1 smp://LcJU@localhost:7001" - alice <## "SMP server test failed at Connect, error: BROKER smp://LcJU@localhost:7001 NETWORK" - alice <## "Possibly, certificate fingerprint in server address is incorrect" - -testAsyncInitiatingOffline :: HasCallStack => FilePath -> IO () -testAsyncInitiatingOffline tmp = do - putStrLn "testAsyncInitiatingOffline" - inv <- withNewTestChat tmp "alice" aliceProfile $ \alice -> do - threadDelay 250000 - alice ##> "/c" - getInvitation alice - withNewTestChat tmp "bob" bobProfile $ \bob -> do - threadDelay 250000 - bob ##> ("/c " <> inv) - bob <## "confirmation sent!" - withTestChat tmp "alice" $ \alice -> do - concurrently_ - (bob <## "alice (Alice): contact is connected") - (alice <## "bob (Bob): contact is connected") - -testAsyncAcceptingOffline :: HasCallStack => FilePath -> IO () -testAsyncAcceptingOffline tmp = do - putStrLn "testAsyncAcceptingOffline" - inv <- withNewTestChat tmp "alice" aliceProfile $ \alice -> do - alice ##> "/c" - getInvitation alice - withNewTestChat tmp "bob" bobProfile $ \bob -> do - threadDelay 250000 - bob ##> ("/c " <> inv) - bob <## "confirmation sent!" - withTestChat tmp "alice" $ \alice -> do - withTestChat tmp "bob" $ \bob -> do - concurrently_ - (bob <## "alice (Alice): contact is connected") - (alice <## "bob (Bob): contact is connected") - -testFullAsync :: HasCallStack => FilePath -> IO () -testFullAsync tmp = do - putStrLn "testFullAsync" - inv <- withNewTestChat tmp "alice" aliceProfile $ \alice -> do - threadDelay 250000 - alice ##> "/c" - getInvitation alice - withNewTestChat tmp "bob" bobProfile $ \bob -> do - threadDelay 250000 - bob ##> ("/c " <> inv) - bob <## "confirmation sent!" - withTestChat tmp "alice" $ \_ -> pure () -- connecting... notification in UI - withTestChat tmp "bob" $ \_ -> pure () -- connecting... notification in UI - withTestChat tmp "alice" $ \alice -> do - alice <## "1 contacts connected (use /cs for the list)" - alice <## "bob (Bob): contact is connected" - withTestChat tmp "bob" $ \bob -> do - bob <## "1 contacts connected (use /cs for the list)" - bob <## "alice (Alice): contact is connected" - -testFullAsyncV1 :: HasCallStack => FilePath -> IO () -testFullAsyncV1 tmp = do - putStrLn "testFullAsyncV1" - inv <- withNewAlice $ \alice -> do - putStrLn "1" - alice ##> "/c" - putStrLn "2" - getInvitation alice - putStrLn "3" - withNewBob $ \bob -> do - putStrLn "4" - bob ##> ("/c " <> inv) - putStrLn "5" - bob <## "confirmation sent!" - putStrLn "6" - withAlice $ \_ -> pure () - putStrLn "7" - withBob $ \_ -> pure () - putStrLn "8" - withAlice $ \alice -> do - putStrLn "9" - alice <## "1 contacts connected (use /cs for the list)" - putStrLn "10" - withBob $ \_ -> pure () - putStrLn "11" - withAlice $ \alice -> do - putStrLn "12" - alice <## "1 contacts connected (use /cs for the list)" - putStrLn "13" - alice <## "bob (Bob): contact is connected" - putStrLn "14" - withBob $ \bob -> do - putStrLn "15" - bob <## "1 contacts connected (use /cs for the list)" - putStrLn "16" - bob <## "alice (Alice): contact is connected" - where - withNewAlice = withNewTestChatV1 tmp "alice" aliceProfile - withAlice = withTestChatV1 tmp "alice" - withNewBob = withNewTestChatV1 tmp "bob" bobProfile - withBob = withTestChatV1 tmp "bob" - -testFullAsyncV1toV2 :: HasCallStack => FilePath -> IO () -testFullAsyncV1toV2 tmp = do - putStrLn "testFullAsyncV1toV2" - inv <- withNewAlice $ \alice -> do - putStrLn "1" - alice ##> "/c" - putStrLn "2" - getInvitation alice - putStrLn "3" - withNewBob $ \bob -> do - putStrLn "4" - bob ##> ("/c " <> inv) - putStrLn "5" - bob <## "confirmation sent!" - withAlice $ \_ -> pure () - putStrLn "6" - withBob $ \_ -> pure () - putStrLn "7" - withAlice $ \alice -> do - putStrLn "8" - alice <## "1 contacts connected (use /cs for the list)" - putStrLn "9" - withBob $ \_ -> pure () - putStrLn "10" - withAlice $ \alice -> do - putStrLn "11" - alice <## "1 contacts connected (use /cs for the list)" - putStrLn "12" - alice <## "bob (Bob): contact is connected" - putStrLn "13" - withBob $ \bob -> do - putStrLn "14" - bob <## "1 contacts connected (use /cs for the list)" - putStrLn "15" - bob <## "alice (Alice): contact is connected" - where - withNewAlice = withNewTestChat tmp "alice" aliceProfile - withAlice = withTestChat tmp "alice" - withNewBob = withNewTestChatV1 tmp "bob" bobProfile - withBob = withTestChatV1 tmp "bob" - -testFullAsyncV2toV1 :: HasCallStack => FilePath -> IO () -testFullAsyncV2toV1 tmp = do - putStrLn "testFullAsyncV2toV1" - inv <- withNewAlice $ \alice -> do - putStrLn "1" - alice ##> "/c" - putStrLn "2" - getInvitation alice - putStrLn "3" - withNewBob $ \bob -> do - putStrLn "4" - bob ##> ("/c " <> inv) - putStrLn "5" - bob <## "confirmation sent!" - putStrLn "6" - withAlice $ \_ -> pure () - putStrLn "7" - withBob $ \_ -> pure () - putStrLn "8" - withAlice $ \alice -> do - putStrLn "9" - alice <## "1 contacts connected (use /cs for the list)" - putStrLn "10" - withBob $ \_ -> pure () - putStrLn "11" - withAlice $ \alice -> do - putStrLn "12" - alice <## "1 contacts connected (use /cs for the list)" - putStrLn "13" - alice <## "bob (Bob): contact is connected" - putStrLn "14" - withBob $ \bob -> do - putStrLn "15" - bob <## "1 contacts connected (use /cs for the list)" - putStrLn "16" - bob <## "alice (Alice): contact is connected" - where - withNewAlice = withNewTestChatV1 tmp "alice" aliceProfile - {-# INLINE withNewAlice #-} - withAlice = withTestChatV1 tmp "alice" - {-# INLINE withAlice #-} - withNewBob = withNewTestChat tmp "bob" bobProfile - {-# INLINE withNewBob #-} - withBob = withTestChat tmp "bob" - {-# INLINE withBob #-} - -testAsyncFileTransferSenderRestarts :: HasCallStack => FilePath -> IO () -testAsyncFileTransferSenderRestarts tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChat tmp "alice" aliceProfile $ \alice -> do - connectUsers alice bob - startFileTransfer' alice bob "test_1MB.pdf" "1017.7 KiB / 1042157 bytes" - threadDelay 100000 - withTestChatContactConnected tmp "alice" $ \alice -> do - alice <## "completed sending file 1 (test_1MB.pdf) to bob" - bob <## "completed receiving file 1 (test_1MB.pdf) from alice" - src <- B.readFile "./tests/fixtures/test_1MB.pdf" - dest <- B.readFile "./tests/tmp/test_1MB.pdf" - dest `shouldBe` src - -testAsyncFileTransferReceiverRestarts :: HasCallStack => FilePath -> IO () -testAsyncFileTransferReceiverRestarts tmp = do - withNewTestChat tmp "alice" aliceProfile $ \alice -> do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - connectUsers alice bob - startFileTransfer' alice bob "test_1MB.pdf" "1017.7 KiB / 1042157 bytes" - threadDelay 100000 - withTestChatContactConnected tmp "bob" $ \bob -> do - alice <## "completed sending file 1 (test_1MB.pdf) to bob" - bob <## "completed receiving file 1 (test_1MB.pdf) from alice" - src <- B.readFile "./tests/fixtures/test_1MB.pdf" - dest <- B.readFile "./tests/tmp/test_1MB.pdf" - dest `shouldBe` src - -testAsyncFileTransfer :: HasCallStack => FilePath -> IO () -testAsyncFileTransfer tmp = do - withNewTestChat tmp "alice" aliceProfile $ \alice -> - withNewTestChat tmp "bob" bobProfile $ \bob -> - connectUsers alice bob - withTestChatContactConnected tmp "alice" $ \alice -> do - alice ##> "/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"type\":\"text\", \"text\": \"hi, sending a file\"}}" - alice <# "@bob hi, sending a file" - alice <# "/f @bob ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - withTestChatContactConnected tmp "bob" $ \bob -> do - bob <# "alice> hi, sending a file" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1 ./tests/tmp" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - -- withTestChatContactConnected' tmp "alice" -- TODO not needed in v2 - -- withTestChatContactConnected' tmp "bob" -- TODO not needed in v2 - withTestChatContactConnected' tmp "alice" - withTestChatContactConnected' tmp "bob" - withTestChatContactConnected tmp "alice" $ \alice -> do - alice <## "started sending file 1 (test.jpg) to bob" - alice <## "completed sending file 1 (test.jpg) to bob" - withTestChatContactConnected tmp "bob" $ \bob -> do - bob <## "started receiving file 1 (test.jpg) from alice" - bob <## "completed receiving file 1 (test.jpg) from alice" - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/test.jpg" - dest `shouldBe` src - -testAsyncFileTransferV1 :: HasCallStack => FilePath -> IO () -testAsyncFileTransferV1 tmp = do - withNewTestChatV1 tmp "alice" aliceProfile $ \alice -> - withNewTestChatV1 tmp "bob" bobProfile $ \bob -> - connectUsers alice bob - withTestChatContactConnectedV1 tmp "alice" $ \alice -> do - alice ##> "/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"type\":\"text\", \"text\": \"hi, sending a file\"}}" - alice <# "@bob hi, sending a file" - alice <# "/f @bob ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - withTestChatContactConnectedV1 tmp "bob" $ \bob -> do - bob <# "alice> hi, sending a file" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1 ./tests/tmp" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - withTestChatContactConnectedV1' tmp "alice" -- TODO not needed in v2 - withTestChatContactConnectedV1' tmp "bob" -- TODO not needed in v2 - withTestChatContactConnectedV1' tmp "alice" - withTestChatContactConnectedV1' tmp "bob" - withTestChatContactConnectedV1 tmp "alice" $ \alice -> do - alice <## "started sending file 1 (test.jpg) to bob" - alice <## "completed sending file 1 (test.jpg) to bob" - withTestChatContactConnectedV1 tmp "bob" $ \bob -> do - bob <## "started receiving file 1 (test.jpg) from alice" - bob <## "completed receiving file 1 (test.jpg) from alice" - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/test.jpg" - dest `shouldBe` src - -testAsyncGroupFileTransfer :: HasCallStack => FilePath -> IO () -testAsyncGroupFileTransfer tmp = do - withNewTestChat tmp "alice" aliceProfile $ \alice -> - withNewTestChat tmp "bob" bobProfile $ \bob -> - withNewTestChat tmp "cath" cathProfile $ \cath -> - createGroup3 "team" alice bob cath - withTestChatGroup3Connected tmp "alice" $ \alice -> do - alice ##> "/_send #1 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"text\"}}" - alice <# "/f #team ./tests/fixtures/test.jpg" - alice <## "use /fc 1 to cancel sending" - withTestChatGroup3Connected tmp "bob" $ \bob -> do - bob <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - bob ##> "/fr 1 ./tests/tmp/" - bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" - withTestChatGroup3Connected tmp "cath" $ \cath -> do - cath <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - cath <## "use /fr 1 [/ | ] to receive it" - cath ##> "/fr 1 ./tests/tmp/" - cath <## "saving file 1 from alice to ./tests/tmp/test_1.jpg" - withTestChatGroup3Connected' tmp "alice" - withTestChatGroup3Connected' tmp "bob" - withTestChatGroup3Connected' tmp "cath" - -- withTestChatGroup3Connected' tmp "alice" -- TODO not needed in v2 - -- withTestChatGroup3Connected' tmp "bob" -- TODO not needed in v2 - -- withTestChatGroup3Connected' tmp "cath" -- TODO not needed in v2 - withTestChatGroup3Connected' tmp "alice" - withTestChatGroup3Connected tmp "bob" $ \bob -> do - bob <## "started receiving file 1 (test.jpg) from alice" - withTestChatGroup3Connected tmp "cath" $ \cath -> do - cath <## "started receiving file 1 (test.jpg) from alice" - withTestChatGroup3Connected tmp "alice" $ \alice -> do - alice - <### [ "started sending file 1 (test.jpg) to bob", - "completed sending file 1 (test.jpg) to bob", - "started sending file 1 (test.jpg) to cath", - "completed sending file 1 (test.jpg) to cath" - ] - withTestChatGroup3Connected tmp "bob" $ \bob -> do - bob <## "completed receiving file 1 (test.jpg) from alice" - withTestChatGroup3Connected tmp "cath" $ \cath -> do - cath <## "completed receiving file 1 (test.jpg) from alice" - src <- B.readFile "./tests/fixtures/test.jpg" - dest <- B.readFile "./tests/tmp/test.jpg" - dest `shouldBe` src - dest2 <- B.readFile "./tests/tmp/test_1.jpg" - dest2 `shouldBe` src - -testCallType :: CallType -testCallType = CallType {media = CMVideo, capabilities = CallCapabilities {encryption = True}} - -testWebRTCSession :: WebRTCSession -testWebRTCSession = - WebRTCSession - { rtcSession = "{}", - rtcIceCandidates = "[]" - } - -testWebRTCCallOffer :: WebRTCCallOffer -testWebRTCCallOffer = - WebRTCCallOffer - { callType = testCallType, - rtcSession = testWebRTCSession - } - -serialize :: ToJSON a => a -> String -serialize = B.unpack . LB.toStrict . J.encode - -repeatM_ :: Int -> IO a -> IO () -repeatM_ n a = forM_ [1 .. n] $ const a - -testNegotiateCall :: HasCallStack => FilePath -> IO () -testNegotiateCall = - testChat2 aliceProfile bobProfile $ \alice bob -> do - connectUsers alice bob - -- just for testing db query - alice ##> "/_call get" - -- alice invite bob to call - alice ##> ("/_call invite @2 " <> serialize testCallType) - alice <## "ok" - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: calling...")]) - bob <## "alice wants to connect with you via WebRTC video call (e2e encrypted)" - repeatM_ 3 $ getTermLine bob - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: calling...")]) - -- bob accepts call by sending WebRTC offer - bob ##> ("/_call offer @2 " <> serialize testWebRTCCallOffer) - bob <## "ok" - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: accepted")]) - alice <## "bob accepted your WebRTC video call (e2e encrypted)" - repeatM_ 3 $ getTermLine alice - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: accepted")]) - -- alice confirms call by sending WebRTC answer - alice ##> ("/_call answer @2 " <> serialize testWebRTCSession) - alice <## "ok" - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: connecting...")]) - bob <## "alice continued the WebRTC call" - repeatM_ 3 $ getTermLine bob - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: connecting...")]) - -- participants can update calls as connected - alice ##> "/_call status @2 connected" - alice <## "ok" - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: in progress (00:00)")]) - bob ##> "/_call status @2 connected" - bob <## "ok" - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: in progress (00:00)")]) - -- either party can end the call - bob ##> "/_call end @2" - bob <## "ok" - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: ended (00:00)")]) - alice <## "call with bob ended" - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: ended (00:00)")]) - -testMaintenanceMode :: HasCallStack => FilePath -> IO () -testMaintenanceMode tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChatOpts tmp testOpts {maintenance = True} "alice" aliceProfile $ \alice -> do - alice ##> "/c" - alice <## "error: chat not started" - alice ##> "/_start" - alice <## "chat started" - connectUsers alice bob - alice #> "@bob hi" - bob <# "alice> hi" - alice ##> "/_db export {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" - alice <## "error: chat not stopped" - alice ##> "/_stop" - alice <## "chat stopped" - alice ##> "/_start" - alice <## "chat started" - -- chat works after start - alice <## "1 contacts connected (use /cs for the list)" - alice #> "@bob hi again" - bob <# "alice> hi again" - bob #> "@alice hello" - alice <# "bob> hello" - -- export / delete / import - alice ##> "/_stop" - alice <## "chat stopped" - alice ##> "/_db export {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" - alice <## "ok" - doesFileExist "./tests/tmp/alice-chat.zip" `shouldReturn` True - alice ##> "/_db import {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" - alice <## "ok" - -- cannot start chat after import - alice ##> "/_start" - alice <## "error: chat store changed, please restart chat" - -- works after full restart - withTestChat tmp "alice" $ \alice -> testChatWorking alice bob - -testChatWorking :: HasCallStack => TestCC -> TestCC -> IO () -testChatWorking alice bob = do - alice <## "1 contacts connected (use /cs for the list)" - alice #> "@bob hello again" - bob <# "alice> hello again" - bob #> "@alice hello too" - alice <# "bob> hello too" - -testMaintenanceModeWithFiles :: HasCallStack => FilePath -> IO () -testMaintenanceModeWithFiles tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChatOpts tmp testOpts {maintenance = True} "alice" aliceProfile $ \alice -> do - alice ##> "/_start" - alice <## "chat started" - alice ##> "/_files_folder ./tests/tmp/alice_files" - alice <## "ok" - connectUsers alice bob - startFileTransferWithDest' bob alice "test.jpg" "136.5 KiB / 139737 bytes" Nothing - bob <## "completed sending file 1 (test.jpg) to alice" - alice <## "completed receiving file 1 (test.jpg) from bob" - src <- B.readFile "./tests/fixtures/test.jpg" - B.readFile "./tests/tmp/alice_files/test.jpg" `shouldReturn` src - threadDelay 500000 - alice ##> "/_stop" - alice <## "chat stopped" - alice ##> "/_db export {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" - alice <## "ok" - alice ##> "/_db delete" - alice <## "ok" - -- cannot start chat after delete - alice ##> "/_start" - alice <## "error: chat store changed, please restart chat" - doesDirectoryExist "./tests/tmp/alice_files" `shouldReturn` False - alice ##> "/_db import {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" - alice <## "ok" - B.readFile "./tests/tmp/alice_files/test.jpg" `shouldReturn` src - -- works after full restart - withTestChat tmp "alice" $ \alice -> testChatWorking alice bob - -testDatabaseEncryption :: HasCallStack => FilePath -> IO () -testDatabaseEncryption tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChatOpts tmp testOpts {maintenance = True} "alice" aliceProfile $ \alice -> do - alice ##> "/_start" - alice <## "chat started" - connectUsers alice bob - alice #> "@bob hi" - bob <# "alice> hi" - alice ##> "/db encrypt mykey" - alice <## "error: chat not stopped" - alice ##> "/db decrypt mykey" - alice <## "error: chat not stopped" - alice ##> "/_stop" - alice <## "chat stopped" - alice ##> "/db decrypt mykey" - alice <## "error: chat database is not encrypted" - alice ##> "/db encrypt mykey" - alice <## "ok" - alice ##> "/_start" - alice <## "error: chat store changed, please restart chat" - withTestChatOpts tmp testOpts {maintenance = True, dbKey = "mykey"} "alice" $ \alice -> do - alice ##> "/_start" - alice <## "chat started" - testChatWorking alice bob - alice ##> "/_stop" - alice <## "chat stopped" - alice ##> "/db key wrongkey nextkey" - alice <## "error encrypting database: wrong passphrase or invalid database file" - alice ##> "/db key mykey nextkey" - alice <## "ok" - alice ##> "/_db encryption {\"currentKey\":\"nextkey\",\"newKey\":\"anotherkey\"}" - alice <## "ok" - withTestChatOpts tmp testOpts {maintenance = True, dbKey = "anotherkey"} "alice" $ \alice -> do - alice ##> "/_start" - alice <## "chat started" - testChatWorking alice bob - alice ##> "/_stop" - alice <## "chat stopped" - alice ##> "/db decrypt anotherkey" - alice <## "ok" - withTestChat tmp "alice" $ \alice -> testChatWorking alice bob - -testMuteContact :: HasCallStack => FilePath -> IO () -testMuteContact = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice #> "@bob hello" - bob <# "alice> hello" - bob ##> "/mute alice" - bob <## "ok" - alice #> "@bob hi" - (bob "/contacts" - bob <## "alice (Alice) (muted, you can /unmute @alice)" - bob ##> "/unmute alice" - bob <## "ok" - bob ##> "/contacts" - bob <## "alice (Alice)" - alice #> "@bob hi again" - bob <# "alice> hi again" - -testMuteGroup :: HasCallStack => FilePath -> IO () -testMuteGroup = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - createGroup3 "team" alice bob cath - threadDelay 1000000 - alice #> "#team hello!" - concurrently_ - (bob <# "#team alice> hello!") - (cath <# "#team alice> hello!") - bob ##> "/mute #team" - bob <## "ok" - alice #> "#team hi" - concurrently_ - (bob hi") - bob ##> "/gs" - bob <## "#team (muted, you can /unmute #team)" - bob ##> "/unmute #team" - bob <## "ok" - alice #> "#team hi again" - concurrently_ - (bob <# "#team alice> hi again") - (cath <# "#team alice> hi again") - bob ##> "/gs" - bob <## "#team" - -testCreateSecondUser :: HasCallStack => FilePath -> IO () -testCreateSecondUser = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - connectUsers alice bob - - alice ##> "/create user alisa" - showActiveUser alice "alisa" - - -- connect using second user - connectUsers alice bob - alice #> "@bob hello" - bob <# "alisa> hello" - bob #> "@alisa hey" - alice <# "bob> hey" - - alice ##> "/user" - showActiveUser alice "alisa" - - alice ##> "/users" - alice <## "alice (Alice)" - alice <## "alisa (active)" - - -- receive message to first user - bob #> "@alice hey alice" - (alice, "alice") $<# "bob> hey alice" - - connectUsers alice cath - - -- set active user to first user - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - - alice ##> "/user" - showActiveUser alice "alice (Alice)" - - alice ##> "/users" - alice <## "alice (Alice) (active)" - alice <## "alisa" - - alice <##> bob - - cath #> "@alisa hey alisa" - (alice, "alisa") $<# "cath> hey alisa" - alice ##> "@cath hi cath" - alice <## "no contact cath" - - -- set active user to second user - alice ##> "/_user 2" - showActiveUser alice "alisa" - -testUsersSubscribeAfterRestart :: HasCallStack => FilePath -> IO () -testUsersSubscribeAfterRestart tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChat tmp "alice" aliceProfile $ \alice -> do - connectUsers alice bob - alice <##> bob - - alice ##> "/create user alisa" - showActiveUser alice "alisa" - connectUsers alice bob - alice <##> bob - - withTestChat tmp "alice" $ \alice -> do - -- second user is active - alice <## "1 contacts connected (use /cs for the list)" - alice <## "[user: alice] 1 contacts connected (use /cs for the list)" - - -- second user receives message - alice <##> bob - - -- first user receives message - bob #> "@alice hey alice" - (alice, "alice") $<# "bob> hey alice" - -testMultipleUserAddresses :: HasCallStack => FilePath -> IO () -testMultipleUserAddresses = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - alice ##> "/ad" - cLinkAlice <- getContactLink alice True - bob ##> ("/c " <> cLinkAlice) - alice <#? bob - alice @@@ [("<@bob", "")] - alice ##> "/ac bob" - alice <## "bob (Bob): accepting contact request..." - concurrently_ - (bob <## "alice (Alice): contact is connected") - (alice <## "bob (Bob): contact is connected") - threadDelay 100000 - alice @@@ [("@bob", "Voice messages: enabled")] - alice <##> bob - - alice ##> "/create user alisa" - showActiveUser alice "alisa" - - -- connect using second user address - alice ##> "/ad" - cLinkAlisa <- getContactLink alice True - bob ##> ("/c " <> cLinkAlisa) - alice <#? bob - alice #$> ("/_get chats 2 pcc=on", chats, [("<@bob", "")]) - alice ##> "/ac bob" - alice <## "bob (Bob): accepting contact request..." - concurrently_ - (bob <## "alisa: contact is connected") - (alice <## "bob (Bob): contact is connected") - threadDelay 100000 - alice #$> ("/_get chats 2 pcc=on", chats, [("@bob", "Voice messages: enabled")]) - alice <##> bob - - bob #> "@alice hey alice" - (alice, "alice") $<# "bob> hey alice" - - -- delete first user address - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - - alice ##> "/da" - alice <## "Your chat address is deleted - accepted contacts will remain connected." - alice <## "To create a new chat address use /ad" - - -- second user receives request when not active - cath ##> ("/c " <> cLinkAlisa) - cath <## "connection request sent!" - alice <## "[user: alisa] cath (Catherine) wants to connect to you!" - alice <## "to accept: /ac cath" - alice <## "to reject: /rc cath (the sender will NOT be notified)" - - -- accept request - alice ##> "/user alisa" - showActiveUser alice "alisa" - - alice ##> "/ac cath" - alice <## "cath (Catherine): accepting contact request..." - concurrently_ - (cath <## "alisa: contact is connected") - (alice <## "cath (Catherine): contact is connected") - threadDelay 100000 - alice #$> ("/_get chats 2 pcc=on", chats, [("@cath", "Voice messages: enabled"), ("@bob", "hey")]) - alice <##> cath - - -- first user doesn't have cath as contact - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice @@@ [("@bob", "hey alice")] - -testCreateUserDefaultServers :: HasCallStack => FilePath -> IO () -testCreateUserDefaultServers = - testChat2 aliceProfile bobProfile $ - \alice _ -> do - alice #$> ("/smp smp://2345-w==@smp2.example.im;smp://3456-w==@smp3.example.im:5224", id, "ok") - alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") - - alice ##> "/create user alisa" - showActiveUser alice "alisa" - - alice #$> ("/smp", id, "smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001") - - -- with same_smp=off - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") - - alice ##> "/create user same_smp=off alisa2" - showActiveUser alice "alisa2" - - alice #$> ("/smp", id, "smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001") - -testCreateUserSameServers :: HasCallStack => FilePath -> IO () -testCreateUserSameServers = - testChat2 aliceProfile bobProfile $ - \alice _ -> do - alice #$> ("/smp smp://2345-w==@smp2.example.im;smp://3456-w==@smp3.example.im:5224", id, "ok") - alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") - - alice ##> "/create user same_smp=on alisa" - showActiveUser alice "alisa" - - alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") - -testDeleteUser :: HasCallStack => FilePath -> IO () -testDeleteUser = - testChat4 aliceProfile bobProfile cathProfile danProfile $ - \alice bob cath dan -> do - connectUsers alice bob - - -- cannot delete active user - - alice ##> "/_delete user 1 del_smp=off" - alice <## "cannot delete active user" - - -- delete user without deleting SMP queues - - alice ##> "/create user alisa" - showActiveUser alice "alisa" - - connectUsers alice cath - alice <##> cath - - alice ##> "/users" - alice <## "alice (Alice)" - alice <## "alisa (active)" - - alice ##> "/_delete user 1 del_smp=off" - alice <## "ok" - - alice ##> "/users" - alice <## "alisa (active)" - - bob #> "@alice hey" - -- no connection authorization error - connection wasn't deleted - (alice "/delete user alisa" - alice <## "cannot delete active user" - - alice ##> "/users" - alice <## "alisa (active)" - - alice <##> cath - - -- delete user deleting SMP queues - - alice ##> "/create user alisa2" - showActiveUser alice "alisa2" - - connectUsers alice dan - alice <##> dan - - alice ##> "/users" - alice <## "alisa" - alice <## "alisa2 (active)" - - alice ##> "/delete user alisa" - alice <### ["ok", "completed deleting user"] - - alice ##> "/users" - alice <## "alisa2 (active)" - - cath #> "@alisa hey" - cath <## "[alisa, contactId: 2, connId: 1] error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" - (alice dan - -testUsersDifferentCIExpirationTTL :: HasCallStack => FilePath -> IO () -testUsersDifferentCIExpirationTTL tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChatCfg tmp cfg "alice" aliceProfile $ \alice -> do - -- first user messages - connectUsers alice bob - - alice #> "@bob alice 1" - bob <# "alice> alice 1" - bob #> "@alice alice 2" - alice <# "bob> alice 2" - - -- second user messages - alice ##> "/create user alisa" - showActiveUser alice "alisa" - connectUsers alice bob - - alice #> "@bob alisa 1" - bob <# "alisa> alisa 1" - bob #> "@alisa alisa 2" - alice <# "bob> alisa 2" - - -- set ttl for first user - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/_ttl 1 1", id, "ok") - - -- set ttl for second user - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/_ttl 2 3", id, "ok") - - -- first user messages - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/ttl", id, "old messages are set to be deleted after: 1 second(s)") - - alice #> "@bob alice 3" - bob <# "alice> alice 3" - bob #> "@alice alice 4" - alice <# "bob> alice 4" - - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "alice 1"), (0, "alice 2"), (1, "alice 3"), (0, "alice 4")]) - - -- second user messages - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/ttl", id, "old messages are set to be deleted after: 3 second(s)") - - alice #> "@bob alisa 3" - bob <# "alisa> alisa 3" - bob #> "@alisa alisa 4" - alice <# "bob> alisa 4" - - alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) - - threadDelay 2000000 - - -- messages both before and after setting chat item ttl are deleted - -- first user messages - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/_get chat @2 count=100", chat, []) - - -- second user messages - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) - - threadDelay 2000000 - - alice #$> ("/_get chat @4 count=100", chat, []) - where - cfg = testCfg {ciExpirationInterval = 500000} - -testUsersRestartCIExpiration :: HasCallStack => FilePath -> IO () -testUsersRestartCIExpiration tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChatCfg tmp cfg "alice" aliceProfile $ \alice -> do - -- set ttl for first user - alice #$> ("/_ttl 1 1", id, "ok") - connectUsers alice bob - - -- create second user and set ttl - alice ##> "/create user alisa" - showActiveUser alice "alisa" - alice #$> ("/_ttl 2 3", id, "ok") - connectUsers alice bob - - -- first user messages - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - - alice #> "@bob alice 1" - bob <# "alice> alice 1" - bob #> "@alice alice 2" - alice <# "bob> alice 2" - - -- second user messages - alice ##> "/user alisa" - showActiveUser alice "alisa" - - alice #> "@bob alisa 1" - bob <# "alisa> alisa 1" - bob #> "@alisa alisa 2" - alice <# "bob> alisa 2" - - -- first user will be active on restart - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - - withTestChatCfg tmp cfg "alice" $ \alice -> do - alice <## "1 contacts connected (use /cs for the list)" - alice <## "[user: alisa] 1 contacts connected (use /cs for the list)" - - -- first user messages - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/ttl", id, "old messages are set to be deleted after: 1 second(s)") - - alice #> "@bob alice 3" - bob <# "alice> alice 3" - bob #> "@alice alice 4" - alice <# "bob> alice 4" - - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "alice 1"), (0, "alice 2"), (1, "alice 3"), (0, "alice 4")]) - - -- second user messages - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/ttl", id, "old messages are set to be deleted after: 3 second(s)") - - alice #> "@bob alisa 3" - bob <# "alisa> alisa 3" - bob #> "@alisa alisa 4" - alice <# "bob> alisa 4" - - alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) - - threadDelay 2000000 - - -- messages both before and after restart are deleted - -- first user messages - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/_get chat @2 count=100", chat, []) - - -- second user messages - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) - - threadDelay 2000000 - - alice #$> ("/_get chat @4 count=100", chat, []) - where - cfg = testCfg {ciExpirationInterval = 500000} - -testEnableCIExpirationOnlyForOneUser :: HasCallStack => FilePath -> IO () -testEnableCIExpirationOnlyForOneUser tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChatCfg tmp cfg "alice" aliceProfile $ \alice -> do - -- first user messages - connectUsers alice bob - - alice #> "@bob alice 1" - bob <# "alice> alice 1" - bob #> "@alice alice 2" - alice <# "bob> alice 2" - - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "alice 1"), (0, "alice 2")]) - - -- second user messages before first user sets ttl - alice ##> "/create user alisa" - showActiveUser alice "alisa" - connectUsers alice bob - - alice #> "@bob alisa 1" - bob <# "alisa> alisa 1" - bob #> "@alisa alisa 2" - alice <# "bob> alisa 2" - - -- set ttl for first user - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/_ttl 1 1", id, "ok") - - -- second user messages after first user sets ttl - alice ##> "/user alisa" - showActiveUser alice "alisa" - - alice #> "@bob alisa 3" - bob <# "alisa> alisa 3" - bob #> "@alisa alisa 4" - alice <# "bob> alisa 4" - - alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) - - threadDelay 2000000 - - -- messages are deleted for first user - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/_get chat @2 count=100", chat, []) - - -- messages are not deleted for second user - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) - - withTestChatCfg tmp cfg "alice" $ \alice -> do - alice <## "1 contacts connected (use /cs for the list)" - alice <## "[user: alice] 1 contacts connected (use /cs for the list)" - - -- messages are not deleted for second user after restart - alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) - - alice #> "@bob alisa 5" - bob <# "alisa> alisa 5" - bob #> "@alisa alisa 6" - alice <# "bob> alisa 6" - - threadDelay 2000000 - - -- new messages are not deleted for second user - alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4"), (1, "alisa 5"), (0, "alisa 6")]) - where - cfg = testCfg {ciExpirationInterval = 500000} - -testDisableCIExpirationOnlyForOneUser :: HasCallStack => FilePath -> IO () -testDisableCIExpirationOnlyForOneUser tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChatCfg tmp cfg "alice" aliceProfile $ \alice -> do - -- set ttl for first user - alice #$> ("/_ttl 1 1", id, "ok") - connectUsers alice bob - - -- create second user and set ttl - alice ##> "/create user alisa" - showActiveUser alice "alisa" - alice #$> ("/_ttl 2 1", id, "ok") - connectUsers alice bob - - -- first user disables expiration - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/ttl none", id, "ok") - alice #$> ("/ttl", id, "old messages are not being deleted") - - -- second user still has ttl configured - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/ttl", id, "old messages are set to be deleted after: 1 second(s)") - - alice #> "@bob alisa 1" - bob <# "alisa> alisa 1" - bob #> "@alisa alisa 2" - alice <# "bob> alisa 2" - - alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2")]) - - threadDelay 2000000 - - -- second user messages are deleted - alice #$> ("/_get chat @4 count=100", chat, []) - - withTestChatCfg tmp cfg "alice" $ \alice -> do - alice <## "1 contacts connected (use /cs for the list)" - alice <## "[user: alice] 1 contacts connected (use /cs for the list)" - - -- second user still has ttl configured after restart - alice #$> ("/ttl", id, "old messages are set to be deleted after: 1 second(s)") - - alice #> "@bob alisa 3" - bob <# "alisa> alisa 3" - bob #> "@alisa alisa 4" - alice <# "bob> alisa 4" - - alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 3"), (0, "alisa 4")]) - - threadDelay 2000000 - - -- second user messages are deleted - alice #$> ("/_get chat @4 count=100", chat, []) - where - cfg = testCfg {ciExpirationInterval = 500000} - -testUsersTimedMessages :: HasCallStack => FilePath -> IO () -testUsersTimedMessages tmp = do - withNewTestChat tmp "bob" bobProfile $ \bob -> do - withNewTestChat tmp "alice" aliceProfile $ \alice -> do - connectUsers alice bob - configureTimedMessages alice bob "2" "1" - - -- create second user and configure timed messages for contact - alice ##> "/create user alisa" - showActiveUser alice "alisa" - connectUsers alice bob - configureTimedMessages alice bob "4" "2" - - -- first user messages - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - - alice #> "@bob alice 1" - bob <# "alice> alice 1" - bob #> "@alice alice 2" - alice <# "bob> alice 2" - - -- second user messages - alice ##> "/user alisa" - showActiveUser alice "alisa" - - alice #> "@bob alisa 1" - bob <# "alisa> alisa 1" - bob #> "@alisa alisa 2" - alice <# "bob> alisa 2" - - -- messages are deleted after ttl - threadDelay 500000 - - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/_get chat @2 count=100", chat, [(1, "alice 1"), (0, "alice 2")]) - - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 1"), (0, "alisa 2")]) - - threadDelay 1000000 - - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/_get chat @2 count=100", chat, []) - - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 1"), (0, "alisa 2")]) - - threadDelay 1000000 - - alice ##> "/user" - showActiveUser alice "alisa" - alice #$> ("/_get chat @4 count=100", chat, []) - - -- first user messages - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - - alice #> "@bob alice 3" - bob <# "alice> alice 3" - bob #> "@alice alice 4" - alice <# "bob> alice 4" - - -- second user messages - alice ##> "/user alisa" - showActiveUser alice "alisa" - - alice #> "@bob alisa 3" - bob <# "alisa> alisa 3" - bob #> "@alisa alisa 4" - alice <# "bob> alisa 4" - - withTestChat tmp "alice" $ \alice -> do - alice <## "1 contacts connected (use /cs for the list)" - alice <## "[user: alice] 1 contacts connected (use /cs for the list)" - - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/_get chat @2 count=100", chat, [(1, "alice 3"), (0, "alice 4")]) - - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 3"), (0, "alisa 4")]) - - -- messages are deleted after restart - threadDelay 1000000 - - alice ##> "/user alice" - showActiveUser alice "alice (Alice)" - alice #$> ("/_get chat @2 count=100", chat, []) - - alice ##> "/user alisa" - showActiveUser alice "alisa" - alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 3"), (0, "alisa 4")]) - - threadDelay 1000000 - - alice ##> "/user" - showActiveUser alice "alisa" - alice #$> ("/_get chat @4 count=100", chat, []) - where - configureTimedMessages :: HasCallStack => TestCC -> TestCC -> String -> String -> IO () - configureTimedMessages alice bob bobId ttl = do - aliceName <- userName alice - alice ##> ("/_set prefs @" <> bobId <> " {\"timedMessages\": {\"allow\": \"yes\", \"ttl\": " <> ttl <> "}}") - alice <## "you updated preferences for bob:" - alice <## ("Disappearing messages: off (you allow: yes (" <> ttl <> " sec), contact allows: no)") - bob <## (aliceName <> " updated preferences for you:") - bob <## ("Disappearing messages: off (you allow: no, contact allows: yes (" <> ttl <> " sec))") - bob ##> ("/set disappear @" <> aliceName <> " yes") - bob <## ("you updated preferences for " <> aliceName <> ":") - bob <## ("Disappearing messages: enabled (you allow: yes (" <> ttl <> " sec), contact allows: yes (" <> ttl <> " sec))") - alice <## "bob updated preferences for you:" - alice <## ("Disappearing messages: enabled (you allow: yes (" <> ttl <> " sec), contact allows: yes (" <> ttl <> " sec))") - alice #$> ("/clear bob", id, "bob: all messages are removed locally ONLY") -- to remove feature items - -testSetChatItemTTL :: HasCallStack => FilePath -> IO () -testSetChatItemTTL = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice #> "@bob 1" - bob <# "alice> 1" - bob #> "@alice 2" - alice <# "bob> 2" - -- chat item with file - alice #$> ("/_files_folder ./tests/tmp/app_files", id, "ok") - copyFile "./tests/fixtures/test.jpg" "./tests/tmp/app_files/test.jpg" - alice ##> "/_send @2 json {\"filePath\": \"test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" - alice <# "/f @bob test.jpg" - alice <## "use /fc 1 to cancel sending" - bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" - bob <## "use /fr 1 [/ | ] to receive it" - -- above items should be deleted after we set ttl - threadDelay 3000000 - alice #> "@bob 3" - bob <# "alice> 3" - bob #> "@alice 4" - alice <# "bob> 4" - alice #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((1, "1"), Nothing), ((0, "2"), Nothing), ((1, ""), Just "test.jpg"), ((1, "3"), Nothing), ((0, "4"), Nothing)]) - checkActionDeletesFile "./tests/tmp/app_files/test.jpg" $ - alice #$> ("/_ttl 1 2", id, "ok") - alice #$> ("/_get chat @2 count=100", chat, [(1, "3"), (0, "4")]) -- when expiration is turned on, first cycle is synchronous - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "1"), (1, "2"), (0, ""), (0, "3"), (1, "4")]) - alice #$> ("/_ttl 1", id, "old messages are set to be deleted after: 2 second(s)") - alice #$> ("/ttl week", id, "ok") - alice #$> ("/ttl", id, "old messages are set to be deleted after: one week") - alice #$> ("/ttl none", id, "ok") - alice #$> ("/ttl", id, "old messages are not being deleted") - -testGroupLink :: HasCallStack => FilePath -> IO () -testGroupLink = - testChat3 aliceProfile bobProfile cathProfile $ - \alice bob cath -> do - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/show link #team" - alice <## "no group link, to create: /create link #team" - alice ##> "/create link #team" - _ <- getGroupLink alice "team" True - alice ##> "/delete link #team" - alice <## "Group link is deleted - joined members will remain connected." - alice <## "To create a new group link use /create link #team" - alice ##> "/create link #team" - gLink <- getGroupLink alice "team" True - alice ##> "/show link #team" - _ <- getGroupLink alice "team" False - alice ##> "/create link #team" - alice <## "you already have link for this group, to show: /show link #team" - bob ##> ("/c " <> gLink) - bob <## "connection request sent!" - alice <## "bob (Bob): accepting request to join group #team..." - concurrentlyN_ - [ do - alice <## "bob (Bob): contact is connected" - alice <## "bob invited to group #team via your group link" - alice <## "#team: bob joined the group", - do - bob <## "alice (Alice): contact is connected" - bob <## "#team: you joined the group" - ] - threadDelay 100000 - alice #$> ("/_get chat #1 count=100", chat, [(0, "invited via your group link"), (0, "connected")]) - -- contacts connected via group link are not in chat previews - alice @@@ [("#team", "connected")] - bob @@@ [("#team", "connected")] - alice <##> bob - alice @@@ [("@bob", "hey"), ("#team", "connected")] - - -- user address doesn't interfere - alice ##> "/ad" - cLink <- getContactLink alice True - cath ##> ("/c " <> cLink) - alice <#? cath - alice ##> "/ac cath" - alice <## "cath (Catherine): accepting contact request..." - concurrently_ - (cath <## "alice (Alice): contact is connected") - (alice <## "cath (Catherine): contact is connected") - alice <##> cath - - -- third member - cath ##> ("/c " <> gLink) - cath <## "connection request sent!" - alice <## "cath_1 (Catherine): accepting request to join group #team..." - -- if contact existed it is merged - concurrentlyN_ - [ alice - <### [ "cath_1 (Catherine): contact is connected", - "contact cath_1 is merged into cath", - "use @cath to send messages", - EndsWith "invited to group #team via your group link", - EndsWith "joined the group" - ], - cath - <### [ "alice_1 (Alice): contact is connected", - "contact alice_1 is merged into alice", - "use @alice to send messages", - "#team: you joined the group", - "#team: member bob (Bob) is connected" - ], - do - bob <## "#team: alice added cath (Catherine) to the group (connecting...)" - bob <## "#team: new member cath is connected" - ] - alice #> "#team hello" - concurrently_ - (bob <# "#team alice> hello") - (cath <# "#team alice> hello") - bob #> "#team hi there" - concurrently_ - (alice <# "#team bob> hi there") - (cath <# "#team bob> hi there") - cath #> "#team hey team" - concurrently_ - (alice <# "#team cath> hey team") - (bob <# "#team cath> hey team") - - -- leaving team removes link - alice ##> "/l team" - concurrentlyN_ - [ do - alice <## "#team: you left the group" - alice <## "use /d #team to delete the group", - bob <## "#team: alice left the group", - cath <## "#team: alice left the group" - ] - alice ##> "/show link #team" - alice <## "no group link, to create: /create link #team" - -testGroupLinkDeleteGroupRejoin :: HasCallStack => FilePath -> IO () -testGroupLinkDeleteGroupRejoin = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/create link #team" - gLink <- getGroupLink alice "team" True - bob ##> ("/c " <> gLink) - bob <## "connection request sent!" - alice <## "bob (Bob): accepting request to join group #team..." - concurrentlyN_ - [ do - alice <## "bob (Bob): contact is connected" - alice <## "bob invited to group #team via your group link" - alice <## "#team: bob joined the group", - do - bob <## "alice (Alice): contact is connected" - bob <## "#team: you joined the group" - ] - -- use contact so it's not deleted when deleting group - bob <##> alice - bob ##> "/l team" - concurrentlyN_ - [ do - bob <## "#team: you left the group" - bob <## "use /d #team to delete the group", - alice <## "#team: bob left the group" - ] - bob ##> "/d #team" - bob <## "#team: you deleted the group" - -- re-join via same link - bob ##> ("/c " <> gLink) - bob <## "connection request sent!" - alice <## "bob_1 (Bob): accepting request to join group #team..." - concurrentlyN_ - [ alice - <### [ "bob_1 (Bob): contact is connected", - "contact bob_1 is merged into bob", - "use @bob to send messages", - EndsWith "invited to group #team via your group link", - EndsWith "joined the group" - ], - bob - <### [ "alice_1 (Alice): contact is connected", - "contact alice_1 is merged into alice", - "use @alice to send messages", - "#team: you joined the group" - ] - ] - alice #> "#team hello" - bob <# "#team alice> hello" - bob #> "#team hi there" - alice <# "#team bob> hi there" - -testGroupLinkContactUsed :: HasCallStack => FilePath -> IO () -testGroupLinkContactUsed = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/create link #team" - gLink <- getGroupLink alice "team" True - bob ##> ("/c " <> gLink) - bob <## "connection request sent!" - alice <## "bob (Bob): accepting request to join group #team..." - concurrentlyN_ - [ do - alice <## "bob (Bob): contact is connected" - alice <## "bob invited to group #team via your group link" - alice <## "#team: bob joined the group", - do - bob <## "alice (Alice): contact is connected" - bob <## "#team: you joined the group" - ] - -- sending/receiving a message marks contact as used - threadDelay 100000 - alice @@@ [("#team", "connected")] - bob @@@ [("#team", "connected")] - alice #> "@bob hello" - bob <# "alice> hello" - alice #$> ("/clear bob", id, "bob: all messages are removed locally ONLY") - alice @@@ [("@bob", ""), ("#team", "connected")] - bob #$> ("/clear alice", id, "alice: all messages are removed locally ONLY") - bob @@@ [("@alice", ""), ("#team", "connected")] - -testGroupLinkIncognitoMembership :: HasCallStack => FilePath -> IO () -testGroupLinkIncognitoMembership = - testChat4 aliceProfile bobProfile cathProfile danProfile $ - \alice bob cath dan -> do - -- bob connected incognito to alice - alice ##> "/c" - inv <- getInvitation alice - bob #$> ("/incognito on", id, "ok") - bob ##> ("/c " <> inv) - bob <## "confirmation sent!" - bobIncognito <- getTermLine bob - concurrentlyN_ - [ do - bob <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) - bob <## "use /i alice to print out this incognito profile again", - alice <## (bobIncognito <> ": contact is connected") - ] - bob #$> ("/incognito off", id, "ok") - -- alice creates group - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - -- alice invites bob - alice ##> ("/a team " <> bobIncognito) - concurrentlyN_ - [ alice <## ("invitation to join the group #team sent to " <> bobIncognito), - do - bob <## "#team: alice invites you to join the group as admin" - bob <## ("use /j team to join incognito as " <> bobIncognito) - ] - bob ##> "/j team" - concurrently_ - (alice <## ("#team: " <> bobIncognito <> " joined the group")) - (bob <## ("#team: you joined the group incognito as " <> bobIncognito)) - -- bob creates group link, cath joins - bob ##> "/create link #team" - gLink <- getGroupLink bob "team" True - cath ##> ("/c " <> gLink) - cath <## "connection request sent!" - bob <## "cath (Catherine): accepting request to join group #team..." - _ <- getTermLine bob - concurrentlyN_ - [ do - bob <## ("cath (Catherine): contact is connected, your incognito profile for this contact is " <> bobIncognito) - bob <## "use /i cath to print out this incognito profile again" - bob <## "cath invited to group #team via your group link" - bob <## "#team: cath joined the group", - do - cath <## (bobIncognito <> ": contact is connected") - cath <## "#team: you joined the group" - cath <## "#team: member alice (Alice) is connected", - do - alice <## ("#team: " <> bobIncognito <> " added cath (Catherine) to the group (connecting...)") - alice <## "#team: new member cath is connected" - ] - bob ?#> "@cath hi, I'm incognito" - cath <# (bobIncognito <> "> hi, I'm incognito") - cath #> ("@" <> bobIncognito <> " hey, I'm cath") - bob ?<# "cath> hey, I'm cath" - -- dan joins incognito - dan #$> ("/incognito on", id, "ok") - dan ##> ("/c " <> gLink) - danIncognito <- getTermLine dan - dan <## "connection request sent incognito!" - bob <## (danIncognito <> ": accepting request to join group #team...") - _ <- getTermLine bob - _ <- getTermLine dan - concurrentlyN_ - [ do - bob <## (danIncognito <> ": contact is connected, your incognito profile for this contact is " <> bobIncognito) - bob <## ("use /i " <> danIncognito <> " to print out this incognito profile again") - bob <## (danIncognito <> " invited to group #team via your group link") - bob <## ("#team: " <> danIncognito <> " joined the group"), - do - dan <## (bobIncognito <> ": contact is connected, your incognito profile for this contact is " <> danIncognito) - dan <## ("use /i " <> bobIncognito <> " to print out this incognito profile again") - dan <## ("#team: you joined the group incognito as " <> danIncognito) - dan - <### [ "#team: member alice (Alice) is connected", - "#team: member cath (Catherine) is connected" - ], - do - alice <## ("#team: " <> bobIncognito <> " added " <> danIncognito <> " to the group (connecting...)") - alice <## ("#team: new member " <> danIncognito <> " is connected"), - do - cath <## ("#team: " <> bobIncognito <> " added " <> danIncognito <> " to the group (connecting...)") - cath <## ("#team: new member " <> danIncognito <> " is connected") - ] - dan #$> ("/incognito off", id, "ok") - bob ?#> ("@" <> danIncognito <> " hi, I'm incognito") - dan ?<# (bobIncognito <> "> hi, I'm incognito") - dan ?#> ("@" <> bobIncognito <> " hey, me too") - bob ?<# (danIncognito <> "> hey, me too") - alice #> "#team hello" - concurrentlyN_ - [ bob ?<# "#team alice> hello", - cath <# "#team alice> hello", - dan ?<# "#team alice> hello" - ] - bob ?#> "#team hi there" - concurrentlyN_ - [ alice <# ("#team " <> bobIncognito <> "> hi there"), - cath <# ("#team " <> bobIncognito <> "> hi there"), - dan ?<# ("#team " <> bobIncognito <> "> hi there") - ] - cath #> "#team hey" - concurrentlyN_ - [ alice <# "#team cath> hey", - bob ?<# "#team cath> hey", - dan ?<# "#team cath> hey" - ] - dan ?#> "#team how is it going?" - concurrentlyN_ - [ alice <# ("#team " <> danIncognito <> "> how is it going?"), - bob ?<# ("#team " <> danIncognito <> "> how is it going?"), - cath <# ("#team " <> danIncognito <> "> how is it going?") - ] - -testGroupLinkUnusedHostContactDeleted :: HasCallStack => FilePath -> IO () -testGroupLinkUnusedHostContactDeleted = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - -- create group 1 - alice ##> "/g team" - alice <## "group #team is created" - alice <## "to add members use /a team or /create link #team" - alice ##> "/create link #team" - gLinkTeam <- getGroupLink alice "team" True - bob ##> ("/c " <> gLinkTeam) - bob <## "connection request sent!" - alice <## "bob (Bob): accepting request to join group #team..." - concurrentlyN_ - [ do - alice <## "bob (Bob): contact is connected" - alice <## "bob invited to group #team via your group link" - alice <## "#team: bob joined the group", - do - bob <## "alice (Alice): contact is connected" - bob <## "#team: you joined the group" - ] - -- create group 2 - alice ##> "/g club" - alice <## "group #club is created" - alice <## "to add members use /a club or /create link #club" - alice ##> "/create link #club" - gLinkClub <- getGroupLink alice "club" True - bob ##> ("/c " <> gLinkClub) - bob <## "connection request sent!" - alice <## "bob_1 (Bob): accepting request to join group #club..." - concurrentlyN_ - [ alice - <### [ "bob_1 (Bob): contact is connected", - "contact bob_1 is merged into bob", - "use @bob to send messages", - EndsWith "invited to group #club via your group link", - EndsWith "joined the group" - ], - bob - <### [ "alice_1 (Alice): contact is connected", - "contact alice_1 is merged into alice", - "use @alice to send messages", - "#club: you joined the group" - ] - ] - -- list contacts - bob ##> "/contacts" - bob <## "alice (Alice)" - -- delete group 1, host contact and profile are kept - bobLeaveDeleteGroup alice bob "team" - bob ##> "/contacts" - bob <## "alice (Alice)" - bob `hasContactProfiles` ["alice", "bob"] - -- delete group 2, unused host contact and profile are deleted - bobLeaveDeleteGroup alice bob "club" - bob ##> "/contacts" - (bob TestCC -> TestCC -> String -> IO () - bobLeaveDeleteGroup alice bob group = do - bob ##> ("/l " <> group) - concurrentlyN_ - [ do - bob <## ("#" <> group <> ": you left the group") - bob <## ("use /d #" <> group <> " to delete the group"), - alice <## ("#" <> group <> ": bob left the group") - ] - bob ##> ("/d #" <> group) - bob <## ("#" <> group <> ": you deleted the group") - -testGroupLinkIncognitoUnusedHostContactsDeleted :: HasCallStack => FilePath -> IO () -testGroupLinkIncognitoUnusedHostContactsDeleted = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - bob #$> ("/incognito on", id, "ok") - bobIncognitoTeam <- createGroupBobIncognito alice bob "team" "alice" - bobIncognitoClub <- createGroupBobIncognito alice bob "club" "alice_1" - bobIncognitoTeam `shouldNotBe` bobIncognitoClub - -- list contacts - bob ##> "/contacts" - bob <## "i alice (Alice)" - bob <## "i alice_1 (Alice)" - bob `hasContactProfiles` ["alice", "alice", "bob", T.pack bobIncognitoTeam, T.pack bobIncognitoClub] - -- delete group 1, unused host contact and profile are deleted - bobLeaveDeleteGroup alice bob "team" bobIncognitoTeam - bob ##> "/contacts" - bob <## "i alice_1 (Alice)" - bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognitoClub] - -- delete group 2, unused host contact and profile are deleted - bobLeaveDeleteGroup alice bob "club" bobIncognitoClub - bob ##> "/contacts" - (bob TestCC -> TestCC -> String -> String -> IO String - createGroupBobIncognito alice bob group bobsAliceContact = do - alice ##> ("/g " <> group) - alice <## ("group #" <> group <> " is created") - alice <## ("to add members use /a " <> group <> " or /create link #" <> group) - alice ##> ("/create link #" <> group) - gLinkTeam <- getGroupLink alice group True - bob ##> ("/c " <> gLinkTeam) - bobIncognito <- getTermLine bob - bob <## "connection request sent incognito!" - alice <## (bobIncognito <> ": accepting request to join group #" <> group <> "...") - _ <- getTermLine bob - concurrentlyN_ - [ do - alice <## (bobIncognito <> ": contact is connected") - alice <## (bobIncognito <> " invited to group #" <> group <> " via your group link") - alice <## ("#" <> group <> ": " <> bobIncognito <> " joined the group"), - do - bob <## (bobsAliceContact <> " (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) - bob <## ("use /i " <> bobsAliceContact <> " to print out this incognito profile again") - bob <## ("#" <> group <> ": you joined the group incognito as " <> bobIncognito) - ] - pure bobIncognito - bobLeaveDeleteGroup :: HasCallStack => TestCC -> TestCC -> String -> String -> IO () - bobLeaveDeleteGroup alice bob group bobIncognito = do - bob ##> ("/l " <> group) - concurrentlyN_ - [ do - bob <## ("#" <> group <> ": you left the group") - bob <## ("use /d #" <> group <> " to delete the group"), - alice <## ("#" <> group <> ": " <> bobIncognito <> " left the group") - ] - bob ##> ("/d #" <> group) - bob <## ("#" <> group <> ": you deleted the group") - -testSwitchContact :: HasCallStack => FilePath -> IO () -testSwitchContact = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - connectUsers alice bob - alice #$> ("/switch bob", id, "ok") - bob <## "alice started changing address for you" - alice <## "bob: you started changing address" - bob <## "alice changed address for you" - alice <## "bob: you changed address" - alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "started changing address..."), (1, "you changed address")]) - bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "started changing address for you..."), (0, "changed address for you")]) - alice <##> bob - -testSwitchGroupMember :: HasCallStack => FilePath -> IO () -testSwitchGroupMember = - testChat2 aliceProfile bobProfile $ - \alice bob -> do - createGroup2 "team" alice bob - alice #$> ("/switch #team bob", id, "ok") - bob <## "#team: alice started changing address for you" - alice <## "#team: you started changing address for bob" - bob <## "#team: alice changed address for you" - alice <## "#team: you changed address for bob" - alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "started changing address for bob..."), (1, "you changed address for bob")]) - bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "started changing address for you..."), (0, "changed address for you")]) - alice #> "#team hey" - bob <# "#team alice> hey" - bob #> "#team hi" - alice <# "#team bob> hi" - -testMarkContactVerified :: HasCallStack => FilePath -> IO () -testMarkContactVerified = - testChat2 aliceProfile bobProfile $ \alice bob -> do - connectUsers alice bob - alice ##> "/i bob" - bobInfo alice - alice <## "connection not verified, use /code command to see security code" - alice ##> "/code bob" - bCode <- getTermLine alice - bob ##> "/code alice" - aCode <- getTermLine bob - bCode `shouldBe` aCode - alice ##> "/verify bob 123" - alice <##. "connection not verified, current code is " - alice ##> ("/verify bob " <> aCode) - alice <## "connection verified" - alice ##> "/i bob" - bobInfo alice - alice <## "connection verified" - alice ##> "/verify bob" - alice <##. "connection not verified, current code is " - alice ##> "/i bob" - bobInfo alice - alice <## "connection not verified, use /code command to see security code" - where - bobInfo :: HasCallStack => TestCC -> IO () - bobInfo alice = do - alice <## "contact ID: 2" - alice <## "receiving messages via: localhost" - alice <## "sending messages via: localhost" - alice <## "you've shared main profile with this contact" - -testMarkGroupMemberVerified :: HasCallStack => FilePath -> IO () -testMarkGroupMemberVerified = - testChat2 aliceProfile bobProfile $ \alice bob -> do - createGroup2 "team" alice bob - alice ##> "/i #team bob" - bobInfo alice - alice <## "connection not verified, use /code command to see security code" - alice ##> "/code #team bob" - bCode <- getTermLine alice - bob ##> "/code #team alice" - aCode <- getTermLine bob - bCode `shouldBe` aCode - alice ##> "/verify #team bob 123" - alice <##. "connection not verified, current code is " - alice ##> ("/verify #team bob " <> aCode) - alice <## "connection verified" - alice ##> "/i #team bob" - bobInfo alice - alice <## "connection verified" - alice ##> "/verify #team bob" - alice <##. "connection not verified, current code is " - alice ##> "/i #team bob" - bobInfo alice - alice <## "connection not verified, use /code command to see security code" - where - bobInfo :: HasCallStack => TestCC -> IO () - bobInfo alice = do - alice <## "group ID: 1" - alice <## "member ID: 2" - alice <## "receiving messages via: localhost" - alice <## "sending messages via: localhost" - -withTestChatContactConnected :: HasCallStack => FilePath -> String -> (HasCallStack => TestCC -> IO a) -> IO a -withTestChatContactConnected tmp dbPrefix action = - withTestChat tmp dbPrefix $ \cc -> do - cc <## "1 contacts connected (use /cs for the list)" - action cc - -withTestChatContactConnected' :: HasCallStack => FilePath -> String -> IO () -withTestChatContactConnected' tmp dbPrefix = withTestChatContactConnected tmp dbPrefix $ \_ -> pure () - -withTestChatContactConnectedV1 :: HasCallStack => FilePath -> String -> (HasCallStack => TestCC -> IO a) -> IO a -withTestChatContactConnectedV1 tmp dbPrefix action = - withTestChatV1 tmp dbPrefix $ \cc -> do - cc <## "1 contacts connected (use /cs for the list)" - action cc - -withTestChatContactConnectedV1' :: HasCallStack => FilePath -> String -> IO () -withTestChatContactConnectedV1' tmp dbPrefix = withTestChatContactConnectedV1 tmp dbPrefix $ \_ -> pure () - -withTestChatGroup3Connected :: HasCallStack => FilePath -> String -> (HasCallStack => TestCC -> IO a) -> IO a -withTestChatGroup3Connected tmp dbPrefix action = do - withTestChat tmp dbPrefix $ \cc -> do - cc <## "2 contacts connected (use /cs for the list)" - cc <## "#team: connected to server(s)" - action cc - -withTestChatGroup3Connected' :: HasCallStack => FilePath -> String -> IO () -withTestChatGroup3Connected' tmp dbPrefix = withTestChatGroup3Connected tmp dbPrefix $ \_ -> pure () - -startFileTransfer :: HasCallStack => TestCC -> TestCC -> IO () -startFileTransfer alice bob = - startFileTransfer' alice bob "test.jpg" "136.5 KiB / 139737 bytes" - -startFileTransfer' :: HasCallStack => TestCC -> TestCC -> String -> String -> IO () -startFileTransfer' cc1 cc2 fileName fileSize = startFileTransferWithDest' cc1 cc2 fileName fileSize $ Just "./tests/tmp" - -startFileTransferWithDest' :: HasCallStack => TestCC -> TestCC -> String -> String -> Maybe String -> IO () -startFileTransferWithDest' cc1 cc2 fileName fileSize fileDest_ = do - name1 <- userName cc1 - name2 <- userName cc2 - cc1 #> ("/f @" <> name2 <> " ./tests/fixtures/" <> fileName) - cc1 <## "use /fc 1 to cancel sending" - cc2 <# (name1 <> "> sends file " <> fileName <> " (" <> fileSize <> ")") - cc2 <## "use /fr 1 [/ | ] to receive it" - cc2 ##> ("/fr 1" <> maybe "" (" " <>) fileDest_) - cc2 <## ("saving file 1 from " <> name1 <> " to " <> maybe id () fileDest_ fileName) - concurrently_ - (cc2 <## ("started receiving file 1 (" <> fileName <> ") from " <> name1)) - (cc1 <## ("started sending file 1 (" <> fileName <> ") to " <> name2)) - -checkPartialTransfer :: HasCallStack => String -> IO () -checkPartialTransfer fileName = do - src <- B.readFile $ "./tests/fixtures/" <> fileName - dest <- B.readFile $ "./tests/tmp/" <> fileName - B.unpack src `shouldStartWith` B.unpack dest - B.length src > B.length dest `shouldBe` True - -checkActionDeletesFile :: HasCallStack => FilePath -> IO () -> IO () -checkActionDeletesFile file action = do - fileExistsBefore <- doesFileExist file - fileExistsBefore `shouldBe` True - action - fileExistsAfter <- doesFileExist file - fileExistsAfter `shouldBe` False - -waitFileExists :: HasCallStack => FilePath -> IO () -waitFileExists f = unlessM (doesFileExist f) $ waitFileExists f - -connectUsers :: HasCallStack => TestCC -> TestCC -> IO () -connectUsers cc1 cc2 = do - name1 <- showName cc1 - name2 <- showName cc2 - cc1 ##> "/c" - inv <- getInvitation cc1 - cc2 ##> ("/c " <> inv) - cc2 <## "confirmation sent!" - concurrently_ - (cc2 <## (name1 <> ": contact is connected")) - (cc1 <## (name2 <> ": contact is connected")) - -showName :: TestCC -> IO String -showName (TestCC ChatController {currentUser} _ _ _ _) = do - Just User {localDisplayName, profile = LocalProfile {fullName}} <- readTVarIO currentUser - pure . T.unpack $ localDisplayName <> optionalFullName localDisplayName fullName - -createGroup2 :: HasCallStack => String -> TestCC -> TestCC -> IO () -createGroup2 gName cc1 cc2 = do - connectUsers cc1 cc2 - name2 <- userName cc2 - cc1 ##> ("/g " <> gName) - cc1 <## ("group #" <> gName <> " is created") - cc1 <## ("to add members use /a " <> gName <> " or /create link #" <> gName) - addMember gName cc1 cc2 GRAdmin - cc2 ##> ("/j " <> gName) - concurrently_ - (cc1 <## ("#" <> gName <> ": " <> name2 <> " joined the group")) - (cc2 <## ("#" <> gName <> ": you joined the group")) - -createGroup3 :: HasCallStack => String -> TestCC -> TestCC -> TestCC -> IO () -createGroup3 gName cc1 cc2 cc3 = do - createGroup2 gName cc1 cc2 - connectUsers cc1 cc3 - name3 <- userName cc3 - sName2 <- showName cc2 - sName3 <- showName cc3 - addMember gName cc1 cc3 GRAdmin - cc3 ##> ("/j " <> gName) - concurrentlyN_ - [ cc1 <## ("#" <> gName <> ": " <> name3 <> " joined the group"), - do - cc3 <## ("#" <> gName <> ": you joined the group") - cc3 <## ("#" <> gName <> ": member " <> sName2 <> " is connected"), - do - cc2 <## ("#" <> gName <> ": alice added " <> sName3 <> " to the group (connecting...)") - cc2 <## ("#" <> gName <> ": new member " <> name3 <> " is connected") - ] - -addMember :: HasCallStack => String -> TestCC -> TestCC -> GroupMemberRole -> IO () -addMember gName inviting invitee role = do - name1 <- userName inviting - memName <- userName invitee - inviting ##> ("/a " <> gName <> " " <> memName <> " " <> B.unpack (strEncode role)) - concurrentlyN_ - [ inviting <## ("invitation to join the group #" <> gName <> " sent to " <> memName), - do - invitee <## ("#" <> gName <> ": " <> name1 <> " invites you to join the group as " <> B.unpack (strEncode role)) - invitee <## ("use /j " <> gName <> " to accept") - ] - --- | test sending direct messages -(<##>) :: HasCallStack => TestCC -> TestCC -> IO () -cc1 <##> cc2 = do - name1 <- userName cc1 - name2 <- userName cc2 - cc1 #> ("@" <> name2 <> " hi") - cc2 <# (name1 <> "> hi") - cc2 #> ("@" <> name1 <> " hey") - cc1 <# (name2 <> "> hey") - -(##>) :: HasCallStack => TestCC -> String -> IO () -cc ##> cmd = do - cc `send` cmd - cc <## cmd - -(#>) :: HasCallStack => TestCC -> String -> IO () -cc #> cmd = do - cc `send` cmd - cc <# cmd - -(?#>) :: HasCallStack => TestCC -> String -> IO () -cc ?#> cmd = do - cc `send` cmd - cc <# ("i " <> cmd) - -(#$>) :: (Eq a, Show a, HasCallStack) => TestCC -> (String, String -> a, a) -> Expectation -cc #$> (cmd, f, res) = do - cc ##> cmd - (f <$> getTermLine cc) `shouldReturn` res - -chat :: String -> [(Int, String)] -chat = map (\(a, _, _) -> a) . chat'' - -chat' :: String -> [((Int, String), Maybe (Int, String))] -chat' = map (\(a, b, _) -> (a, b)) . chat'' - -chatF :: String -> [((Int, String), Maybe String)] -chatF = map (\(a, _, c) -> (a, c)) . chat'' - -chat'' :: String -> [((Int, String), Maybe (Int, String), Maybe String)] -chat'' = read - -chatFeatures :: [(Int, String)] -chatFeatures = map (\(a, _, _) -> a) chatFeatures'' - -chatFeatures' :: [((Int, String), Maybe (Int, String))] -chatFeatures' = map (\(a, b, _) -> (a, b)) chatFeatures'' - -chatFeaturesF :: [((Int, String), Maybe String)] -chatFeaturesF = map (\(a, _, c) -> (a, c)) chatFeatures'' - -chatFeatures'' :: [((Int, String), Maybe (Int, String), Maybe String)] -chatFeatures'' = [((0, "Disappearing messages: off"), Nothing, Nothing), ((0, "Full deletion: off"), Nothing, Nothing), ((0, "Voice messages: enabled"), Nothing, Nothing)] - -groupFeatures :: [(Int, String)] -groupFeatures = map (\(a, _, _) -> a) groupFeatures'' - -groupFeatures'' :: [((Int, String), Maybe (Int, String), Maybe String)] -groupFeatures'' = [((0, "Disappearing messages: off"), Nothing, Nothing), ((0, "Direct messages: on"), Nothing, Nothing), ((0, "Full deletion: off"), Nothing, Nothing), ((0, "Voice messages: on"), Nothing, Nothing)] - -itemId :: Int -> String -itemId i = show $ length chatFeatures + i - -(@@@) :: HasCallStack => TestCC -> [(String, String)] -> Expectation -(@@@) = getChats mapChats - -mapChats :: [(String, String, Maybe ConnStatus)] -> [(String, String)] -mapChats = map $ \(ldn, msg, _) -> (ldn, msg) - -chats :: String -> [(String, String)] -chats = mapChats . read - -(@@@!) :: HasCallStack => TestCC -> [(String, String, Maybe ConnStatus)] -> Expectation -(@@@!) = getChats id - -getChats :: HasCallStack => (Eq a, Show a) => ([(String, String, Maybe ConnStatus)] -> [a]) -> TestCC -> [a] -> Expectation -getChats f cc res = do - cc ##> "/_get chats 1 pcc=on" - line <- getTermLine cc - f (read line) `shouldMatchList` res - -send :: TestCC -> String -> IO () -send TestCC {chatController = cc} cmd = atomically $ writeTBQueue (inputQ cc) cmd - -(<##) :: HasCallStack => TestCC -> String -> Expectation -cc <## line = do - l <- getTermLine cc - when (l /= line) $ print ("expected: " <> line, ", got: " <> l) - l `shouldBe` line - -(<##.) :: HasCallStack => TestCC -> String -> Expectation -cc <##. line = do - l <- getTermLine cc - let prefix = line `isPrefixOf` l - unless prefix $ print ("expected to start from: " <> line, ", got: " <> l) - prefix `shouldBe` True - -(<#.) :: HasCallStack => TestCC -> String -> Expectation -cc <#. line = do - l <- dropTime <$> getTermLine cc - let prefix = line `isPrefixOf` l - unless prefix $ print ("expected to start from: " <> line, ", got: " <> l) - prefix `shouldBe` True - -(<##..) :: HasCallStack => TestCC -> [String] -> Expectation -cc <##.. ls = do - l <- getTermLine cc - let prefix = any (`isPrefixOf` l) ls - unless prefix $ print ("expected to start from one of: " <> show ls, ", got: " <> l) - prefix `shouldBe` True - -data ConsoleResponse = ConsoleString String | WithTime String | EndsWith String - deriving (Show) - -instance IsString ConsoleResponse where fromString = ConsoleString - --- this assumes that the string can only match one option -getInAnyOrder :: HasCallStack => (String -> String) -> TestCC -> [ConsoleResponse] -> Expectation -getInAnyOrder _ _ [] = pure () -getInAnyOrder f cc ls = do - line <- f <$> getTermLine cc - let rest = filter (not . expected line) ls - if length rest < length ls - then getInAnyOrder f cc rest - else error $ "unexpected output: " <> line - where - expected :: String -> ConsoleResponse -> Bool - expected l = \case - ConsoleString s -> l == s - WithTime s -> dropTime_ l == Just s - EndsWith s -> s `isSuffixOf` l - -(<###) :: HasCallStack => TestCC -> [ConsoleResponse] -> Expectation -(<###) = getInAnyOrder id - -(<##?) :: HasCallStack => TestCC -> [ConsoleResponse] -> Expectation -(<##?) = getInAnyOrder dropTime - -(<#) :: HasCallStack => TestCC -> String -> Expectation -cc <# line = (dropTime <$> getTermLine cc) `shouldReturn` line - -(?<#) :: HasCallStack => TestCC -> String -> Expectation -cc ?<# line = (dropTime <$> getTermLine cc) `shouldReturn` "i " <> line - -($<#) :: HasCallStack => (TestCC, String) -> String -> Expectation -(cc, uName) $<# line = (dropTime . dropUser uName <$> getTermLine cc) `shouldReturn` line - -( TestCC -> Expectation -( TestCC -> TestCC -> Expectation -cc1 <#? cc2 = do - name <- userName cc2 - sName <- showName cc2 - cc2 <## "connection request sent!" - cc1 <## (sName <> " wants to connect to you!") - cc1 <## ("to accept: /ac " <> name) - cc1 <## ("to reject: /rc " <> name <> " (the sender will NOT be notified)") - -dropUser :: HasCallStack => String -> String -> String -dropUser uName msg = fromMaybe err $ dropUser_ uName msg - where - err = error $ "invalid user: " <> msg - -dropUser_ :: String -> String -> Maybe String -dropUser_ uName msg = do - let userPrefix = "[user: " <> uName <> "] " - if userPrefix `isPrefixOf` msg - then Just $ drop (length userPrefix) msg - else Nothing - -dropTime :: HasCallStack => String -> String -dropTime msg = fromMaybe err $ dropTime_ msg - where - err = error $ "invalid time: " <> msg - -dropTime_ :: String -> Maybe String -dropTime_ msg = case splitAt 6 msg of - ([m, m', ':', s, s', ' '], text) -> - if all isDigit [m, m', s, s'] then Just text else Nothing - _ -> Nothing - -getInvitation :: HasCallStack => TestCC -> IO String -getInvitation cc = do - cc <## "pass this invitation link to your contact (via another channel):" - cc <## "" - inv <- getTermLine cc - cc <## "" - cc <## "and ask them to connect: /c " - pure inv - -getContactLink :: HasCallStack => TestCC -> Bool -> IO String -getContactLink cc created = do - cc <## if created then "Your new chat address is created!" else "Your chat address:" - cc <## "" - link <- getTermLine cc - cc <## "" - cc <## "Anybody can send you contact requests with: /c " - cc <## "to show it again: /sa" - cc <## "to delete it: /da (accepted contacts will remain connected)" - pure link - -getGroupLink :: HasCallStack => TestCC -> String -> Bool -> IO String -getGroupLink cc gName created = do - cc <## if created then "Group link is created!" else "Group link:" - cc <## "" - link <- getTermLine cc - cc <## "" - cc <## "Anybody can connect to you and join group with: /c " - cc <## ("to show it again: /show link #" <> gName) - cc <## ("to delete it: /delete link #" <> gName <> " (joined members will remain connected to you)") - pure link - -hasContactProfiles :: HasCallStack => TestCC -> [ContactName] -> Expectation -hasContactProfiles cc names = - getContactProfiles cc >>= \ps -> ps `shouldMatchList` names - -getContactProfiles :: TestCC -> IO [ContactName] -getContactProfiles cc = do - user_ <- readTVarIO (currentUser $ chatController cc) - case user_ of - Nothing -> pure [] - Just user -> do - profiles <- withTransaction (chatStore $ chatController cc) $ \db -> getUserContactProfiles db user - pure $ map (\Profile {displayName} -> displayName) profiles - -lastItemId :: HasCallStack => TestCC -> IO String -lastItemId cc = do - cc ##> "/last_item_id" - getTermLine cc - -showActiveUser :: HasCallStack => TestCC -> String -> Expectation -showActiveUser cc name = do - cc <## ("user profile: " <> name) - cc <## "use /p [] to change it" - cc <## "(the updated profile will be sent to all your contacts)" + chatDirectTests + chatGroupTests + chatFileTests + chatProfileTests diff --git a/tests/ChatTests/Direct.hs b/tests/ChatTests/Direct.hs new file mode 100644 index 000000000..a37edad46 --- /dev/null +++ b/tests/ChatTests/Direct.hs @@ -0,0 +1,1594 @@ +{-# LANGUAGE DuplicateRecordFields #-} +{-# LANGUAGE OverloadedStrings #-} +{-# LANGUAGE PostfixOperators #-} +{-# LANGUAGE RankNTypes #-} + +module ChatTests.Direct where + +import ChatClient +import ChatTests.Utils +import Control.Concurrent (threadDelay) +import Control.Concurrent.Async (concurrently_) +import Control.Monad (forM_) +import Data.Aeson (ToJSON) +import qualified Data.Aeson as J +import qualified Data.ByteString.Char8 as B +import qualified Data.ByteString.Lazy.Char8 as LB +import Simplex.Chat.Call +import Simplex.Chat.Controller (ChatConfig (..)) +import Simplex.Chat.Options (ChatOpts (..)) +import Simplex.Chat.Types (authErrDisableCount, sameVerificationCode, verificationCode) +import qualified Simplex.Messaging.Crypto as C +import System.Directory (copyFile, doesDirectoryExist, doesFileExist) +import Test.Hspec + +chatDirectTests :: SpecWith FilePath +chatDirectTests = do + describe "direct messages" $ do + describe "add contact and send/receive message" testAddContact + it "deleting contact deletes profile" testDeleteContactDeletesProfile + it "direct message quoted replies" testDirectMessageQuotedReply + it "direct message update" testDirectMessageUpdate + it "direct message delete" testDirectMessageDelete + it "direct live message" testDirectLiveMessage + it "repeat AUTH errors disable contact" testRepeatAuthErrorsDisableContact + describe "SMP servers" $ do + it "get and set SMP servers" testGetSetSMPServers + it "test SMP server connection" testTestSMPServerConnection + describe "async connection handshake" $ do + it "connect when initiating client goes offline" testAsyncInitiatingOffline + it "connect when accepting client goes offline" testAsyncAcceptingOffline + describe "connect, fully asynchronous (when clients are never simultaneously online)" $ do + it "v2" testFullAsync + describe "webrtc calls api" $ do + it "negotiate call" testNegotiateCall + describe "maintenance mode" $ do + it "start/stop/export/import chat" testMaintenanceMode + it "export/import chat with files" testMaintenanceModeWithFiles + it "encrypt/decrypt database" testDatabaseEncryption + describe "mute/unmute messages" $ do + it "mute/unmute contact" testMuteContact + it "mute/unmute group" testMuteGroup + describe "multiple users" $ do + it "create second user" testCreateSecondUser + it "multiple users subscribe and receive messages after restart" testUsersSubscribeAfterRestart + it "both users have contact link" testMultipleUserAddresses + it "create user with default servers" testCreateUserDefaultServers + it "create user with same servers" testCreateUserSameServers + it "delete user" testDeleteUser + it "users have different chat item TTL configuration, chat items expire" testUsersDifferentCIExpirationTTL + it "chat items expire after restart for all users according to per user configuration" testUsersRestartCIExpiration + it "chat items only expire for users who configured expiration" testEnableCIExpirationOnlyForOneUser + it "disabling chat item expiration doesn't disable it for other users" testDisableCIExpirationOnlyForOneUser + it "both users have configured timed messages with contacts, messages expire, restart" testUsersTimedMessages + describe "chat item expiration" $ do + it "set chat item TTL" testSetChatItemTTL + describe "queue rotation" $ do + it "switch contact to a different queue" testSwitchContact + it "switch group member to a different queue" testSwitchGroupMember + describe "connection verification code" $ do + it "verificationCode function converts ByteString to series of digits" $ \_ -> + verificationCode (C.sha256Hash "abcd") `shouldBe` "61889 38426 63934 09576 96390 79389 84124 85253 63658 69469 70853 37788 95900 68296 20156 25" + it "sameVerificationCode function should ignore spaces" $ \_ -> + sameVerificationCode "123 456 789" "12345 6789" `shouldBe` True + it "mark contact verified" testMarkContactVerified + it "mark group member verified" testMarkGroupMemberVerified + +testAddContact :: HasCallStack => SpecWith FilePath +testAddContact = versionTestMatrix2 runTestAddContact + where + runTestAddContact alice bob = do + alice ##> "/_connect 1" + inv <- getInvitation alice + bob ##> ("/_connect 1 " <> inv) + bob <## "confirmation sent!" + concurrently_ + (bob <## "alice (Alice): contact is connected") + (alice <## "bob (Bob): contact is connected") + threadDelay 100000 + chatsEmpty alice bob + alice #> "@bob hello there 🙂" + bob <# "alice> hello there 🙂" + alice ##> "/_unread chat @2 on" + alice <## "ok" + alice ##> "/_unread chat @2 off" + alice <## "ok" + chatsOneMessage alice bob + bob #> "@alice hello there" + alice <# "bob> hello there" + bob #> "@alice how are you?" + alice <# "bob> how are you?" + chatsManyMessages alice bob + -- test adding the same contact one more time - local name will be different + alice ##> "/c" + inv' <- getInvitation alice + bob ##> ("/c " <> inv') + bob <## "confirmation sent!" + concurrently_ + (bob <## "alice_1 (Alice): contact is connected") + (alice <## "bob_1 (Bob): contact is connected") + alice #> "@bob_1 hello" + bob <# "alice_1> hello" + bob #> "@alice_1 hi" + alice <# "bob_1> hi" + alice @@@ [("@bob_1", "hi"), ("@bob", "how are you?")] + bob @@@ [("@alice_1", "hi"), ("@alice", "how are you?")] + -- test deleting contact + alice ##> "/d bob_1" + alice <## "bob_1: contact is deleted" + alice ##> "@bob_1 hey" + alice <## "no contact bob_1" + alice @@@ [("@bob", "how are you?")] + alice `hasContactProfiles` ["alice", "bob"] + bob @@@ [("@alice_1", "hi"), ("@alice", "how are you?")] + bob `hasContactProfiles` ["alice", "alice", "bob"] + -- test clearing chat + alice #$> ("/clear bob", id, "bob: all messages are removed locally ONLY") + alice #$> ("/_get chat @2 count=100", chat, []) + bob #$> ("/clear alice", id, "alice: all messages are removed locally ONLY") + bob #$> ("/_get chat @2 count=100", chat, []) + chatsEmpty alice bob = do + alice @@@ [("@bob", "Voice messages: enabled")] + alice #$> ("/_get chat @2 count=100", chat, chatFeatures) + bob @@@ [("@alice", "Voice messages: enabled")] + bob #$> ("/_get chat @2 count=100", chat, chatFeatures) + chatsOneMessage alice bob = do + alice @@@ [("@bob", "hello there 🙂")] + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hello there 🙂")]) + bob @@@ [("@alice", "hello there 🙂")] + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hello there 🙂")]) + chatsManyMessages alice bob = do + alice @@@ [("@bob", "how are you?")] + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hello there 🙂"), (0, "hello there"), (0, "how are you?")]) + bob @@@ [("@alice", "how are you?")] + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hello there 🙂"), (1, "hello there"), (1, "how are you?")]) + -- pagination + alice #$> ("/_get chat @2 after=" <> itemId 1 <> " count=100", chat, [(0, "hello there"), (0, "how are you?")]) + alice #$> ("/_get chat @2 before=" <> itemId 2 <> " count=100", chat, chatFeatures <> [(1, "hello there 🙂")]) + -- search + alice #$> ("/_get chat @2 count=100 search=ello ther", chat, [(1, "hello there 🙂"), (0, "hello there")]) + -- read messages + alice #$> ("/_read chat @2 from=1 to=100", id, "ok") + bob #$> ("/_read chat @2 from=1 to=100", id, "ok") + alice #$> ("/_read chat @2", id, "ok") + bob #$> ("/_read chat @2", id, "ok") + +testDeleteContactDeletesProfile :: HasCallStack => FilePath -> IO () +testDeleteContactDeletesProfile = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice <##> bob + -- alice deletes contact, profile is deleted + alice ##> "/d bob" + alice <## "bob: contact is deleted" + alice ##> "/_contacts 1" + (alice "/d alice" + bob <## "alice: contact is deleted" + bob ##> "/contacts" + (bob FilePath -> IO () +testDirectMessageQuotedReply = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice ##> "/_send @2 text hello! how are you?" + alice <# "@bob hello! how are you?" + bob <# "alice> hello! how are you?" + bob #> "@alice hi!" + alice <# "bob> hi!" + bob `send` "> @alice (hello) all good - you?" + bob <# "@alice > hello! how are you?" + bob <## " all good - you?" + alice <# "bob> > hello! how are you?" + alice <## " all good - you?" + bob #$> ("/_get chat @2 count=1", chat', [((1, "all good - you?"), Just (0, "hello! how are you?"))]) + alice #$> ("/_get chat @2 count=1", chat', [((0, "all good - you?"), Just (1, "hello! how are you?"))]) + bob `send` ">> @alice (all good) will tell more" + bob <# "@alice >> all good - you?" + bob <## " will tell more" + alice <# "bob> >> all good - you?" + alice <## " will tell more" + bob #$> ("/_get chat @2 count=1", chat', [((1, "will tell more"), Just (1, "all good - you?"))]) + alice #$> ("/_get chat @2 count=1", chat', [((0, "will tell more"), Just (0, "all good - you?"))]) + +testDirectMessageUpdate :: HasCallStack => FilePath -> IO () +testDirectMessageUpdate = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + + -- msg id 1 + alice #> "@bob hello 🙂" + bob <# "alice> hello 🙂" + + -- msg id 2 + bob `send` "> @alice (hello) hi alice" + bob <# "@alice > hello 🙂" + bob <## " hi alice" + alice <# "bob> > hello 🙂" + alice <## " hi alice" + + alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "hello 🙂"), Nothing), ((0, "hi alice"), Just (1, "hello 🙂"))]) + bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hello 🙂"), Nothing), ((1, "hi alice"), Just (0, "hello 🙂"))]) + + alice ##> ("/_update item @2 " <> itemId 1 <> " text hey 👋") + alice <# "@bob [edited] hey 👋" + bob <# "alice> [edited] hey 👋" + + alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "hey 👋"), Nothing), ((0, "hi alice"), Just (1, "hello 🙂"))]) + bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hey 👋"), Nothing), ((1, "hi alice"), Just (0, "hello 🙂"))]) + + -- msg id 3 + bob `send` "> @alice (hey) hey alice" + bob <# "@alice > hey 👋" + bob <## " hey alice" + alice <# "bob> > hey 👋" + alice <## " hey alice" + + alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "hey 👋"), Nothing), ((0, "hi alice"), Just (1, "hello 🙂")), ((0, "hey alice"), Just (1, "hey 👋"))]) + bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hey 👋"), Nothing), ((1, "hi alice"), Just (0, "hello 🙂")), ((1, "hey alice"), Just (0, "hey 👋"))]) + + alice ##> ("/_update item @2 " <> itemId 1 <> " text greetings 🤝") + alice <# "@bob [edited] greetings 🤝" + bob <# "alice> [edited] greetings 🤝" + + alice #$> ("/_update item @2 " <> itemId 2 <> " text updating bob's message", id, "cannot update this item") + + alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "greetings 🤝"), Nothing), ((0, "hi alice"), Just (1, "hello 🙂")), ((0, "hey alice"), Just (1, "hey 👋"))]) + bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "greetings 🤝"), Nothing), ((1, "hi alice"), Just (0, "hello 🙂")), ((1, "hey alice"), Just (0, "hey 👋"))]) + + bob ##> ("/_update item @2 " <> itemId 2 <> " text hey Alice") + bob <# "@alice [edited] > hello 🙂" + bob <## " hey Alice" + alice <# "bob> [edited] > hello 🙂" + alice <## " hey Alice" + + bob ##> ("/_update item @2 " <> itemId 3 <> " text greetings Alice") + bob <# "@alice [edited] > hey 👋" + bob <## " greetings Alice" + alice <# "bob> [edited] > hey 👋" + alice <## " greetings Alice" + + alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((1, "greetings 🤝"), Nothing), ((0, "hey Alice"), Just (1, "hello 🙂")), ((0, "greetings Alice"), Just (1, "hey 👋"))]) + bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "greetings 🤝"), Nothing), ((1, "hey Alice"), Just (0, "hello 🙂")), ((1, "greetings Alice"), Just (0, "hey 👋"))]) + +testDirectMessageDelete :: HasCallStack => FilePath -> IO () +testDirectMessageDelete = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + + -- alice, bob: msg id 1 + alice #> "@bob hello 🙂" + bob <# "alice> hello 🙂" + + -- alice, bob: msg id 2 + bob `send` "> @alice (hello 🙂) hey alic" + bob <# "@alice > hello 🙂" + bob <## " hey alic" + alice <# "bob> > hello 🙂" + alice <## " hey alic" + + -- alice: deletes msg ids 1,2 + alice #$> ("/_delete item @2 " <> itemId 1 <> " internal", id, "message deleted") + alice #$> ("/_delete item @2 " <> itemId 2 <> " internal", id, "message deleted") + + alice @@@ [("@bob", "Voice messages: enabled")] + alice #$> ("/_get chat @2 count=100", chat, chatFeatures) + + -- alice: msg id 1 + bob ##> ("/_update item @2 " <> itemId 2 <> " text hey alice") + bob <# "@alice [edited] > hello 🙂" + bob <## " hey alice" + alice <# "bob> [edited] hey alice" + alice @@@ [("@bob", "hey alice")] + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hey alice")]) + + -- bob: marks deleted msg id 2 + bob #$> ("/_delete item @2 " <> itemId 2 <> " broadcast", id, "message marked deleted") + bob @@@ [("@alice", "hey alice [marked deleted]")] + alice <# "bob> [marked deleted] hey alice" + alice @@@ [("@bob", "hey alice [marked deleted]")] + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hey alice [marked deleted]")]) + + -- alice: deletes msg id 1 that was broadcast deleted by bob + alice #$> ("/_delete item @2 " <> itemId 1 <> " internal", id, "message deleted") + alice @@@ [("@bob", "Voice messages: enabled")] + alice #$> ("/_get chat @2 count=100", chat, chatFeatures) + + -- alice: msg id 1, bob: msg id 3 (quoting message alice deleted locally) + bob `send` "> @alice (hello 🙂) do you receive my messages?" + bob <# "@alice > hello 🙂" + bob <## " do you receive my messages?" + alice <# "bob> > hello 🙂" + alice <## " do you receive my messages?" + alice @@@ [("@bob", "do you receive my messages?")] + alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "do you receive my messages?"), Just (1, "hello 🙂"))]) + alice #$> ("/_delete item @2 " <> itemId 1 <> " broadcast", id, "cannot delete this item") + + -- alice: msg id 2, bob: msg id 4 + bob #> "@alice how are you?" + alice <# "bob> how are you?" + + -- alice: deletes msg id 2 + alice #$> ("/_delete item @2 " <> itemId 2 <> " internal", id, "message deleted") + + -- bob: marks deleted msg id 4 (that alice deleted locally) + bob #$> ("/_delete item @2 " <> itemId 4 <> " broadcast", id, "message marked deleted") + alice <## "bob> [deleted - original message not found]" + + alice @@@ [("@bob", "do you receive my messages?")] + alice #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "do you receive my messages?"), Just (1, "hello 🙂"))]) + bob @@@ [("@alice", "how are you? [marked deleted]")] + bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hello 🙂"), Nothing), ((1, "hey alice [marked deleted]"), Just (0, "hello 🙂")), ((1, "do you receive my messages?"), Just (0, "hello 🙂")), ((1, "how are you? [marked deleted]"), Nothing)]) + + -- bob: deletes msg ids 2,4 (that he has marked deleted) + bob #$> ("/_delete item @2 " <> itemId 2 <> " internal", id, "message deleted") + bob #$> ("/_delete item @2 " <> itemId 4 <> " internal", id, "message deleted") + bob #$> ("/_get chat @2 count=100", chat', chatFeatures' <> [((0, "hello 🙂"), Nothing), ((1, "do you receive my messages?"), Just (0, "hello 🙂"))]) + +testDirectLiveMessage :: HasCallStack => FilePath -> IO () +testDirectLiveMessage = + testChat2 aliceProfile bobProfile $ \alice bob -> do + connectUsers alice bob + -- non-empty live message is sent instantly + alice `send` "/live @bob hello" + bob <# "alice> [LIVE started] use /show [on/off/4] hello" + alice ##> ("/_update item @2 " <> itemId 1 <> " text hello there") + alice <# "@bob [LIVE] hello there" + bob <# "alice> [LIVE ended] hello there" + -- empty live message is also sent instantly + alice `send` "/live @bob" + bob <# "alice> [LIVE started] use /show [on/off/5]" + alice ##> ("/_update item @2 " <> itemId 2 <> " text hello 2") + alice <# "@bob [LIVE] hello 2" + bob <# "alice> [LIVE ended] hello 2" + +testRepeatAuthErrorsDisableContact :: HasCallStack => FilePath -> IO () +testRepeatAuthErrorsDisableContact = + testChat2 aliceProfile bobProfile $ \alice bob -> do + connectUsers alice bob + alice <##> bob + bob ##> "/d alice" + bob <## "alice: contact is deleted" + forM_ [1 .. authErrDisableCount] $ \_ -> sendAuth alice + alice <## "[bob] connection is disabled, to enable: /enable bob, to delete: /d bob" + alice ##> "@bob hey" + alice <## "bob: disabled, to enable: /enable bob, to delete: /d bob" + alice ##> "/enable bob" + alice <## "ok" + sendAuth alice + where + sendAuth alice = do + alice #> "@bob hey" + alice <## "[bob, contactId: 2, connId: 1] error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" + +testGetSetSMPServers :: HasCallStack => FilePath -> IO () +testGetSetSMPServers = + testChat2 aliceProfile bobProfile $ + \alice _ -> do + alice #$> ("/_smp 1", id, "smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001") + alice #$> ("/smp smp://1234-w==@smp1.example.im", id, "ok") + alice #$> ("/smp", id, "smp://1234-w==@smp1.example.im") + alice #$> ("/smp smp://1234-w==:password@smp1.example.im", id, "ok") + alice #$> ("/smp", id, "smp://1234-w==:password@smp1.example.im") + alice #$> ("/smp smp://2345-w==@smp2.example.im;smp://3456-w==@smp3.example.im:5224", id, "ok") + alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") + alice #$> ("/smp default", id, "ok") + alice #$> ("/smp", id, "smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001") + +testTestSMPServerConnection :: HasCallStack => FilePath -> IO () +testTestSMPServerConnection = + testChat2 aliceProfile bobProfile $ + \alice _ -> do + alice ##> "/smp test 1 smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=@localhost:7001" + alice <## "SMP server test passed" + -- to test with password: + -- alice <## "SMP server test failed at CreateQueue, error: SMP AUTH" + -- alice <## "Server requires authorization to create queues, check password" + alice ##> "/smp test 1 smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001" + alice <## "SMP server test passed" + alice ##> "/smp test 1 smp://LcJU@localhost:7001" + alice <## "SMP server test failed at Connect, error: BROKER smp://LcJU@localhost:7001 NETWORK" + alice <## "Possibly, certificate fingerprint in server address is incorrect" + +testAsyncInitiatingOffline :: HasCallStack => FilePath -> IO () +testAsyncInitiatingOffline tmp = do + putStrLn "testAsyncInitiatingOffline" + inv <- withNewTestChat tmp "alice" aliceProfile $ \alice -> do + threadDelay 250000 + alice ##> "/c" + getInvitation alice + withNewTestChat tmp "bob" bobProfile $ \bob -> do + threadDelay 250000 + bob ##> ("/c " <> inv) + bob <## "confirmation sent!" + withTestChat tmp "alice" $ \alice -> do + concurrently_ + (bob <## "alice (Alice): contact is connected") + (alice <## "bob (Bob): contact is connected") + +testAsyncAcceptingOffline :: HasCallStack => FilePath -> IO () +testAsyncAcceptingOffline tmp = do + putStrLn "testAsyncAcceptingOffline" + inv <- withNewTestChat tmp "alice" aliceProfile $ \alice -> do + alice ##> "/c" + getInvitation alice + withNewTestChat tmp "bob" bobProfile $ \bob -> do + threadDelay 250000 + bob ##> ("/c " <> inv) + bob <## "confirmation sent!" + withTestChat tmp "alice" $ \alice -> do + withTestChat tmp "bob" $ \bob -> do + concurrently_ + (bob <## "alice (Alice): contact is connected") + (alice <## "bob (Bob): contact is connected") + +testFullAsync :: HasCallStack => FilePath -> IO () +testFullAsync tmp = do + putStrLn "testFullAsync" + inv <- withNewTestChat tmp "alice" aliceProfile $ \alice -> do + threadDelay 250000 + alice ##> "/c" + getInvitation alice + withNewTestChat tmp "bob" bobProfile $ \bob -> do + threadDelay 250000 + bob ##> ("/c " <> inv) + bob <## "confirmation sent!" + withTestChat tmp "alice" $ \_ -> pure () -- connecting... notification in UI + withTestChat tmp "bob" $ \_ -> pure () -- connecting... notification in UI + withTestChat tmp "alice" $ \alice -> do + alice <## "1 contacts connected (use /cs for the list)" + alice <## "bob (Bob): contact is connected" + withTestChat tmp "bob" $ \bob -> do + bob <## "1 contacts connected (use /cs for the list)" + bob <## "alice (Alice): contact is connected" + +testFullAsyncV1 :: HasCallStack => FilePath -> IO () +testFullAsyncV1 tmp = do + putStrLn "testFullAsyncV1" + inv <- withNewAlice $ \alice -> do + putStrLn "1" + alice ##> "/c" + putStrLn "2" + getInvitation alice + putStrLn "3" + withNewBob $ \bob -> do + putStrLn "4" + bob ##> ("/c " <> inv) + putStrLn "5" + bob <## "confirmation sent!" + putStrLn "6" + withAlice $ \_ -> pure () + putStrLn "7" + withBob $ \_ -> pure () + putStrLn "8" + withAlice $ \alice -> do + putStrLn "9" + alice <## "1 contacts connected (use /cs for the list)" + putStrLn "10" + withBob $ \_ -> pure () + putStrLn "11" + withAlice $ \alice -> do + putStrLn "12" + alice <## "1 contacts connected (use /cs for the list)" + putStrLn "13" + alice <## "bob (Bob): contact is connected" + putStrLn "14" + withBob $ \bob -> do + putStrLn "15" + bob <## "1 contacts connected (use /cs for the list)" + putStrLn "16" + bob <## "alice (Alice): contact is connected" + where + withNewAlice = withNewTestChatV1 tmp "alice" aliceProfile + withAlice = withTestChatV1 tmp "alice" + withNewBob = withNewTestChatV1 tmp "bob" bobProfile + withBob = withTestChatV1 tmp "bob" + +testFullAsyncV1toV2 :: HasCallStack => FilePath -> IO () +testFullAsyncV1toV2 tmp = do + putStrLn "testFullAsyncV1toV2" + inv <- withNewAlice $ \alice -> do + putStrLn "1" + alice ##> "/c" + putStrLn "2" + getInvitation alice + putStrLn "3" + withNewBob $ \bob -> do + putStrLn "4" + bob ##> ("/c " <> inv) + putStrLn "5" + bob <## "confirmation sent!" + withAlice $ \_ -> pure () + putStrLn "6" + withBob $ \_ -> pure () + putStrLn "7" + withAlice $ \alice -> do + putStrLn "8" + alice <## "1 contacts connected (use /cs for the list)" + putStrLn "9" + withBob $ \_ -> pure () + putStrLn "10" + withAlice $ \alice -> do + putStrLn "11" + alice <## "1 contacts connected (use /cs for the list)" + putStrLn "12" + alice <## "bob (Bob): contact is connected" + putStrLn "13" + withBob $ \bob -> do + putStrLn "14" + bob <## "1 contacts connected (use /cs for the list)" + putStrLn "15" + bob <## "alice (Alice): contact is connected" + where + withNewAlice = withNewTestChat tmp "alice" aliceProfile + withAlice = withTestChat tmp "alice" + withNewBob = withNewTestChatV1 tmp "bob" bobProfile + withBob = withTestChatV1 tmp "bob" + +testFullAsyncV2toV1 :: HasCallStack => FilePath -> IO () +testFullAsyncV2toV1 tmp = do + putStrLn "testFullAsyncV2toV1" + inv <- withNewAlice $ \alice -> do + putStrLn "1" + alice ##> "/c" + putStrLn "2" + getInvitation alice + putStrLn "3" + withNewBob $ \bob -> do + putStrLn "4" + bob ##> ("/c " <> inv) + putStrLn "5" + bob <## "confirmation sent!" + putStrLn "6" + withAlice $ \_ -> pure () + putStrLn "7" + withBob $ \_ -> pure () + putStrLn "8" + withAlice $ \alice -> do + putStrLn "9" + alice <## "1 contacts connected (use /cs for the list)" + putStrLn "10" + withBob $ \_ -> pure () + putStrLn "11" + withAlice $ \alice -> do + putStrLn "12" + alice <## "1 contacts connected (use /cs for the list)" + putStrLn "13" + alice <## "bob (Bob): contact is connected" + putStrLn "14" + withBob $ \bob -> do + putStrLn "15" + bob <## "1 contacts connected (use /cs for the list)" + putStrLn "16" + bob <## "alice (Alice): contact is connected" + where + withNewAlice = withNewTestChatV1 tmp "alice" aliceProfile + {-# INLINE withNewAlice #-} + withAlice = withTestChatV1 tmp "alice" + {-# INLINE withAlice #-} + withNewBob = withNewTestChat tmp "bob" bobProfile + {-# INLINE withNewBob #-} + withBob = withTestChat tmp "bob" + {-# INLINE withBob #-} + +testCallType :: CallType +testCallType = CallType {media = CMVideo, capabilities = CallCapabilities {encryption = True}} + +testWebRTCSession :: WebRTCSession +testWebRTCSession = + WebRTCSession + { rtcSession = "{}", + rtcIceCandidates = "[]" + } + +testWebRTCCallOffer :: WebRTCCallOffer +testWebRTCCallOffer = + WebRTCCallOffer + { callType = testCallType, + rtcSession = testWebRTCSession + } + +serialize :: ToJSON a => a -> String +serialize = B.unpack . LB.toStrict . J.encode + +repeatM_ :: Int -> IO a -> IO () +repeatM_ n a = forM_ [1 .. n] $ const a + +testNegotiateCall :: HasCallStack => FilePath -> IO () +testNegotiateCall = + testChat2 aliceProfile bobProfile $ \alice bob -> do + connectUsers alice bob + -- just for testing db query + alice ##> "/_call get" + -- alice invite bob to call + alice ##> ("/_call invite @2 " <> serialize testCallType) + alice <## "ok" + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: calling...")]) + bob <## "alice wants to connect with you via WebRTC video call (e2e encrypted)" + repeatM_ 3 $ getTermLine bob + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: calling...")]) + -- bob accepts call by sending WebRTC offer + bob ##> ("/_call offer @2 " <> serialize testWebRTCCallOffer) + bob <## "ok" + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: accepted")]) + alice <## "bob accepted your WebRTC video call (e2e encrypted)" + repeatM_ 3 $ getTermLine alice + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: accepted")]) + -- alice confirms call by sending WebRTC answer + alice ##> ("/_call answer @2 " <> serialize testWebRTCSession) + alice <## "ok" + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: connecting...")]) + bob <## "alice continued the WebRTC call" + repeatM_ 3 $ getTermLine bob + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: connecting...")]) + -- participants can update calls as connected + alice ##> "/_call status @2 connected" + alice <## "ok" + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: in progress (00:00)")]) + bob ##> "/_call status @2 connected" + bob <## "ok" + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: in progress (00:00)")]) + -- either party can end the call + bob ##> "/_call end @2" + bob <## "ok" + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "incoming call: ended (00:00)")]) + alice <## "call with bob ended" + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "outgoing call: ended (00:00)")]) + +testMaintenanceMode :: HasCallStack => FilePath -> IO () +testMaintenanceMode tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChatOpts tmp testOpts {maintenance = True} "alice" aliceProfile $ \alice -> do + alice ##> "/c" + alice <## "error: chat not started" + alice ##> "/_start" + alice <## "chat started" + connectUsers alice bob + alice #> "@bob hi" + bob <# "alice> hi" + alice ##> "/_db export {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" + alice <## "error: chat not stopped" + alice ##> "/_stop" + alice <## "chat stopped" + alice ##> "/_start" + alice <## "chat started" + -- chat works after start + alice <## "1 contacts connected (use /cs for the list)" + alice #> "@bob hi again" + bob <# "alice> hi again" + bob #> "@alice hello" + alice <# "bob> hello" + -- export / delete / import + alice ##> "/_stop" + alice <## "chat stopped" + alice ##> "/_db export {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" + alice <## "ok" + doesFileExist "./tests/tmp/alice-chat.zip" `shouldReturn` True + alice ##> "/_db import {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" + alice <## "ok" + -- cannot start chat after import + alice ##> "/_start" + alice <## "error: chat store changed, please restart chat" + -- works after full restart + withTestChat tmp "alice" $ \alice -> testChatWorking alice bob + +testChatWorking :: HasCallStack => TestCC -> TestCC -> IO () +testChatWorking alice bob = do + alice <## "1 contacts connected (use /cs for the list)" + alice #> "@bob hello again" + bob <# "alice> hello again" + bob #> "@alice hello too" + alice <# "bob> hello too" + +testMaintenanceModeWithFiles :: HasCallStack => FilePath -> IO () +testMaintenanceModeWithFiles tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChatOpts tmp testOpts {maintenance = True} "alice" aliceProfile $ \alice -> do + alice ##> "/_start" + alice <## "chat started" + alice ##> "/_files_folder ./tests/tmp/alice_files" + alice <## "ok" + connectUsers alice bob + startFileTransferWithDest' bob alice "test.jpg" "136.5 KiB / 139737 bytes" Nothing + bob <## "completed sending file 1 (test.jpg) to alice" + alice <## "completed receiving file 1 (test.jpg) from bob" + src <- B.readFile "./tests/fixtures/test.jpg" + B.readFile "./tests/tmp/alice_files/test.jpg" `shouldReturn` src + threadDelay 500000 + alice ##> "/_stop" + alice <## "chat stopped" + alice ##> "/_db export {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" + alice <## "ok" + alice ##> "/_db delete" + alice <## "ok" + -- cannot start chat after delete + alice ##> "/_start" + alice <## "error: chat store changed, please restart chat" + doesDirectoryExist "./tests/tmp/alice_files" `shouldReturn` False + alice ##> "/_db import {\"archivePath\": \"./tests/tmp/alice-chat.zip\"}" + alice <## "ok" + B.readFile "./tests/tmp/alice_files/test.jpg" `shouldReturn` src + -- works after full restart + withTestChat tmp "alice" $ \alice -> testChatWorking alice bob + +testDatabaseEncryption :: HasCallStack => FilePath -> IO () +testDatabaseEncryption tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChatOpts tmp testOpts {maintenance = True} "alice" aliceProfile $ \alice -> do + alice ##> "/_start" + alice <## "chat started" + connectUsers alice bob + alice #> "@bob hi" + bob <# "alice> hi" + alice ##> "/db encrypt mykey" + alice <## "error: chat not stopped" + alice ##> "/db decrypt mykey" + alice <## "error: chat not stopped" + alice ##> "/_stop" + alice <## "chat stopped" + alice ##> "/db decrypt mykey" + alice <## "error: chat database is not encrypted" + alice ##> "/db encrypt mykey" + alice <## "ok" + alice ##> "/_start" + alice <## "error: chat store changed, please restart chat" + withTestChatOpts tmp testOpts {maintenance = True, dbKey = "mykey"} "alice" $ \alice -> do + alice ##> "/_start" + alice <## "chat started" + testChatWorking alice bob + alice ##> "/_stop" + alice <## "chat stopped" + alice ##> "/db key wrongkey nextkey" + alice <## "error encrypting database: wrong passphrase or invalid database file" + alice ##> "/db key mykey nextkey" + alice <## "ok" + alice ##> "/_db encryption {\"currentKey\":\"nextkey\",\"newKey\":\"anotherkey\"}" + alice <## "ok" + withTestChatOpts tmp testOpts {maintenance = True, dbKey = "anotherkey"} "alice" $ \alice -> do + alice ##> "/_start" + alice <## "chat started" + testChatWorking alice bob + alice ##> "/_stop" + alice <## "chat stopped" + alice ##> "/db decrypt anotherkey" + alice <## "ok" + withTestChat tmp "alice" $ \alice -> testChatWorking alice bob + +testMuteContact :: HasCallStack => FilePath -> IO () +testMuteContact = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice #> "@bob hello" + bob <# "alice> hello" + bob ##> "/mute alice" + bob <## "ok" + alice #> "@bob hi" + (bob "/contacts" + bob <## "alice (Alice) (muted, you can /unmute @alice)" + bob ##> "/unmute alice" + bob <## "ok" + bob ##> "/contacts" + bob <## "alice (Alice)" + alice #> "@bob hi again" + bob <# "alice> hi again" + +testMuteGroup :: HasCallStack => FilePath -> IO () +testMuteGroup = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + threadDelay 1000000 + alice #> "#team hello!" + concurrently_ + (bob <# "#team alice> hello!") + (cath <# "#team alice> hello!") + bob ##> "/mute #team" + bob <## "ok" + alice #> "#team hi" + concurrently_ + (bob hi") + bob ##> "/gs" + bob <## "#team (muted, you can /unmute #team)" + bob ##> "/unmute #team" + bob <## "ok" + alice #> "#team hi again" + concurrently_ + (bob <# "#team alice> hi again") + (cath <# "#team alice> hi again") + bob ##> "/gs" + bob <## "#team" + +testCreateSecondUser :: HasCallStack => FilePath -> IO () +testCreateSecondUser = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + connectUsers alice bob + + alice ##> "/create user alisa" + showActiveUser alice "alisa" + + -- connect using second user + connectUsers alice bob + alice #> "@bob hello" + bob <# "alisa> hello" + bob #> "@alisa hey" + alice <# "bob> hey" + + alice ##> "/user" + showActiveUser alice "alisa" + + alice ##> "/users" + alice <## "alice (Alice)" + alice <## "alisa (active)" + + -- receive message to first user + bob #> "@alice hey alice" + (alice, "alice") $<# "bob> hey alice" + + connectUsers alice cath + + -- set active user to first user + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + + alice ##> "/user" + showActiveUser alice "alice (Alice)" + + alice ##> "/users" + alice <## "alice (Alice) (active)" + alice <## "alisa" + + alice <##> bob + + cath #> "@alisa hey alisa" + (alice, "alisa") $<# "cath> hey alisa" + alice ##> "@cath hi cath" + alice <## "no contact cath" + + -- set active user to second user + alice ##> "/_user 2" + showActiveUser alice "alisa" + +testUsersSubscribeAfterRestart :: HasCallStack => FilePath -> IO () +testUsersSubscribeAfterRestart tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChat tmp "alice" aliceProfile $ \alice -> do + connectUsers alice bob + alice <##> bob + + alice ##> "/create user alisa" + showActiveUser alice "alisa" + connectUsers alice bob + alice <##> bob + + withTestChat tmp "alice" $ \alice -> do + -- second user is active + alice <## "1 contacts connected (use /cs for the list)" + alice <## "[user: alice] 1 contacts connected (use /cs for the list)" + + -- second user receives message + alice <##> bob + + -- first user receives message + bob #> "@alice hey alice" + (alice, "alice") $<# "bob> hey alice" + +testMultipleUserAddresses :: HasCallStack => FilePath -> IO () +testMultipleUserAddresses = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + alice ##> "/ad" + cLinkAlice <- getContactLink alice True + bob ##> ("/c " <> cLinkAlice) + alice <#? bob + alice @@@ [("<@bob", "")] + alice ##> "/ac bob" + alice <## "bob (Bob): accepting contact request..." + concurrently_ + (bob <## "alice (Alice): contact is connected") + (alice <## "bob (Bob): contact is connected") + threadDelay 100000 + alice @@@ [("@bob", "Voice messages: enabled")] + alice <##> bob + + alice ##> "/create user alisa" + showActiveUser alice "alisa" + + -- connect using second user address + alice ##> "/ad" + cLinkAlisa <- getContactLink alice True + bob ##> ("/c " <> cLinkAlisa) + alice <#? bob + alice #$> ("/_get chats 2 pcc=on", chats, [("<@bob", "")]) + alice ##> "/ac bob" + alice <## "bob (Bob): accepting contact request..." + concurrently_ + (bob <## "alisa: contact is connected") + (alice <## "bob (Bob): contact is connected") + threadDelay 100000 + alice #$> ("/_get chats 2 pcc=on", chats, [("@bob", "Voice messages: enabled")]) + alice <##> bob + + bob #> "@alice hey alice" + (alice, "alice") $<# "bob> hey alice" + + -- delete first user address + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + + alice ##> "/da" + alice <## "Your chat address is deleted - accepted contacts will remain connected." + alice <## "To create a new chat address use /ad" + + -- second user receives request when not active + cath ##> ("/c " <> cLinkAlisa) + cath <## "connection request sent!" + alice <## "[user: alisa] cath (Catherine) wants to connect to you!" + alice <## "to accept: /ac cath" + alice <## "to reject: /rc cath (the sender will NOT be notified)" + + -- accept request + alice ##> "/user alisa" + showActiveUser alice "alisa" + + alice ##> "/ac cath" + alice <## "cath (Catherine): accepting contact request..." + concurrently_ + (cath <## "alisa: contact is connected") + (alice <## "cath (Catherine): contact is connected") + threadDelay 100000 + alice #$> ("/_get chats 2 pcc=on", chats, [("@cath", "Voice messages: enabled"), ("@bob", "hey")]) + alice <##> cath + + -- first user doesn't have cath as contact + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice @@@ [("@bob", "hey alice")] + +testCreateUserDefaultServers :: HasCallStack => FilePath -> IO () +testCreateUserDefaultServers = + testChat2 aliceProfile bobProfile $ + \alice _ -> do + alice #$> ("/smp smp://2345-w==@smp2.example.im;smp://3456-w==@smp3.example.im:5224", id, "ok") + alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") + + alice ##> "/create user alisa" + showActiveUser alice "alisa" + + alice #$> ("/smp", id, "smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001") + + -- with same_smp=off + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") + + alice ##> "/create user same_smp=off alisa2" + showActiveUser alice "alisa2" + + alice #$> ("/smp", id, "smp://LcJUMfVhwD8yxjAiSaDzzGF3-kLG4Uh0Fl_ZIjrRwjI=:server_password@localhost:7001") + +testCreateUserSameServers :: HasCallStack => FilePath -> IO () +testCreateUserSameServers = + testChat2 aliceProfile bobProfile $ + \alice _ -> do + alice #$> ("/smp smp://2345-w==@smp2.example.im;smp://3456-w==@smp3.example.im:5224", id, "ok") + alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") + + alice ##> "/create user same_smp=on alisa" + showActiveUser alice "alisa" + + alice #$> ("/smp", id, "smp://2345-w==@smp2.example.im, smp://3456-w==@smp3.example.im:5224") + +testDeleteUser :: HasCallStack => FilePath -> IO () +testDeleteUser = + testChat4 aliceProfile bobProfile cathProfile danProfile $ + \alice bob cath dan -> do + connectUsers alice bob + + -- cannot delete active user + + alice ##> "/_delete user 1 del_smp=off" + alice <## "cannot delete active user" + + -- delete user without deleting SMP queues + + alice ##> "/create user alisa" + showActiveUser alice "alisa" + + connectUsers alice cath + alice <##> cath + + alice ##> "/users" + alice <## "alice (Alice)" + alice <## "alisa (active)" + + alice ##> "/_delete user 1 del_smp=off" + alice <## "ok" + + alice ##> "/users" + alice <## "alisa (active)" + + bob #> "@alice hey" + -- no connection authorization error - connection wasn't deleted + (alice "/delete user alisa" + alice <## "cannot delete active user" + + alice ##> "/users" + alice <## "alisa (active)" + + alice <##> cath + + -- delete user deleting SMP queues + + alice ##> "/create user alisa2" + showActiveUser alice "alisa2" + + connectUsers alice dan + alice <##> dan + + alice ##> "/users" + alice <## "alisa" + alice <## "alisa2 (active)" + + alice ##> "/delete user alisa" + alice <### ["ok", "completed deleting user"] + + alice ##> "/users" + alice <## "alisa2 (active)" + + cath #> "@alisa hey" + cath <## "[alisa, contactId: 2, connId: 1] error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" + (alice dan + +testUsersDifferentCIExpirationTTL :: HasCallStack => FilePath -> IO () +testUsersDifferentCIExpirationTTL tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChatCfg tmp cfg "alice" aliceProfile $ \alice -> do + -- first user messages + connectUsers alice bob + + alice #> "@bob alice 1" + bob <# "alice> alice 1" + bob #> "@alice alice 2" + alice <# "bob> alice 2" + + -- second user messages + alice ##> "/create user alisa" + showActiveUser alice "alisa" + connectUsers alice bob + + alice #> "@bob alisa 1" + bob <# "alisa> alisa 1" + bob #> "@alisa alisa 2" + alice <# "bob> alisa 2" + + -- set ttl for first user + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/_ttl 1 1", id, "ok") + + -- set ttl for second user + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/_ttl 2 3", id, "ok") + + -- first user messages + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/ttl", id, "old messages are set to be deleted after: 1 second(s)") + + alice #> "@bob alice 3" + bob <# "alice> alice 3" + bob #> "@alice alice 4" + alice <# "bob> alice 4" + + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "alice 1"), (0, "alice 2"), (1, "alice 3"), (0, "alice 4")]) + + -- second user messages + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/ttl", id, "old messages are set to be deleted after: 3 second(s)") + + alice #> "@bob alisa 3" + bob <# "alisa> alisa 3" + bob #> "@alisa alisa 4" + alice <# "bob> alisa 4" + + alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) + + threadDelay 2000000 + + -- messages both before and after setting chat item ttl are deleted + -- first user messages + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/_get chat @2 count=100", chat, []) + + -- second user messages + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) + + threadDelay 2000000 + + alice #$> ("/_get chat @4 count=100", chat, []) + where + cfg = testCfg {ciExpirationInterval = 500000} + +testUsersRestartCIExpiration :: HasCallStack => FilePath -> IO () +testUsersRestartCIExpiration tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChatCfg tmp cfg "alice" aliceProfile $ \alice -> do + -- set ttl for first user + alice #$> ("/_ttl 1 1", id, "ok") + connectUsers alice bob + + -- create second user and set ttl + alice ##> "/create user alisa" + showActiveUser alice "alisa" + alice #$> ("/_ttl 2 3", id, "ok") + connectUsers alice bob + + -- first user messages + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + + alice #> "@bob alice 1" + bob <# "alice> alice 1" + bob #> "@alice alice 2" + alice <# "bob> alice 2" + + -- second user messages + alice ##> "/user alisa" + showActiveUser alice "alisa" + + alice #> "@bob alisa 1" + bob <# "alisa> alisa 1" + bob #> "@alisa alisa 2" + alice <# "bob> alisa 2" + + -- first user will be active on restart + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + + withTestChatCfg tmp cfg "alice" $ \alice -> do + alice <## "1 contacts connected (use /cs for the list)" + alice <## "[user: alisa] 1 contacts connected (use /cs for the list)" + + -- first user messages + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/ttl", id, "old messages are set to be deleted after: 1 second(s)") + + alice #> "@bob alice 3" + bob <# "alice> alice 3" + bob #> "@alice alice 4" + alice <# "bob> alice 4" + + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "alice 1"), (0, "alice 2"), (1, "alice 3"), (0, "alice 4")]) + + -- second user messages + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/ttl", id, "old messages are set to be deleted after: 3 second(s)") + + alice #> "@bob alisa 3" + bob <# "alisa> alisa 3" + bob #> "@alisa alisa 4" + alice <# "bob> alisa 4" + + alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) + + threadDelay 2000000 + + -- messages both before and after restart are deleted + -- first user messages + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/_get chat @2 count=100", chat, []) + + -- second user messages + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) + + threadDelay 2000000 + + alice #$> ("/_get chat @4 count=100", chat, []) + where + cfg = testCfg {ciExpirationInterval = 500000} + +testEnableCIExpirationOnlyForOneUser :: HasCallStack => FilePath -> IO () +testEnableCIExpirationOnlyForOneUser tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChatCfg tmp cfg "alice" aliceProfile $ \alice -> do + -- first user messages + connectUsers alice bob + + alice #> "@bob alice 1" + bob <# "alice> alice 1" + bob #> "@alice alice 2" + alice <# "bob> alice 2" + + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "alice 1"), (0, "alice 2")]) + + -- second user messages before first user sets ttl + alice ##> "/create user alisa" + showActiveUser alice "alisa" + connectUsers alice bob + + alice #> "@bob alisa 1" + bob <# "alisa> alisa 1" + bob #> "@alisa alisa 2" + alice <# "bob> alisa 2" + + -- set ttl for first user + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/_ttl 1 1", id, "ok") + + -- second user messages after first user sets ttl + alice ##> "/user alisa" + showActiveUser alice "alisa" + + alice #> "@bob alisa 3" + bob <# "alisa> alisa 3" + bob #> "@alisa alisa 4" + alice <# "bob> alisa 4" + + alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) + + threadDelay 2000000 + + -- messages are deleted for first user + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/_get chat @2 count=100", chat, []) + + -- messages are not deleted for second user + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) + + withTestChatCfg tmp cfg "alice" $ \alice -> do + alice <## "1 contacts connected (use /cs for the list)" + alice <## "[user: alice] 1 contacts connected (use /cs for the list)" + + -- messages are not deleted for second user after restart + alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4")]) + + alice #> "@bob alisa 5" + bob <# "alisa> alisa 5" + bob #> "@alisa alisa 6" + alice <# "bob> alisa 6" + + threadDelay 2000000 + + -- new messages are not deleted for second user + alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2"), (1, "alisa 3"), (0, "alisa 4"), (1, "alisa 5"), (0, "alisa 6")]) + where + cfg = testCfg {ciExpirationInterval = 500000} + +testDisableCIExpirationOnlyForOneUser :: HasCallStack => FilePath -> IO () +testDisableCIExpirationOnlyForOneUser tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChatCfg tmp cfg "alice" aliceProfile $ \alice -> do + -- set ttl for first user + alice #$> ("/_ttl 1 1", id, "ok") + connectUsers alice bob + + -- create second user and set ttl + alice ##> "/create user alisa" + showActiveUser alice "alisa" + alice #$> ("/_ttl 2 1", id, "ok") + connectUsers alice bob + + -- first user disables expiration + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/ttl none", id, "ok") + alice #$> ("/ttl", id, "old messages are not being deleted") + + -- second user still has ttl configured + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/ttl", id, "old messages are set to be deleted after: 1 second(s)") + + alice #> "@bob alisa 1" + bob <# "alisa> alisa 1" + bob #> "@alisa alisa 2" + alice <# "bob> alisa 2" + + alice #$> ("/_get chat @4 count=100", chat, chatFeatures <> [(1, "alisa 1"), (0, "alisa 2")]) + + threadDelay 2000000 + + -- second user messages are deleted + alice #$> ("/_get chat @4 count=100", chat, []) + + withTestChatCfg tmp cfg "alice" $ \alice -> do + alice <## "1 contacts connected (use /cs for the list)" + alice <## "[user: alice] 1 contacts connected (use /cs for the list)" + + -- second user still has ttl configured after restart + alice #$> ("/ttl", id, "old messages are set to be deleted after: 1 second(s)") + + alice #> "@bob alisa 3" + bob <# "alisa> alisa 3" + bob #> "@alisa alisa 4" + alice <# "bob> alisa 4" + + alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 3"), (0, "alisa 4")]) + + threadDelay 2000000 + + -- second user messages are deleted + alice #$> ("/_get chat @4 count=100", chat, []) + where + cfg = testCfg {ciExpirationInterval = 500000} + +testUsersTimedMessages :: HasCallStack => FilePath -> IO () +testUsersTimedMessages tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChat tmp "alice" aliceProfile $ \alice -> do + connectUsers alice bob + configureTimedMessages alice bob "2" "1" + + -- create second user and configure timed messages for contact + alice ##> "/create user alisa" + showActiveUser alice "alisa" + connectUsers alice bob + configureTimedMessages alice bob "4" "2" + + -- first user messages + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + + alice #> "@bob alice 1" + bob <# "alice> alice 1" + bob #> "@alice alice 2" + alice <# "bob> alice 2" + + -- second user messages + alice ##> "/user alisa" + showActiveUser alice "alisa" + + alice #> "@bob alisa 1" + bob <# "alisa> alisa 1" + bob #> "@alisa alisa 2" + alice <# "bob> alisa 2" + + -- messages are deleted after ttl + threadDelay 500000 + + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/_get chat @2 count=100", chat, [(1, "alice 1"), (0, "alice 2")]) + + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 1"), (0, "alisa 2")]) + + threadDelay 1000000 + + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/_get chat @2 count=100", chat, []) + + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 1"), (0, "alisa 2")]) + + threadDelay 1000000 + + alice ##> "/user" + showActiveUser alice "alisa" + alice #$> ("/_get chat @4 count=100", chat, []) + + -- first user messages + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + + alice #> "@bob alice 3" + bob <# "alice> alice 3" + bob #> "@alice alice 4" + alice <# "bob> alice 4" + + -- second user messages + alice ##> "/user alisa" + showActiveUser alice "alisa" + + alice #> "@bob alisa 3" + bob <# "alisa> alisa 3" + bob #> "@alisa alisa 4" + alice <# "bob> alisa 4" + + withTestChat tmp "alice" $ \alice -> do + alice <## "1 contacts connected (use /cs for the list)" + alice <## "[user: alice] 1 contacts connected (use /cs for the list)" + + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/_get chat @2 count=100", chat, [(1, "alice 3"), (0, "alice 4")]) + + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 3"), (0, "alisa 4")]) + + -- messages are deleted after restart + threadDelay 1000000 + + alice ##> "/user alice" + showActiveUser alice "alice (Alice)" + alice #$> ("/_get chat @2 count=100", chat, []) + + alice ##> "/user alisa" + showActiveUser alice "alisa" + alice #$> ("/_get chat @4 count=100", chat, [(1, "alisa 3"), (0, "alisa 4")]) + + threadDelay 1000000 + + alice ##> "/user" + showActiveUser alice "alisa" + alice #$> ("/_get chat @4 count=100", chat, []) + where + configureTimedMessages :: HasCallStack => TestCC -> TestCC -> String -> String -> IO () + configureTimedMessages alice bob bobId ttl = do + aliceName <- userName alice + alice ##> ("/_set prefs @" <> bobId <> " {\"timedMessages\": {\"allow\": \"yes\", \"ttl\": " <> ttl <> "}}") + alice <## "you updated preferences for bob:" + alice <## ("Disappearing messages: off (you allow: yes (" <> ttl <> " sec), contact allows: no)") + bob <## (aliceName <> " updated preferences for you:") + bob <## ("Disappearing messages: off (you allow: no, contact allows: yes (" <> ttl <> " sec))") + bob ##> ("/set disappear @" <> aliceName <> " yes") + bob <## ("you updated preferences for " <> aliceName <> ":") + bob <## ("Disappearing messages: enabled (you allow: yes (" <> ttl <> " sec), contact allows: yes (" <> ttl <> " sec))") + alice <## "bob updated preferences for you:" + alice <## ("Disappearing messages: enabled (you allow: yes (" <> ttl <> " sec), contact allows: yes (" <> ttl <> " sec))") + alice #$> ("/clear bob", id, "bob: all messages are removed locally ONLY") -- to remove feature items + +testSetChatItemTTL :: HasCallStack => FilePath -> IO () +testSetChatItemTTL = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice #> "@bob 1" + bob <# "alice> 1" + bob #> "@alice 2" + alice <# "bob> 2" + -- chat item with file + alice #$> ("/_files_folder ./tests/tmp/app_files", id, "ok") + copyFile "./tests/fixtures/test.jpg" "./tests/tmp/app_files/test.jpg" + alice ##> "/_send @2 json {\"filePath\": \"test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" + alice <# "/f @bob test.jpg" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + -- above items should be deleted after we set ttl + threadDelay 3000000 + alice #> "@bob 3" + bob <# "alice> 3" + bob #> "@alice 4" + alice <# "bob> 4" + alice #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((1, "1"), Nothing), ((0, "2"), Nothing), ((1, ""), Just "test.jpg"), ((1, "3"), Nothing), ((0, "4"), Nothing)]) + checkActionDeletesFile "./tests/tmp/app_files/test.jpg" $ + alice #$> ("/_ttl 1 2", id, "ok") + alice #$> ("/_get chat @2 count=100", chat, [(1, "3"), (0, "4")]) -- when expiration is turned on, first cycle is synchronous + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "1"), (1, "2"), (0, ""), (0, "3"), (1, "4")]) + alice #$> ("/_ttl 1", id, "old messages are set to be deleted after: 2 second(s)") + alice #$> ("/ttl week", id, "ok") + alice #$> ("/ttl", id, "old messages are set to be deleted after: one week") + alice #$> ("/ttl none", id, "ok") + alice #$> ("/ttl", id, "old messages are not being deleted") + +testSwitchContact :: HasCallStack => FilePath -> IO () +testSwitchContact = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice #$> ("/switch bob", id, "ok") + bob <## "alice started changing address for you" + alice <## "bob: you started changing address" + bob <## "alice changed address for you" + alice <## "bob: you changed address" + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "started changing address..."), (1, "you changed address")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "started changing address for you..."), (0, "changed address for you")]) + alice <##> bob + +testSwitchGroupMember :: HasCallStack => FilePath -> IO () +testSwitchGroupMember = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + createGroup2 "team" alice bob + alice #$> ("/switch #team bob", id, "ok") + bob <## "#team: alice started changing address for you" + alice <## "#team: you started changing address for bob" + bob <## "#team: alice changed address for you" + alice <## "#team: you changed address for bob" + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "started changing address for bob..."), (1, "you changed address for bob")]) + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "started changing address for you..."), (0, "changed address for you")]) + alice #> "#team hey" + bob <# "#team alice> hey" + bob #> "#team hi" + alice <# "#team bob> hi" + +testMarkContactVerified :: HasCallStack => FilePath -> IO () +testMarkContactVerified = + testChat2 aliceProfile bobProfile $ \alice bob -> do + connectUsers alice bob + alice ##> "/i bob" + bobInfo alice + alice <## "connection not verified, use /code command to see security code" + alice ##> "/code bob" + bCode <- getTermLine alice + bob ##> "/code alice" + aCode <- getTermLine bob + bCode `shouldBe` aCode + alice ##> "/verify bob 123" + alice <##. "connection not verified, current code is " + alice ##> ("/verify bob " <> aCode) + alice <## "connection verified" + alice ##> "/i bob" + bobInfo alice + alice <## "connection verified" + alice ##> "/verify bob" + alice <##. "connection not verified, current code is " + alice ##> "/i bob" + bobInfo alice + alice <## "connection not verified, use /code command to see security code" + where + bobInfo :: HasCallStack => TestCC -> IO () + bobInfo alice = do + alice <## "contact ID: 2" + alice <## "receiving messages via: localhost" + alice <## "sending messages via: localhost" + alice <## "you've shared main profile with this contact" + +testMarkGroupMemberVerified :: HasCallStack => FilePath -> IO () +testMarkGroupMemberVerified = + testChat2 aliceProfile bobProfile $ \alice bob -> do + createGroup2 "team" alice bob + alice ##> "/i #team bob" + bobInfo alice + alice <## "connection not verified, use /code command to see security code" + alice ##> "/code #team bob" + bCode <- getTermLine alice + bob ##> "/code #team alice" + aCode <- getTermLine bob + bCode `shouldBe` aCode + alice ##> "/verify #team bob 123" + alice <##. "connection not verified, current code is " + alice ##> ("/verify #team bob " <> aCode) + alice <## "connection verified" + alice ##> "/i #team bob" + bobInfo alice + alice <## "connection verified" + alice ##> "/verify #team bob" + alice <##. "connection not verified, current code is " + alice ##> "/i #team bob" + bobInfo alice + alice <## "connection not verified, use /code command to see security code" + where + bobInfo :: HasCallStack => TestCC -> IO () + bobInfo alice = do + alice <## "group ID: 1" + alice <## "member ID: 2" + alice <## "receiving messages via: localhost" + alice <## "sending messages via: localhost" diff --git a/tests/ChatTests/Files.hs b/tests/ChatTests/Files.hs new file mode 100644 index 000000000..06211756d --- /dev/null +++ b/tests/ChatTests/Files.hs @@ -0,0 +1,934 @@ +{-# LANGUAGE OverloadedStrings #-} +{-# LANGUAGE PostfixOperators #-} + +module ChatTests.Files where + +import ChatClient +import ChatTests.Utils +import Control.Concurrent (threadDelay) +import Control.Concurrent.Async (concurrently_) +import qualified Data.ByteString.Char8 as B +import Simplex.Chat.Controller (ChatConfig (..), InlineFilesConfig (..), defaultInlineFilesConfig) +import Simplex.Chat.Options (ChatOpts (..)) +import Simplex.Messaging.Util (unlessM) +import System.Directory (copyFile, doesFileExist) +import System.FilePath (()) +import Test.Hspec + +chatFileTests :: SpecWith FilePath +chatFileTests = do + describe "sending and receiving files" $ do + describe "send and receive file" $ fileTestMatrix2 runTestFileTransfer + it "send and receive file inline (without accepting)" testInlineFileTransfer + it "accept inline file transfer, sender cancels during transfer" testAcceptInlineFileSndCancelDuringTransfer + it "send and receive small file inline (default config)" testSmallInlineFileTransfer + it "small file sent without acceptance is ignored in terminal by default" testSmallInlineFileIgnored + it "receive file inline with inline=on option" testReceiveInline + describe "send and receive a small file" $ fileTestMatrix2 runTestSmallFileTransfer + describe "sender cancelled file transfer before transfer" $ fileTestMatrix2 runTestFileSndCancelBeforeTransfer + it "sender cancelled file transfer during transfer" testFileSndCancelDuringTransfer + it "recipient cancelled file transfer" testFileRcvCancel + describe "send and receive file to group" $ fileTestMatrix3 runTestGroupFileTransfer + it "send and receive file inline to group (without accepting)" testInlineGroupFileTransfer + it "send and receive small file inline to group (default config)" testSmallInlineGroupFileTransfer + it "small file sent without acceptance is ignored in terminal by default" testSmallInlineGroupFileIgnored + describe "sender cancelled group file transfer before transfer" $ fileTestMatrix3 runTestGroupFileSndCancelBeforeTransfer + describe "messages with files" $ do + describe "send and receive message with file" $ fileTestMatrix2 runTestMessageWithFile + it "send and receive image" testSendImage + it "files folder: send and receive image" testFilesFoldersSendImage + it "files folder: sender deleted file during transfer" testFilesFoldersImageSndDelete + it "files folder: recipient deleted file during transfer" testFilesFoldersImageRcvDelete + it "send and receive image with text and quote" testSendImageWithTextAndQuote + describe "send and receive image to group" testGroupSendImage + it "send and receive image with text and quote to group" testGroupSendImageWithTextAndQuote + describe "async sending and receiving files" $ do + it "send and receive file, sender restarts" testAsyncFileTransferSenderRestarts + it "send and receive file, receiver restarts" testAsyncFileTransferReceiverRestarts + xdescribe "send and receive file, fully asynchronous" $ do + it "v2" testAsyncFileTransfer + it "v1" testAsyncFileTransferV1 + xit "send and receive file to group, fully asynchronous" testAsyncGroupFileTransfer + +runTestFileTransfer :: HasCallStack => TestCC -> TestCC -> IO () +runTestFileTransfer alice bob = do + connectUsers alice bob + startFileTransfer' alice bob "test.pdf" "266.0 KiB / 272376 bytes" + concurrentlyN_ + [ do + bob #> "@alice receiving here..." + bob <## "completed receiving file 1 (test.pdf) from alice", + alice + <### [ WithTime "bob> receiving here...", + "completed sending file 1 (test.pdf) to bob" + ] + ] + src <- B.readFile "./tests/fixtures/test.pdf" + dest <- B.readFile "./tests/tmp/test.pdf" + dest `shouldBe` src + +testInlineFileTransfer :: HasCallStack => FilePath -> IO () +testInlineFileTransfer = + testChatCfg2 cfg aliceProfile bobProfile $ \alice bob -> do + connectUsers alice bob + bob ##> "/_files_folder ./tests/tmp/" + bob <## "ok" + alice ##> "/_send @2 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/test.jpg\"}" + alice <# "@bob voice message (00:10)" + alice <# "/f @bob ./tests/fixtures/test.jpg" + -- below is not shown in "sent" mode + -- alice <## "use /fc 1 to cancel sending" + bob <# "alice> voice message (00:10)" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + -- below is not shown in "sent" mode + -- bob <## "use /fr 1 [/ | ] to receive it" + bob <## "started receiving file 1 (test.jpg) from alice" + concurrently_ + (alice <## "completed sending file 1 (test.jpg) to bob") + (bob <## "completed receiving file 1 (test.jpg) from alice") + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/test.jpg" + dest `shouldBe` src + where + cfg = testCfg {inlineFiles = defaultInlineFilesConfig {offerChunks = 100, sendChunks = 100, receiveChunks = 100}} + +testAcceptInlineFileSndCancelDuringTransfer :: HasCallStack => FilePath -> IO () +testAcceptInlineFileSndCancelDuringTransfer = + testChatCfg2 cfg aliceProfile bobProfile $ \alice bob -> do + connectUsers alice bob + bob ##> "/_files_folder ./tests/tmp/" + bob <## "ok" + alice #> "/f @bob ./tests/fixtures/test_1MB.pdf" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> sends file test_1MB.pdf (1017.7 KiB / 1042157 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1 inline=on" + bob <## "saving file 1 from alice to test_1MB.pdf" + alice <## "started sending file 1 (test_1MB.pdf) to bob" + bob <## "started receiving file 1 (test_1MB.pdf) from alice" + alice ##> "/fc 1" -- test that inline file cancel doesn't delete contact connection + concurrentlyN_ + [ do + alice <##. "cancelled sending file 1 (test_1MB.pdf)" + alice <## "completed sending file 1 (test_1MB.pdf) to bob", + do + bob <## "completed receiving file 1 (test_1MB.pdf) from alice" + bob <## "alice cancelled sending file 1 (test_1MB.pdf)" + ] + alice #> "@bob hi" + bob <# "alice> hi" + bob #> "@alice hey" + alice <# "bob> hey" + where + cfg = testCfg {inlineFiles = defaultInlineFilesConfig {offerChunks = 100, receiveChunks = 50}} + +testSmallInlineFileTransfer :: HasCallStack => FilePath -> IO () +testSmallInlineFileTransfer = + testChat2 aliceProfile bobProfile $ \alice bob -> do + connectUsers alice bob + bob ##> "/_files_folder ./tests/tmp/" + bob <## "ok" + alice ##> "/_send @2 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" + alice <# "@bob voice message (00:10)" + alice <# "/f @bob ./tests/fixtures/logo.jpg" + -- below is not shown in "sent" mode + -- alice <## "use /fc 1 to cancel sending" + bob <# "alice> voice message (00:10)" + bob <# "alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" + -- below is not shown in "sent" mode + -- bob <## "use /fr 1 [/ | ] to receive it" + bob <## "started receiving file 1 (logo.jpg) from alice" + concurrently_ + (alice <## "completed sending file 1 (logo.jpg) to bob") + (bob <## "completed receiving file 1 (logo.jpg) from alice") + src <- B.readFile "./tests/fixtures/logo.jpg" + dest <- B.readFile "./tests/tmp/logo.jpg" + dest `shouldBe` src + +testSmallInlineFileIgnored :: HasCallStack => FilePath -> IO () +testSmallInlineFileIgnored tmp = do + withNewTestChat tmp "alice" aliceProfile $ \alice -> + withNewTestChatOpts tmp testOpts {allowInstantFiles = False} "bob" bobProfile $ \bob -> do + connectUsers alice bob + bob ##> "/_files_folder ./tests/tmp/" + bob <## "ok" + alice ##> "/_send @2 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" + alice <# "@bob voice message (00:10)" + alice <# "/f @bob ./tests/fixtures/logo.jpg" + -- below is not shown in "sent" mode + -- alice <## "use /fc 1 to cancel sending" + bob <# "alice> voice message (00:10)" + bob <# "alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob <## "A small file sent without acceptance - you can enable receiving such files with -f option." + -- below is not shown in "sent" mode + -- bob <## "use /fr 1 [/ | ] to receive it" + alice <## "completed sending file 1 (logo.jpg) to bob" + bob ##> "/fr 1" + bob <## "file is already being received: logo.jpg" + +testReceiveInline :: HasCallStack => FilePath -> IO () +testReceiveInline = + testChatCfg2 cfg aliceProfile bobProfile $ \alice bob -> do + connectUsers alice bob + alice #> "/f @bob ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1 inline=on ./tests/tmp" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + alice <## "started sending file 1 (test.jpg) to bob" + alice <## "completed sending file 1 (test.jpg) to bob" + bob <## "started receiving file 1 (test.jpg) from alice" + bob <## "completed receiving file 1 (test.jpg) from alice" + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/test.jpg" + dest `shouldBe` src + where + cfg = testCfg {inlineFiles = defaultInlineFilesConfig {offerChunks = 10, receiveChunks = 5}} + +runTestSmallFileTransfer :: HasCallStack => TestCC -> TestCC -> IO () +runTestSmallFileTransfer alice bob = do + connectUsers alice bob + alice #> "/f @bob ./tests/fixtures/test.txt" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> sends file test.txt (11 bytes / 11 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1 ./tests/tmp" + bob <## "saving file 1 from alice to ./tests/tmp/test.txt" + concurrentlyN_ + [ do + bob <## "started receiving file 1 (test.txt) from alice" + bob <## "completed receiving file 1 (test.txt) from alice", + do + alice <## "started sending file 1 (test.txt) to bob" + alice <## "completed sending file 1 (test.txt) to bob" + ] + src <- B.readFile "./tests/fixtures/test.txt" + dest <- B.readFile "./tests/tmp/test.txt" + dest `shouldBe` src + +runTestFileSndCancelBeforeTransfer :: HasCallStack => TestCC -> TestCC -> IO () +runTestFileSndCancelBeforeTransfer alice bob = do + connectUsers alice bob + alice #> "/f @bob ./tests/fixtures/test.txt" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> sends file test.txt (11 bytes / 11 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + alice ##> "/fc 1" + concurrentlyN_ + [ alice <##. "cancelled sending file 1 (test.txt)", + bob <## "alice cancelled sending file 1 (test.txt)" + ] + alice ##> "/fs 1" + alice + <##.. [ "sending file 1 (test.txt): no file transfers", + "sending file 1 (test.txt) cancelled: bob" + ] + alice <## "file transfer cancelled" + bob ##> "/fs 1" + bob <## "receiving file 1 (test.txt) cancelled" + bob ##> "/fr 1 ./tests/tmp" + bob <## "file cancelled: test.txt" + +testFileSndCancelDuringTransfer :: HasCallStack => FilePath -> IO () +testFileSndCancelDuringTransfer = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + startFileTransfer' alice bob "test_1MB.pdf" "1017.7 KiB / 1042157 bytes" + alice ##> "/fc 1" + concurrentlyN_ + [ do + alice <## "cancelled sending file 1 (test_1MB.pdf) to bob" + alice ##> "/fs 1" + alice <## "sending file 1 (test_1MB.pdf) cancelled: bob" + alice <## "file transfer cancelled", + do + bob <## "alice cancelled sending file 1 (test_1MB.pdf)" + bob ##> "/fs 1" + bob <## "receiving file 1 (test_1MB.pdf) cancelled, received part path: ./tests/tmp/test_1MB.pdf" + ] + checkPartialTransfer "test_1MB.pdf" + +testFileRcvCancel :: HasCallStack => FilePath -> IO () +testFileRcvCancel = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + startFileTransfer alice bob + bob ##> "/fs 1" + getTermLine bob >>= (`shouldStartWith` "receiving file 1 (test.jpg) progress") + waitFileExists "./tests/tmp/test.jpg" + bob ##> "/fc 1" + concurrentlyN_ + [ do + bob <## "cancelled receiving file 1 (test.jpg) from alice" + bob ##> "/fs 1" + bob <## "receiving file 1 (test.jpg) cancelled, received part path: ./tests/tmp/test.jpg", + do + alice <## "bob cancelled receiving file 1 (test.jpg)" + alice ##> "/fs 1" + alice <## "sending file 1 (test.jpg) cancelled: bob" + ] + checkPartialTransfer "test.jpg" + +runTestGroupFileTransfer :: HasCallStack => TestCC -> TestCC -> TestCC -> IO () +runTestGroupFileTransfer alice bob cath = do + createGroup3 "team" alice bob cath + alice #> "/f #team ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + concurrentlyN_ + [ do + bob <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it", + do + cath <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + cath <## "use /fr 1 [/ | ] to receive it" + ] + alice ##> "/fs 1" + getTermLine alice >>= (`shouldStartWith` "sending file 1 (test.jpg): no file transfers") + bob ##> "/fr 1 ./tests/tmp/" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + concurrentlyN_ + [ do + alice <## "started sending file 1 (test.jpg) to bob" + alice <## "completed sending file 1 (test.jpg) to bob" + alice ##> "/fs 1" + alice <## "sending file 1 (test.jpg) complete: bob", + do + bob <## "started receiving file 1 (test.jpg) from alice" + bob <## "completed receiving file 1 (test.jpg) from alice" + ] + cath ##> "/fr 1 ./tests/tmp/" + cath <## "saving file 1 from alice to ./tests/tmp/test_1.jpg" + concurrentlyN_ + [ do + alice <## "started sending file 1 (test.jpg) to cath" + alice <## "completed sending file 1 (test.jpg) to cath" + alice ##> "/fs 1" + getTermLine alice >>= (`shouldStartWith` "sending file 1 (test.jpg) complete"), + do + cath <## "started receiving file 1 (test.jpg) from alice" + cath <## "completed receiving file 1 (test.jpg) from alice" + ] + src <- B.readFile "./tests/fixtures/test.jpg" + dest1 <- B.readFile "./tests/tmp/test.jpg" + dest2 <- B.readFile "./tests/tmp/test_1.jpg" + dest1 `shouldBe` src + dest2 `shouldBe` src + +testInlineGroupFileTransfer :: HasCallStack => FilePath -> IO () +testInlineGroupFileTransfer = + testChatCfg3 cfg aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + bob ##> "/_files_folder ./tests/tmp/bob/" + bob <## "ok" + cath ##> "/_files_folder ./tests/tmp/cath/" + cath <## "ok" + alice ##> "/_send #1 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" + alice <# "#team voice message (00:10)" + alice <# "/f #team ./tests/fixtures/logo.jpg" + -- below is not shown in "sent" mode + -- alice <## "use /fc 1 to cancel sending" + concurrentlyN_ + [ do + alice + <### [ "completed sending file 1 (logo.jpg) to bob", + "completed sending file 1 (logo.jpg) to cath" + ] + alice ##> "/fs 1" + alice <##. "sending file 1 (logo.jpg) complete", + do + bob <# "#team alice> voice message (00:10)" + bob <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" + bob <## "started receiving file 1 (logo.jpg) from alice" + bob <## "completed receiving file 1 (logo.jpg) from alice", + do + cath <# "#team alice> voice message (00:10)" + cath <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" + cath <## "started receiving file 1 (logo.jpg) from alice" + cath <## "completed receiving file 1 (logo.jpg) from alice" + ] + src <- B.readFile "./tests/fixtures/logo.jpg" + dest1 <- B.readFile "./tests/tmp/bob/logo.jpg" + dest2 <- B.readFile "./tests/tmp/cath/logo.jpg" + dest1 `shouldBe` src + dest2 `shouldBe` src + where + cfg = testCfg {inlineFiles = defaultInlineFilesConfig {offerChunks = 100, sendChunks = 100, totalSendChunks = 100, receiveChunks = 100}} + +testSmallInlineGroupFileTransfer :: HasCallStack => FilePath -> IO () +testSmallInlineGroupFileTransfer = + testChatCfg3 testCfg aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + bob ##> "/_files_folder ./tests/tmp/bob/" + bob <## "ok" + cath ##> "/_files_folder ./tests/tmp/cath/" + cath <## "ok" + alice ##> "/_send #1 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" + alice <# "#team voice message (00:10)" + alice <# "/f #team ./tests/fixtures/logo.jpg" + -- below is not shown in "sent" mode + -- alice <## "use /fc 1 to cancel sending" + concurrentlyN_ + [ do + alice + <### [ "completed sending file 1 (logo.jpg) to bob", + "completed sending file 1 (logo.jpg) to cath" + ] + alice ##> "/fs 1" + alice <##. "sending file 1 (logo.jpg) complete", + do + bob <# "#team alice> voice message (00:10)" + bob <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" + bob <## "started receiving file 1 (logo.jpg) from alice" + bob <## "completed receiving file 1 (logo.jpg) from alice", + do + cath <# "#team alice> voice message (00:10)" + cath <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" + cath <## "started receiving file 1 (logo.jpg) from alice" + cath <## "completed receiving file 1 (logo.jpg) from alice" + ] + src <- B.readFile "./tests/fixtures/logo.jpg" + dest1 <- B.readFile "./tests/tmp/bob/logo.jpg" + dest2 <- B.readFile "./tests/tmp/cath/logo.jpg" + dest1 `shouldBe` src + dest2 `shouldBe` src + +testSmallInlineGroupFileIgnored :: HasCallStack => FilePath -> IO () +testSmallInlineGroupFileIgnored tmp = do + withNewTestChat tmp "alice" aliceProfile $ \alice -> + withNewTestChatOpts tmp testOpts {allowInstantFiles = False} "bob" bobProfile $ \bob -> do + withNewTestChatOpts tmp testOpts {allowInstantFiles = False} "cath" cathProfile $ \cath -> do + createGroup3 "team" alice bob cath + bob ##> "/_files_folder ./tests/tmp/bob/" + bob <## "ok" + cath ##> "/_files_folder ./tests/tmp/cath/" + cath <## "ok" + alice ##> "/_send #1 json {\"msgContent\":{\"type\":\"voice\", \"duration\":10, \"text\":\"\"}, \"filePath\":\"./tests/fixtures/logo.jpg\"}" + alice <# "#team voice message (00:10)" + alice <# "/f #team ./tests/fixtures/logo.jpg" + -- below is not shown in "sent" mode + -- alice <## "use /fc 1 to cancel sending" + concurrentlyN_ + [ do + alice + <### [ "completed sending file 1 (logo.jpg) to bob", + "completed sending file 1 (logo.jpg) to cath" + ] + alice ##> "/fs 1" + alice <##. "sending file 1 (logo.jpg) complete", + do + bob <# "#team alice> voice message (00:10)" + bob <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob <## "A small file sent without acceptance - you can enable receiving such files with -f option." + bob ##> "/fr 1" + bob <## "file is already being received: logo.jpg", + do + cath <# "#team alice> voice message (00:10)" + cath <# "#team alice> sends file logo.jpg (31.3 KiB / 32080 bytes)" + cath <## "use /fr 1 [/ | ] to receive it" + cath <## "A small file sent without acceptance - you can enable receiving such files with -f option." + cath ##> "/fr 1" + cath <## "file is already being received: logo.jpg" + ] + +runTestGroupFileSndCancelBeforeTransfer :: HasCallStack => TestCC -> TestCC -> TestCC -> IO () +runTestGroupFileSndCancelBeforeTransfer alice bob cath = do + createGroup3 "team" alice bob cath + alice #> "/f #team ./tests/fixtures/test.txt" + alice <## "use /fc 1 to cancel sending" + concurrentlyN_ + [ do + bob <# "#team alice> sends file test.txt (11 bytes / 11 bytes)" + bob <## "use /fr 1 [/ | ] to receive it", + do + cath <# "#team alice> sends file test.txt (11 bytes / 11 bytes)" + cath <## "use /fr 1 [/ | ] to receive it" + ] + alice ##> "/fc 1" + concurrentlyN_ + [ alice <## "cancelled sending file 1 (test.txt)", + bob <## "alice cancelled sending file 1 (test.txt)", + cath <## "alice cancelled sending file 1 (test.txt)" + ] + alice ##> "/fs 1" + alice <## "sending file 1 (test.txt): no file transfers" + alice <## "file transfer cancelled" + bob ##> "/fs 1" + bob <## "receiving file 1 (test.txt) cancelled" + bob ##> "/fr 1 ./tests/tmp" + bob <## "file cancelled: test.txt" + +runTestMessageWithFile :: HasCallStack => TestCC -> TestCC -> IO () +runTestMessageWithFile alice bob = do + connectUsers alice bob + alice ##> "/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"type\": \"text\", \"text\": \"hi, sending a file\"}}" + alice <# "@bob hi, sending a file" + alice <# "/f @bob ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> hi, sending a file" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1 ./tests/tmp" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + concurrently_ + (bob <## "started receiving file 1 (test.jpg) from alice") + (alice <## "started sending file 1 (test.jpg) to bob") + concurrently_ + (bob <## "completed receiving file 1 (test.jpg) from alice") + (alice <## "completed sending file 1 (test.jpg) to bob") + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/test.jpg" + dest `shouldBe` src + alice #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((1, "hi, sending a file"), Just "./tests/fixtures/test.jpg")]) + bob #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((0, "hi, sending a file"), Just "./tests/tmp/test.jpg")]) + +testSendImage :: HasCallStack => FilePath -> IO () +testSendImage = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice ##> "/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" + alice <# "/f @bob ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1 ./tests/tmp" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + concurrently_ + (bob <## "started receiving file 1 (test.jpg) from alice") + (alice <## "started sending file 1 (test.jpg) to bob") + concurrently_ + (bob <## "completed receiving file 1 (test.jpg) from alice") + (alice <## "completed sending file 1 (test.jpg) to bob") + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/test.jpg" + dest `shouldBe` src + alice #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((1, ""), Just "./tests/fixtures/test.jpg")]) + bob #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((0, ""), Just "./tests/tmp/test.jpg")]) + -- deleting contact without files folder set should not remove file + bob ##> "/d alice" + bob <## "alice: contact is deleted" + fileExists <- doesFileExist "./tests/tmp/test.jpg" + fileExists `shouldBe` True + +testFilesFoldersSendImage :: HasCallStack => FilePath -> IO () +testFilesFoldersSendImage = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice #$> ("/_files_folder ./tests/fixtures", id, "ok") + bob #$> ("/_files_folder ./tests/tmp/app_files", id, "ok") + alice ##> "/_send @2 json {\"filePath\": \"test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" + alice <# "/f @bob test.jpg" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1" + bob <## "saving file 1 from alice to test.jpg" + concurrently_ + (bob <## "started receiving file 1 (test.jpg) from alice") + (alice <## "started sending file 1 (test.jpg) to bob") + concurrently_ + (bob <## "completed receiving file 1 (test.jpg) from alice") + (alice <## "completed sending file 1 (test.jpg) to bob") + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/app_files/test.jpg" + dest `shouldBe` src + alice #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((1, ""), Just "test.jpg")]) + bob #$> ("/_get chat @2 count=100", chatF, chatFeaturesF <> [((0, ""), Just "test.jpg")]) + -- deleting contact with files folder set should remove file + checkActionDeletesFile "./tests/tmp/app_files/test.jpg" $ do + bob ##> "/d alice" + bob <## "alice: contact is deleted" + +testFilesFoldersImageSndDelete :: HasCallStack => FilePath -> IO () +testFilesFoldersImageSndDelete = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice #$> ("/_files_folder ./tests/tmp/alice_app_files", id, "ok") + copyFile "./tests/fixtures/test_1MB.pdf" "./tests/tmp/alice_app_files/test_1MB.pdf" + bob #$> ("/_files_folder ./tests/tmp/bob_app_files", id, "ok") + alice ##> "/_send @2 json {\"filePath\": \"test_1MB.pdf\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" + alice <# "/f @bob test_1MB.pdf" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> sends file test_1MB.pdf (1017.7 KiB / 1042157 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1" + bob <## "saving file 1 from alice to test_1MB.pdf" + concurrently_ + (bob <## "started receiving file 1 (test_1MB.pdf) from alice") + (alice <## "started sending file 1 (test_1MB.pdf) to bob") + -- deleting contact should cancel and remove file + checkActionDeletesFile "./tests/tmp/alice_app_files/test_1MB.pdf" $ do + alice ##> "/d bob" + alice <## "bob: contact is deleted" + bob ##> "/fs 1" + bob <##. "receiving file 1 (test_1MB.pdf) progress" + -- deleting contact should remove cancelled file + checkActionDeletesFile "./tests/tmp/bob_app_files/test_1MB.pdf" $ do + bob ##> "/d alice" + bob <## "alice: contact is deleted" + +testFilesFoldersImageRcvDelete :: HasCallStack => FilePath -> IO () +testFilesFoldersImageRcvDelete = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice #$> ("/_files_folder ./tests/fixtures", id, "ok") + bob #$> ("/_files_folder ./tests/tmp/app_files", id, "ok") + alice ##> "/_send @2 json {\"filePath\": \"test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" + alice <# "/f @bob test.jpg" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1" + bob <## "saving file 1 from alice to test.jpg" + concurrently_ + (bob <## "started receiving file 1 (test.jpg) from alice") + (alice <## "started sending file 1 (test.jpg) to bob") + -- deleting contact should cancel and remove file + waitFileExists "./tests/tmp/app_files/test.jpg" + checkActionDeletesFile "./tests/tmp/app_files/test.jpg" $ do + bob ##> "/d alice" + bob <## "alice: contact is deleted" + alice <## "bob cancelled receiving file 1 (test.jpg)" + alice ##> "/fs 1" + alice <## "sending file 1 (test.jpg) cancelled: bob" + +testSendImageWithTextAndQuote :: HasCallStack => FilePath -> IO () +testSendImageWithTextAndQuote = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + bob #> "@alice hi alice" + alice <# "bob> hi alice" + alice ##> ("/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"quotedItemId\": " <> itemId 1 <> ", \"msgContent\": {\"text\":\"hey bob\",\"type\":\"image\",\"image\":\"\"}}") + alice <# "@bob > hi alice" + alice <## " hey bob" + alice <# "/f @bob ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + bob <# "alice> > hi alice" + bob <## " hey bob" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1 ./tests/tmp" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + concurrently_ + (bob <## "started receiving file 1 (test.jpg) from alice") + (alice <## "started sending file 1 (test.jpg) to bob") + concurrently_ + (bob <## "completed receiving file 1 (test.jpg) from alice") + (alice <## "completed sending file 1 (test.jpg) to bob") + src <- B.readFile "./tests/fixtures/test.jpg" + B.readFile "./tests/tmp/test.jpg" `shouldReturn` src + alice #$> ("/_get chat @2 count=100", chat'', chatFeatures'' <> [((0, "hi alice"), Nothing, Nothing), ((1, "hey bob"), Just (0, "hi alice"), Just "./tests/fixtures/test.jpg")]) + alice @@@ [("@bob", "hey bob")] + bob #$> ("/_get chat @2 count=100", chat'', chatFeatures'' <> [((1, "hi alice"), Nothing, Nothing), ((0, "hey bob"), Just (1, "hi alice"), Just "./tests/tmp/test.jpg")]) + bob @@@ [("@alice", "hey bob")] + -- quoting (file + text) with file uses quoted text + bob ##> ("/_send @2 json {\"filePath\": \"./tests/fixtures/test.pdf\", \"quotedItemId\": " <> itemId 2 <> ", \"msgContent\": {\"text\":\"\",\"type\":\"file\"}}") + bob <# "@alice > hey bob" + bob <## " test.pdf" + bob <# "/f @alice ./tests/fixtures/test.pdf" + bob <## "use /fc 2 to cancel sending" + alice <# "bob> > hey bob" + alice <## " test.pdf" + alice <# "bob> sends file test.pdf (266.0 KiB / 272376 bytes)" + alice <## "use /fr 2 [/ | ] to receive it" + alice ##> "/fr 2 ./tests/tmp" + alice <## "saving file 2 from bob to ./tests/tmp/test.pdf" + concurrently_ + (alice <## "started receiving file 2 (test.pdf) from bob") + (bob <## "started sending file 2 (test.pdf) to alice") + concurrently_ + (alice <## "completed receiving file 2 (test.pdf) from bob") + (bob <## "completed sending file 2 (test.pdf) to alice") + txtSrc <- B.readFile "./tests/fixtures/test.pdf" + B.readFile "./tests/tmp/test.pdf" `shouldReturn` txtSrc + -- quoting (file without text) with file uses file name + alice ##> ("/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"quotedItemId\": " <> itemId 3 <> ", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}") + alice <# "@bob > test.pdf" + alice <## " test.jpg" + alice <# "/f @bob ./tests/fixtures/test.jpg" + alice <## "use /fc 3 to cancel sending" + bob <# "alice> > test.pdf" + bob <## " test.jpg" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 3 [/ | ] to receive it" + bob ##> "/fr 3 ./tests/tmp" + bob <## "saving file 3 from alice to ./tests/tmp/test_1.jpg" + concurrently_ + (bob <## "started receiving file 3 (test.jpg) from alice") + (alice <## "started sending file 3 (test.jpg) to bob") + concurrently_ + (bob <## "completed receiving file 3 (test.jpg) from alice") + (alice <## "completed sending file 3 (test.jpg) to bob") + B.readFile "./tests/tmp/test_1.jpg" `shouldReturn` src + +testGroupSendImage :: SpecWith FilePath +testGroupSendImage = versionTestMatrix3 runTestGroupSendImage + where + runTestGroupSendImage :: HasCallStack => TestCC -> TestCC -> TestCC -> IO () + runTestGroupSendImage alice bob cath = do + createGroup3 "team" alice bob cath + threadDelay 1000000 + alice ##> "/_send #1 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"image\",\"image\":\"\"}}" + alice <# "/f #team ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + concurrentlyN_ + [ do + bob <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it", + do + cath <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + cath <## "use /fr 1 [/ | ] to receive it" + ] + bob ##> "/fr 1 ./tests/tmp/" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + concurrentlyN_ + [ do + alice <## "started sending file 1 (test.jpg) to bob" + alice <## "completed sending file 1 (test.jpg) to bob", + do + bob <## "started receiving file 1 (test.jpg) from alice" + bob <## "completed receiving file 1 (test.jpg) from alice" + ] + cath ##> "/fr 1 ./tests/tmp/" + cath <## "saving file 1 from alice to ./tests/tmp/test_1.jpg" + concurrentlyN_ + [ do + alice <## "started sending file 1 (test.jpg) to cath" + alice <## "completed sending file 1 (test.jpg) to cath", + do + cath <## "started receiving file 1 (test.jpg) from alice" + cath <## "completed receiving file 1 (test.jpg) from alice" + ] + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/test.jpg" + dest `shouldBe` src + dest2 <- B.readFile "./tests/tmp/test_1.jpg" + dest2 `shouldBe` src + alice #$> ("/_get chat #1 count=1", chatF, [((1, ""), Just "./tests/fixtures/test.jpg")]) + bob #$> ("/_get chat #1 count=1", chatF, [((0, ""), Just "./tests/tmp/test.jpg")]) + cath #$> ("/_get chat #1 count=1", chatF, [((0, ""), Just "./tests/tmp/test_1.jpg")]) + +testGroupSendImageWithTextAndQuote :: HasCallStack => FilePath -> IO () +testGroupSendImageWithTextAndQuote = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + threadDelay 1000000 + bob #> "#team hi team" + concurrently_ + (alice <# "#team bob> hi team") + (cath <# "#team bob> hi team") + threadDelay 1000000 + msgItemId <- lastItemId alice + alice ##> ("/_send #1 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"quotedItemId\": " <> msgItemId <> ", \"msgContent\": {\"text\":\"hey bob\",\"type\":\"image\",\"image\":\"\"}}") + alice <# "#team > bob hi team" + alice <## " hey bob" + alice <# "/f #team ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + concurrentlyN_ + [ do + bob <# "#team alice> > bob hi team" + bob <## " hey bob" + bob <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it", + do + cath <# "#team alice> > bob hi team" + cath <## " hey bob" + cath <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + cath <## "use /fr 1 [/ | ] to receive it" + ] + bob ##> "/fr 1 ./tests/tmp/" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + concurrentlyN_ + [ do + alice <## "started sending file 1 (test.jpg) to bob" + alice <## "completed sending file 1 (test.jpg) to bob", + do + bob <## "started receiving file 1 (test.jpg) from alice" + bob <## "completed receiving file 1 (test.jpg) from alice" + ] + cath ##> "/fr 1 ./tests/tmp/" + cath <## "saving file 1 from alice to ./tests/tmp/test_1.jpg" + concurrentlyN_ + [ do + alice <## "started sending file 1 (test.jpg) to cath" + alice <## "completed sending file 1 (test.jpg) to cath", + do + cath <## "started receiving file 1 (test.jpg) from alice" + cath <## "completed receiving file 1 (test.jpg) from alice" + ] + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/test.jpg" + dest `shouldBe` src + dest2 <- B.readFile "./tests/tmp/test_1.jpg" + dest2 `shouldBe` src + alice #$> ("/_get chat #1 count=2", chat'', [((0, "hi team"), Nothing, Nothing), ((1, "hey bob"), Just (0, "hi team"), Just "./tests/fixtures/test.jpg")]) + alice @@@ [("#team", "hey bob"), ("@bob", "sent invitation to join group team as admin"), ("@cath", "sent invitation to join group team as admin")] + bob #$> ("/_get chat #1 count=2", chat'', [((1, "hi team"), Nothing, Nothing), ((0, "hey bob"), Just (1, "hi team"), Just "./tests/tmp/test.jpg")]) + bob @@@ [("#team", "hey bob"), ("@alice", "received invitation to join group team as admin")] + cath #$> ("/_get chat #1 count=2", chat'', [((0, "hi team"), Nothing, Nothing), ((0, "hey bob"), Just (0, "hi team"), Just "./tests/tmp/test_1.jpg")]) + cath @@@ [("#team", "hey bob"), ("@alice", "received invitation to join group team as admin")] + +testAsyncFileTransferSenderRestarts :: HasCallStack => FilePath -> IO () +testAsyncFileTransferSenderRestarts tmp = do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + withNewTestChat tmp "alice" aliceProfile $ \alice -> do + connectUsers alice bob + startFileTransfer' alice bob "test_1MB.pdf" "1017.7 KiB / 1042157 bytes" + threadDelay 100000 + withTestChatContactConnected tmp "alice" $ \alice -> do + alice <## "completed sending file 1 (test_1MB.pdf) to bob" + bob <## "completed receiving file 1 (test_1MB.pdf) from alice" + src <- B.readFile "./tests/fixtures/test_1MB.pdf" + dest <- B.readFile "./tests/tmp/test_1MB.pdf" + dest `shouldBe` src + +testAsyncFileTransferReceiverRestarts :: HasCallStack => FilePath -> IO () +testAsyncFileTransferReceiverRestarts tmp = do + withNewTestChat tmp "alice" aliceProfile $ \alice -> do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + connectUsers alice bob + startFileTransfer' alice bob "test_1MB.pdf" "1017.7 KiB / 1042157 bytes" + threadDelay 100000 + withTestChatContactConnected tmp "bob" $ \bob -> do + alice <## "completed sending file 1 (test_1MB.pdf) to bob" + bob <## "completed receiving file 1 (test_1MB.pdf) from alice" + src <- B.readFile "./tests/fixtures/test_1MB.pdf" + dest <- B.readFile "./tests/tmp/test_1MB.pdf" + dest `shouldBe` src + +testAsyncFileTransfer :: HasCallStack => FilePath -> IO () +testAsyncFileTransfer tmp = do + withNewTestChat tmp "alice" aliceProfile $ \alice -> + withNewTestChat tmp "bob" bobProfile $ \bob -> + connectUsers alice bob + withTestChatContactConnected tmp "alice" $ \alice -> do + alice ##> "/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"type\":\"text\", \"text\": \"hi, sending a file\"}}" + alice <# "@bob hi, sending a file" + alice <# "/f @bob ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + withTestChatContactConnected tmp "bob" $ \bob -> do + bob <# "alice> hi, sending a file" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1 ./tests/tmp" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + -- withTestChatContactConnected' tmp "alice" -- TODO not needed in v2 + -- withTestChatContactConnected' tmp "bob" -- TODO not needed in v2 + withTestChatContactConnected' tmp "alice" + withTestChatContactConnected' tmp "bob" + withTestChatContactConnected tmp "alice" $ \alice -> do + alice <## "started sending file 1 (test.jpg) to bob" + alice <## "completed sending file 1 (test.jpg) to bob" + withTestChatContactConnected tmp "bob" $ \bob -> do + bob <## "started receiving file 1 (test.jpg) from alice" + bob <## "completed receiving file 1 (test.jpg) from alice" + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/test.jpg" + dest `shouldBe` src + +testAsyncFileTransferV1 :: HasCallStack => FilePath -> IO () +testAsyncFileTransferV1 tmp = do + withNewTestChatV1 tmp "alice" aliceProfile $ \alice -> + withNewTestChatV1 tmp "bob" bobProfile $ \bob -> + connectUsers alice bob + withTestChatContactConnectedV1 tmp "alice" $ \alice -> do + alice ##> "/_send @2 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"type\":\"text\", \"text\": \"hi, sending a file\"}}" + alice <# "@bob hi, sending a file" + alice <# "/f @bob ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + withTestChatContactConnectedV1 tmp "bob" $ \bob -> do + bob <# "alice> hi, sending a file" + bob <# "alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1 ./tests/tmp" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + withTestChatContactConnectedV1' tmp "alice" -- TODO not needed in v2 + withTestChatContactConnectedV1' tmp "bob" -- TODO not needed in v2 + withTestChatContactConnectedV1' tmp "alice" + withTestChatContactConnectedV1' tmp "bob" + withTestChatContactConnectedV1 tmp "alice" $ \alice -> do + alice <## "started sending file 1 (test.jpg) to bob" + alice <## "completed sending file 1 (test.jpg) to bob" + withTestChatContactConnectedV1 tmp "bob" $ \bob -> do + bob <## "started receiving file 1 (test.jpg) from alice" + bob <## "completed receiving file 1 (test.jpg) from alice" + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/test.jpg" + dest `shouldBe` src + +testAsyncGroupFileTransfer :: HasCallStack => FilePath -> IO () +testAsyncGroupFileTransfer tmp = do + withNewTestChat tmp "alice" aliceProfile $ \alice -> + withNewTestChat tmp "bob" bobProfile $ \bob -> + withNewTestChat tmp "cath" cathProfile $ \cath -> + createGroup3 "team" alice bob cath + withTestChatGroup3Connected tmp "alice" $ \alice -> do + alice ##> "/_send #1 json {\"filePath\": \"./tests/fixtures/test.jpg\", \"msgContent\": {\"text\":\"\",\"type\":\"text\"}}" + alice <# "/f #team ./tests/fixtures/test.jpg" + alice <## "use /fc 1 to cancel sending" + withTestChatGroup3Connected tmp "bob" $ \bob -> do + bob <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + bob <## "use /fr 1 [/ | ] to receive it" + bob ##> "/fr 1 ./tests/tmp/" + bob <## "saving file 1 from alice to ./tests/tmp/test.jpg" + withTestChatGroup3Connected tmp "cath" $ \cath -> do + cath <# "#team alice> sends file test.jpg (136.5 KiB / 139737 bytes)" + cath <## "use /fr 1 [/ | ] to receive it" + cath ##> "/fr 1 ./tests/tmp/" + cath <## "saving file 1 from alice to ./tests/tmp/test_1.jpg" + withTestChatGroup3Connected' tmp "alice" + withTestChatGroup3Connected' tmp "bob" + withTestChatGroup3Connected' tmp "cath" + -- withTestChatGroup3Connected' tmp "alice" -- TODO not needed in v2 + -- withTestChatGroup3Connected' tmp "bob" -- TODO not needed in v2 + -- withTestChatGroup3Connected' tmp "cath" -- TODO not needed in v2 + withTestChatGroup3Connected' tmp "alice" + withTestChatGroup3Connected tmp "bob" $ \bob -> do + bob <## "started receiving file 1 (test.jpg) from alice" + withTestChatGroup3Connected tmp "cath" $ \cath -> do + cath <## "started receiving file 1 (test.jpg) from alice" + withTestChatGroup3Connected tmp "alice" $ \alice -> do + alice + <### [ "started sending file 1 (test.jpg) to bob", + "completed sending file 1 (test.jpg) to bob", + "started sending file 1 (test.jpg) to cath", + "completed sending file 1 (test.jpg) to cath" + ] + withTestChatGroup3Connected tmp "bob" $ \bob -> do + bob <## "completed receiving file 1 (test.jpg) from alice" + withTestChatGroup3Connected tmp "cath" $ \cath -> do + cath <## "completed receiving file 1 (test.jpg) from alice" + src <- B.readFile "./tests/fixtures/test.jpg" + dest <- B.readFile "./tests/tmp/test.jpg" + dest `shouldBe` src + dest2 <- B.readFile "./tests/tmp/test_1.jpg" + dest2 `shouldBe` src + +startFileTransfer :: HasCallStack => TestCC -> TestCC -> IO () +startFileTransfer alice bob = + startFileTransfer' alice bob "test.jpg" "136.5 KiB / 139737 bytes" + +startFileTransfer' :: HasCallStack => TestCC -> TestCC -> String -> String -> IO () +startFileTransfer' cc1 cc2 fileName fileSize = startFileTransferWithDest' cc1 cc2 fileName fileSize $ Just "./tests/tmp" + +checkPartialTransfer :: HasCallStack => String -> IO () +checkPartialTransfer fileName = do + src <- B.readFile $ "./tests/fixtures/" <> fileName + dest <- B.readFile $ "./tests/tmp/" <> fileName + B.unpack src `shouldStartWith` B.unpack dest + B.length src > B.length dest `shouldBe` True + +waitFileExists :: HasCallStack => FilePath -> IO () +waitFileExists f = unlessM (doesFileExist f) $ waitFileExists f diff --git a/tests/ChatTests/Groups.hs b/tests/ChatTests/Groups.hs new file mode 100644 index 000000000..116f02917 --- /dev/null +++ b/tests/ChatTests/Groups.hs @@ -0,0 +1,1781 @@ +{-# LANGUAGE OverloadedStrings #-} +{-# LANGUAGE PostfixOperators #-} + +module ChatTests.Groups where + +import ChatClient +import ChatTests.Utils +import Control.Concurrent (threadDelay) +import Control.Concurrent.Async (concurrently_) +import Control.Monad (when) +import qualified Data.Text as T +import Simplex.Chat.Types (GroupMemberRole (..)) +import Test.Hspec + +chatGroupTests :: SpecWith FilePath +chatGroupTests = do + describe "chat groups" $ do + describe "add contacts, create group and send/receive messages" testGroup + it "add contacts, create group and send/receive messages, check messages" testGroupCheckMessages + it "create and join group with 4 members" testGroup2 + it "create and delete group" testGroupDelete + it "create group with the same displayName" testGroupSameName + it "invitee delete group when in status invited" testGroupDeleteWhenInvited + it "re-add member in status invited" testGroupReAddInvited + it "re-add member in status invited, change role" testGroupReAddInvitedChangeRole + it "delete contact before they accept group invitation, contact joins group" testGroupDeleteInvitedContact + it "member profile is kept when deleting group if other groups have this member" testDeleteGroupMemberProfileKept + it "remove contact from group and add again" testGroupRemoveAdd + it "list groups containing group invitations" testGroupList + it "group message quoted replies" testGroupMessageQuotedReply + it "group message update" testGroupMessageUpdate + it "group message delete" testGroupMessageDelete + it "group live message" testGroupLiveMessage + it "update group profile" testUpdateGroupProfile + it "update member role" testUpdateMemberRole + it "unused contacts are deleted after all their groups are deleted" testGroupDeleteUnusedContacts + it "group description is shown as the first message to new members" testGroupDescription + describe "async group connections" $ do + xit "create and join group when clients go offline" testGroupAsync + describe "group links" $ do + it "create group link, join via group link" testGroupLink + it "delete group, re-join via same link" testGroupLinkDeleteGroupRejoin + it "sending message to contact created via group link marks it used" testGroupLinkContactUsed + it "create group link, join via group link - incognito membership" testGroupLinkIncognitoMembership + it "unused host contact is deleted after all groups with it are deleted" testGroupLinkUnusedHostContactDeleted + it "leaving groups with unused host contacts deletes incognito profiles" testGroupLinkIncognitoUnusedHostContactsDeleted + +testGroup :: HasCallStack => SpecWith FilePath +testGroup = versionTestMatrix3 runTestGroup + where + runTestGroup alice bob cath = testGroupShared alice bob cath False + +testGroupCheckMessages :: HasCallStack => FilePath -> IO () +testGroupCheckMessages = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> testGroupShared alice bob cath True + +testGroupShared :: HasCallStack => TestCC -> TestCC -> TestCC -> Bool -> IO () +testGroupShared alice bob cath checkMessages = do + connectUsers alice bob + connectUsers alice cath + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as admin" + bob <## "use /j team to accept" + ] + bob ##> "/j team" + concurrently_ + (alice <## "#team: bob joined the group") + (bob <## "#team: you joined the group") + when checkMessages $ threadDelay 1000000 -- for deterministic order of messages and "connected" events + alice ##> "/a team cath" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to cath", + do + cath <## "#team: alice invites you to join the group as admin" + cath <## "use /j team to accept" + ] + cath ##> "/j team" + concurrentlyN_ + [ alice <## "#team: cath joined the group", + do + cath <## "#team: you joined the group" + cath <## "#team: member bob (Bob) is connected", + do + bob <## "#team: alice added cath (Catherine) to the group (connecting...)" + bob <## "#team: new member cath is connected" + ] + when checkMessages $ threadDelay 1000000 -- for deterministic order of messages and "connected" events + alice #> "#team hello" + msgItem1 <- lastItemId alice + concurrently_ + (bob <# "#team alice> hello") + (cath <# "#team alice> hello") + when checkMessages $ threadDelay 1000000 -- server assigns timestamps with one second precision + bob #> "#team hi there" + concurrently_ + (alice <# "#team bob> hi there") + (cath <# "#team bob> hi there") + when checkMessages $ threadDelay 1000000 + cath #> "#team hey team" + concurrently_ + (alice <# "#team cath> hey team") + (bob <# "#team cath> hey team") + msgItem2 <- lastItemId alice + bob <##> cath + when checkMessages $ getReadChats msgItem1 msgItem2 + -- list groups + alice ##> "/gs" + alice <## "#team" + -- list group members + alice ##> "/ms team" + alice + <### [ "alice (Alice): owner, you, created group", + "bob (Bob): admin, invited, connected", + "cath (Catherine): admin, invited, connected" + ] + -- list contacts + alice ##> "/contacts" + alice <## "bob (Bob)" + alice <## "cath (Catherine)" + -- test observer role + -- to be enabled once the role is enabled in parser + -- alice ##> "/mr team bob observer" + -- concurrentlyN_ + -- [ alice <## "#team: you changed the role of bob from admin to observer", + -- bob <## "#team: alice changed your role from admin to observer", + -- cath <## "#team: alice changed the role of bob from admin to observer" + -- ] + -- bob ##> "#team hello" + -- bob <## "you don't have permission to send messages to this group" + -- bob ##> "/rm team cath" + -- bob <## "you have insufficient permissions for this action, the required role is admin" + -- cath #> "#team hello" + -- concurrentlyN_ + -- [ alice <# "#team cath> hello", + -- bob <# "#team cath> hello" + -- ] + -- alice ##> "/mr team bob admin" + -- concurrentlyN_ + -- [ alice <## "#team: you changed the role of bob from observer to admin", + -- bob <## "#team: alice changed your role from observer to admin", + -- cath <## "#team: alice changed the role of bob from observer to admin" + -- ] + -- remove member + bob ##> "/rm team cath" + concurrentlyN_ + [ bob <## "#team: you removed cath from the group", + alice <## "#team: bob removed cath from the group", + do + cath <## "#team: bob removed you from the group" + cath <## "use /d #team to delete the group" + ] + bob #> "#team hi" + concurrently_ + (alice <# "#team bob> hi") + (cath "#team hello" + concurrently_ + (bob <# "#team alice> hello") + (cath "#team hello" + cath <## "you are no longer a member of the group" + bob <##> cath + -- delete contact + alice ##> "/d bob" + alice <## "bob: contact is deleted" + alice ##> "@bob hey" + alice <## "no contact bob" + when checkMessages $ threadDelay 1000000 + alice #> "#team checking connection" + bob <# "#team alice> checking connection" + when checkMessages $ threadDelay 1000000 + bob #> "#team received" + alice <# "#team bob> received" + when checkMessages $ do + alice @@@ [("@cath", "sent invitation to join group team as admin"), ("#team", "received")] + bob @@@ [("@alice", "received invitation to join group team as admin"), ("@cath", "hey"), ("#team", "received")] + -- test clearing chat + alice #$> ("/clear #team", id, "#team: all messages are removed locally ONLY") + alice #$> ("/_get chat #1 count=100", chat, []) + bob #$> ("/clear #team", id, "#team: all messages are removed locally ONLY") + bob #$> ("/_get chat #1 count=100", chat, []) + cath #$> ("/clear #team", id, "#team: all messages are removed locally ONLY") + cath #$> ("/_get chat #1 count=100", chat, []) + where + getReadChats :: HasCallStack => String -> String -> IO () + getReadChats msgItem1 msgItem2 = do + alice @@@ [("#team", "hey team"), ("@cath", "sent invitation to join group team as admin"), ("@bob", "sent invitation to join group team as admin")] + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (0, "connected"), (1, "hello"), (0, "hi there"), (0, "hey team")]) + -- "before" and "after" define a chat item id across all chats, + -- so we take into account group event items as well as sent group invitations in direct chats + alice #$> ("/_get chat #1 after=" <> msgItem1 <> " count=100", chat, [(0, "hi there"), (0, "hey team")]) + alice #$> ("/_get chat #1 before=" <> msgItem2 <> " count=100", chat, [(0, "connected"), (0, "connected"), (1, "hello"), (0, "hi there")]) + alice #$> ("/_get chat #1 count=100 search=team", chat, [(0, "hey team")]) + bob @@@ [("@cath", "hey"), ("#team", "hey team"), ("@alice", "received invitation to join group team as admin")] + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "added cath (Catherine)"), (0, "connected"), (0, "hello"), (1, "hi there"), (0, "hey team")]) + cath @@@ [("@bob", "hey"), ("#team", "hey team"), ("@alice", "received invitation to join group team as admin")] + cath #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "connected"), (0, "hello"), (0, "hi there"), (1, "hey team")]) + alice #$> ("/_read chat #1 from=1 to=100", id, "ok") + bob #$> ("/_read chat #1 from=1 to=100", id, "ok") + cath #$> ("/_read chat #1 from=1 to=100", id, "ok") + alice #$> ("/_read chat #1", id, "ok") + bob #$> ("/_read chat #1", id, "ok") + cath #$> ("/_read chat #1", id, "ok") + alice #$> ("/_unread chat #1 on", id, "ok") + alice #$> ("/_unread chat #1 off", id, "ok") + +testGroup2 :: HasCallStack => FilePath -> IO () +testGroup2 = + testChat4 aliceProfile bobProfile cathProfile danProfile $ + \alice bob cath dan -> do + connectUsers alice bob + connectUsers alice cath + connectUsers bob dan + connectUsers alice dan + alice ##> "/g club" + alice <## "group #club is created" + alice <## "to add members use /a club or /create link #club" + alice ##> "/a club bob" + concurrentlyN_ + [ alice <## "invitation to join the group #club sent to bob", + do + bob <## "#club: alice invites you to join the group as admin" + bob <## "use /j club to accept" + ] + alice ##> "/a club cath" + concurrentlyN_ + [ alice <## "invitation to join the group #club sent to cath", + do + cath <## "#club: alice invites you to join the group as admin" + cath <## "use /j club to accept" + ] + bob ##> "/j club" + concurrently_ + (alice <## "#club: bob joined the group") + (bob <## "#club: you joined the group") + cath ##> "/j club" + concurrentlyN_ + [ alice <## "#club: cath joined the group", + do + cath <## "#club: you joined the group" + cath <## "#club: member bob (Bob) is connected", + do + bob <## "#club: alice added cath (Catherine) to the group (connecting...)" + bob <## "#club: new member cath is connected" + ] + bob ##> "/a club dan" + concurrentlyN_ + [ bob <## "invitation to join the group #club sent to dan", + do + dan <## "#club: bob invites you to join the group as admin" + dan <## "use /j club to accept" + ] + dan ##> "/j club" + concurrentlyN_ + [ bob <## "#club: dan joined the group", + do + dan <## "#club: you joined the group" + dan + <### [ "#club: member alice_1 (Alice) is connected", + "contact alice_1 is merged into alice", + "use @alice to send messages", + "#club: member cath (Catherine) is connected" + ], + do + alice <## "#club: bob added dan_1 (Daniel) to the group (connecting...)" + alice <## "#club: new member dan_1 is connected" + alice <## "contact dan_1 is merged into dan" + alice <## "use @dan to send messages", + do + cath <## "#club: bob added dan (Daniel) to the group (connecting...)" + cath <## "#club: new member dan is connected" + ] + alice #> "#club hello" + concurrentlyN_ + [ bob <# "#club alice> hello", + cath <# "#club alice> hello", + dan <# "#club alice> hello" + ] + bob #> "#club hi there" + concurrentlyN_ + [ alice <# "#club bob> hi there", + cath <# "#club bob> hi there", + dan <# "#club bob> hi there" + ] + cath #> "#club hey" + concurrentlyN_ + [ alice <# "#club cath> hey", + bob <# "#club cath> hey", + dan <# "#club cath> hey" + ] + dan #> "#club how is it going?" + concurrentlyN_ + [ alice <# "#club dan> how is it going?", + bob <# "#club dan> how is it going?", + cath <# "#club dan> how is it going?" + ] + bob <##> cath + dan <##> cath + dan <##> alice + -- show last messages + alice ##> "/t #club 8" + alice -- these strings are expected in any order because of sorting by time and rounding of time for sent + <##? [ "#club bob> connected", + "#club cath> connected", + "#club bob> added dan (Daniel)", + "#club dan> connected", + "#club hello", + "#club bob> hi there", + "#club cath> hey", + "#club dan> how is it going?" + ] + alice ##> "/t @dan 2" + alice + <##? [ "dan> hi", + "@dan hey" + ] + alice ##> "/t 21" + alice + <##? [ "@bob sent invitation to join group club as admin", + "@cath sent invitation to join group club as admin", + "#club bob> connected", + "#club cath> connected", + "#club bob> added dan (Daniel)", + "#club dan> connected", + "#club hello", + "#club bob> hi there", + "#club cath> hey", + "#club dan> how is it going?", + "dan> hi", + "@dan hey", + "dan> Disappearing messages: off", + "dan> Full deletion: off", + "dan> Voice messages: enabled", + "bob> Disappearing messages: off", + "bob> Full deletion: off", + "bob> Voice messages: enabled", + "cath> Disappearing messages: off", + "cath> Full deletion: off", + "cath> Voice messages: enabled" + ] + -- remove member + cath ##> "/rm club dan" + concurrentlyN_ + [ cath <## "#club: you removed dan from the group", + alice <## "#club: cath removed dan from the group", + bob <## "#club: cath removed dan from the group", + do + dan <## "#club: cath removed you from the group" + dan <## "use /d #club to delete the group" + ] + alice #> "#club hello" + concurrentlyN_ + [ bob <# "#club alice> hello", + cath <# "#club alice> hello", + (dan "#club hi there" + concurrentlyN_ + [ alice <# "#club bob> hi there", + cath <# "#club bob> hi there", + (dan "#club hey" + concurrentlyN_ + [ alice <# "#club cath> hey", + bob <# "#club cath> hey", + (dan "#club how is it going?" + dan <## "you are no longer a member of the group" + dan ##> "/d #club" + dan <## "#club: you deleted the group" + dan <##> cath + dan <##> alice + -- member leaves + bob ##> "/l club" + concurrentlyN_ + [ do + bob <## "#club: you left the group" + bob <## "use /d #club to delete the group", + alice <## "#club: bob left the group", + cath <## "#club: bob left the group" + ] + alice #> "#club hello" + concurrently_ + (cath <# "#club alice> hello") + (bob "#club hey" + concurrently_ + (alice <# "#club cath> hey") + (bob "#club how is it going?" + bob <## "you are no longer a member of the group" + bob ##> "/d #club" + bob <## "#club: you deleted the group" + bob <##> cath + bob <##> alice + +testGroupDelete :: HasCallStack => FilePath -> IO () +testGroupDelete = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + alice ##> "/d #team" + concurrentlyN_ + [ alice <## "#team: you deleted the group", + do + bob <## "#team: alice deleted the group" + bob <## "use /d #team to delete the local copy of the group", + do + cath <## "#team: alice deleted the group" + cath <## "use /d #team to delete the local copy of the group" + ] + alice ##> "#team hi" + alice <## "no group #team" + bob ##> "/d #team" + bob <## "#team: you deleted the group" + cath ##> "#team hi" + cath <## "you are no longer a member of the group" + cath ##> "/d #team" + cath <## "#team: you deleted the group" + alice <##> bob + alice <##> cath + -- unused group contacts are deleted + bob ##> "@cath hi" + bob <## "no contact cath" + (cath "@bob hi" + cath <## "no contact bob" + (bob FilePath -> IO () +testGroupSameName = + testChat2 aliceProfile bobProfile $ + \alice _ -> do + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/g team" + alice <## "group #team_1 (team) is created" + alice <## "to add members use /a team_1 or /create link #team_1" + +testGroupDeleteWhenInvited :: HasCallStack => FilePath -> IO () +testGroupDeleteWhenInvited = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as admin" + bob <## "use /j team to accept" + ] + bob ##> "/d #team" + bob <## "#team: you deleted the group" + -- alice doesn't receive notification that bob deleted group, + -- but she can re-add bob + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as admin" + bob <## "use /j team to accept" + ] + +testGroupReAddInvited :: HasCallStack => FilePath -> IO () +testGroupReAddInvited = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as admin" + bob <## "use /j team to accept" + ] + -- alice re-adds bob, he sees it as the same group + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as admin" + bob <## "use /j team to accept" + ] + -- if alice removes bob and then re-adds him, she uses a new connection request + -- and he sees it as a new group with a different local display name + alice ##> "/rm team bob" + alice <## "#team: you removed bob from the group" + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team_1 (team): alice invites you to join the group as admin" + bob <## "use /j team_1 to accept" + ] + +testGroupReAddInvitedChangeRole :: HasCallStack => FilePath -> IO () +testGroupReAddInvitedChangeRole = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as admin" + bob <## "use /j team to accept" + ] + -- alice re-adds bob, he sees it as the same group + alice ##> "/a team bob owner" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as owner" + bob <## "use /j team to accept" + ] + -- bob joins as owner + bob ##> "/j team" + concurrently_ + (alice <## "#team: bob joined the group") + (bob <## "#team: you joined the group") + bob ##> "/d #team" + concurrentlyN_ + [ bob <## "#team: you deleted the group", + do + alice <## "#team: bob deleted the group" + alice <## "use /d #team to delete the local copy of the group" + ] + bob ##> "#team hi" + bob <## "no group #team" + alice ##> "/d #team" + alice <## "#team: you deleted the group" + +testGroupDeleteInvitedContact :: HasCallStack => FilePath -> IO () +testGroupDeleteInvitedContact = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as admin" + bob <## "use /j team to accept" + ] + alice ##> "/d bob" + alice <## "bob: contact is deleted" + bob ##> "/j team" + concurrently_ + (alice <## "#team: bob joined the group") + (bob <## "#team: you joined the group") + alice #> "#team hello" + bob <# "#team alice> hello" + bob #> "#team hi there" + alice <# "#team bob> hi there" + alice ##> "@bob hey" + alice <## "no contact bob" + bob #> "@alice hey" + bob <## "[alice, contactId: 2, connId: 1] error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" + (alice FilePath -> IO () +testDeleteGroupMemberProfileKept = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + -- group 1 + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as admin" + bob <## "use /j team to accept" + ] + bob ##> "/j team" + concurrently_ + (alice <## "#team: bob joined the group") + (bob <## "#team: you joined the group") + alice #> "#team hello" + bob <# "#team alice> hello" + bob #> "#team hi there" + alice <# "#team bob> hi there" + -- group 2 + alice ##> "/g club" + alice <## "group #club is created" + alice <## "to add members use /a club or /create link #club" + alice ##> "/a club bob" + concurrentlyN_ + [ alice <## "invitation to join the group #club sent to bob", + do + bob <## "#club: alice invites you to join the group as admin" + bob <## "use /j club to accept" + ] + bob ##> "/j club" + concurrently_ + (alice <## "#club: bob joined the group") + (bob <## "#club: you joined the group") + alice #> "#club hello" + bob <# "#club alice> hello" + bob #> "#club hi there" + alice <# "#club bob> hi there" + -- delete contact + alice ##> "/d bob" + alice <## "bob: contact is deleted" + alice ##> "@bob hey" + alice <## "no contact bob" + bob #> "@alice hey" + bob <## "[alice, contactId: 2, connId: 1] error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" + (alice "/d #team" + concurrentlyN_ + [ alice <## "#team: you deleted the group", + do + bob <## "#team: alice deleted the group" + bob <## "use /d #team to delete the local copy of the group" + ] + alice ##> "#team hi" + alice <## "no group #team" + bob ##> "/d #team" + bob <## "#team: you deleted the group" + -- group 2 still works + alice #> "#club checking connection" + bob <# "#club alice> checking connection" + bob #> "#club received" + alice <# "#club bob> received" + +testGroupRemoveAdd :: HasCallStack => FilePath -> IO () +testGroupRemoveAdd = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + -- remove member + alice ##> "/rm team bob" + concurrentlyN_ + [ alice <## "#team: you removed bob from the group", + do + bob <## "#team: alice removed you from the group" + bob <## "use /d #team to delete the group", + cath <## "#team: alice removed bob from the group" + ] + alice ##> "/a team bob" + alice <## "invitation to join the group #team sent to bob" + bob <## "#team_1 (team): alice invites you to join the group as admin" + bob <## "use /j team_1 to accept" + bob ##> "/j team_1" + concurrentlyN_ + [ alice <## "#team: bob joined the group", + do + bob <## "#team_1: you joined the group" + bob <## "#team_1: member cath_1 (Catherine) is connected" + bob <## "contact cath_1 is merged into cath" + bob <## "use @cath to send messages", + do + cath <## "#team: alice added bob_1 (Bob) to the group (connecting...)" + cath <## "#team: new member bob_1 is connected" + cath <## "contact bob_1 is merged into bob" + cath <## "use @bob to send messages" + ] + alice #> "#team hi" + concurrently_ + (bob <# "#team_1 alice> hi") + (cath <# "#team alice> hi") + bob #> "#team_1 hey" + concurrently_ + (alice <# "#team bob> hey") + (cath <# "#team bob> hey") + cath #> "#team hello" + concurrently_ + (alice <# "#team cath> hello") + (bob <# "#team_1 cath> hello") + +testGroupList :: HasCallStack => FilePath -> IO () +testGroupList = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + createGroup2 "team" alice bob + alice ##> "/g tennis" + alice <## "group #tennis is created" + alice <## "to add members use /a tennis or /create link #tennis" + alice ##> "/a tennis bob" + concurrentlyN_ + [ alice <## "invitation to join the group #tennis sent to bob", + do + bob <## "#tennis: alice invites you to join the group as admin" + bob <## "use /j tennis to accept" + ] + -- alice sees both groups + alice ##> "/gs" + alice <### ["#team", "#tennis"] + -- bob sees #tennis as invitation + bob ##> "/gs" + bob + <### [ "#team", + "#tennis - you are invited (/j tennis to join, /d #tennis to delete invitation)" + ] + -- after deleting invitation bob sees only one group + bob ##> "/d #tennis" + bob <## "#tennis: you deleted the group" + bob ##> "/gs" + bob <## "#team" + +testGroupMessageQuotedReply :: HasCallStack => FilePath -> IO () +testGroupMessageQuotedReply = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + threadDelay 1000000 + alice #> "#team hello! how are you?" + concurrently_ + (bob <# "#team alice> hello! how are you?") + (cath <# "#team alice> hello! how are you?") + threadDelay 1000000 + bob `send` "> #team @alice (hello) hello, all good, you?" + bob <# "#team > alice hello! how are you?" + bob <## " hello, all good, you?" + concurrently_ + ( do + alice <# "#team bob> > alice hello! how are you?" + alice <## " hello, all good, you?" + ) + ( do + cath <# "#team bob> > alice hello! how are you?" + cath <## " hello, all good, you?" + ) + bob #$> ("/_get chat #1 count=2", chat', [((0, "hello! how are you?"), Nothing), ((1, "hello, all good, you?"), Just (0, "hello! how are you?"))]) + alice #$> ("/_get chat #1 count=2", chat', [((1, "hello! how are you?"), Nothing), ((0, "hello, all good, you?"), Just (1, "hello! how are you?"))]) + cath #$> ("/_get chat #1 count=2", chat', [((0, "hello! how are you?"), Nothing), ((0, "hello, all good, you?"), Just (0, "hello! how are you?"))]) + bob `send` "> #team bob (hello, all good) will tell more" + bob <# "#team > bob hello, all good, you?" + bob <## " will tell more" + concurrently_ + ( do + alice <# "#team bob> > bob hello, all good, you?" + alice <## " will tell more" + ) + ( do + cath <# "#team bob> > bob hello, all good, you?" + cath <## " will tell more" + ) + bob #$> ("/_get chat #1 count=1", chat', [((1, "will tell more"), Just (1, "hello, all good, you?"))]) + alice #$> ("/_get chat #1 count=1", chat', [((0, "will tell more"), Just (0, "hello, all good, you?"))]) + cath #$> ("/_get chat #1 count=1", chat', [((0, "will tell more"), Just (0, "hello, all good, you?"))]) + threadDelay 1000000 + cath `send` "> #team bob (hello) hi there!" + cath <# "#team > bob hello, all good, you?" + cath <## " hi there!" + concurrently_ + ( do + alice <# "#team cath> > bob hello, all good, you?" + alice <## " hi there!" + ) + ( do + bob <# "#team cath> > bob hello, all good, you?" + bob <## " hi there!" + ) + cath #$> ("/_get chat #1 count=1", chat', [((1, "hi there!"), Just (0, "hello, all good, you?"))]) + alice #$> ("/_get chat #1 count=1", chat', [((0, "hi there!"), Just (0, "hello, all good, you?"))]) + bob #$> ("/_get chat #1 count=1", chat', [((0, "hi there!"), Just (1, "hello, all good, you?"))]) + alice `send` "> #team (will tell) go on" + alice <# "#team > bob will tell more" + alice <## " go on" + concurrently_ + ( do + bob <# "#team alice> > bob will tell more" + bob <## " go on" + ) + ( do + cath <# "#team alice> > bob will tell more" + cath <## " go on" + ) + +testGroupMessageUpdate :: HasCallStack => FilePath -> IO () +testGroupMessageUpdate = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + threadDelay 1000000 + -- alice, bob: msg id 5, cath: msg id 4 (after group invitations & group events) + alice #> "#team hello!" + concurrently_ + (bob <# "#team alice> hello!") + (cath <# "#team alice> hello!") + + msgItemId1 <- lastItemId alice + alice ##> ("/_update item #1 " <> msgItemId1 <> " text hey 👋") + alice <# "#team [edited] hey 👋" + concurrently_ + (bob <# "#team alice> [edited] hey 👋") + (cath <# "#team alice> [edited] hey 👋") + + alice #$> ("/_get chat #1 count=1", chat', [((1, "hey 👋"), Nothing)]) + bob #$> ("/_get chat #1 count=1", chat', [((0, "hey 👋"), Nothing)]) + cath #$> ("/_get chat #1 count=1", chat', [((0, "hey 👋"), Nothing)]) + + threadDelay 1000000 + -- alice, bob: msg id 6, cath: msg id 5 + bob `send` "> #team @alice (hey) hi alice" + bob <# "#team > alice hey 👋" + bob <## " hi alice" + concurrently_ + ( do + alice <# "#team bob> > alice hey 👋" + alice <## " hi alice" + ) + ( do + cath <# "#team bob> > alice hey 👋" + cath <## " hi alice" + ) + + alice #$> ("/_get chat #1 count=2", chat', [((1, "hey 👋"), Nothing), ((0, "hi alice"), Just (1, "hey 👋"))]) + bob #$> ("/_get chat #1 count=2", chat', [((0, "hey 👋"), Nothing), ((1, "hi alice"), Just (0, "hey 👋"))]) + cath #$> ("/_get chat #1 count=2", chat', [((0, "hey 👋"), Nothing), ((0, "hi alice"), Just (0, "hey 👋"))]) + + alice ##> ("/_update item #1 " <> msgItemId1 <> " text greetings 🤝") + alice <# "#team [edited] greetings 🤝" + concurrently_ + (bob <# "#team alice> [edited] greetings 🤝") + (cath <# "#team alice> [edited] greetings 🤝") + + msgItemId2 <- lastItemId alice + alice #$> ("/_update item #1 " <> msgItemId2 <> " text updating bob's message", id, "cannot update this item") + + threadDelay 1000000 + cath `send` "> #team @alice (greetings) greetings!" + cath <# "#team > alice greetings 🤝" + cath <## " greetings!" + concurrently_ + ( do + alice <# "#team cath> > alice greetings 🤝" + alice <## " greetings!" + ) + ( do + bob <# "#team cath> > alice greetings 🤝" + bob <## " greetings!" + ) + + alice #$> ("/_get chat #1 count=3", chat', [((1, "greetings 🤝"), Nothing), ((0, "hi alice"), Just (1, "hey 👋")), ((0, "greetings!"), Just (1, "greetings 🤝"))]) + bob #$> ("/_get chat #1 count=3", chat', [((0, "greetings 🤝"), Nothing), ((1, "hi alice"), Just (0, "hey 👋")), ((0, "greetings!"), Just (0, "greetings 🤝"))]) + cath #$> ("/_get chat #1 count=3", chat', [((0, "greetings 🤝"), Nothing), ((0, "hi alice"), Just (0, "hey 👋")), ((1, "greetings!"), Just (0, "greetings 🤝"))]) + +testGroupMessageDelete :: HasCallStack => FilePath -> IO () +testGroupMessageDelete = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + threadDelay 1000000 + -- alice, bob: msg id 5, cath: msg id 4 (after group invitations & group events) + alice #> "#team hello!" + concurrently_ + (bob <# "#team alice> hello!") + (cath <# "#team alice> hello!") + + msgItemId1 <- lastItemId alice + alice #$> ("/_delete item #1 " <> msgItemId1 <> " internal", id, "message deleted") + + alice #$> ("/_get chat #1 count=1", chat, [(0, "connected")]) + bob #$> ("/_get chat #1 count=1", chat, [(0, "hello!")]) + cath #$> ("/_get chat #1 count=1", chat, [(0, "hello!")]) + + threadDelay 1000000 + -- alice: msg id 5, bob: msg id 6, cath: msg id 5 + bob `send` "> #team @alice (hello) hi alic" + bob <# "#team > alice hello!" + bob <## " hi alic" + concurrently_ + ( do + alice <# "#team bob> > alice hello!" + alice <## " hi alic" + ) + ( do + cath <# "#team bob> > alice hello!" + cath <## " hi alic" + ) + + alice #$> ("/_get chat #1 count=1", chat', [((0, "hi alic"), Just (1, "hello!"))]) + bob #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((1, "hi alic"), Just (0, "hello!"))]) + cath #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((0, "hi alic"), Just (0, "hello!"))]) + + msgItemId2 <- lastItemId alice + alice #$> ("/_delete item #1 " <> msgItemId2 <> " internal", id, "message deleted") + + alice #$> ("/_get chat #1 count=1", chat', [((0, "connected"), Nothing)]) + bob #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((1, "hi alic"), Just (0, "hello!"))]) + cath #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((0, "hi alic"), Just (0, "hello!"))]) + + -- alice: msg id 5 + msgItemId3 <- lastItemId bob + bob ##> ("/_update item #1 " <> msgItemId3 <> " text hi alice") + bob <# "#team [edited] > alice hello!" + bob <## " hi alice" + concurrently_ + (alice <# "#team bob> [edited] hi alice") + ( do + cath <# "#team bob> [edited] > alice hello!" + cath <## " hi alice" + ) + + alice #$> ("/_get chat #1 count=1", chat', [((0, "hi alice"), Nothing)]) + bob #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((1, "hi alice"), Just (0, "hello!"))]) + cath #$> ("/_get chat #1 count=2", chat', [((0, "hello!"), Nothing), ((0, "hi alice"), Just (0, "hello!"))]) + + threadDelay 1000000 + -- alice: msg id 6, bob: msg id 7, cath: msg id 6 + cath #> "#team how are you?" + concurrently_ + (alice <# "#team cath> how are you?") + (bob <# "#team cath> how are you?") + + msgItemId4 <- lastItemId cath + cath #$> ("/_delete item #1 " <> msgItemId4 <> " broadcast", id, "message marked deleted") + concurrently_ + (alice <# "#team cath> [marked deleted] how are you?") + (bob <# "#team cath> [marked deleted] how are you?") + + alice ##> "/last_item_id 1" + msgItemId6 <- getTermLine alice + alice #$> ("/_delete item #1 " <> msgItemId6 <> " broadcast", id, "cannot delete this item") + alice #$> ("/_delete item #1 " <> msgItemId6 <> " internal", id, "message deleted") + + alice #$> ("/_get chat #1 count=1", chat', [((0, "how are you? [marked deleted]"), Nothing)]) + bob #$> ("/_get chat #1 count=3", chat', [((0, "hello!"), Nothing), ((1, "hi alice"), Just (0, "hello!")), ((0, "how are you? [marked deleted]"), Nothing)]) + cath #$> ("/_get chat #1 count=3", chat', [((0, "hello!"), Nothing), ((0, "hi alice"), Just (0, "hello!")), ((1, "how are you? [marked deleted]"), Nothing)]) + +testGroupLiveMessage :: HasCallStack => FilePath -> IO () +testGroupLiveMessage = + testChat3 aliceProfile bobProfile cathProfile $ \alice bob cath -> do + createGroup3 "team" alice bob cath + threadDelay 500000 + -- non-empty live message is sent instantly + alice `send` "/live #team hello" + msgItemId1 <- lastItemId alice + bob <#. "#team alice> [LIVE started]" + cath <#. "#team alice> [LIVE started]" + alice ##> ("/_update item #1 " <> msgItemId1 <> " text hello there") + alice <# "#team [LIVE] hello there" + bob <# "#team alice> [LIVE ended] hello there" + cath <# "#team alice> [LIVE ended] hello there" + -- empty live message is also sent instantly + alice `send` "/live #team" + msgItemId2 <- lastItemId alice + bob <#. "#team alice> [LIVE started]" + cath <#. "#team alice> [LIVE started]" + alice ##> ("/_update item #1 " <> msgItemId2 <> " text hello 2") + alice <# "#team [LIVE] hello 2" + bob <# "#team alice> [LIVE ended] hello 2" + cath <# "#team alice> [LIVE ended] hello 2" + +testUpdateGroupProfile :: HasCallStack => FilePath -> IO () +testUpdateGroupProfile = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + threadDelay 1000000 + alice #> "#team hello!" + concurrently_ + (bob <# "#team alice> hello!") + (cath <# "#team alice> hello!") + bob ##> "/gp team my_team" + bob <## "you have insufficient permissions for this action, the required role is owner" + alice ##> "/gp team my_team" + alice <## "changed to #my_team" + concurrentlyN_ + [ do + bob <## "alice updated group #team:" + bob <## "changed to #my_team", + do + cath <## "alice updated group #team:" + cath <## "changed to #my_team" + ] + bob #> "#my_team hi" + concurrently_ + (alice <# "#my_team bob> hi") + (cath <# "#my_team bob> hi") + +testUpdateMemberRole :: HasCallStack => FilePath -> IO () +testUpdateMemberRole = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + connectUsers alice bob + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + addMember "team" alice bob GRAdmin + alice ##> "/mr team bob member" + alice <## "#team: you changed the role of bob from admin to member" + bob <## "#team: alice invites you to join the group as member" + bob <## "use /j team to accept" + bob ##> "/j team" + concurrently_ + (alice <## "#team: bob joined the group") + (bob <## "#team: you joined the group") + connectUsers bob cath + bob ##> "/a team cath" + bob <## "you have insufficient permissions for this action, the required role is admin" + alice ##> "/mr team bob admin" + concurrently_ + (alice <## "#team: you changed the role of bob from member to admin") + (bob <## "#team: alice changed your role from member to admin") + bob ##> "/a team cath owner" + bob <## "you have insufficient permissions for this action, the required role is owner" + addMember "team" bob cath GRMember + cath ##> "/j team" + concurrentlyN_ + [ bob <## "#team: cath joined the group", + do + cath <## "#team: you joined the group" + cath <## "#team: member alice (Alice) is connected", + do + alice <## "#team: bob added cath (Catherine) to the group (connecting...)" + alice <## "#team: new member cath is connected" + ] + alice ##> "/mr team alice admin" + concurrentlyN_ + [ alice <## "#team: you changed your role from owner to admin", + bob <## "#team: alice changed the role from owner to admin", + cath <## "#team: alice changed the role from owner to admin" + ] + alice ##> "/d #team" + alice <## "you have insufficient permissions for this action, the required role is owner" + +testGroupDeleteUnusedContacts :: HasCallStack => FilePath -> IO () +testGroupDeleteUnusedContacts = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + -- create group 1 + createGroup3 "team" alice bob cath + -- create group 2 + alice ##> "/g club" + alice <## "group #club is created" + alice <## "to add members use /a club or /create link #club" + alice ##> "/a club bob" + concurrentlyN_ + [ alice <## "invitation to join the group #club sent to bob", + do + bob <## "#club: alice invites you to join the group as admin" + bob <## "use /j club to accept" + ] + bob ##> "/j club" + concurrently_ + (alice <## "#club: bob joined the group") + (bob <## "#club: you joined the group") + alice ##> "/a club cath" + concurrentlyN_ + [ alice <## "invitation to join the group #club sent to cath", + do + cath <## "#club: alice invites you to join the group as admin" + cath <## "use /j club to accept" + ] + cath ##> "/j club" + concurrentlyN_ + [ alice <## "#club: cath joined the group", + do + cath <## "#club: you joined the group" + cath <## "#club: member bob_1 (Bob) is connected" + cath <## "contact bob_1 is merged into bob" + cath <## "use @bob to send messages", + do + bob <## "#club: alice added cath_1 (Catherine) to the group (connecting...)" + bob <## "#club: new member cath_1 is connected" + bob <## "contact cath_1 is merged into cath" + bob <## "use @cath to send messages" + ] + -- list contacts + bob ##> "/contacts" + bob <## "alice (Alice)" + bob <## "cath (Catherine)" + cath ##> "/contacts" + cath <## "alice (Alice)" + cath <## "bob (Bob)" + -- delete group 1, contacts and profiles are kept + deleteGroup alice bob cath "team" + bob ##> "/contacts" + bob <## "alice (Alice)" + bob <## "cath (Catherine)" + bob `hasContactProfiles` ["alice", "bob", "cath"] + cath ##> "/contacts" + cath <## "alice (Alice)" + cath <## "bob (Bob)" + cath `hasContactProfiles` ["alice", "bob", "cath"] + -- delete group 2, unused contacts and profiles are deleted + deleteGroup alice bob cath "club" + bob ##> "/contacts" + bob <## "alice (Alice)" + bob `hasContactProfiles` ["alice", "bob"] + cath ##> "/contacts" + cath <## "alice (Alice)" + cath `hasContactProfiles` ["alice", "cath"] + where + deleteGroup :: HasCallStack => TestCC -> TestCC -> TestCC -> String -> IO () + deleteGroup alice bob cath group = do + alice ##> ("/d #" <> group) + concurrentlyN_ + [ alice <## ("#" <> group <> ": you deleted the group"), + do + bob <## ("#" <> group <> ": alice deleted the group") + bob <## ("use /d #" <> group <> " to delete the local copy of the group"), + do + cath <## ("#" <> group <> ": alice deleted the group") + cath <## ("use /d #" <> group <> " to delete the local copy of the group") + ] + bob ##> ("/d #" <> group) + bob <## ("#" <> group <> ": you deleted the group") + cath ##> ("/d #" <> group) + cath <## ("#" <> group <> ": you deleted the group") + +testGroupDescription :: HasCallStack => FilePath -> IO () +testGroupDescription = testChat4 aliceProfile bobProfile cathProfile danProfile $ \alice bob cath dan -> do + connectUsers alice bob + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + addMember "team" alice bob GRAdmin + bob ##> "/j team" + concurrentlyN_ + [ alice <## "#team: bob joined the group", + bob <## "#team: you joined the group" + ] + alice ##> "/group_profile team" + alice <## "#team" + groupInfo alice + alice ##> "/group_descr team Welcome to the team!" + alice <## "description changed to:" + alice <## "Welcome to the team!" + bob <## "alice updated group #team:" + bob <## "description changed to:" + bob <## "Welcome to the team!" + alice ##> "/group_profile team" + alice <## "#team" + alice <## "description:" + alice <## "Welcome to the team!" + groupInfo alice + connectUsers alice cath + addMember "team" alice cath GRMember + cath ##> "/j team" + concurrentlyN_ + [ alice <## "#team: cath joined the group", + do + cath <## "#team: you joined the group" + cath <# "#team alice> Welcome to the team!" + cath <## "#team: member bob (Bob) is connected", + do + bob <## "#team: alice added cath (Catherine) to the group (connecting...)" + bob <## "#team: new member cath is connected" + ] + connectUsers bob dan + addMember "team" bob dan GRMember + dan ##> "/j team" + concurrentlyN_ + [ bob <## "#team: dan joined the group", + do + dan <## "#team: you joined the group" + dan <# "#team bob> Welcome to the team!" + dan + <### [ "#team: member alice (Alice) is connected", + "#team: member cath (Catherine) is connected" + ], + bobAddedDan alice, + bobAddedDan cath + ] + where + groupInfo :: HasCallStack => TestCC -> IO () + groupInfo alice = do + alice <## "group preferences:" + alice <## "Disappearing messages: off" + alice <## "Direct messages: on" + alice <## "Full deletion: off" + alice <## "Voice messages: on" + bobAddedDan :: HasCallStack => TestCC -> IO () + bobAddedDan cc = do + cc <## "#team: bob added dan (Daniel) to the group (connecting...)" + cc <## "#team: new member dan is connected" + +testGroupAsync :: HasCallStack => FilePath -> IO () +testGroupAsync tmp = do + print (0 :: Integer) + withNewTestChat tmp "alice" aliceProfile $ \alice -> do + withNewTestChat tmp "bob" bobProfile $ \bob -> do + connectUsers alice bob + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/a team bob" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to bob", + do + bob <## "#team: alice invites you to join the group as admin" + bob <## "use /j team to accept" + ] + bob ##> "/j team" + concurrently_ + (alice <## "#team: bob joined the group") + (bob <## "#team: you joined the group") + alice #> "#team hello bob" + bob <# "#team alice> hello bob" + print (1 :: Integer) + withTestChat tmp "alice" $ \alice -> do + withNewTestChat tmp "cath" cathProfile $ \cath -> do + alice <## "1 contacts connected (use /cs for the list)" + alice <## "#team: connected to server(s)" + connectUsers alice cath + alice ##> "/a team cath" + concurrentlyN_ + [ alice <## "invitation to join the group #team sent to cath", + do + cath <## "#team: alice invites you to join the group as admin" + cath <## "use /j team to accept" + ] + cath ##> "/j team" + concurrentlyN_ + [ alice <## "#team: cath joined the group", + cath <## "#team: you joined the group" + ] + alice #> "#team hello cath" + cath <# "#team alice> hello cath" + print (2 :: Integer) + withTestChat tmp "bob" $ \bob -> do + withTestChat tmp "cath" $ \cath -> do + concurrentlyN_ + [ do + bob <## "1 contacts connected (use /cs for the list)" + bob <## "#team: connected to server(s)" + bob <## "#team: alice added cath (Catherine) to the group (connecting...)" + bob <# "#team alice> hello cath" + bob <## "#team: new member cath is connected", + do + cath <## "2 contacts connected (use /cs for the list)" + cath <## "#team: connected to server(s)" + cath <## "#team: member bob (Bob) is connected" + ] + threadDelay 500000 + print (3 :: Integer) + withTestChat tmp "bob" $ \bob -> do + withNewTestChat tmp "dan" danProfile $ \dan -> do + bob <## "2 contacts connected (use /cs for the list)" + bob <## "#team: connected to server(s)" + connectUsers bob dan + bob ##> "/a team dan" + concurrentlyN_ + [ bob <## "invitation to join the group #team sent to dan", + do + dan <## "#team: bob invites you to join the group as admin" + dan <## "use /j team to accept" + ] + dan ##> "/j team" + concurrentlyN_ + [ bob <## "#team: dan joined the group", + dan <## "#team: you joined the group" + ] + threadDelay 1000000 + threadDelay 1000000 + print (4 :: Integer) + withTestChat tmp "alice" $ \alice -> do + withTestChat tmp "cath" $ \cath -> do + withTestChat tmp "dan" $ \dan -> do + concurrentlyN_ + [ do + alice <## "2 contacts connected (use /cs for the list)" + alice <## "#team: connected to server(s)" + alice <## "#team: bob added dan (Daniel) to the group (connecting...)" + alice <## "#team: new member dan is connected", + do + cath <## "2 contacts connected (use /cs for the list)" + cath <## "#team: connected to server(s)" + cath <## "#team: bob added dan (Daniel) to the group (connecting...)" + cath <## "#team: new member dan is connected", + do + dan <## "3 contacts connected (use /cs for the list)" + dan <## "#team: connected to server(s)" + dan <## "#team: member alice (Alice) is connected" + dan <## "#team: member cath (Catherine) is connected" + ] + threadDelay 1000000 + print (5 :: Integer) + withTestChat tmp "alice" $ \alice -> do + withTestChat tmp "bob" $ \bob -> do + withTestChat tmp "cath" $ \cath -> do + withTestChat tmp "dan" $ \dan -> do + concurrentlyN_ + [ do + alice <## "3 contacts connected (use /cs for the list)" + alice <## "#team: connected to server(s)", + do + bob <## "3 contacts connected (use /cs for the list)" + bob <## "#team: connected to server(s)", + do + cath <## "3 contacts connected (use /cs for the list)" + cath <## "#team: connected to server(s)", + do + dan <## "3 contacts connected (use /cs for the list)" + dan <## "#team: connected to server(s)" + ] + alice #> "#team hello" + concurrentlyN_ + [ bob <# "#team alice> hello", + cath <# "#team alice> hello", + dan <# "#team alice> hello" + ] + bob #> "#team hi there" + concurrentlyN_ + [ alice <# "#team bob> hi there", + cath <# "#team bob> hi there", + dan <# "#team bob> hi there" + ] + cath #> "#team hey" + concurrentlyN_ + [ alice <# "#team cath> hey", + bob <# "#team cath> hey", + dan <# "#team cath> hey" + ] + dan #> "#team how is it going?" + concurrentlyN_ + [ alice <# "#team dan> how is it going?", + bob <# "#team dan> how is it going?", + cath <# "#team dan> how is it going?" + ] + bob <##> cath + dan <##> cath + dan <##> alice + +testGroupLink :: HasCallStack => FilePath -> IO () +testGroupLink = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/show link #team" + alice <## "no group link, to create: /create link #team" + alice ##> "/create link #team" + _ <- getGroupLink alice "team" True + alice ##> "/delete link #team" + alice <## "Group link is deleted - joined members will remain connected." + alice <## "To create a new group link use /create link #team" + alice ##> "/create link #team" + gLink <- getGroupLink alice "team" True + alice ##> "/show link #team" + _ <- getGroupLink alice "team" False + alice ##> "/create link #team" + alice <## "you already have link for this group, to show: /show link #team" + bob ##> ("/c " <> gLink) + bob <## "connection request sent!" + alice <## "bob (Bob): accepting request to join group #team..." + concurrentlyN_ + [ do + alice <## "bob (Bob): contact is connected" + alice <## "bob invited to group #team via your group link" + alice <## "#team: bob joined the group", + do + bob <## "alice (Alice): contact is connected" + bob <## "#team: you joined the group" + ] + threadDelay 100000 + alice #$> ("/_get chat #1 count=100", chat, [(0, "invited via your group link"), (0, "connected")]) + -- contacts connected via group link are not in chat previews + alice @@@ [("#team", "connected")] + bob @@@ [("#team", "connected")] + alice <##> bob + alice @@@ [("@bob", "hey"), ("#team", "connected")] + + -- user address doesn't interfere + alice ##> "/ad" + cLink <- getContactLink alice True + cath ##> ("/c " <> cLink) + alice <#? cath + alice ##> "/ac cath" + alice <## "cath (Catherine): accepting contact request..." + concurrently_ + (cath <## "alice (Alice): contact is connected") + (alice <## "cath (Catherine): contact is connected") + alice <##> cath + + -- third member + cath ##> ("/c " <> gLink) + cath <## "connection request sent!" + alice <## "cath_1 (Catherine): accepting request to join group #team..." + -- if contact existed it is merged + concurrentlyN_ + [ alice + <### [ "cath_1 (Catherine): contact is connected", + "contact cath_1 is merged into cath", + "use @cath to send messages", + EndsWith "invited to group #team via your group link", + EndsWith "joined the group" + ], + cath + <### [ "alice_1 (Alice): contact is connected", + "contact alice_1 is merged into alice", + "use @alice to send messages", + "#team: you joined the group", + "#team: member bob (Bob) is connected" + ], + do + bob <## "#team: alice added cath (Catherine) to the group (connecting...)" + bob <## "#team: new member cath is connected" + ] + alice #> "#team hello" + concurrently_ + (bob <# "#team alice> hello") + (cath <# "#team alice> hello") + bob #> "#team hi there" + concurrently_ + (alice <# "#team bob> hi there") + (cath <# "#team bob> hi there") + cath #> "#team hey team" + concurrently_ + (alice <# "#team cath> hey team") + (bob <# "#team cath> hey team") + + -- leaving team removes link + alice ##> "/l team" + concurrentlyN_ + [ do + alice <## "#team: you left the group" + alice <## "use /d #team to delete the group", + bob <## "#team: alice left the group", + cath <## "#team: alice left the group" + ] + alice ##> "/show link #team" + alice <## "no group link, to create: /create link #team" + +testGroupLinkDeleteGroupRejoin :: HasCallStack => FilePath -> IO () +testGroupLinkDeleteGroupRejoin = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/create link #team" + gLink <- getGroupLink alice "team" True + bob ##> ("/c " <> gLink) + bob <## "connection request sent!" + alice <## "bob (Bob): accepting request to join group #team..." + concurrentlyN_ + [ do + alice <## "bob (Bob): contact is connected" + alice <## "bob invited to group #team via your group link" + alice <## "#team: bob joined the group", + do + bob <## "alice (Alice): contact is connected" + bob <## "#team: you joined the group" + ] + -- use contact so it's not deleted when deleting group + bob <##> alice + bob ##> "/l team" + concurrentlyN_ + [ do + bob <## "#team: you left the group" + bob <## "use /d #team to delete the group", + alice <## "#team: bob left the group" + ] + bob ##> "/d #team" + bob <## "#team: you deleted the group" + -- re-join via same link + bob ##> ("/c " <> gLink) + bob <## "connection request sent!" + alice <## "bob_1 (Bob): accepting request to join group #team..." + concurrentlyN_ + [ alice + <### [ "bob_1 (Bob): contact is connected", + "contact bob_1 is merged into bob", + "use @bob to send messages", + EndsWith "invited to group #team via your group link", + EndsWith "joined the group" + ], + bob + <### [ "alice_1 (Alice): contact is connected", + "contact alice_1 is merged into alice", + "use @alice to send messages", + "#team: you joined the group" + ] + ] + alice #> "#team hello" + bob <# "#team alice> hello" + bob #> "#team hi there" + alice <# "#team bob> hi there" + +testGroupLinkContactUsed :: HasCallStack => FilePath -> IO () +testGroupLinkContactUsed = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/create link #team" + gLink <- getGroupLink alice "team" True + bob ##> ("/c " <> gLink) + bob <## "connection request sent!" + alice <## "bob (Bob): accepting request to join group #team..." + concurrentlyN_ + [ do + alice <## "bob (Bob): contact is connected" + alice <## "bob invited to group #team via your group link" + alice <## "#team: bob joined the group", + do + bob <## "alice (Alice): contact is connected" + bob <## "#team: you joined the group" + ] + -- sending/receiving a message marks contact as used + threadDelay 100000 + alice @@@ [("#team", "connected")] + bob @@@ [("#team", "connected")] + alice #> "@bob hello" + bob <# "alice> hello" + alice #$> ("/clear bob", id, "bob: all messages are removed locally ONLY") + alice @@@ [("@bob", ""), ("#team", "connected")] + bob #$> ("/clear alice", id, "alice: all messages are removed locally ONLY") + bob @@@ [("@alice", ""), ("#team", "connected")] + +testGroupLinkIncognitoMembership :: HasCallStack => FilePath -> IO () +testGroupLinkIncognitoMembership = + testChat4 aliceProfile bobProfile cathProfile danProfile $ + \alice bob cath dan -> do + -- bob connected incognito to alice + alice ##> "/c" + inv <- getInvitation alice + bob #$> ("/incognito on", id, "ok") + bob ##> ("/c " <> inv) + bob <## "confirmation sent!" + bobIncognito <- getTermLine bob + concurrentlyN_ + [ do + bob <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) + bob <## "use /i alice to print out this incognito profile again", + alice <## (bobIncognito <> ": contact is connected") + ] + bob #$> ("/incognito off", id, "ok") + -- alice creates group + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + -- alice invites bob + alice ##> ("/a team " <> bobIncognito) + concurrentlyN_ + [ alice <## ("invitation to join the group #team sent to " <> bobIncognito), + do + bob <## "#team: alice invites you to join the group as admin" + bob <## ("use /j team to join incognito as " <> bobIncognito) + ] + bob ##> "/j team" + concurrently_ + (alice <## ("#team: " <> bobIncognito <> " joined the group")) + (bob <## ("#team: you joined the group incognito as " <> bobIncognito)) + -- bob creates group link, cath joins + bob ##> "/create link #team" + gLink <- getGroupLink bob "team" True + cath ##> ("/c " <> gLink) + cath <## "connection request sent!" + bob <## "cath (Catherine): accepting request to join group #team..." + _ <- getTermLine bob + concurrentlyN_ + [ do + bob <## ("cath (Catherine): contact is connected, your incognito profile for this contact is " <> bobIncognito) + bob <## "use /i cath to print out this incognito profile again" + bob <## "cath invited to group #team via your group link" + bob <## "#team: cath joined the group", + do + cath <## (bobIncognito <> ": contact is connected") + cath <## "#team: you joined the group" + cath <## "#team: member alice (Alice) is connected", + do + alice <## ("#team: " <> bobIncognito <> " added cath (Catherine) to the group (connecting...)") + alice <## "#team: new member cath is connected" + ] + bob ?#> "@cath hi, I'm incognito" + cath <# (bobIncognito <> "> hi, I'm incognito") + cath #> ("@" <> bobIncognito <> " hey, I'm cath") + bob ?<# "cath> hey, I'm cath" + -- dan joins incognito + dan #$> ("/incognito on", id, "ok") + dan ##> ("/c " <> gLink) + danIncognito <- getTermLine dan + dan <## "connection request sent incognito!" + bob <## (danIncognito <> ": accepting request to join group #team...") + _ <- getTermLine bob + _ <- getTermLine dan + concurrentlyN_ + [ do + bob <## (danIncognito <> ": contact is connected, your incognito profile for this contact is " <> bobIncognito) + bob <## ("use /i " <> danIncognito <> " to print out this incognito profile again") + bob <## (danIncognito <> " invited to group #team via your group link") + bob <## ("#team: " <> danIncognito <> " joined the group"), + do + dan <## (bobIncognito <> ": contact is connected, your incognito profile for this contact is " <> danIncognito) + dan <## ("use /i " <> bobIncognito <> " to print out this incognito profile again") + dan <## ("#team: you joined the group incognito as " <> danIncognito) + dan + <### [ "#team: member alice (Alice) is connected", + "#team: member cath (Catherine) is connected" + ], + do + alice <## ("#team: " <> bobIncognito <> " added " <> danIncognito <> " to the group (connecting...)") + alice <## ("#team: new member " <> danIncognito <> " is connected"), + do + cath <## ("#team: " <> bobIncognito <> " added " <> danIncognito <> " to the group (connecting...)") + cath <## ("#team: new member " <> danIncognito <> " is connected") + ] + dan #$> ("/incognito off", id, "ok") + bob ?#> ("@" <> danIncognito <> " hi, I'm incognito") + dan ?<# (bobIncognito <> "> hi, I'm incognito") + dan ?#> ("@" <> bobIncognito <> " hey, me too") + bob ?<# (danIncognito <> "> hey, me too") + alice #> "#team hello" + concurrentlyN_ + [ bob ?<# "#team alice> hello", + cath <# "#team alice> hello", + dan ?<# "#team alice> hello" + ] + bob ?#> "#team hi there" + concurrentlyN_ + [ alice <# ("#team " <> bobIncognito <> "> hi there"), + cath <# ("#team " <> bobIncognito <> "> hi there"), + dan ?<# ("#team " <> bobIncognito <> "> hi there") + ] + cath #> "#team hey" + concurrentlyN_ + [ alice <# "#team cath> hey", + bob ?<# "#team cath> hey", + dan ?<# "#team cath> hey" + ] + dan ?#> "#team how is it going?" + concurrentlyN_ + [ alice <# ("#team " <> danIncognito <> "> how is it going?"), + bob ?<# ("#team " <> danIncognito <> "> how is it going?"), + cath <# ("#team " <> danIncognito <> "> how is it going?") + ] + +testGroupLinkUnusedHostContactDeleted :: HasCallStack => FilePath -> IO () +testGroupLinkUnusedHostContactDeleted = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + -- create group 1 + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> "/create link #team" + gLinkTeam <- getGroupLink alice "team" True + bob ##> ("/c " <> gLinkTeam) + bob <## "connection request sent!" + alice <## "bob (Bob): accepting request to join group #team..." + concurrentlyN_ + [ do + alice <## "bob (Bob): contact is connected" + alice <## "bob invited to group #team via your group link" + alice <## "#team: bob joined the group", + do + bob <## "alice (Alice): contact is connected" + bob <## "#team: you joined the group" + ] + -- create group 2 + alice ##> "/g club" + alice <## "group #club is created" + alice <## "to add members use /a club or /create link #club" + alice ##> "/create link #club" + gLinkClub <- getGroupLink alice "club" True + bob ##> ("/c " <> gLinkClub) + bob <## "connection request sent!" + alice <## "bob_1 (Bob): accepting request to join group #club..." + concurrentlyN_ + [ alice + <### [ "bob_1 (Bob): contact is connected", + "contact bob_1 is merged into bob", + "use @bob to send messages", + EndsWith "invited to group #club via your group link", + EndsWith "joined the group" + ], + bob + <### [ "alice_1 (Alice): contact is connected", + "contact alice_1 is merged into alice", + "use @alice to send messages", + "#club: you joined the group" + ] + ] + -- list contacts + bob ##> "/contacts" + bob <## "alice (Alice)" + -- delete group 1, host contact and profile are kept + bobLeaveDeleteGroup alice bob "team" + bob ##> "/contacts" + bob <## "alice (Alice)" + bob `hasContactProfiles` ["alice", "bob"] + -- delete group 2, unused host contact and profile are deleted + bobLeaveDeleteGroup alice bob "club" + bob ##> "/contacts" + (bob TestCC -> TestCC -> String -> IO () + bobLeaveDeleteGroup alice bob group = do + bob ##> ("/l " <> group) + concurrentlyN_ + [ do + bob <## ("#" <> group <> ": you left the group") + bob <## ("use /d #" <> group <> " to delete the group"), + alice <## ("#" <> group <> ": bob left the group") + ] + bob ##> ("/d #" <> group) + bob <## ("#" <> group <> ": you deleted the group") + +testGroupLinkIncognitoUnusedHostContactsDeleted :: HasCallStack => FilePath -> IO () +testGroupLinkIncognitoUnusedHostContactsDeleted = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + bob #$> ("/incognito on", id, "ok") + bobIncognitoTeam <- createGroupBobIncognito alice bob "team" "alice" + bobIncognitoClub <- createGroupBobIncognito alice bob "club" "alice_1" + bobIncognitoTeam `shouldNotBe` bobIncognitoClub + -- list contacts + bob ##> "/contacts" + bob <## "i alice (Alice)" + bob <## "i alice_1 (Alice)" + bob `hasContactProfiles` ["alice", "alice", "bob", T.pack bobIncognitoTeam, T.pack bobIncognitoClub] + -- delete group 1, unused host contact and profile are deleted + bobLeaveDeleteGroup alice bob "team" bobIncognitoTeam + bob ##> "/contacts" + bob <## "i alice_1 (Alice)" + bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognitoClub] + -- delete group 2, unused host contact and profile are deleted + bobLeaveDeleteGroup alice bob "club" bobIncognitoClub + bob ##> "/contacts" + (bob TestCC -> TestCC -> String -> String -> IO String + createGroupBobIncognito alice bob group bobsAliceContact = do + alice ##> ("/g " <> group) + alice <## ("group #" <> group <> " is created") + alice <## ("to add members use /a " <> group <> " or /create link #" <> group) + alice ##> ("/create link #" <> group) + gLinkTeam <- getGroupLink alice group True + bob ##> ("/c " <> gLinkTeam) + bobIncognito <- getTermLine bob + bob <## "connection request sent incognito!" + alice <## (bobIncognito <> ": accepting request to join group #" <> group <> "...") + _ <- getTermLine bob + concurrentlyN_ + [ do + alice <## (bobIncognito <> ": contact is connected") + alice <## (bobIncognito <> " invited to group #" <> group <> " via your group link") + alice <## ("#" <> group <> ": " <> bobIncognito <> " joined the group"), + do + bob <## (bobsAliceContact <> " (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) + bob <## ("use /i " <> bobsAliceContact <> " to print out this incognito profile again") + bob <## ("#" <> group <> ": you joined the group incognito as " <> bobIncognito) + ] + pure bobIncognito + bobLeaveDeleteGroup :: HasCallStack => TestCC -> TestCC -> String -> String -> IO () + bobLeaveDeleteGroup alice bob group bobIncognito = do + bob ##> ("/l " <> group) + concurrentlyN_ + [ do + bob <## ("#" <> group <> ": you left the group") + bob <## ("use /d #" <> group <> " to delete the group"), + alice <## ("#" <> group <> ": " <> bobIncognito <> " left the group") + ] + bob ##> ("/d #" <> group) + bob <## ("#" <> group <> ": you deleted the group") diff --git a/tests/ChatTests/Profiles.hs b/tests/ChatTests/Profiles.hs new file mode 100644 index 000000000..35f1ff81d --- /dev/null +++ b/tests/ChatTests/Profiles.hs @@ -0,0 +1,1328 @@ +{-# LANGUAGE OverloadedStrings #-} +{-# LANGUAGE PostfixOperators #-} + +module ChatTests.Profiles where + +import ChatClient +import ChatTests.Utils +import Control.Concurrent (threadDelay) +import Control.Concurrent.Async (concurrently_) +import qualified Data.Text as T +import Simplex.Chat.Types (ConnStatus (..), GroupMemberRole (..)) +import System.Directory (copyFile, createDirectoryIfMissing) +import Test.Hspec + +chatProfileTests :: SpecWith FilePath +chatProfileTests = do + describe "user profiles" $ do + it "update user profile and notify contacts" testUpdateProfile + it "update user profile with image" testUpdateProfileImage + describe "user contact link" $ do + describe "create and connect via contact link" testUserContactLink + it "auto accept contact requests" testUserContactLinkAutoAccept + it "deduplicate contact requests" testDeduplicateContactRequests + it "deduplicate contact requests with profile change" testDeduplicateContactRequestsProfileChange + it "reject contact and delete contact link" testRejectContactAndDeleteUserContact + it "delete connection requests when contact link deleted" testDeleteConnectionRequests + it "auto-reply message" testAutoReplyMessage + it "auto-reply message in incognito" testAutoReplyMessageInIncognito + describe "incognito mode" $ do + it "connect incognito via invitation link" testConnectIncognitoInvitationLink + it "connect incognito via contact address" testConnectIncognitoContactAddress + it "accept contact request incognito" testAcceptContactRequestIncognito + it "join group incognito" testJoinGroupIncognito + it "can't invite contact to whom user connected incognito to a group" testCantInviteContactIncognito + it "can't see global preferences update" testCantSeeGlobalPrefsUpdateIncognito + it "deleting contact first, group second deletes incognito profile" testDeleteContactThenGroupDeletesIncognitoProfile + it "deleting group first, contact second deletes incognito profile" testDeleteGroupThenContactDeletesIncognitoProfile + describe "contact aliases" $ do + it "set contact alias" testSetAlias + it "set connection alias" testSetConnectionAlias + describe "preferences" $ do + it "set contact preferences" testSetContactPrefs + it "feature offers" testFeatureOffers + it "update group preferences" testUpdateGroupPrefs + it "allow full deletion to contact" testAllowFullDeletionContact + it "allow full deletion to group" testAllowFullDeletionGroup + it "prohibit direct messages to group members" testProhibitDirectMessages + it "enable timed messages with contact" testEnableTimedMessagesContact + it "enable timed messages in group" testEnableTimedMessagesGroup + it "timed messages enabled globally, contact turns on" testTimedMessagesEnabledGlobally + +testUpdateProfile :: HasCallStack => FilePath -> IO () +testUpdateProfile = + testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + createGroup3 "team" alice bob cath + alice ##> "/p" + alice <## "user profile: alice (Alice)" + alice <## "use /p [] to change it" + alice <## "(the updated profile will be sent to all your contacts)" + alice ##> "/p alice" + concurrentlyN_ + [ alice <## "user full name removed (your contacts are notified)", + bob <## "contact alice removed full name", + cath <## "contact alice removed full name" + ] + alice ##> "/p alice Alice Jones" + concurrentlyN_ + [ alice <## "user full name changed to Alice Jones (your contacts are notified)", + bob <## "contact alice updated full name: Alice Jones", + cath <## "contact alice updated full name: Alice Jones" + ] + cath ##> "/p cate" + concurrentlyN_ + [ cath <## "user profile is changed to cate (your contacts are notified)", + do + alice <## "contact cath changed to cate" + alice <## "use @cate to send messages", + do + bob <## "contact cath changed to cate" + bob <## "use @cate to send messages" + ] + cath ##> "/p cat Cate" + concurrentlyN_ + [ cath <## "user profile is changed to cat (Cate) (your contacts are notified)", + do + alice <## "contact cate changed to cat (Cate)" + alice <## "use @cat to send messages", + do + bob <## "contact cate changed to cat (Cate)" + bob <## "use @cat to send messages" + ] + +testUpdateProfileImage :: HasCallStack => FilePath -> IO () +testUpdateProfileImage = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice ##> "/profile_image " + alice <## "profile image updated" + alice ##> "/profile_image" + alice <## "profile image removed" + alice ##> "/_profile 1 {\"displayName\": \"alice2\", \"fullName\": \"\"}" + alice <## "user profile is changed to alice2 (your contacts are notified)" + bob <## "contact alice changed to alice2" + bob <## "use @alice2 to send messages" + (bob do + alice ##> "/ad" + cLink <- getContactLink alice True + bob ##> ("/c " <> cLink) + alice <#? bob + alice @@@ [("<@bob", "")] + alice ##> "/ac bob" + alice <## "bob (Bob): accepting contact request..." + concurrently_ + (bob <## "alice (Alice): contact is connected") + (alice <## "bob (Bob): contact is connected") + threadDelay 100000 + alice @@@ [("@bob", "Voice messages: enabled")] + alice <##> bob + + cath ##> ("/c " <> cLink) + alice <#? cath + alice @@@ [("<@cath", ""), ("@bob", "hey")] + alice ##> "/ac cath" + alice <## "cath (Catherine): accepting contact request..." + concurrently_ + (cath <## "alice (Alice): contact is connected") + (alice <## "cath (Catherine): contact is connected") + threadDelay 100000 + alice @@@ [("@cath", "Voice messages: enabled"), ("@bob", "hey")] + alice <##> cath + +testUserContactLinkAutoAccept :: HasCallStack => FilePath -> IO () +testUserContactLinkAutoAccept = + testChat4 aliceProfile bobProfile cathProfile danProfile $ + \alice bob cath dan -> do + alice ##> "/ad" + cLink <- getContactLink alice True + + bob ##> ("/c " <> cLink) + alice <#? bob + alice @@@ [("<@bob", "")] + alice ##> "/ac bob" + alice <## "bob (Bob): accepting contact request..." + concurrently_ + (bob <## "alice (Alice): contact is connected") + (alice <## "bob (Bob): contact is connected") + threadDelay 100000 + alice @@@ [("@bob", "Voice messages: enabled")] + alice <##> bob + + alice ##> "/auto_accept on" + alice <## "auto_accept on" + + cath ##> ("/c " <> cLink) + cath <## "connection request sent!" + alice <## "cath (Catherine): accepting contact request..." + concurrently_ + (cath <## "alice (Alice): contact is connected") + (alice <## "cath (Catherine): contact is connected") + threadDelay 100000 + alice @@@ [("@cath", "Voice messages: enabled"), ("@bob", "hey")] + alice <##> cath + + alice ##> "/auto_accept off" + alice <## "auto_accept off" + + dan ##> ("/c " <> cLink) + alice <#? dan + alice @@@ [("<@dan", ""), ("@cath", "hey"), ("@bob", "hey")] + alice ##> "/ac dan" + alice <## "dan (Daniel): accepting contact request..." + concurrently_ + (dan <## "alice (Alice): contact is connected") + (alice <## "dan (Daniel): contact is connected") + threadDelay 100000 + alice @@@ [("@dan", "Voice messages: enabled"), ("@cath", "hey"), ("@bob", "hey")] + alice <##> dan + +testDeduplicateContactRequests :: HasCallStack => FilePath -> IO () +testDeduplicateContactRequests = testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + alice ##> "/ad" + cLink <- getContactLink alice True + + bob ##> ("/c " <> cLink) + alice <#? bob + alice @@@ [("<@bob", "")] + bob @@@! [(":1", "", Just ConnJoined)] + + bob ##> ("/c " <> cLink) + alice <#? bob + bob ##> ("/c " <> cLink) + alice <#? bob + alice @@@ [("<@bob", "")] + bob @@@! [(":3", "", Just ConnJoined), (":2", "", Just ConnJoined), (":1", "", Just ConnJoined)] + + alice ##> "/ac bob" + alice <## "bob (Bob): accepting contact request..." + concurrently_ + (bob <## "alice (Alice): contact is connected") + (alice <## "bob (Bob): contact is connected") + + bob ##> ("/c " <> cLink) + bob <## "alice (Alice): contact already exists" + alice @@@ [("@bob", "Voice messages: enabled")] + bob @@@ [("@alice", "Voice messages: enabled"), (":2", ""), (":1", "")] + bob ##> "/_delete :1" + bob <## "connection :1 deleted" + bob ##> "/_delete :2" + bob <## "connection :2 deleted" + + alice <##> bob + alice @@@ [("@bob", "hey")] + bob @@@ [("@alice", "hey")] + + bob ##> ("/c " <> cLink) + bob <## "alice (Alice): contact already exists" + + alice <##> bob + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hi"), (0, "hey"), (1, "hi"), (0, "hey")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hi"), (1, "hey"), (0, "hi"), (1, "hey")]) + + cath ##> ("/c " <> cLink) + alice <#? cath + alice @@@ [("<@cath", ""), ("@bob", "hey")] + alice ##> "/ac cath" + alice <## "cath (Catherine): accepting contact request..." + concurrently_ + (cath <## "alice (Alice): contact is connected") + (alice <## "cath (Catherine): contact is connected") + threadDelay 100000 + alice @@@ [("@cath", "Voice messages: enabled"), ("@bob", "hey")] + alice <##> cath + +testDeduplicateContactRequestsProfileChange :: HasCallStack => FilePath -> IO () +testDeduplicateContactRequestsProfileChange = testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + alice ##> "/ad" + cLink <- getContactLink alice True + + bob ##> ("/c " <> cLink) + alice <#? bob + alice @@@ [("<@bob", "")] + + bob ##> "/p bob" + bob <## "user full name removed (your contacts are notified)" + bob ##> ("/c " <> cLink) + bob <## "connection request sent!" + alice <## "bob wants to connect to you!" + alice <## "to accept: /ac bob" + alice <## "to reject: /rc bob (the sender will NOT be notified)" + alice @@@ [("<@bob", "")] + + bob ##> "/p bob Bob Ross" + bob <## "user full name changed to Bob Ross (your contacts are notified)" + bob ##> ("/c " <> cLink) + alice <#? bob + alice @@@ [("<@bob", "")] + + bob ##> "/p robert Robert" + bob <## "user profile is changed to robert (Robert) (your contacts are notified)" + bob ##> ("/c " <> cLink) + alice <#? bob + alice @@@ [("<@robert", "")] + + alice ##> "/ac bob" + alice <## "no contact request from bob" + alice ##> "/ac robert" + alice <## "robert (Robert): accepting contact request..." + concurrently_ + (bob <## "alice (Alice): contact is connected") + (alice <## "robert (Robert): contact is connected") + + bob ##> ("/c " <> cLink) + bob <## "alice (Alice): contact already exists" + alice @@@ [("@robert", "Voice messages: enabled")] + bob @@@ [("@alice", "Voice messages: enabled"), (":3", ""), (":2", ""), (":1", "")] + bob ##> "/_delete :1" + bob <## "connection :1 deleted" + bob ##> "/_delete :2" + bob <## "connection :2 deleted" + bob ##> "/_delete :3" + bob <## "connection :3 deleted" + + alice <##> bob + alice @@@ [("@robert", "hey")] + bob @@@ [("@alice", "hey")] + + bob ##> ("/c " <> cLink) + bob <## "alice (Alice): contact already exists" + + alice <##> bob + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hi"), (0, "hey"), (1, "hi"), (0, "hey")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hi"), (1, "hey"), (0, "hi"), (1, "hey")]) + + cath ##> ("/c " <> cLink) + alice <#? cath + alice @@@ [("<@cath", ""), ("@robert", "hey")] + alice ##> "/ac cath" + alice <## "cath (Catherine): accepting contact request..." + concurrently_ + (cath <## "alice (Alice): contact is connected") + (alice <## "cath (Catherine): contact is connected") + threadDelay 100000 + alice @@@ [("@cath", "Voice messages: enabled"), ("@robert", "hey")] + alice <##> cath + +testRejectContactAndDeleteUserContact :: HasCallStack => FilePath -> IO () +testRejectContactAndDeleteUserContact = testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + alice ##> "/_address 1" + cLink <- getContactLink alice True + bob ##> ("/c " <> cLink) + alice <#? bob + alice ##> "/rc bob" + alice <## "bob: contact request rejected" + (bob "/_show_address 1" + cLink' <- getContactLink alice False + alice <## "auto_accept off" + cLink' `shouldBe` cLink + + alice ##> "/_delete_address 1" + alice <## "Your chat address is deleted - accepted contacts will remain connected." + alice <## "To create a new chat address use /ad" + + cath ##> ("/c " <> cLink) + cath <## "error: connection authorization failed - this could happen if connection was deleted, secured with different credentials, or due to a bug - please re-create the connection" + +testDeleteConnectionRequests :: HasCallStack => FilePath -> IO () +testDeleteConnectionRequests = testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + alice ##> "/ad" + cLink <- getContactLink alice True + bob ##> ("/c " <> cLink) + alice <#? bob + cath ##> ("/c " <> cLink) + alice <#? cath + + alice ##> "/da" + alice <## "Your chat address is deleted - accepted contacts will remain connected." + alice <## "To create a new chat address use /ad" + + alice ##> "/ad" + cLink' <- getContactLink alice True + bob ##> ("/c " <> cLink') + -- same names are used here, as they were released at /da + alice <#? bob + cath ##> ("/c " <> cLink') + alice <#? cath + +testAutoReplyMessage :: HasCallStack => FilePath -> IO () +testAutoReplyMessage = testChat2 aliceProfile bobProfile $ + \alice bob -> do + alice ##> "/ad" + cLink <- getContactLink alice True + alice ##> "/_auto_accept 1 on incognito=off text hello!" + alice <## "auto_accept on" + alice <## "auto reply:" + alice <## "hello!" + + bob ##> ("/c " <> cLink) + bob <## "connection request sent!" + alice <## "bob (Bob): accepting contact request..." + concurrentlyN_ + [ do + bob <## "alice (Alice): contact is connected" + bob <# "alice> hello!", + do + alice <## "bob (Bob): contact is connected" + alice <# "@bob hello!" + ] + +testAutoReplyMessageInIncognito :: HasCallStack => FilePath -> IO () +testAutoReplyMessageInIncognito = testChat2 aliceProfile bobProfile $ + \alice bob -> do + alice ##> "/ad" + cLink <- getContactLink alice True + alice ##> "/auto_accept on incognito=on text hello!" + alice <## "auto_accept on, incognito" + alice <## "auto reply:" + alice <## "hello!" + + bob ##> ("/c " <> cLink) + bob <## "connection request sent!" + alice <## "bob (Bob): accepting contact request..." + aliceIncognito <- getTermLine alice + concurrentlyN_ + [ do + bob <## (aliceIncognito <> ": contact is connected") + bob <# (aliceIncognito <> "> hello!"), + do + alice <## ("bob (Bob): contact is connected, your incognito profile for this contact is " <> aliceIncognito) + alice + <### [ "use /i bob to print out this incognito profile again", + WithTime "i @bob hello!" + ] + ] + +testConnectIncognitoInvitationLink :: HasCallStack => FilePath -> IO () +testConnectIncognitoInvitationLink = testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + alice #$> ("/incognito on", id, "ok") + bob #$> ("/incognito on", id, "ok") + alice ##> "/c" + inv <- getInvitation alice + bob ##> ("/c " <> inv) + bob <## "confirmation sent!" + bobIncognito <- getTermLine bob + aliceIncognito <- getTermLine alice + concurrentlyN_ + [ do + bob <## (aliceIncognito <> ": contact is connected, your incognito profile for this contact is " <> bobIncognito) + bob <## ("use /i " <> aliceIncognito <> " to print out this incognito profile again"), + do + alice <## (bobIncognito <> ": contact is connected, your incognito profile for this contact is " <> aliceIncognito) + alice <## ("use /i " <> bobIncognito <> " to print out this incognito profile again") + ] + -- after turning incognito mode off conversation is incognito + alice #$> ("/incognito off", id, "ok") + bob #$> ("/incognito off", id, "ok") + alice ?#> ("@" <> bobIncognito <> " psst, I'm incognito") + bob ?<# (aliceIncognito <> "> psst, I'm incognito") + bob ?#> ("@" <> aliceIncognito <> " me too") + alice ?<# (bobIncognito <> "> me too") + -- new contact is connected non incognito + connectUsers alice cath + alice <##> cath + -- bob is not notified on profile change + alice ##> "/p alice" + concurrentlyN_ + [ alice <## "user full name removed (your contacts are notified)", + cath <## "contact alice removed full name" + ] + alice ?#> ("@" <> bobIncognito <> " do you see that I've changed profile?") + bob ?<# (aliceIncognito <> "> do you see that I've changed profile?") + bob ?#> ("@" <> aliceIncognito <> " no") + alice ?<# (bobIncognito <> "> no") + alice ##> "/_set prefs @2 {}" + alice <## ("your preferences for " <> bobIncognito <> " did not change") + (bob "/_set prefs @2 {\"fullDelete\": {\"allow\": \"always\"}}" + alice <## ("you updated preferences for " <> bobIncognito <> ":") + alice <## "Full deletion: enabled for contact (you allow: always, contact allows: no)" + bob <## (aliceIncognito <> " updated preferences for you:") + bob <## "Full deletion: enabled for you (you allow: no, contact allows: always)" + bob ##> "/_set prefs @2 {}" + bob <## ("your preferences for " <> aliceIncognito <> " did not change") + (alice "/_set prefs @2 {\"fullDelete\": {\"allow\": \"no\"}}" + alice <## ("you updated preferences for " <> bobIncognito <> ":") + alice <## "Full deletion: off (you allow: no, contact allows: no)" + bob <## (aliceIncognito <> " updated preferences for you:") + bob <## "Full deletion: off (you allow: no, contact allows: no)" + -- list contacts + alice ##> "/contacts" + alice + <### [ ConsoleString $ "i " <> bobIncognito, + "cath (Catherine)" + ] + alice `hasContactProfiles` ["alice", T.pack aliceIncognito, T.pack bobIncognito, "cath"] + bob ##> "/contacts" + bob <## ("i " <> aliceIncognito) + bob `hasContactProfiles` ["bob", T.pack aliceIncognito, T.pack bobIncognito] + -- alice deletes contact, incognito profile is deleted + alice ##> ("/d " <> bobIncognito) + alice <## (bobIncognito <> ": contact is deleted") + alice ##> "/contacts" + alice <## "cath (Catherine)" + alice `hasContactProfiles` ["alice", "cath"] + -- bob deletes contact, incognito profile is deleted + bob ##> ("/d " <> aliceIncognito) + bob <## (aliceIncognito <> ": contact is deleted") + bob ##> "/contacts" + (bob FilePath -> IO () +testConnectIncognitoContactAddress = testChat2 aliceProfile bobProfile $ + \alice bob -> do + alice ##> "/ad" + cLink <- getContactLink alice True + bob #$> ("/incognito on", id, "ok") + bob ##> ("/c " <> cLink) + bobIncognito <- getTermLine bob + bob <## "connection request sent incognito!" + alice <## (bobIncognito <> " wants to connect to you!") + alice <## ("to accept: /ac " <> bobIncognito) + alice <## ("to reject: /rc " <> bobIncognito <> " (the sender will NOT be notified)") + alice ##> ("/ac " <> bobIncognito) + alice <## (bobIncognito <> ": accepting contact request...") + _ <- getTermLine bob + concurrentlyN_ + [ do + bob <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) + bob <## "use /i alice to print out this incognito profile again", + alice <## (bobIncognito <> ": contact is connected") + ] + -- after turning incognito mode off conversation is incognito + alice #$> ("/incognito off", id, "ok") + bob #$> ("/incognito off", id, "ok") + alice #> ("@" <> bobIncognito <> " who are you?") + bob ?<# "alice> who are you?" + bob ?#> "@alice I'm Batman" + alice <# (bobIncognito <> "> I'm Batman") + -- list contacts + bob ##> "/contacts" + bob <## "i alice (Alice)" + bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognito] + -- delete contact, incognito profile is deleted + bob ##> "/d alice" + bob <## "alice: contact is deleted" + bob ##> "/contacts" + (bob FilePath -> IO () +testAcceptContactRequestIncognito = testChat2 aliceProfile bobProfile $ + \alice bob -> do + alice ##> "/ad" + cLink <- getContactLink alice True + bob ##> ("/c " <> cLink) + alice <#? bob + alice #$> ("/incognito on", id, "ok") + alice ##> "/ac bob" + alice <## "bob (Bob): accepting contact request..." + aliceIncognito <- getTermLine alice + concurrentlyN_ + [ bob <## (aliceIncognito <> ": contact is connected"), + do + alice <## ("bob (Bob): contact is connected, your incognito profile for this contact is " <> aliceIncognito) + alice <## "use /i bob to print out this incognito profile again" + ] + -- after turning incognito mode off conversation is incognito + alice #$> ("/incognito off", id, "ok") + bob #$> ("/incognito off", id, "ok") + alice ?#> "@bob my profile is totally inconspicuous" + bob <# (aliceIncognito <> "> my profile is totally inconspicuous") + bob #> ("@" <> aliceIncognito <> " I know!") + alice ?<# "bob> I know!" + -- list contacts + alice ##> "/contacts" + alice <## "i bob (Bob)" + alice `hasContactProfiles` ["alice", "bob", T.pack aliceIncognito] + -- delete contact, incognito profile is deleted + alice ##> "/d bob" + alice <## "bob: contact is deleted" + alice ##> "/contacts" + (alice FilePath -> IO () +testJoinGroupIncognito = testChat4 aliceProfile bobProfile cathProfile danProfile $ + \alice bob cath dan -> do + -- non incognito connections + connectUsers alice bob + connectUsers alice dan + connectUsers bob cath + connectUsers bob dan + connectUsers cath dan + -- cath connected incognito to alice + alice ##> "/c" + inv <- getInvitation alice + cath #$> ("/incognito on", id, "ok") + cath ##> ("/c " <> inv) + cath <## "confirmation sent!" + cathIncognito <- getTermLine cath + concurrentlyN_ + [ do + cath <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> cathIncognito) + cath <## "use /i alice to print out this incognito profile again", + alice <## (cathIncognito <> ": contact is connected") + ] + -- alice creates group + alice ##> "/g secret_club" + alice <## "group #secret_club is created" + alice <## "to add members use /a secret_club or /create link #secret_club" + -- alice invites bob + alice ##> "/a secret_club bob" + concurrentlyN_ + [ alice <## "invitation to join the group #secret_club sent to bob", + do + bob <## "#secret_club: alice invites you to join the group as admin" + bob <## "use /j secret_club to accept" + ] + bob ##> "/j secret_club" + concurrently_ + (alice <## "#secret_club: bob joined the group") + (bob <## "#secret_club: you joined the group") + -- alice invites cath + alice ##> ("/a secret_club " <> cathIncognito) + concurrentlyN_ + [ alice <## ("invitation to join the group #secret_club sent to " <> cathIncognito), + do + cath <## "#secret_club: alice invites you to join the group as admin" + cath <## ("use /j secret_club to join incognito as " <> cathIncognito) + ] + -- cath uses the same incognito profile when joining group, disabling incognito mode doesn't affect it + cath #$> ("/incognito off", id, "ok") + cath ##> "/j secret_club" + -- cath and bob don't merge contacts + concurrentlyN_ + [ alice <## ("#secret_club: " <> cathIncognito <> " joined the group"), + do + cath <## ("#secret_club: you joined the group incognito as " <> cathIncognito) + cath <## "#secret_club: member bob_1 (Bob) is connected", + do + bob <## ("#secret_club: alice added " <> cathIncognito <> " to the group (connecting...)") + bob <## ("#secret_club: new member " <> cathIncognito <> " is connected") + ] + -- cath cannot invite to the group because her membership is incognito + cath ##> "/a secret_club dan" + cath <## "you've connected to this group using an incognito profile - prohibited to invite contacts" + -- alice invites dan + alice ##> "/a secret_club dan" + concurrentlyN_ + [ alice <## "invitation to join the group #secret_club sent to dan", + do + dan <## "#secret_club: alice invites you to join the group as admin" + dan <## "use /j secret_club to accept" + ] + dan ##> "/j secret_club" + -- cath and dan don't merge contacts + concurrentlyN_ + [ alice <## "#secret_club: dan joined the group", + do + dan <## "#secret_club: you joined the group" + dan + <### [ ConsoleString $ "#secret_club: member " <> cathIncognito <> " is connected", + "#secret_club: member bob_1 (Bob) is connected", + "contact bob_1 is merged into bob", + "use @bob to send messages" + ], + do + bob <## "#secret_club: alice added dan_1 (Daniel) to the group (connecting...)" + bob <## "#secret_club: new member dan_1 is connected" + bob <## "contact dan_1 is merged into dan" + bob <## "use @dan to send messages", + do + cath <## "#secret_club: alice added dan_1 (Daniel) to the group (connecting...)" + cath <## "#secret_club: new member dan_1 is connected" + ] + -- send messages - group is incognito for cath + alice #> "#secret_club hello" + concurrentlyN_ + [ bob <# "#secret_club alice> hello", + cath ?<# "#secret_club alice> hello", + dan <# "#secret_club alice> hello" + ] + bob #> "#secret_club hi there" + concurrentlyN_ + [ alice <# "#secret_club bob> hi there", + cath ?<# "#secret_club bob_1> hi there", + dan <# "#secret_club bob> hi there" + ] + cath ?#> "#secret_club hey" + concurrentlyN_ + [ alice <# ("#secret_club " <> cathIncognito <> "> hey"), + bob <# ("#secret_club " <> cathIncognito <> "> hey"), + dan <# ("#secret_club " <> cathIncognito <> "> hey") + ] + dan #> "#secret_club how is it going?" + concurrentlyN_ + [ alice <# "#secret_club dan> how is it going?", + bob <# "#secret_club dan> how is it going?", + cath ?<# "#secret_club dan_1> how is it going?" + ] + -- cath and bob can send messages via new direct connection, cath is incognito + bob #> ("@" <> cathIncognito <> " hi, I'm bob") + cath ?<# "bob_1> hi, I'm bob" + cath ?#> "@bob_1 hey, I'm incognito" + bob <# (cathIncognito <> "> hey, I'm incognito") + -- cath and dan can send messages via new direct connection, cath is incognito + dan #> ("@" <> cathIncognito <> " hi, I'm dan") + cath ?<# "dan_1> hi, I'm dan" + cath ?#> "@dan_1 hey, I'm incognito" + dan <# (cathIncognito <> "> hey, I'm incognito") + -- non incognito connections are separate + bob <##> cath + dan <##> cath + -- list groups + cath ##> "/gs" + cath <## "i #secret_club" + -- list group members + alice ##> "/ms secret_club" + alice + <### [ "alice (Alice): owner, you, created group", + "bob (Bob): admin, invited, connected", + ConsoleString $ cathIncognito <> ": admin, invited, connected", + "dan (Daniel): admin, invited, connected" + ] + bob ##> "/ms secret_club" + bob + <### [ "alice (Alice): owner, host, connected", + "bob (Bob): admin, you, connected", + ConsoleString $ cathIncognito <> ": admin, connected", + "dan (Daniel): admin, connected" + ] + cath ##> "/ms secret_club" + cath + <### [ "alice (Alice): owner, host, connected", + "bob_1 (Bob): admin, connected", + ConsoleString $ "i " <> cathIncognito <> ": admin, you, connected", + "dan_1 (Daniel): admin, connected" + ] + dan ##> "/ms secret_club" + dan + <### [ "alice (Alice): owner, host, connected", + "bob (Bob): admin, connected", + ConsoleString $ cathIncognito <> ": admin, connected", + "dan (Daniel): admin, you, connected" + ] + -- remove member + bob ##> ("/rm secret_club " <> cathIncognito) + concurrentlyN_ + [ bob <## ("#secret_club: you removed " <> cathIncognito <> " from the group"), + alice <## ("#secret_club: bob removed " <> cathIncognito <> " from the group"), + dan <## ("#secret_club: bob removed " <> cathIncognito <> " from the group"), + do + cath <## "#secret_club: bob_1 removed you from the group" + cath <## "use /d #secret_club to delete the group" + ] + bob #> "#secret_club hi" + concurrentlyN_ + [ alice <# "#secret_club bob> hi", + dan <# "#secret_club bob> hi", + (cath "#secret_club hello" + concurrentlyN_ + [ bob <# "#secret_club alice> hello", + dan <# "#secret_club alice> hello", + (cath "#secret_club hello" + cath <## "you are no longer a member of the group" + -- cath can still message members directly + bob #> ("@" <> cathIncognito <> " I removed you from group") + cath ?<# "bob_1> I removed you from group" + cath ?#> "@bob_1 ok" + bob <# (cathIncognito <> "> ok") + +testCantInviteContactIncognito :: HasCallStack => FilePath -> IO () +testCantInviteContactIncognito = testChat2 aliceProfile bobProfile $ + \alice bob -> do + -- alice connected incognito to bob + alice #$> ("/incognito on", id, "ok") + alice ##> "/c" + inv <- getInvitation alice + bob ##> ("/c " <> inv) + bob <## "confirmation sent!" + aliceIncognito <- getTermLine alice + concurrentlyN_ + [ bob <## (aliceIncognito <> ": contact is connected"), + do + alice <## ("bob (Bob): contact is connected, your incognito profile for this contact is " <> aliceIncognito) + alice <## "use /i bob to print out this incognito profile again" + ] + -- alice creates group non incognito + alice #$> ("/incognito off", id, "ok") + alice ##> "/g club" + alice <## "group #club is created" + alice <## "to add members use /a club or /create link #club" + alice ##> "/a club bob" + alice <## "you're using your main profile for this group - prohibited to invite contacts to whom you are connected incognito" + -- bob doesn't receive invitation + (bob FilePath -> IO () +testCantSeeGlobalPrefsUpdateIncognito = testChat3 aliceProfile bobProfile cathProfile $ + \alice bob cath -> do + alice #$> ("/incognito on", id, "ok") + alice ##> "/c" + invIncognito <- getInvitation alice + alice #$> ("/incognito off", id, "ok") + alice ##> "/c" + inv <- getInvitation alice + bob ##> ("/c " <> invIncognito) + bob <## "confirmation sent!" + aliceIncognito <- getTermLine alice + cath ##> ("/c " <> inv) + cath <## "confirmation sent!" + concurrentlyN_ + [ bob <## (aliceIncognito <> ": contact is connected"), + do + alice <## ("bob (Bob): contact is connected, your incognito profile for this contact is " <> aliceIncognito) + alice <## "use /i bob to print out this incognito profile again", + do + cath <## "alice (Alice): contact is connected" + ] + alice <## "cath (Catherine): contact is connected" + alice ##> "/_profile 1 {\"displayName\": \"alice\", \"fullName\": \"\", \"preferences\": {\"fullDelete\": {\"allow\": \"always\"}}}" + alice <## "user full name removed (your contacts are notified)" + alice <## "updated preferences:" + alice <## "Full deletion allowed: always" + (alice "/_set prefs @2 {\"fullDelete\": {\"allow\": \"always\"}}" + bob <## ("you updated preferences for " <> aliceIncognito <> ":") + bob <## "Full deletion: enabled for contact (you allow: always, contact allows: no)" + alice <## "bob updated preferences for you:" + alice <## "Full deletion: enabled for you (you allow: no, contact allows: always)" + alice ##> "/_set prefs @2 {\"fullDelete\": {\"allow\": \"yes\"}}" + alice <## "you updated preferences for bob:" + alice <## "Full deletion: enabled (you allow: yes, contact allows: always)" + bob <## (aliceIncognito <> " updated preferences for you:") + bob <## "Full deletion: enabled (you allow: always, contact allows: yes)" + (cath "/_set prefs @3 {\"fullDelete\": {\"allow\": \"always\"}}" + alice <## "your preferences for cath did not change" + alice ##> "/_set prefs @3 {\"fullDelete\": {\"allow\": \"yes\"}}" + alice <## "you updated preferences for cath:" + alice <## "Full deletion: off (you allow: yes, contact allows: no)" + cath <## "alice updated preferences for you:" + cath <## "Full deletion: off (you allow: default (no), contact allows: yes)" + +testDeleteContactThenGroupDeletesIncognitoProfile :: HasCallStack => FilePath -> IO () +testDeleteContactThenGroupDeletesIncognitoProfile = testChat2 aliceProfile bobProfile $ + \alice bob -> do + -- bob connects incognito to alice + alice ##> "/c" + inv <- getInvitation alice + bob #$> ("/incognito on", id, "ok") + bob ##> ("/c " <> inv) + bob <## "confirmation sent!" + bobIncognito <- getTermLine bob + concurrentlyN_ + [ alice <## (bobIncognito <> ": contact is connected"), + do + bob <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) + bob <## "use /i alice to print out this incognito profile again" + ] + -- bob joins group using incognito profile + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> ("/a team " <> bobIncognito) + concurrentlyN_ + [ alice <## ("invitation to join the group #team sent to " <> bobIncognito), + do + bob <## "#team: alice invites you to join the group as admin" + bob <## ("use /j team to join incognito as " <> bobIncognito) + ] + bob ##> "/j team" + concurrently_ + (alice <## ("#team: " <> bobIncognito <> " joined the group")) + (bob <## ("#team: you joined the group incognito as " <> bobIncognito)) + bob ##> "/contacts" + bob <## "i alice (Alice)" + bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognito] + -- delete contact + bob ##> "/d alice" + bob <## "alice: contact is deleted" + bob ##> "/contacts" + (bob "/l team" + concurrentlyN_ + [ do + bob <## "#team: you left the group" + bob <## "use /d #team to delete the group", + alice <## ("#team: " <> bobIncognito <> " left the group") + ] + bob ##> "/d #team" + bob <## "#team: you deleted the group" + bob `hasContactProfiles` ["bob"] + +testDeleteGroupThenContactDeletesIncognitoProfile :: HasCallStack => FilePath -> IO () +testDeleteGroupThenContactDeletesIncognitoProfile = testChat2 aliceProfile bobProfile $ + \alice bob -> do + -- bob connects incognito to alice + alice ##> "/c" + inv <- getInvitation alice + bob #$> ("/incognito on", id, "ok") + bob ##> ("/c " <> inv) + bob <## "confirmation sent!" + bobIncognito <- getTermLine bob + concurrentlyN_ + [ alice <## (bobIncognito <> ": contact is connected"), + do + bob <## ("alice (Alice): contact is connected, your incognito profile for this contact is " <> bobIncognito) + bob <## "use /i alice to print out this incognito profile again" + ] + -- bob joins group using incognito profile + alice ##> "/g team" + alice <## "group #team is created" + alice <## "to add members use /a team or /create link #team" + alice ##> ("/a team " <> bobIncognito) + concurrentlyN_ + [ alice <## ("invitation to join the group #team sent to " <> bobIncognito), + do + bob <## "#team: alice invites you to join the group as admin" + bob <## ("use /j team to join incognito as " <> bobIncognito) + ] + bob ##> "/j team" + concurrently_ + (alice <## ("#team: " <> bobIncognito <> " joined the group")) + (bob <## ("#team: you joined the group incognito as " <> bobIncognito)) + bob ##> "/contacts" + bob <## "i alice (Alice)" + bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognito] + -- delete group + bob ##> "/l team" + concurrentlyN_ + [ do + bob <## "#team: you left the group" + bob <## "use /d #team to delete the group", + alice <## ("#team: " <> bobIncognito <> " left the group") + ] + bob ##> "/d #team" + bob <## "#team: you deleted the group" + bob `hasContactProfiles` ["alice", "bob", T.pack bobIncognito] + -- delete contact + bob ##> "/d alice" + bob <## "alice: contact is deleted" + bob ##> "/contacts" + (bob FilePath -> IO () +testSetAlias = testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice #$> ("/_set alias @2 my friend bob", id, "contact bob alias updated: my friend bob") + alice ##> "/contacts" + alice <## "bob (Bob) (alias: my friend bob)" + alice #$> ("/_set alias @2", id, "contact bob alias removed") + alice ##> "/contacts" + alice <## "bob (Bob)" + +testSetConnectionAlias :: HasCallStack => FilePath -> IO () +testSetConnectionAlias = testChat2 aliceProfile bobProfile $ + \alice bob -> do + alice ##> "/c" + inv <- getInvitation alice + alice @@@ [(":1", "")] + alice ##> "/_set alias :1 friend" + alice <## "connection 1 alias updated: friend" + bob ##> ("/c " <> inv) + bob <## "confirmation sent!" + concurrently_ + (alice <## "bob (Bob): contact is connected") + (bob <## "alice (Alice): contact is connected") + threadDelay 100000 + alice @@@ [("@bob", "Voice messages: enabled")] + alice ##> "/contacts" + alice <## "bob (Bob) (alias: friend)" + +testSetContactPrefs :: HasCallStack => FilePath -> IO () +testSetContactPrefs = testChat2 aliceProfile bobProfile $ + \alice bob -> do + alice #$> ("/_files_folder ./tests/tmp/alice", id, "ok") + bob #$> ("/_files_folder ./tests/tmp/bob", id, "ok") + createDirectoryIfMissing True "./tests/tmp/alice" + createDirectoryIfMissing True "./tests/tmp/bob" + copyFile "./tests/fixtures/test.txt" "./tests/tmp/alice/test.txt" + copyFile "./tests/fixtures/test.txt" "./tests/tmp/bob/test.txt" + bob ##> "/_profile 1 {\"displayName\": \"bob\", \"fullName\": \"Bob\", \"preferences\": {\"voice\": {\"allow\": \"no\"}}}" + bob <## "profile image removed" + bob <## "updated preferences:" + bob <## "Voice messages allowed: no" + (bob "/_set prefs @2 {}" + alice <## "your preferences for bob did not change" + (bob ("/_get chat @2 count=100", chat, startFeatures) + bob #$> ("/_get chat @2 count=100", chat, startFeatures) + let sendVoice = "/_send @2 json {\"filePath\": \"test.txt\", \"msgContent\": {\"type\": \"voice\", \"text\": \"\", \"duration\": 10}}" + voiceNotAllowed = "bad chat command: feature not allowed Voice messages" + alice ##> sendVoice + alice <## voiceNotAllowed + bob ##> sendVoice + bob <## voiceNotAllowed + -- alice ##> "/_set prefs @2 {\"voice\": {\"allow\": \"always\"}}" + alice ##> "/set voice @bob always" + alice <## "you updated preferences for bob:" + alice <## "Voice messages: enabled for contact (you allow: always, contact allows: no)" + alice #$> ("/_get chat @2 count=100", chat, startFeatures <> [(1, "Voice messages: enabled for contact")]) + bob <## "alice updated preferences for you:" + bob <## "Voice messages: enabled for you (you allow: default (no), contact allows: always)" + bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you")]) + alice ##> sendVoice + alice <## voiceNotAllowed + bob ##> sendVoice + bob <# "@alice voice message (00:10)" + bob <# "/f @alice test.txt" + bob <## "completed sending file 1 (test.txt) to alice" + alice <# "bob> voice message (00:10)" + alice <# "bob> sends file test.txt (11 bytes / 11 bytes)" + alice <## "started receiving file 1 (test.txt) from bob" + alice <## "completed receiving file 1 (test.txt) from bob" + (bob "/_profile 1 {\"displayName\": \"alice\", \"fullName\": \"Alice\", \"preferences\": {\"voice\": {\"allow\": \"no\"}}}" + alice ##> "/set voice no" + alice <## "updated preferences:" + alice <## "Voice messages allowed: no" + (alice "/_set prefs @2 {\"voice\": {\"allow\": \"yes\"}}" + alice <## "you updated preferences for bob:" + alice <## "Voice messages: off (you allow: yes, contact allows: no)" + alice #$> ("/_get chat @2 count=100", chat, startFeatures <> [(1, "Voice messages: enabled for contact"), (0, "voice message (00:10)"), (1, "Voice messages: off")]) + bob <## "alice updated preferences for you:" + bob <## "Voice messages: off (you allow: default (no), contact allows: yes)" + bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you"), (1, "voice message (00:10)"), (0, "Voice messages: off")]) + (bob "/_profile 1 {\"displayName\": \"bob\", \"fullName\": \"\", \"preferences\": {\"voice\": {\"allow\": \"yes\"}}}" + bob <## "user full name removed (your contacts are notified)" + bob <## "updated preferences:" + bob <## "Voice messages allowed: yes" + bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you"), (1, "voice message (00:10)"), (0, "Voice messages: off"), (1, "Voice messages: enabled")]) + (bob ("/_get chat @2 count=100", chat, startFeatures <> [(1, "Voice messages: enabled for contact"), (0, "voice message (00:10)"), (1, "Voice messages: off"), (0, "Voice messages: enabled")]) + (alice "/_set prefs @2 {}" + bob <## "your preferences for alice did not change" + -- no change + bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you"), (1, "voice message (00:10)"), (0, "Voice messages: off"), (1, "Voice messages: enabled")]) + (bob "/_set prefs @2 {\"voice\": {\"allow\": \"no\"}}" + alice <## "you updated preferences for bob:" + alice <## "Voice messages: off (you allow: no, contact allows: yes)" + alice #$> ("/_get chat @2 count=100", chat, startFeatures <> [(1, "Voice messages: enabled for contact"), (0, "voice message (00:10)"), (1, "Voice messages: off"), (0, "Voice messages: enabled"), (1, "Voice messages: off")]) + bob <## "alice updated preferences for you:" + bob <## "Voice messages: off (you allow: default (yes), contact allows: no)" + bob #$> ("/_get chat @2 count=100", chat, startFeatures <> [(0, "Voice messages: enabled for you"), (1, "voice message (00:10)"), (0, "Voice messages: off"), (1, "Voice messages: enabled"), (0, "Voice messages: off")]) + +testFeatureOffers :: HasCallStack => FilePath -> IO () +testFeatureOffers = testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice ##> "/set delete @bob yes" + alice <## "you updated preferences for bob:" + alice <## "Full deletion: off (you allow: yes, contact allows: no)" + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Full deletion")]) + bob <## "alice updated preferences for you:" + bob <## "Full deletion: off (you allow: default (no), contact allows: yes)" + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Full deletion")]) + alice ##> "/set delete @bob no" + alice <## "you updated preferences for bob:" + alice <## "Full deletion: off (you allow: no, contact allows: no)" + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Full deletion"), (1, "you cancelled Full deletion")]) + bob <## "alice updated preferences for you:" + bob <## "Full deletion: off (you allow: default (no), contact allows: no)" + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Full deletion"), (0, "cancelled Full deletion")]) + +testUpdateGroupPrefs :: HasCallStack => FilePath -> IO () +testUpdateGroupPrefs = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + createGroup2 "team" alice bob + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected")]) + threadDelay 500000 + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected")]) + alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"fullDelete\": {\"enable\": \"on\"}, \"directMessages\": {\"enable\": \"on\"}}}" + alice <## "updated group preferences:" + alice <## "Full deletion: on" + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on")]) + bob <## "alice updated group #team:" + bob <## "updated group preferences:" + bob <## "Full deletion: on" + threadDelay 500000 + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Full deletion: on")]) + alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"fullDelete\": {\"enable\": \"off\"}, \"voice\": {\"enable\": \"off\"}, \"directMessages\": {\"enable\": \"on\"}}}" + alice <## "updated group preferences:" + alice <## "Full deletion: off" + alice <## "Voice messages: off" + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on"), (1, "Full deletion: off"), (1, "Voice messages: off")]) + bob <## "alice updated group #team:" + bob <## "updated group preferences:" + bob <## "Full deletion: off" + bob <## "Voice messages: off" + threadDelay 500000 + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Full deletion: on"), (0, "Full deletion: off"), (0, "Voice messages: off")]) + -- alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"fullDelete\": {\"enable\": \"off\"}, \"voice\": {\"enable\": \"on\"}}}" + alice ##> "/set voice #team on" + alice <## "updated group preferences:" + alice <## "Voice messages: on" + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on"), (1, "Full deletion: off"), (1, "Voice messages: off"), (1, "Voice messages: on")]) + bob <## "alice updated group #team:" + bob <## "updated group preferences:" + bob <## "Voice messages: on" + threadDelay 500000 + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Full deletion: on"), (0, "Full deletion: off"), (0, "Voice messages: off"), (0, "Voice messages: on")]) + threadDelay 500000 + alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"fullDelete\": {\"enable\": \"off\"}, \"voice\": {\"enable\": \"on\"}, \"directMessages\": {\"enable\": \"on\"}}}" + -- no update + threadDelay 500000 + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on"), (1, "Full deletion: off"), (1, "Voice messages: off"), (1, "Voice messages: on")]) + alice #> "#team hey" + bob <# "#team alice> hey" + threadDelay 1000000 + bob #> "#team hi" + alice <# "#team bob> hi" + threadDelay 500000 + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Full deletion: on"), (1, "Full deletion: off"), (1, "Voice messages: off"), (1, "Voice messages: on"), (1, "hey"), (0, "hi")]) + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Full deletion: on"), (0, "Full deletion: off"), (0, "Voice messages: off"), (0, "Voice messages: on"), (0, "hey"), (1, "hi")]) + +testAllowFullDeletionContact :: HasCallStack => FilePath -> IO () +testAllowFullDeletionContact = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice <##> bob + alice ##> "/set delete @bob always" + alice <## "you updated preferences for bob:" + alice <## "Full deletion: enabled for contact (you allow: always, contact allows: no)" + bob <## "alice updated preferences for you:" + bob <## "Full deletion: enabled for you (you allow: default (no), contact allows: always)" + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hi"), (0, "hey"), (1, "Full deletion: enabled for contact")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hi"), (1, "hey"), (0, "Full deletion: enabled for you")]) + bob #$> ("/_delete item @2 " <> itemId 2 <> " broadcast", id, "message deleted") + alice <# "bob> [deleted] hey" + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "hi"), (1, "Full deletion: enabled for contact")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "hi"), (0, "Full deletion: enabled for you")]) + +testAllowFullDeletionGroup :: HasCallStack => FilePath -> IO () +testAllowFullDeletionGroup = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + createGroup2 "team" alice bob + threadDelay 1000000 + alice #> "#team hi" + bob <# "#team alice> hi" + threadDelay 1000000 + bob #> "#team hey" + bob ##> "/last_item_id #team" + msgItemId <- getTermLine bob + alice <# "#team bob> hey" + alice ##> "/set delete #team on" + alice <## "updated group preferences:" + alice <## "Full deletion: on" + bob <## "alice updated group #team:" + bob <## "updated group preferences:" + bob <## "Full deletion: on" + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "hi"), (0, "hey"), (1, "Full deletion: on")]) + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "hi"), (1, "hey"), (0, "Full deletion: on")]) + bob #$> ("/_delete item #1 " <> msgItemId <> " broadcast", id, "message deleted") + alice <# "#team bob> [deleted] hey" + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "hi"), (1, "Full deletion: on")]) + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "hi"), (0, "Full deletion: on")]) + +testProhibitDirectMessages :: HasCallStack => FilePath -> IO () +testProhibitDirectMessages = + testChat4 aliceProfile bobProfile cathProfile danProfile $ \alice bob cath dan -> do + createGroup3 "team" alice bob cath + threadDelay 1000000 + alice ##> "/set direct #team off" + alice <## "updated group preferences:" + alice <## "Direct messages: off" + directProhibited bob + directProhibited cath + threadDelay 1000000 + -- still can send direct messages to direct contacts + alice #> "@bob hello again" + bob <# "alice> hello again" + alice #> "@cath hello again" + cath <# "alice> hello again" + bob ##> "@cath hello again" + bob <## "direct messages to indirect contact cath are prohibited" + (cath "/j #team" + concurrentlyN_ + [ cath <## "#team: dan joined the group", + do + dan <## "#team: you joined the group" + dan + <### [ "#team: member alice (Alice) is connected", + "#team: member bob (Bob) is connected" + ], + do + alice <## "#team: cath added dan (Daniel) to the group (connecting...)" + alice <## "#team: new member dan is connected", + do + bob <## "#team: cath added dan (Daniel) to the group (connecting...)" + bob <## "#team: new member dan is connected" + ] + alice ##> "@dan hi" + alice <## "direct messages to indirect contact dan are prohibited" + bob ##> "@dan hi" + bob <## "direct messages to indirect contact dan are prohibited" + (dan "@alice hi" + dan <## "direct messages to indirect contact alice are prohibited" + dan ##> "@bob hi" + dan <## "direct messages to indirect contact bob are prohibited" + dan #> "@cath hi" + cath <# "dan> hi" + cath #> "@dan hi" + dan <# "cath> hi" + where + directProhibited :: HasCallStack => TestCC -> IO () + directProhibited cc = do + cc <## "alice updated group #team:" + cc <## "updated group preferences:" + cc <## "Direct messages: off" + +testEnableTimedMessagesContact :: HasCallStack => FilePath -> IO () +testEnableTimedMessagesContact = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + connectUsers alice bob + alice ##> "/_set prefs @2 {\"timedMessages\": {\"allow\": \"yes\", \"ttl\": 1}}" + alice <## "you updated preferences for bob:" + alice <## "Disappearing messages: off (you allow: yes (1 sec), contact allows: no)" + bob <## "alice updated preferences for you:" + bob <## "Disappearing messages: off (you allow: no, contact allows: yes (1 sec))" + bob ##> "/set disappear @alice yes" + bob <## "you updated preferences for alice:" + bob <## "Disappearing messages: enabled (you allow: yes (1 sec), contact allows: yes (1 sec))" + alice <## "bob updated preferences for you:" + alice <## "Disappearing messages: enabled (you allow: yes (1 sec), contact allows: yes (1 sec))" + alice <##> bob + threadDelay 500000 + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Disappearing messages (1 sec)"), (0, "Disappearing messages: enabled (1 sec)"), (1, "hi"), (0, "hey")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Disappearing messages (1 sec)"), (1, "Disappearing messages: enabled (1 sec)"), (0, "hi"), (1, "hey")]) + threadDelay 1000000 + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Disappearing messages (1 sec)"), (0, "Disappearing messages: enabled (1 sec)")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Disappearing messages (1 sec)"), (1, "Disappearing messages: enabled (1 sec)")]) + -- turn off, messages are not disappearing + bob ##> "/set disappear @alice no" + bob <## "you updated preferences for alice:" + bob <## "Disappearing messages: off (you allow: no, contact allows: yes (1 sec))" + alice <## "bob updated preferences for you:" + alice <## "Disappearing messages: off (you allow: yes (1 sec), contact allows: no)" + alice <##> bob + threadDelay 1500000 + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "you offered Disappearing messages (1 sec)"), (0, "Disappearing messages: enabled (1 sec)"), (0, "Disappearing messages: off"), (1, "hi"), (0, "hey")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "offered Disappearing messages (1 sec)"), (1, "Disappearing messages: enabled (1 sec)"), (1, "Disappearing messages: off"), (0, "hi"), (1, "hey")]) + -- test api + bob ##> "/set disappear @alice yes 30s" + bob <## "you updated preferences for alice:" + bob <## "Disappearing messages: enabled (you allow: yes (30 sec), contact allows: yes (1 sec))" + alice <## "bob updated preferences for you:" + alice <## "Disappearing messages: enabled (you allow: yes (30 sec), contact allows: yes (30 sec))" + bob ##> "/set disappear @alice week" -- "yes" is optional + bob <## "you updated preferences for alice:" + bob <## "Disappearing messages: enabled (you allow: yes (1 week), contact allows: yes (1 sec))" + alice <## "bob updated preferences for you:" + alice <## "Disappearing messages: enabled (you allow: yes (1 week), contact allows: yes (1 week))" + +testEnableTimedMessagesGroup :: HasCallStack => FilePath -> IO () +testEnableTimedMessagesGroup = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + createGroup2 "team" alice bob + threadDelay 1000000 + alice ##> "/_group_profile #1 {\"displayName\": \"team\", \"fullName\": \"team\", \"groupPreferences\": {\"timedMessages\": {\"enable\": \"on\", \"ttl\": 1}, \"directMessages\": {\"enable\": \"on\"}}}" + alice <## "updated group preferences:" + alice <## "Disappearing messages: on (1 sec)" + bob <## "alice updated group #team:" + bob <## "updated group preferences:" + bob <## "Disappearing messages: on (1 sec)" + threadDelay 1000000 + alice #> "#team hi" + bob <# "#team alice> hi" + threadDelay 500000 + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Disappearing messages: on (1 sec)"), (1, "hi")]) + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Disappearing messages: on (1 sec)"), (0, "hi")]) + threadDelay 1000000 + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Disappearing messages: on (1 sec)")]) + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Disappearing messages: on (1 sec)")]) + -- turn off, messages are not disappearing + alice ##> "/set disappear #team off" + alice <## "updated group preferences:" + alice <## "Disappearing messages: off" + bob <## "alice updated group #team:" + bob <## "updated group preferences:" + bob <## "Disappearing messages: off" + threadDelay 1000000 + alice #> "#team hey" + bob <# "#team alice> hey" + threadDelay 1500000 + alice #$> ("/_get chat #1 count=100", chat, [(0, "connected"), (1, "Disappearing messages: on (1 sec)"), (1, "Disappearing messages: off"), (1, "hey")]) + bob #$> ("/_get chat #1 count=100", chat, groupFeatures <> [(0, "connected"), (0, "Disappearing messages: on (1 sec)"), (0, "Disappearing messages: off"), (0, "hey")]) + -- test api + alice ##> "/set disappear #team on 30s" + alice <## "updated group preferences:" + alice <## "Disappearing messages: on (30 sec)" + bob <## "alice updated group #team:" + bob <## "updated group preferences:" + bob <## "Disappearing messages: on (30 sec)" + alice ##> "/set disappear #team week" -- "on" is optional + alice <## "updated group preferences:" + alice <## "Disappearing messages: on (1 week)" + bob <## "alice updated group #team:" + bob <## "updated group preferences:" + bob <## "Disappearing messages: on (1 week)" + +testTimedMessagesEnabledGlobally :: HasCallStack => FilePath -> IO () +testTimedMessagesEnabledGlobally = + testChat2 aliceProfile bobProfile $ + \alice bob -> do + alice ##> "/set disappear yes" + alice <## "updated preferences:" + alice <## "Disappearing messages allowed: yes" + connectUsers alice bob + bob ##> "/_set prefs @2 {\"timedMessages\": {\"allow\": \"yes\", \"ttl\": 1}}" + bob <## "you updated preferences for alice:" + bob <## "Disappearing messages: enabled (you allow: yes (1 sec), contact allows: yes)" + alice <## "bob updated preferences for you:" + alice <## "Disappearing messages: enabled (you allow: yes (1 sec), contact allows: yes (1 sec))" + alice <##> bob + threadDelay 500000 + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "Disappearing messages: enabled (1 sec)"), (1, "hi"), (0, "hey")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "Disappearing messages: enabled (1 sec)"), (0, "hi"), (1, "hey")]) + threadDelay 1000000 + alice #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(0, "Disappearing messages: enabled (1 sec)")]) + bob #$> ("/_get chat @2 count=100", chat, chatFeatures <> [(1, "Disappearing messages: enabled (1 sec)")]) diff --git a/tests/ChatTests/Utils.hs b/tests/ChatTests/Utils.hs new file mode 100644 index 000000000..47ebcafd1 --- /dev/null +++ b/tests/ChatTests/Utils.hs @@ -0,0 +1,442 @@ +{-# LANGUAGE DuplicateRecordFields #-} +{-# LANGUAGE LambdaCase #-} +{-# LANGUAGE NamedFieldPuns #-} +{-# LANGUAGE OverloadedStrings #-} +{-# LANGUAGE RankNTypes #-} + +module ChatTests.Utils where + +import ChatClient +import Control.Concurrent.Async (concurrently_) +import Control.Concurrent.STM +import Control.Monad (unless, when) +import qualified Data.ByteString.Char8 as B +import Data.Char (isDigit) +import Data.List (isPrefixOf, isSuffixOf) +import Data.Maybe (fromMaybe) +import Data.String +import qualified Data.Text as T +import Simplex.Chat.Controller (ChatConfig (..), ChatController (..), InlineFilesConfig (..), defaultInlineFilesConfig) +import Simplex.Chat.Store (getUserContactProfiles) +import Simplex.Chat.Types +import Simplex.Messaging.Agent.Store.SQLite (withTransaction) +import Simplex.Messaging.Encoding.String +import System.Directory (doesFileExist) +import System.FilePath (()) +import Test.Hspec + +defaultPrefs :: Maybe Preferences +defaultPrefs = Just $ toChatPrefs defaultChatPrefs + +aliceProfile :: Profile +aliceProfile = Profile {displayName = "alice", fullName = "Alice", image = Nothing, preferences = defaultPrefs} + +bobProfile :: Profile +bobProfile = Profile {displayName = "bob", fullName = "Bob", image = Just (ImageData ""), preferences = defaultPrefs} + +cathProfile :: Profile +cathProfile = Profile {displayName = "cath", fullName = "Catherine", image = Nothing, preferences = defaultPrefs} + +danProfile :: Profile +danProfile = Profile {displayName = "dan", fullName = "Daniel", image = Nothing, preferences = defaultPrefs} + +versionTestMatrix2 :: (HasCallStack => TestCC -> TestCC -> IO ()) -> SpecWith FilePath +versionTestMatrix2 runTest = do + it "v2" $ testChat2 aliceProfile bobProfile runTest + it "v1" $ testChatCfg2 testCfgV1 aliceProfile bobProfile runTest + it "v1 to v2" $ runTestCfg2 testCfg testCfgV1 runTest + it "v2 to v1" $ runTestCfg2 testCfgV1 testCfg runTest + +versionTestMatrix3 :: (HasCallStack => TestCC -> TestCC -> TestCC -> IO ()) -> SpecWith FilePath +versionTestMatrix3 runTest = do + it "v2" $ testChat3 aliceProfile bobProfile cathProfile runTest + +-- it "v1" $ testChatCfg3 testCfgV1 aliceProfile bobProfile cathProfile runTest +-- it "v1 to v2" $ runTestCfg3 testCfg testCfgV1 testCfgV1 runTest +-- it "v2+v1 to v2" $ runTestCfg3 testCfg testCfg testCfgV1 runTest +-- it "v2 to v1" $ runTestCfg3 testCfgV1 testCfg testCfg runTest +-- it "v2+v1 to v1" $ runTestCfg3 testCfgV1 testCfg testCfgV1 runTest + +inlineCfg :: Integer -> ChatConfig +inlineCfg n = testCfg {inlineFiles = defaultInlineFilesConfig {sendChunks = 0, offerChunks = n, receiveChunks = n}} + +fileTestMatrix2 :: (HasCallStack => TestCC -> TestCC -> IO ()) -> SpecWith FilePath +fileTestMatrix2 runTest = do + it "via connection" $ runTestCfg2 viaConn viaConn runTest + it "inline (accepting)" $ runTestCfg2 inline inline runTest + it "via connection (inline offered)" $ runTestCfg2 inline viaConn runTest + it "via connection (inline supported)" $ runTestCfg2 viaConn inline runTest + where + inline = inlineCfg 100 + viaConn = inlineCfg 0 + +fileTestMatrix3 :: (HasCallStack => TestCC -> TestCC -> TestCC -> IO ()) -> SpecWith FilePath +fileTestMatrix3 runTest = do + it "via connection" $ runTestCfg3 viaConn viaConn viaConn runTest + it "inline" $ runTestCfg3 inline inline inline runTest + it "via connection (inline offered)" $ runTestCfg3 inline viaConn viaConn runTest + it "via connection (inline supported)" $ runTestCfg3 viaConn inline inline runTest + where + inline = inlineCfg 100 + viaConn = inlineCfg 0 + +runTestCfg2 :: ChatConfig -> ChatConfig -> (HasCallStack => TestCC -> TestCC -> IO ()) -> FilePath -> IO () +runTestCfg2 aliceCfg bobCfg runTest tmp = + withNewTestChatCfg tmp aliceCfg "alice" aliceProfile $ \alice -> + withNewTestChatCfg tmp bobCfg "bob" bobProfile $ \bob -> + runTest alice bob + +runTestCfg3 :: ChatConfig -> ChatConfig -> ChatConfig -> (HasCallStack => TestCC -> TestCC -> TestCC -> IO ()) -> FilePath -> IO () +runTestCfg3 aliceCfg bobCfg cathCfg runTest tmp = + withNewTestChatCfg tmp aliceCfg "alice" aliceProfile $ \alice -> + withNewTestChatCfg tmp bobCfg "bob" bobProfile $ \bob -> + withNewTestChatCfg tmp cathCfg "cath" cathProfile $ \cath -> + runTest alice bob cath + +withTestChatGroup3Connected :: HasCallStack => FilePath -> String -> (HasCallStack => TestCC -> IO a) -> IO a +withTestChatGroup3Connected tmp dbPrefix action = do + withTestChat tmp dbPrefix $ \cc -> do + cc <## "2 contacts connected (use /cs for the list)" + cc <## "#team: connected to server(s)" + action cc + +withTestChatGroup3Connected' :: HasCallStack => FilePath -> String -> IO () +withTestChatGroup3Connected' tmp dbPrefix = withTestChatGroup3Connected tmp dbPrefix $ \_ -> pure () + +withTestChatContactConnected :: HasCallStack => FilePath -> String -> (HasCallStack => TestCC -> IO a) -> IO a +withTestChatContactConnected tmp dbPrefix action = + withTestChat tmp dbPrefix $ \cc -> do + cc <## "1 contacts connected (use /cs for the list)" + action cc + +withTestChatContactConnected' :: HasCallStack => FilePath -> String -> IO () +withTestChatContactConnected' tmp dbPrefix = withTestChatContactConnected tmp dbPrefix $ \_ -> pure () + +withTestChatContactConnectedV1 :: HasCallStack => FilePath -> String -> (HasCallStack => TestCC -> IO a) -> IO a +withTestChatContactConnectedV1 tmp dbPrefix action = + withTestChatV1 tmp dbPrefix $ \cc -> do + cc <## "1 contacts connected (use /cs for the list)" + action cc + +withTestChatContactConnectedV1' :: HasCallStack => FilePath -> String -> IO () +withTestChatContactConnectedV1' tmp dbPrefix = withTestChatContactConnectedV1 tmp dbPrefix $ \_ -> pure () + +-- | test sending direct messages +(<##>) :: HasCallStack => TestCC -> TestCC -> IO () +cc1 <##> cc2 = do + name1 <- userName cc1 + name2 <- userName cc2 + cc1 #> ("@" <> name2 <> " hi") + cc2 <# (name1 <> "> hi") + cc2 #> ("@" <> name1 <> " hey") + cc1 <# (name2 <> "> hey") + +(##>) :: HasCallStack => TestCC -> String -> IO () +cc ##> cmd = do + cc `send` cmd + cc <## cmd + +(#>) :: HasCallStack => TestCC -> String -> IO () +cc #> cmd = do + cc `send` cmd + cc <# cmd + +(?#>) :: HasCallStack => TestCC -> String -> IO () +cc ?#> cmd = do + cc `send` cmd + cc <# ("i " <> cmd) + +(#$>) :: (Eq a, Show a, HasCallStack) => TestCC -> (String, String -> a, a) -> Expectation +cc #$> (cmd, f, res) = do + cc ##> cmd + (f <$> getTermLine cc) `shouldReturn` res + +chat :: String -> [(Int, String)] +chat = map (\(a, _, _) -> a) . chat'' + +chat' :: String -> [((Int, String), Maybe (Int, String))] +chat' = map (\(a, b, _) -> (a, b)) . chat'' + +chatF :: String -> [((Int, String), Maybe String)] +chatF = map (\(a, _, c) -> (a, c)) . chat'' + +chat'' :: String -> [((Int, String), Maybe (Int, String), Maybe String)] +chat'' = read + +chatFeatures :: [(Int, String)] +chatFeatures = map (\(a, _, _) -> a) chatFeatures'' + +chatFeatures' :: [((Int, String), Maybe (Int, String))] +chatFeatures' = map (\(a, b, _) -> (a, b)) chatFeatures'' + +chatFeaturesF :: [((Int, String), Maybe String)] +chatFeaturesF = map (\(a, _, c) -> (a, c)) chatFeatures'' + +chatFeatures'' :: [((Int, String), Maybe (Int, String), Maybe String)] +chatFeatures'' = [((0, "Disappearing messages: off"), Nothing, Nothing), ((0, "Full deletion: off"), Nothing, Nothing), ((0, "Voice messages: enabled"), Nothing, Nothing)] + +groupFeatures :: [(Int, String)] +groupFeatures = map (\(a, _, _) -> a) groupFeatures'' + +groupFeatures'' :: [((Int, String), Maybe (Int, String), Maybe String)] +groupFeatures'' = [((0, "Disappearing messages: off"), Nothing, Nothing), ((0, "Direct messages: on"), Nothing, Nothing), ((0, "Full deletion: off"), Nothing, Nothing), ((0, "Voice messages: on"), Nothing, Nothing)] + +itemId :: Int -> String +itemId i = show $ length chatFeatures + i + +(@@@) :: HasCallStack => TestCC -> [(String, String)] -> Expectation +(@@@) = getChats mapChats + +mapChats :: [(String, String, Maybe ConnStatus)] -> [(String, String)] +mapChats = map $ \(ldn, msg, _) -> (ldn, msg) + +chats :: String -> [(String, String)] +chats = mapChats . read + +(@@@!) :: HasCallStack => TestCC -> [(String, String, Maybe ConnStatus)] -> Expectation +(@@@!) = getChats id + +getChats :: HasCallStack => (Eq a, Show a) => ([(String, String, Maybe ConnStatus)] -> [a]) -> TestCC -> [a] -> Expectation +getChats f cc res = do + cc ##> "/_get chats 1 pcc=on" + line <- getTermLine cc + f (read line) `shouldMatchList` res + +send :: TestCC -> String -> IO () +send TestCC {chatController = cc} cmd = atomically $ writeTBQueue (inputQ cc) cmd + +(<##) :: HasCallStack => TestCC -> String -> Expectation +cc <## line = do + l <- getTermLine cc + when (l /= line) $ print ("expected: " <> line, ", got: " <> l) + l `shouldBe` line + +(<##.) :: HasCallStack => TestCC -> String -> Expectation +cc <##. line = do + l <- getTermLine cc + let prefix = line `isPrefixOf` l + unless prefix $ print ("expected to start from: " <> line, ", got: " <> l) + prefix `shouldBe` True + +(<#.) :: HasCallStack => TestCC -> String -> Expectation +cc <#. line = do + l <- dropTime <$> getTermLine cc + let prefix = line `isPrefixOf` l + unless prefix $ print ("expected to start from: " <> line, ", got: " <> l) + prefix `shouldBe` True + +(<##..) :: HasCallStack => TestCC -> [String] -> Expectation +cc <##.. ls = do + l <- getTermLine cc + let prefix = any (`isPrefixOf` l) ls + unless prefix $ print ("expected to start from one of: " <> show ls, ", got: " <> l) + prefix `shouldBe` True + +data ConsoleResponse = ConsoleString String | WithTime String | EndsWith String + deriving (Show) + +instance IsString ConsoleResponse where fromString = ConsoleString + +-- this assumes that the string can only match one option +getInAnyOrder :: HasCallStack => (String -> String) -> TestCC -> [ConsoleResponse] -> Expectation +getInAnyOrder _ _ [] = pure () +getInAnyOrder f cc ls = do + line <- f <$> getTermLine cc + let rest = filter (not . expected line) ls + if length rest < length ls + then getInAnyOrder f cc rest + else error $ "unexpected output: " <> line + where + expected :: String -> ConsoleResponse -> Bool + expected l = \case + ConsoleString s -> l == s + WithTime s -> dropTime_ l == Just s + EndsWith s -> s `isSuffixOf` l + +(<###) :: HasCallStack => TestCC -> [ConsoleResponse] -> Expectation +(<###) = getInAnyOrder id + +(<##?) :: HasCallStack => TestCC -> [ConsoleResponse] -> Expectation +(<##?) = getInAnyOrder dropTime + +(<#) :: HasCallStack => TestCC -> String -> Expectation +cc <# line = (dropTime <$> getTermLine cc) `shouldReturn` line + +(?<#) :: HasCallStack => TestCC -> String -> Expectation +cc ?<# line = (dropTime <$> getTermLine cc) `shouldReturn` "i " <> line + +($<#) :: HasCallStack => (TestCC, String) -> String -> Expectation +(cc, uName) $<# line = (dropTime . dropUser uName <$> getTermLine cc) `shouldReturn` line + +( TestCC -> Expectation +( TestCC -> TestCC -> Expectation +cc1 <#? cc2 = do + name <- userName cc2 + sName <- showName cc2 + cc2 <## "connection request sent!" + cc1 <## (sName <> " wants to connect to you!") + cc1 <## ("to accept: /ac " <> name) + cc1 <## ("to reject: /rc " <> name <> " (the sender will NOT be notified)") + +dropUser :: HasCallStack => String -> String -> String +dropUser uName msg = fromMaybe err $ dropUser_ uName msg + where + err = error $ "invalid user: " <> msg + +dropUser_ :: String -> String -> Maybe String +dropUser_ uName msg = do + let userPrefix = "[user: " <> uName <> "] " + if userPrefix `isPrefixOf` msg + then Just $ drop (length userPrefix) msg + else Nothing + +dropTime :: HasCallStack => String -> String +dropTime msg = fromMaybe err $ dropTime_ msg + where + err = error $ "invalid time: " <> msg + +dropTime_ :: String -> Maybe String +dropTime_ msg = case splitAt 6 msg of + ([m, m', ':', s, s', ' '], text) -> + if all isDigit [m, m', s, s'] then Just text else Nothing + _ -> Nothing + +getInvitation :: HasCallStack => TestCC -> IO String +getInvitation cc = do + cc <## "pass this invitation link to your contact (via another channel):" + cc <## "" + inv <- getTermLine cc + cc <## "" + cc <## "and ask them to connect: /c " + pure inv + +getContactLink :: HasCallStack => TestCC -> Bool -> IO String +getContactLink cc created = do + cc <## if created then "Your new chat address is created!" else "Your chat address:" + cc <## "" + link <- getTermLine cc + cc <## "" + cc <## "Anybody can send you contact requests with: /c " + cc <## "to show it again: /sa" + cc <## "to delete it: /da (accepted contacts will remain connected)" + pure link + +getGroupLink :: HasCallStack => TestCC -> String -> Bool -> IO String +getGroupLink cc gName created = do + cc <## if created then "Group link is created!" else "Group link:" + cc <## "" + link <- getTermLine cc + cc <## "" + cc <## "Anybody can connect to you and join group with: /c " + cc <## ("to show it again: /show link #" <> gName) + cc <## ("to delete it: /delete link #" <> gName <> " (joined members will remain connected to you)") + pure link + +hasContactProfiles :: HasCallStack => TestCC -> [ContactName] -> Expectation +hasContactProfiles cc names = + getContactProfiles cc >>= \ps -> ps `shouldMatchList` names + +getContactProfiles :: TestCC -> IO [ContactName] +getContactProfiles cc = do + user_ <- readTVarIO (currentUser $ chatController cc) + case user_ of + Nothing -> pure [] + Just user -> do + profiles <- withTransaction (chatStore $ chatController cc) $ \db -> getUserContactProfiles db user + pure $ map (\Profile {displayName} -> displayName) profiles + +lastItemId :: HasCallStack => TestCC -> IO String +lastItemId cc = do + cc ##> "/last_item_id" + getTermLine cc + +showActiveUser :: HasCallStack => TestCC -> String -> Expectation +showActiveUser cc name = do + cc <## ("user profile: " <> name) + cc <## "use /p [] to change it" + cc <## "(the updated profile will be sent to all your contacts)" + +connectUsers :: HasCallStack => TestCC -> TestCC -> IO () +connectUsers cc1 cc2 = do + name1 <- showName cc1 + name2 <- showName cc2 + cc1 ##> "/c" + inv <- getInvitation cc1 + cc2 ##> ("/c " <> inv) + cc2 <## "confirmation sent!" + concurrently_ + (cc2 <## (name1 <> ": contact is connected")) + (cc1 <## (name2 <> ": contact is connected")) + +showName :: TestCC -> IO String +showName (TestCC ChatController {currentUser} _ _ _ _) = do + Just User {localDisplayName, profile = LocalProfile {fullName}} <- readTVarIO currentUser + pure . T.unpack $ localDisplayName <> optionalFullName localDisplayName fullName + +createGroup2 :: HasCallStack => String -> TestCC -> TestCC -> IO () +createGroup2 gName cc1 cc2 = do + connectUsers cc1 cc2 + name2 <- userName cc2 + cc1 ##> ("/g " <> gName) + cc1 <## ("group #" <> gName <> " is created") + cc1 <## ("to add members use /a " <> gName <> " or /create link #" <> gName) + addMember gName cc1 cc2 GRAdmin + cc2 ##> ("/j " <> gName) + concurrently_ + (cc1 <## ("#" <> gName <> ": " <> name2 <> " joined the group")) + (cc2 <## ("#" <> gName <> ": you joined the group")) + +createGroup3 :: HasCallStack => String -> TestCC -> TestCC -> TestCC -> IO () +createGroup3 gName cc1 cc2 cc3 = do + createGroup2 gName cc1 cc2 + connectUsers cc1 cc3 + name3 <- userName cc3 + sName2 <- showName cc2 + sName3 <- showName cc3 + addMember gName cc1 cc3 GRAdmin + cc3 ##> ("/j " <> gName) + concurrentlyN_ + [ cc1 <## ("#" <> gName <> ": " <> name3 <> " joined the group"), + do + cc3 <## ("#" <> gName <> ": you joined the group") + cc3 <## ("#" <> gName <> ": member " <> sName2 <> " is connected"), + do + cc2 <## ("#" <> gName <> ": alice added " <> sName3 <> " to the group (connecting...)") + cc2 <## ("#" <> gName <> ": new member " <> name3 <> " is connected") + ] + +addMember :: HasCallStack => String -> TestCC -> TestCC -> GroupMemberRole -> IO () +addMember gName inviting invitee role = do + name1 <- userName inviting + memName <- userName invitee + inviting ##> ("/a " <> gName <> " " <> memName <> " " <> B.unpack (strEncode role)) + concurrentlyN_ + [ inviting <## ("invitation to join the group #" <> gName <> " sent to " <> memName), + do + invitee <## ("#" <> gName <> ": " <> name1 <> " invites you to join the group as " <> B.unpack (strEncode role)) + invitee <## ("use /j " <> gName <> " to accept") + ] + +checkActionDeletesFile :: HasCallStack => FilePath -> IO () -> IO () +checkActionDeletesFile file action = do + fileExistsBefore <- doesFileExist file + fileExistsBefore `shouldBe` True + action + fileExistsAfter <- doesFileExist file + fileExistsAfter `shouldBe` False + +startFileTransferWithDest' :: HasCallStack => TestCC -> TestCC -> String -> String -> Maybe String -> IO () +startFileTransferWithDest' cc1 cc2 fileName fileSize fileDest_ = do + name1 <- userName cc1 + name2 <- userName cc2 + cc1 #> ("/f @" <> name2 <> " ./tests/fixtures/" <> fileName) + cc1 <## "use /fc 1 to cancel sending" + cc2 <# (name1 <> "> sends file " <> fileName <> " (" <> fileSize <> ")") + cc2 <## "use /fr 1 [/ | ] to receive it" + cc2 ##> ("/fr 1" <> maybe "" (" " <>) fileDest_) + cc2 <## ("saving file 1 from " <> name1 <> " to " <> maybe id () fileDest_ fileName) + concurrently_ + (cc2 <## ("started receiving file 1 (" <> fileName <> ") from " <> name1)) + (cc1 <## ("started sending file 1 (" <> fileName <> ") to " <> name2)) diff --git a/tests/MobileTests.hs b/tests/MobileTests.hs index b18cba744..83e91f044 100644 --- a/tests/MobileTests.hs +++ b/tests/MobileTests.hs @@ -2,7 +2,7 @@ module MobileTests where -import ChatTests +import ChatTests.Utils import Control.Monad.Except import Simplex.Chat.Mobile import Simplex.Chat.Store