mirror of
https://github.com/simplex-chat/simplex-chat.git
synced 2026-04-01 03:05:55 +00:00
816 lines
37 KiB
Haskell
816 lines
37 KiB
Haskell
{-# LANGUAGE DataKinds #-}
|
|
{-# LANGUAGE DuplicateRecordFields #-}
|
|
{-# LANGUAGE FlexibleContexts #-}
|
|
{-# LANGUAGE GADTs #-}
|
|
{-# LANGUAGE LambdaCase #-}
|
|
{-# LANGUAGE NamedFieldPuns #-}
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
{-# LANGUAGE RankNTypes #-}
|
|
{-# LANGUAGE RecordWildCards #-}
|
|
{-# LANGUAGE ScopedTypeVariables #-}
|
|
{-# LANGUAGE TupleSections #-}
|
|
|
|
module Simplex.Chat where
|
|
|
|
import Control.Applicative ((<|>))
|
|
import Control.Logger.Simple
|
|
import Control.Monad.Except
|
|
import Control.Monad.IO.Unlift
|
|
import Control.Monad.Reader
|
|
import Crypto.Random (drgNew)
|
|
import Data.Attoparsec.ByteString.Char8 (Parser)
|
|
import qualified Data.Attoparsec.ByteString.Char8 as A
|
|
import Data.Bifunctor (first)
|
|
import Data.ByteString.Char8 (ByteString)
|
|
import qualified Data.ByteString.Char8 as B
|
|
import Data.Functor (($>))
|
|
import Data.List (find)
|
|
import Data.Maybe (isJust, mapMaybe)
|
|
import Data.Text (Text)
|
|
import qualified Data.Text as T
|
|
import Data.Text.Encoding (encodeUtf8)
|
|
import Numeric.Natural
|
|
import Simplex.Chat.Controller
|
|
import Simplex.Chat.Help
|
|
import Simplex.Chat.Input
|
|
import Simplex.Chat.Notification
|
|
import Simplex.Chat.Options (ChatOpts (..))
|
|
import Simplex.Chat.Protocol
|
|
import Simplex.Chat.Store
|
|
import Simplex.Chat.Styled (plain)
|
|
import Simplex.Chat.Terminal
|
|
import Simplex.Chat.Types
|
|
import Simplex.Chat.View
|
|
import Simplex.Messaging.Agent
|
|
import Simplex.Messaging.Agent.Env.SQLite (AgentConfig (..), defaultAgentConfig)
|
|
import Simplex.Messaging.Agent.Protocol
|
|
import qualified Simplex.Messaging.Crypto as C
|
|
import Simplex.Messaging.Parsers (parseAll)
|
|
import Simplex.Messaging.Util (raceAny_)
|
|
import System.Exit (exitFailure, exitSuccess)
|
|
import System.IO (hFlush, stdout)
|
|
import Text.Read (readMaybe)
|
|
import UnliftIO.Async (race_)
|
|
import qualified UnliftIO.Exception as E
|
|
import UnliftIO.STM
|
|
|
|
data ChatCommand
|
|
= ChatHelp
|
|
| MarkdownHelp
|
|
| AddContact
|
|
| Connect SMPQueueInfo
|
|
| DeleteContact ContactName
|
|
| SendMessage ContactName ByteString
|
|
| NewGroup GroupProfile
|
|
| AddMember GroupName ContactName GroupMemberRole
|
|
| JoinGroup GroupName
|
|
| RemoveMember GroupName ContactName
|
|
| MemberRole GroupName ContactName GroupMemberRole
|
|
| LeaveGroup GroupName
|
|
| DeleteGroup GroupName
|
|
| ListMembers GroupName
|
|
| SendGroupMessage GroupName ByteString
|
|
| UpdateProfile Profile
|
|
| ShowProfile
|
|
| QuitChat
|
|
deriving (Show)
|
|
|
|
data ChatConfig = ChatConfig
|
|
{ agentConfig :: AgentConfig,
|
|
dbPoolSize :: Int,
|
|
tbqSize :: Natural
|
|
}
|
|
|
|
defaultChatConfig :: ChatConfig
|
|
defaultChatConfig =
|
|
ChatConfig
|
|
{ agentConfig =
|
|
defaultAgentConfig
|
|
{ tcpPort = undefined, -- agent does not listen to TCP
|
|
smpServers = undefined, -- filled in from options
|
|
dbFile = undefined, -- filled in from options
|
|
dbPoolSize = 1
|
|
},
|
|
dbPoolSize = 1,
|
|
tbqSize = 16
|
|
}
|
|
|
|
logCfg :: LogConfig
|
|
logCfg = LogConfig {lc_file = Nothing, lc_stderr = True}
|
|
|
|
simplexChat :: WithTerminal t => ChatConfig -> ChatOpts -> t -> IO ()
|
|
simplexChat cfg opts t =
|
|
-- setLogLevel LogInfo -- LogError
|
|
-- withGlobalLogging logCfg $ do
|
|
initializeNotifications
|
|
>>= newChatController cfg opts t
|
|
>>= runSimplexChat
|
|
|
|
newChatController :: WithTerminal t => ChatConfig -> ChatOpts -> t -> (Notification -> IO ()) -> IO ChatController
|
|
newChatController ChatConfig {agentConfig = cfg, dbPoolSize, tbqSize} ChatOpts {dbFile, smpServers} t sendNotification = do
|
|
chatStore <- createStore (dbFile <> ".chat.db") dbPoolSize
|
|
currentUser <- newTVarIO =<< getCreateActiveUser chatStore
|
|
chatTerminal <- newChatTerminal t
|
|
smpAgent <- getSMPAgentClient cfg {dbFile = dbFile <> ".agent.db", smpServers}
|
|
idsDrg <- newTVarIO =<< drgNew
|
|
inputQ <- newTBQueueIO tbqSize
|
|
notifyQ <- newTBQueueIO tbqSize
|
|
chatLock <- newTMVarIO ()
|
|
pure ChatController {..}
|
|
|
|
runSimplexChat :: ChatController -> IO ()
|
|
runSimplexChat = runReaderT (race_ runTerminalInput runChatController)
|
|
|
|
runChatController :: (MonadUnliftIO m, MonadReader ChatController m) => m ()
|
|
runChatController =
|
|
raceAny_
|
|
[ inputSubscriber,
|
|
agentSubscriber,
|
|
notificationSubscriber
|
|
]
|
|
|
|
withLock :: MonadUnliftIO m => TMVar () -> m () -> m ()
|
|
withLock lock =
|
|
E.bracket_
|
|
(void . atomically $ takeTMVar lock)
|
|
(atomically $ putTMVar lock ())
|
|
|
|
inputSubscriber :: (MonadUnliftIO m, MonadReader ChatController m) => m ()
|
|
inputSubscriber = do
|
|
q <- asks inputQ
|
|
l <- asks chatLock
|
|
forever $
|
|
atomically (readTBQueue q) >>= \case
|
|
InputControl _ -> pure ()
|
|
InputCommand s ->
|
|
case parseAll chatCommandP . encodeUtf8 $ T.pack s of
|
|
Left e -> printToView [plain s, "invalid input: " <> plain e]
|
|
Right cmd -> do
|
|
case cmd of
|
|
SendMessage c msg -> showSentMessage c msg
|
|
SendGroupMessage g msg -> showSentGroupMessage g msg
|
|
_ -> printToView [plain s]
|
|
user <- readTVarIO =<< asks currentUser
|
|
withLock l . void . runExceptT $
|
|
processChatCommand user cmd `catchError` showChatError
|
|
|
|
processChatCommand :: ChatMonad m => User -> ChatCommand -> m ()
|
|
processChatCommand user@User {userId, profile} = \case
|
|
ChatHelp -> printToView chatHelpInfo
|
|
MarkdownHelp -> printToView markdownInfo
|
|
AddContact -> do
|
|
(connId, qInfo) <- withAgent createConnection
|
|
withStore $ \st -> createDirectConnection st userId connId
|
|
showInvitation qInfo
|
|
Connect qInfo -> do
|
|
connId <- withAgent $ \a -> joinConnection a qInfo . directMessage $ XInfo profile
|
|
withStore $ \st -> createDirectConnection st userId connId
|
|
DeleteContact cName ->
|
|
withStore (\st -> getContactGroupNames st userId cName) >>= \case
|
|
[] -> do
|
|
conns <- withStore $ \st -> getContactConnections st userId cName
|
|
withAgent $ \a -> forM_ conns $ \Connection {agentConnId} ->
|
|
deleteConnection a agentConnId `catchError` \(_ :: AgentErrorType) -> pure ()
|
|
withStore $ \st -> deleteContact st userId cName
|
|
unsetActive $ ActiveC cName
|
|
showContactDeleted cName
|
|
gs -> showContactGroups cName gs
|
|
SendMessage cName msg -> do
|
|
contact <- withStore $ \st -> getContact st userId cName
|
|
let msgEvent = XMsgNew $ MsgContent MTText [] [MsgContentBody {contentType = SimplexContentType XCText, contentData = msg}]
|
|
sendDirectMessage (contactConnId contact) msgEvent
|
|
setActive $ ActiveC cName
|
|
NewGroup gProfile -> do
|
|
gVar <- asks idsDrg
|
|
group <- withStore $ \st -> createNewGroup st gVar user gProfile
|
|
showGroupCreated group
|
|
AddMember gName cName memRole -> do
|
|
(group, contact) <- withStore $ \st -> (,) <$> getGroup st user gName <*> getContact st userId cName
|
|
let Group {groupId, groupProfile, membership, members} = group
|
|
userRole = memberRole membership
|
|
userMemberId = memberId membership
|
|
when (userRole < GRAdmin || userRole < memRole) $ chatError CEGroupUserRole
|
|
when (memberStatus membership == GSMemInvited) $ chatError (CEGroupNotJoined gName)
|
|
unless (memberActive membership) $ chatError CEGroupMemberNotActive
|
|
when (isJust $ contactMember contact members) $ chatError (CEGroupDuplicateMember cName)
|
|
gVar <- asks idsDrg
|
|
(agentConnId, qInfo) <- withAgent createConnection
|
|
GroupMember {memberId} <- withStore $ \st -> createContactGroupMember st gVar user groupId contact memRole agentConnId
|
|
let msg = XGrpInv $ GroupInvitation (userMemberId, userRole) (memberId, memRole) qInfo groupProfile
|
|
sendDirectMessage (contactConnId contact) msg
|
|
showSentGroupInvitation gName cName
|
|
setActive $ ActiveG gName
|
|
JoinGroup gName -> do
|
|
ReceivedGroupInvitation {fromMember, userMember, queueInfo} <- withStore $ \st -> getGroupInvitation st user gName
|
|
agentConnId <- withAgent $ \a -> joinConnection a queueInfo . directMessage . XGrpAcpt $ memberId userMember
|
|
withStore $ \st -> do
|
|
createMemberConnection st userId fromMember agentConnId
|
|
updateGroupMemberStatus st userId fromMember GSMemAccepted
|
|
updateGroupMemberStatus st userId userMember GSMemAccepted
|
|
MemberRole _gName _cName _mRole -> pure ()
|
|
RemoveMember gName cName -> do
|
|
Group {membership, members} <- withStore $ \st -> getGroup st user gName
|
|
case find ((== cName) . (localDisplayName :: GroupMember -> ContactName)) members of
|
|
Nothing -> chatError $ CEGroupMemberNotFound cName
|
|
Just member -> do
|
|
let userRole = memberRole membership
|
|
when (userRole < GRAdmin || userRole < memberRole member) $ chatError CEGroupUserRole
|
|
sendGroupMessage members . XGrpMemDel $ memberId member
|
|
deleteMemberConnection member
|
|
withStore $ \st -> updateGroupMemberStatus st userId member GSMemRemoved
|
|
showDeletedMember gName Nothing (Just member)
|
|
LeaveGroup gName -> do
|
|
Group {membership, members} <- withStore $ \st -> getGroup st user gName
|
|
sendGroupMessage members XGrpLeave
|
|
mapM_ deleteMemberConnection members
|
|
withStore $ \st -> updateGroupMemberStatus st userId membership GSMemLeft
|
|
showLeftMemberUser gName
|
|
DeleteGroup gName -> do
|
|
g@Group {membership, members} <- withStore $ \st -> getGroup st user gName
|
|
let s = memberStatus membership
|
|
canDelete =
|
|
memberRole membership == GROwner
|
|
|| (s == GSMemRemoved || s == GSMemLeft || s == GSMemGroupDeleted)
|
|
unless canDelete $ chatError CEGroupUserRole
|
|
when (memberActive membership) $ sendGroupMessage members XGrpDel
|
|
mapM_ deleteMemberConnection members
|
|
withStore $ \st -> deleteGroup st user g
|
|
showGroupDeletedUser gName
|
|
ListMembers gName -> do
|
|
group <- withStore $ \st -> getGroup st user gName
|
|
showGroupMembers group
|
|
SendGroupMessage gName msg -> do
|
|
-- TODO save sent messages
|
|
-- TODO save pending message delivery for members without connections
|
|
Group {members, membership} <- withStore $ \st -> getGroup st user gName
|
|
unless (memberActive membership) $ chatError CEGroupMemberUserRemoved
|
|
let msgEvent = XMsgNew $ MsgContent MTText [] [MsgContentBody {contentType = SimplexContentType XCText, contentData = msg}]
|
|
sendGroupMessage members msgEvent
|
|
setActive $ ActiveG gName
|
|
UpdateProfile p -> unless (p == profile) $ do
|
|
user' <- withStore $ \st -> updateUserProfile st user p
|
|
asks currentUser >>= atomically . (`writeTVar` user')
|
|
contacts <- withStore (`getUserContacts` user)
|
|
forM_ contacts $ \ct -> sendDirectMessage (contactConnId ct) $ XInfo p
|
|
showUserProfileUpdated user user'
|
|
ShowProfile -> showUserProfile profile
|
|
QuitChat -> liftIO exitSuccess
|
|
where
|
|
contactMember :: Contact -> [GroupMember] -> Maybe GroupMember
|
|
contactMember Contact {contactId} =
|
|
find $ \GroupMember {memberContactId = cId, memberStatus = s} ->
|
|
cId == Just contactId && s /= GSMemRemoved && s /= GSMemLeft
|
|
|
|
agentSubscriber :: (MonadUnliftIO m, MonadReader ChatController m) => m ()
|
|
agentSubscriber = do
|
|
q <- asks $ subQ . smpAgent
|
|
l <- asks chatLock
|
|
subscribeUserConnections
|
|
forever $ do
|
|
(_, connId, msg) <- atomically $ readTBQueue q
|
|
user <- readTVarIO =<< asks currentUser
|
|
-- TODO handle errors properly
|
|
withLock l . void . runExceptT $
|
|
processAgentMessage user connId msg `catchError` (liftIO . print)
|
|
|
|
subscribeUserConnections :: (MonadUnliftIO m, MonadReader ChatController m) => m ()
|
|
subscribeUserConnections = void . runExceptT $ do
|
|
user <- readTVarIO =<< asks currentUser
|
|
subscribeContacts user
|
|
subscribeGroups user
|
|
subscribePendingConnections user
|
|
where
|
|
subscribeContacts user = do
|
|
contacts <- withStore (`getUserContacts` user)
|
|
forM_ contacts $ \ct@Contact {localDisplayName = c} ->
|
|
(subscribe (contactConnId ct) >> showContactSubscribed c) `catchError` showContactSubError c
|
|
subscribeGroups user = do
|
|
groups <- withStore (`getUserGroups` user)
|
|
forM_ groups $ \Group {members, membership, localDisplayName = g} -> do
|
|
let connectedMembers = mapMaybe (\m -> (m,) <$> memberConnId m) members
|
|
if null connectedMembers
|
|
then
|
|
if memberActive membership
|
|
then showGroupEmpty g
|
|
else showGroupRemoved g
|
|
else do
|
|
forM_ connectedMembers $ \(GroupMember {localDisplayName = c}, cId) ->
|
|
subscribe cId `catchError` showMemberSubError g c
|
|
showGroupSubscribed g
|
|
subscribePendingConnections user = do
|
|
connections <- withStore (`getPendingConnections` user)
|
|
forM_ connections $ \Connection {agentConnId} ->
|
|
subscribe agentConnId `catchError` \_ -> pure ()
|
|
subscribe cId = withAgent (`subscribeConnection` cId)
|
|
|
|
processAgentMessage :: forall m. ChatMonad m => User -> ConnId -> ACommand 'Agent -> m ()
|
|
processAgentMessage user@User {userId, profile} agentConnId agentMessage = unless (sent agentMessage) $ do
|
|
chatDirection <- withStore $ \st -> getConnectionChatDirection st user agentConnId
|
|
forM_ (agentMsgConnStatus agentMessage) $ \status ->
|
|
withStore $ \st -> updateConnectionStatus st (fromConnection chatDirection) status
|
|
case chatDirection of
|
|
ReceivedDirectMessage conn maybeContact ->
|
|
processDirectMessage agentMessage conn maybeContact
|
|
ReceivedGroupMessage conn gName m ->
|
|
processGroupMessage agentMessage conn gName m
|
|
where
|
|
sent :: ACommand 'Agent -> Bool
|
|
sent SENT {} = True
|
|
sent _ = False
|
|
|
|
isMember :: MemberId -> Group -> Bool
|
|
isMember memId Group {membership, members} =
|
|
memberId membership == memId || isJust (find ((== memId) . memberId) members)
|
|
|
|
contactIsReady :: Contact -> Bool
|
|
contactIsReady Contact {activeConn} = connStatus activeConn == ConnReady
|
|
|
|
memberIsReady :: GroupMember -> Bool
|
|
memberIsReady GroupMember {activeConn} = maybe False ((== ConnReady) . connStatus) activeConn
|
|
|
|
agentMsgConnStatus :: ACommand 'Agent -> Maybe ConnStatus
|
|
agentMsgConnStatus = \case
|
|
REQ _ _ -> Just ConnRequested
|
|
INFO _ -> Just ConnSndReady
|
|
CON -> Just ConnReady
|
|
_ -> Nothing
|
|
|
|
processDirectMessage :: ACommand 'Agent -> Connection -> Maybe Contact -> m ()
|
|
processDirectMessage agentMsg conn = \case
|
|
Nothing -> case agentMsg of
|
|
REQ confId connInfo -> do
|
|
saveConnInfo conn connInfo
|
|
acceptAgentConnection conn confId $ XInfo profile
|
|
INFO connInfo ->
|
|
saveConnInfo conn connInfo
|
|
_ -> pure ()
|
|
Just ct@Contact {localDisplayName = c} -> case agentMsg of
|
|
MSG meta msgBody -> do
|
|
ChatMessage {chatMsgEvent} <- liftEither $ parseChatMessage msgBody
|
|
case chatMsgEvent of
|
|
XMsgNew (MsgContent MTText [] body) -> newTextMessage c meta $ find (isSimplexContentType XCText) body
|
|
XInfo p -> xInfo ct p
|
|
XGrpInv gInv -> processGroupInvitation ct gInv
|
|
XInfoProbe probe -> xInfoProbe ct probe
|
|
XInfoProbeCheck probeHash -> xInfoProbeCheck ct probeHash
|
|
XInfoProbeOk probe -> xInfoProbeOk ct probe
|
|
_ -> pure ()
|
|
REQ confId connInfo -> do
|
|
-- confirming direct connection with a member
|
|
ChatMessage {chatMsgEvent} <- liftEither $ parseChatMessage connInfo
|
|
case chatMsgEvent of
|
|
XGrpMemInfo _memId _memProfile -> do
|
|
-- TODO check member ID
|
|
-- TODO update member profile
|
|
acceptAgentConnection conn confId XOk
|
|
_ -> messageError "REQ from member must have x.grp.mem.info"
|
|
INFO connInfo -> do
|
|
ChatMessage {chatMsgEvent} <- liftEither $ parseChatMessage connInfo
|
|
case chatMsgEvent of
|
|
XGrpMemInfo _memId _memProfile -> do
|
|
-- TODO check member ID
|
|
-- TODO update member profile
|
|
pure ()
|
|
XOk -> pure ()
|
|
_ -> messageError "INFO from member must have x.grp.mem.info or x.ok"
|
|
CON ->
|
|
withStore (\st -> getViaGroupMember st user ct) >>= \case
|
|
Nothing -> do
|
|
showContactConnected ct
|
|
setActive $ ActiveC c
|
|
showToast (c <> "> ") "connected"
|
|
Just (gName, m) ->
|
|
when (memberIsReady m) $ do
|
|
notifyMemberConnected gName m
|
|
when (memberCategory m == GCPreMember) $ probeMatchingContacts ct
|
|
END -> do
|
|
showContactAnotherClient c
|
|
showToast (c <> "> ") "connected to another client"
|
|
unsetActive $ ActiveC c
|
|
DOWN -> do
|
|
showContactDisconnected c
|
|
showToast (c <> "> ") "disconnected"
|
|
UP -> do
|
|
showContactSubscribed c
|
|
showToast (c <> "> ") "is active"
|
|
setActive $ ActiveC c
|
|
_ -> pure ()
|
|
|
|
processGroupMessage :: ACommand 'Agent -> Connection -> GroupName -> GroupMember -> m ()
|
|
processGroupMessage agentMsg conn gName m = case agentMsg of
|
|
REQ confId connInfo -> do
|
|
ChatMessage {chatMsgEvent} <- liftEither $ parseChatMessage connInfo
|
|
case memberCategory m of
|
|
GCInviteeMember ->
|
|
case chatMsgEvent of
|
|
XGrpAcpt memId
|
|
| memId == memberId m -> do
|
|
withStore $ \st -> updateGroupMemberStatus st userId m GSMemAccepted
|
|
acceptAgentConnection conn confId XOk
|
|
| otherwise -> messageError "x.grp.acpt: memberId is different from expected"
|
|
_ -> messageError "REQ from invited member must have x.grp.acpt"
|
|
_ ->
|
|
case chatMsgEvent of
|
|
XGrpMemInfo memId _memProfile
|
|
| memId == memberId m -> do
|
|
-- TODO update member profile
|
|
Group {membership} <- withStore $ \st -> getGroup st user gName
|
|
acceptAgentConnection conn confId $ XGrpMemInfo (memberId membership) profile
|
|
| otherwise -> messageError "x.grp.mem.info: memberId is different from expected"
|
|
_ -> messageError "REQ from member must have x.grp.mem.info"
|
|
INFO connInfo -> do
|
|
ChatMessage {chatMsgEvent} <- liftEither $ parseChatMessage connInfo
|
|
case chatMsgEvent of
|
|
XGrpMemInfo memId _memProfile
|
|
| memId == memberId m -> do
|
|
-- TODO update member profile
|
|
pure ()
|
|
| otherwise -> messageError "x.grp.mem.info: memberId is different from expected"
|
|
XOk -> pure ()
|
|
_ -> messageError "INFO from member must have x.grp.mem.info"
|
|
pure ()
|
|
CON -> do
|
|
group@Group {members, membership} <- withStore $ \st -> getGroup st user gName
|
|
withStore $ \st -> do
|
|
updateGroupMemberStatus st userId m GSMemConnected
|
|
unless (memberActive membership) $
|
|
updateGroupMemberStatus st userId membership GSMemConnected
|
|
-- TODO forward any pending (GMIntroInvReceived) introductions
|
|
case memberCategory m of
|
|
GCHostMember -> do
|
|
showUserJoinedGroup gName
|
|
setActive $ ActiveG gName
|
|
showToast ("#" <> gName) "you are connected to group"
|
|
GCInviteeMember -> do
|
|
showJoinedGroupMember gName m
|
|
setActive $ ActiveG gName
|
|
showToast ("#" <> gName) $ "member " <> localDisplayName (m :: GroupMember) <> " is connected"
|
|
intros <- withStore $ \st -> createIntroductions st group m
|
|
sendGroupMessage members . XGrpMemNew $ memberInfo m
|
|
forM_ intros $ \intro -> do
|
|
sendDirectMessage agentConnId . XGrpMemIntro . memberInfo $ reMember intro
|
|
withStore $ \st -> updateIntroStatus st intro GMIntroSent
|
|
_ -> do
|
|
-- TODO send probe and decide whether to use existing contact connection or the new contact connection
|
|
-- TODO notify member who forwarded introduction - question - where it is stored? There is via_contact but probably there should be via_member in group_members table
|
|
withStore (\st -> getViaGroupContact st user m) >>= \case
|
|
Nothing -> do
|
|
notifyMemberConnected gName m
|
|
messageError "implementation error: connected member does not have contact"
|
|
Just ct ->
|
|
when (contactIsReady ct) $ do
|
|
notifyMemberConnected gName m
|
|
when (memberCategory m == GCPreMember) $ probeMatchingContacts ct
|
|
MSG meta msgBody -> do
|
|
ChatMessage {chatMsgEvent} <- liftEither $ parseChatMessage msgBody
|
|
case chatMsgEvent of
|
|
XMsgNew (MsgContent MTText [] body) ->
|
|
newGroupTextMessage gName m meta $ find (isSimplexContentType XCText) body
|
|
XGrpMemNew memInfo -> xGrpMemNew gName m memInfo
|
|
XGrpMemIntro memInfo -> xGrpMemIntro gName m memInfo
|
|
XGrpMemInv memId introInv -> xGrpMemInv gName m memId introInv
|
|
XGrpMemFwd memInfo introInv -> xGrpMemFwd gName m memInfo introInv
|
|
XGrpMemDel memId -> xGrpMemDel gName m memId
|
|
XGrpLeave -> xGrpLeave gName m
|
|
XGrpDel -> xGrpDel gName m
|
|
_ -> messageError $ "unsupported message: " <> T.pack (show chatMsgEvent)
|
|
_ -> pure ()
|
|
|
|
notifyMemberConnected :: GroupName -> GroupMember -> m ()
|
|
notifyMemberConnected gName m@GroupMember {localDisplayName} = do
|
|
showConnectedToGroupMember gName m
|
|
setActive $ ActiveG gName
|
|
showToast ("#" <> gName) $ "member " <> localDisplayName <> " is connected"
|
|
|
|
probeMatchingContacts :: Contact -> m ()
|
|
probeMatchingContacts ct = do
|
|
gVar <- asks idsDrg
|
|
(probe, probeId) <- withStore $ \st -> createSentProbe st gVar userId ct
|
|
sendDirectMessage (contactConnId ct) $ XInfoProbe probe
|
|
cs <- withStore (\st -> getMatchingContacts st userId ct)
|
|
let probeHash = C.sha256Hash probe
|
|
forM_ cs $ \c -> sendProbeHash c probeHash probeId `catchError` const (pure ())
|
|
where
|
|
sendProbeHash c probeHash probeId = do
|
|
sendDirectMessage (contactConnId c) $ XInfoProbeCheck probeHash
|
|
withStore $ \st -> createSentProbeHash st userId probeId c
|
|
|
|
messageWarning :: Text -> m ()
|
|
messageWarning = liftIO . print
|
|
|
|
messageError :: Text -> m ()
|
|
messageError = liftIO . print
|
|
|
|
newTextMessage :: ContactName -> MsgMeta -> Maybe MsgContentBody -> m ()
|
|
newTextMessage c meta = \case
|
|
Just MsgContentBody {contentData = bs} -> do
|
|
let text = safeDecodeUtf8 bs
|
|
showReceivedMessage c (snd $ broker meta) text (integrity meta)
|
|
showToast (c <> "> ") text
|
|
setActive $ ActiveC c
|
|
_ -> messageError "x.msg.new: no expected message body"
|
|
|
|
newGroupTextMessage :: GroupName -> GroupMember -> MsgMeta -> Maybe MsgContentBody -> m ()
|
|
newGroupTextMessage gName GroupMember {localDisplayName = c} meta = \case
|
|
Just MsgContentBody {contentData = bs} -> do
|
|
let text = safeDecodeUtf8 bs
|
|
showReceivedGroupMessage gName c (snd $ broker meta) text (integrity meta)
|
|
showToast ("#" <> gName <> " " <> c <> "> ") text
|
|
setActive $ ActiveG gName
|
|
_ -> messageError "x.msg.new: no expected message body"
|
|
|
|
processGroupInvitation :: Contact -> GroupInvitation -> m ()
|
|
processGroupInvitation ct@Contact {localDisplayName} inv@(GroupInvitation (fromMemId, fromRole) (memId, memRole) _ _) = do
|
|
when (fromRole < GRAdmin || fromRole < memRole) $ chatError (CEGroupContactRole localDisplayName)
|
|
when (fromMemId == memId) $ chatError CEGroupDuplicateMemberId
|
|
group <- withStore $ \st -> createGroupInvitation st user ct inv
|
|
showReceivedGroupInvitation group localDisplayName memRole
|
|
|
|
xInfo :: Contact -> Profile -> m ()
|
|
xInfo c@Contact {profile = p} p' = unless (p == p') $ do
|
|
c' <- withStore $ \st -> updateContactProfile st userId c p'
|
|
showContactUpdated c c'
|
|
|
|
xInfoProbe :: Contact -> ByteString -> m ()
|
|
xInfoProbe c2 probe = do
|
|
r <- withStore $ \st -> matchReceivedProbe st userId c2 probe
|
|
forM_ r $ \c1 -> probeMatch c1 c2 probe
|
|
|
|
xInfoProbeCheck :: Contact -> ByteString -> m ()
|
|
xInfoProbeCheck c1 probeHash = do
|
|
r <- withStore $ \st -> matchReceivedProbeHash st userId c1 probeHash
|
|
forM_ r . uncurry $ probeMatch c1
|
|
|
|
probeMatch :: Contact -> Contact -> ByteString -> m ()
|
|
probeMatch c1@Contact {profile = p1} c2@Contact {profile = p2} probe =
|
|
when (p1 == p2) $ do
|
|
sendDirectMessage (contactConnId c1) $ XInfoProbeOk probe
|
|
mergeContacts c1 c2
|
|
|
|
xInfoProbeOk :: Contact -> ByteString -> m ()
|
|
xInfoProbeOk c1 probe = do
|
|
r <- withStore $ \st -> matchSentProbe st userId c1 probe
|
|
forM_ r $ \c2 -> mergeContacts c1 c2
|
|
|
|
mergeContacts :: Contact -> Contact -> m ()
|
|
mergeContacts to from = do
|
|
withStore $ \st -> mergeContactRecords st userId to from
|
|
showContactsMerged to from
|
|
|
|
parseChatMessage :: ByteString -> Either ChatError ChatMessage
|
|
parseChatMessage msgBody = first ChatErrorMessage (parseAll rawChatMessageP msgBody >>= toChatMessage)
|
|
|
|
saveConnInfo :: Connection -> ConnInfo -> m ()
|
|
saveConnInfo activeConn connInfo = do
|
|
ChatMessage {chatMsgEvent} <- liftEither $ parseChatMessage connInfo
|
|
case chatMsgEvent of
|
|
XInfo p ->
|
|
withStore $ \st -> createDirectContact st userId activeConn p
|
|
-- TODO show/log error, other events in SMP confirmation
|
|
_ -> pure ()
|
|
|
|
xGrpMemNew :: GroupName -> GroupMember -> MemberInfo -> m ()
|
|
xGrpMemNew gName m memInfo@(MemberInfo memId _ _) = do
|
|
group@Group {membership} <- withStore $ \st -> getGroup st user gName
|
|
when (memberId membership /= memId) $
|
|
if isMember memId group
|
|
then messageError "x.grp.mem.new error: member already exists"
|
|
else do
|
|
newMember <- withStore $ \st -> createNewGroupMember st user group memInfo GCPostMember GSMemAnnounced
|
|
showJoinedGroupMemberConnecting gName m newMember
|
|
|
|
xGrpMemIntro :: GroupName -> GroupMember -> MemberInfo -> m ()
|
|
xGrpMemIntro gName m memInfo@(MemberInfo memId _ _) =
|
|
case memberCategory m of
|
|
GCHostMember -> do
|
|
group <- withStore $ \st -> getGroup st user gName
|
|
if isMember memId group
|
|
then messageWarning "x.grp.mem.intro ignored: member already exists"
|
|
else do
|
|
(groupConnId, groupQInfo) <- withAgent createConnection
|
|
(directConnId, directQInfo) <- withAgent createConnection
|
|
newMember <- withStore $ \st -> createIntroReMember st user group m memInfo groupConnId directConnId
|
|
let msg = XGrpMemInv memId IntroInvitation {groupQInfo, directQInfo}
|
|
sendDirectMessage agentConnId msg
|
|
withStore $ \st -> updateGroupMemberStatus st userId newMember GSMemIntroInvited
|
|
_ -> messageError "x.grp.mem.intro can be only sent by host member"
|
|
|
|
xGrpMemInv :: GroupName -> GroupMember -> MemberId -> IntroInvitation -> m ()
|
|
xGrpMemInv gName m memId introInv =
|
|
case memberCategory m of
|
|
GCInviteeMember -> do
|
|
group <- withStore $ \st -> getGroup st user gName
|
|
case find ((== memId) . memberId) $ members group of
|
|
Nothing -> messageError "x.grp.mem.inv error: referenced member does not exists"
|
|
Just reMember -> do
|
|
intro <- withStore $ \st -> saveIntroInvitation st reMember m introInv
|
|
case activeConn (reMember :: GroupMember) of
|
|
Nothing -> pure () -- this is not an error, introduction will be forwarded once the member is connected
|
|
Just Connection {agentConnId = reAgentConnId} -> do
|
|
sendDirectMessage reAgentConnId $ XGrpMemFwd (memberInfo m) introInv
|
|
withStore $ \st -> updateIntroStatus st intro GMIntroInvForwarded
|
|
_ -> messageError "x.grp.mem.inv can be only sent by invitee member"
|
|
|
|
xGrpMemFwd :: GroupName -> GroupMember -> MemberInfo -> IntroInvitation -> m ()
|
|
xGrpMemFwd gName m memInfo@(MemberInfo memId _ _) introInv@IntroInvitation {groupQInfo, directQInfo} = do
|
|
group@Group {membership} <- withStore $ \st -> getGroup st user gName
|
|
toMember <- case find ((== memId) . memberId) $ members group of
|
|
-- TODO if the missed messages are correctly sent as soon as there is connection before anything else is sent
|
|
-- the situation when member does not exist is an error
|
|
-- member receiving x.grp.mem.fwd should have also received x.grp.mem.new prior to that.
|
|
-- For now, this branch compensates for the lack of delayed message delivery.
|
|
Nothing -> withStore $ \st -> createNewGroupMember st user group memInfo GCPostMember GSMemAnnounced
|
|
Just m' -> pure m'
|
|
withStore $ \st -> saveMemberInvitation st toMember introInv
|
|
let msg = XGrpMemInfo (memberId membership) profile
|
|
groupConnId <- withAgent $ \a -> joinConnection a groupQInfo $ directMessage msg
|
|
directConnId <- withAgent $ \a -> joinConnection a directQInfo $ directMessage msg
|
|
withStore $ \st -> createIntroToMemberContact st userId m toMember groupConnId directConnId
|
|
|
|
xGrpMemDel :: GroupName -> GroupMember -> MemberId -> m ()
|
|
xGrpMemDel gName m memId = do
|
|
Group {membership, members} <- withStore $ \st -> getGroup st user gName
|
|
if memberId membership == memId
|
|
then do
|
|
mapM_ deleteMemberConnection members
|
|
withStore $ \st -> updateGroupMemberStatus st userId membership GSMemRemoved
|
|
showDeletedMemberUser gName m
|
|
else case find ((== memId) . memberId) members of
|
|
Nothing -> messageError "x.grp.mem.del with unknown member ID"
|
|
Just member -> do
|
|
let mRole = memberRole m
|
|
if mRole < GRAdmin || mRole < memberRole member
|
|
then messageError "x.grp.mem.del with insufficient member permissions"
|
|
else do
|
|
deleteMemberConnection member
|
|
withStore $ \st -> updateGroupMemberStatus st userId member GSMemRemoved
|
|
showDeletedMember gName (Just m) (Just member)
|
|
|
|
xGrpLeave :: GroupName -> GroupMember -> m ()
|
|
xGrpLeave gName m = do
|
|
deleteMemberConnection m
|
|
withStore $ \st -> updateGroupMemberStatus st userId m GSMemLeft
|
|
showLeftMember gName m
|
|
|
|
xGrpDel :: GroupName -> GroupMember -> m ()
|
|
xGrpDel gName m = do
|
|
when (memberRole m /= GROwner) $ chatError CEGroupUserRole
|
|
ms <- withStore $ \st -> do
|
|
Group {members, membership} <- getGroup st user gName
|
|
updateGroupMemberStatus st userId membership GSMemGroupDeleted
|
|
pure members
|
|
mapM_ deleteMemberConnection ms
|
|
showGroupDeleted gName m
|
|
|
|
chatError :: ChatMonad m => ChatErrorType -> m ()
|
|
chatError = throwError . ChatError
|
|
|
|
deleteMemberConnection :: ChatMonad m => GroupMember -> m ()
|
|
deleteMemberConnection m@GroupMember {activeConn} = do
|
|
-- User {userId} <- asks currentUser
|
|
withAgent $ forM_ (memberConnId m) . suspendConnection
|
|
-- withStore $ \st -> deleteGroupMemberConnection st userId m
|
|
forM_ activeConn $ \conn -> withStore $ \st -> updateConnectionStatus st conn ConnDeleted
|
|
|
|
sendDirectMessage :: ChatMonad m => ConnId -> ChatMsgEvent -> m ()
|
|
sendDirectMessage agentConnId chatMsgEvent =
|
|
void . withAgent $ \a -> sendMessage a agentConnId $ directMessage chatMsgEvent
|
|
|
|
directMessage :: ChatMsgEvent -> ByteString
|
|
directMessage chatMsgEvent =
|
|
serializeRawChatMessage $
|
|
rawChatMessage ChatMessage {chatMsgId = Nothing, chatMsgEvent, chatDAG = Nothing}
|
|
|
|
sendGroupMessage :: ChatMonad m => [GroupMember] -> ChatMsgEvent -> m ()
|
|
sendGroupMessage members chatMsgEvent = do
|
|
let msg = directMessage chatMsgEvent
|
|
-- TODO once scheduled delivery is implemented memberActive should be changed to memberCurrent
|
|
withAgent $ \a ->
|
|
forM_ (filter memberActive members) $
|
|
traverse (\connId -> sendMessage a connId msg) . memberConnId
|
|
|
|
acceptAgentConnection :: ChatMonad m => Connection -> ConfirmationId -> ChatMsgEvent -> m ()
|
|
acceptAgentConnection conn@Connection {agentConnId} confId msg = do
|
|
withAgent $ \a -> acceptConnection a agentConnId confId $ directMessage msg
|
|
withStore $ \st -> updateConnectionStatus st conn ConnAccepted
|
|
|
|
getCreateActiveUser :: SQLiteStore -> IO User
|
|
getCreateActiveUser st = do
|
|
user <-
|
|
getUsers st >>= \case
|
|
[] -> newUser
|
|
users -> maybe (selectUser users) pure (find activeUser users)
|
|
putStrLn $ "Current user: " <> userStr user
|
|
pure user
|
|
where
|
|
newUser :: IO User
|
|
newUser = do
|
|
putStrLn
|
|
"No user profiles found, it will be created now.\n\
|
|
\Please choose your display name and your full name.\n\
|
|
\They will be sent to your contacts when you connect.\n\
|
|
\They are only stored on your device and you can change them later."
|
|
loop
|
|
where
|
|
loop = do
|
|
displayName <- getContactName
|
|
fullName <- T.pack <$> getWithPrompt "full name (optional)"
|
|
liftIO (runExceptT $ createUser st Profile {displayName, fullName} True) >>= \case
|
|
Left SEDuplicateName -> do
|
|
putStrLn "chosen display name is already used by another profile on this device, choose another one"
|
|
loop
|
|
Left e -> putStrLn ("database error " <> show e) >> exitFailure
|
|
Right user -> pure user
|
|
selectUser :: [User] -> IO User
|
|
selectUser [user] = do
|
|
liftIO $ setActiveUser st (userId user)
|
|
pure user
|
|
selectUser users = do
|
|
putStrLn "Select user profile:"
|
|
forM_ (zip [1 ..] users) $ \(n :: Int, user) -> putStrLn $ show n <> " - " <> userStr user
|
|
loop
|
|
where
|
|
loop = do
|
|
nStr <- getWithPrompt $ "user profile number (1 .. " <> show (length users) <> ")"
|
|
case readMaybe nStr :: Maybe Int of
|
|
Nothing -> putStrLn "invalid user number" >> loop
|
|
Just n
|
|
| n <= 0 || n > length users -> putStrLn "invalid user number" >> loop
|
|
| otherwise -> do
|
|
let user = users !! (n - 1)
|
|
liftIO $ setActiveUser st (userId user)
|
|
pure user
|
|
userStr :: User -> String
|
|
userStr User {localDisplayName, profile = Profile {fullName}} =
|
|
T.unpack $ localDisplayName <> if T.null fullName || localDisplayName == fullName then "" else " (" <> fullName <> ")"
|
|
getContactName :: IO ContactName
|
|
getContactName = do
|
|
displayName <- getWithPrompt "display name (no spaces)"
|
|
if null displayName || isJust (find (== ' ') displayName)
|
|
then putStrLn "display name has space(s), choose another one" >> getContactName
|
|
else pure $ T.pack displayName
|
|
getWithPrompt :: String -> IO String
|
|
getWithPrompt s = putStr (s <> ": ") >> hFlush stdout >> getLine
|
|
|
|
showToast :: (MonadUnliftIO m, MonadReader ChatController m) => Text -> Text -> m ()
|
|
showToast title text = atomically . (`writeTBQueue` Notification {title, text}) =<< asks notifyQ
|
|
|
|
notificationSubscriber :: (MonadUnliftIO m, MonadReader ChatController m) => m ()
|
|
notificationSubscriber = do
|
|
ChatController {notifyQ, sendNotification} <- ask
|
|
forever $ atomically (readTBQueue notifyQ) >>= liftIO . sendNotification
|
|
|
|
withAgent :: ChatMonad m => (AgentClient -> ExceptT AgentErrorType m a) -> m a
|
|
withAgent action =
|
|
asks smpAgent
|
|
>>= runExceptT . action
|
|
>>= liftEither . first ChatErrorAgent
|
|
|
|
withStore ::
|
|
ChatMonad m =>
|
|
(forall m'. (MonadUnliftIO m', MonadError StoreError m') => SQLiteStore -> m' a) ->
|
|
m a
|
|
withStore action =
|
|
asks chatStore
|
|
>>= runExceptT . action
|
|
>>= liftEither . first ChatErrorStore
|
|
|
|
chatCommandP :: Parser ChatCommand
|
|
chatCommandP =
|
|
("/help" <|> "/h") $> ChatHelp
|
|
<|> ("/group #" <|> "/group " <|> "/g #" <|> "/g ") *> (NewGroup <$> groupProfile)
|
|
<|> ("/add #" <|> "/add " <|> "/a #" <|> "/a ") *> (AddMember <$> displayName <* A.space <*> displayName <*> memberRole)
|
|
<|> ("/join #" <|> "/join " <|> "/j #" <|> "/j ") *> (JoinGroup <$> displayName)
|
|
<|> ("/remove #" <|> "/remove " <|> "/rm #" <|> "/rm ") *> (RemoveMember <$> displayName <* A.space <*> displayName)
|
|
<|> ("/leave #" <|> "/leave " <|> "/l #" <|> "/l ") *> (LeaveGroup <$> displayName)
|
|
<|> ("/delete #" <|> "/d #") *> (DeleteGroup <$> displayName)
|
|
<|> ("/members #" <|> "/members " <|> "/ms #" <|> "/ms ") *> (ListMembers <$> displayName)
|
|
<|> A.char '#' *> (SendGroupMessage <$> displayName <* A.space <*> A.takeByteString)
|
|
<|> ("/connect " <|> "/c ") *> (Connect <$> smpQueueInfoP)
|
|
<|> ("/connect" <|> "/c") $> AddContact
|
|
<|> ("/delete @" <|> "/delete " <|> "/d @" <|> "/d ") *> (DeleteContact <$> displayName)
|
|
<|> A.char '@' *> (SendMessage <$> displayName <*> (A.space *> A.takeByteString))
|
|
<|> ("/markdown" <|> "/m") $> MarkdownHelp
|
|
<|> ("/profile " <|> "/p ") *> (UpdateProfile <$> userProfile)
|
|
<|> ("/profile" <|> "/p") $> ShowProfile
|
|
<|> ("/quit" <|> "/q") $> QuitChat
|
|
where
|
|
displayName = safeDecodeUtf8 <$> (B.cons <$> A.satisfy refChar <*> A.takeTill (== ' '))
|
|
refChar c = c > ' ' && c /= '#' && c /= '@'
|
|
userProfile = do
|
|
cName <- displayName
|
|
fullName <- fullNameP cName
|
|
pure Profile {displayName = cName, fullName}
|
|
groupProfile = do
|
|
gName <- displayName
|
|
fullName <- fullNameP gName
|
|
pure GroupProfile {displayName = gName, fullName}
|
|
fullNameP name = do
|
|
n <- (A.space *> A.takeByteString) <|> pure ""
|
|
pure $ if B.null n then name else safeDecodeUtf8 n
|
|
memberRole =
|
|
(" owner" $> GROwner)
|
|
<|> (" admin" $> GRAdmin)
|
|
<|> (" normal" $> GRMember)
|
|
<|> pure GRAdmin
|