2023-09-27 11:41:02 +03:00
|
|
|
{-# LANGUAGE DuplicateRecordFields #-}
|
|
|
|
{-# LANGUAGE FlexibleContexts #-}
|
|
|
|
{-# LANGUAGE LambdaCase #-}
|
|
|
|
{-# LANGUAGE NamedFieldPuns #-}
|
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
|
|
{-# LANGUAGE ScopedTypeVariables #-}
|
2023-10-07 16:23:24 +03:00
|
|
|
{-# LANGUAGE TupleSections #-}
|
2023-09-27 11:41:02 +03:00
|
|
|
|
|
|
|
module Simplex.Chat.Remote where
|
|
|
|
|
2023-10-07 16:23:24 +03:00
|
|
|
import Control.Logger.Simple
|
2023-10-04 18:36:10 +03:00
|
|
|
import Control.Monad
|
2023-09-27 11:41:02 +03:00
|
|
|
import Control.Monad.Except
|
|
|
|
import Control.Monad.IO.Class
|
2023-10-07 16:23:24 +03:00
|
|
|
import Control.Monad.Reader (asks)
|
2023-10-04 18:36:10 +03:00
|
|
|
import Control.Monad.STM (retry)
|
|
|
|
import Crypto.Random (getRandomBytes)
|
2023-09-27 11:41:02 +03:00
|
|
|
import qualified Data.Aeson as J
|
2023-09-27 18:24:38 +03:00
|
|
|
import qualified Data.Binary.Builder as Binary
|
2023-10-04 18:36:10 +03:00
|
|
|
import Data.ByteString (ByteString)
|
|
|
|
import qualified Data.ByteString.Base64.URL as B64U
|
|
|
|
import qualified Data.ByteString.Char8 as B
|
|
|
|
import Data.List.NonEmpty (NonEmpty (..))
|
2023-10-07 16:23:24 +03:00
|
|
|
import Data.Maybe (fromMaybe)
|
2023-09-27 11:41:02 +03:00
|
|
|
import qualified Data.Map.Strict as M
|
2023-10-07 16:23:24 +03:00
|
|
|
import qualified Data.Text as T
|
2023-09-27 11:41:02 +03:00
|
|
|
import qualified Network.HTTP.Types as HTTP
|
2023-10-07 16:23:24 +03:00
|
|
|
import qualified Network.HTTP.Types.Status as Status
|
2023-09-27 11:41:02 +03:00
|
|
|
import qualified Network.HTTP2.Client as HTTP2Client
|
2023-10-07 16:23:24 +03:00
|
|
|
import qualified Network.HTTP2.Server as HTTP2Server
|
2023-09-29 14:56:56 +03:00
|
|
|
import Network.Socket (SockAddr (..), hostAddressToTuple)
|
2023-09-27 11:41:02 +03:00
|
|
|
import Simplex.Chat.Controller
|
2023-09-29 14:56:56 +03:00
|
|
|
import qualified Simplex.Chat.Remote.Discovery as Discovery
|
2023-09-27 11:41:02 +03:00
|
|
|
import Simplex.Chat.Remote.Types
|
2023-09-29 14:56:56 +03:00
|
|
|
import Simplex.Chat.Store.Remote
|
2023-09-27 11:41:02 +03:00
|
|
|
import Simplex.Chat.Types
|
2023-09-29 14:56:56 +03:00
|
|
|
import qualified Simplex.Messaging.Crypto as C
|
|
|
|
import Simplex.Messaging.Encoding.String (StrEncoding (..))
|
|
|
|
import qualified Simplex.Messaging.TMap as TM
|
|
|
|
import Simplex.Messaging.Transport.Client (TransportHost (..))
|
2023-10-04 18:36:10 +03:00
|
|
|
import Simplex.Messaging.Transport.Credentials (genCredentials, tlsCredentials)
|
2023-09-27 11:41:02 +03:00
|
|
|
import Simplex.Messaging.Transport.HTTP2 (HTTP2Body (..))
|
2023-10-04 18:36:10 +03:00
|
|
|
import Simplex.Messaging.Transport.HTTP2.Client (HTTP2Client)
|
2023-09-27 11:41:02 +03:00
|
|
|
import qualified Simplex.Messaging.Transport.HTTP2.Client as HTTP2
|
2023-09-27 18:24:38 +03:00
|
|
|
import qualified Simplex.Messaging.Transport.HTTP2.Server as HTTP2
|
2023-10-07 16:23:24 +03:00
|
|
|
import Simplex.Messaging.Util (bshow, ifM, tshow)
|
2023-09-27 11:41:02 +03:00
|
|
|
import System.Directory (getFileSize)
|
2023-09-29 14:56:56 +03:00
|
|
|
import UnliftIO
|
2023-09-27 11:41:02 +03:00
|
|
|
|
2023-09-27 18:24:38 +03:00
|
|
|
withRemoteHostSession :: (ChatMonad m) => RemoteHostId -> (RemoteHostSession -> m a) -> m a
|
2023-09-27 11:41:02 +03:00
|
|
|
withRemoteHostSession remoteHostId action = do
|
|
|
|
chatReadVar remoteHostSessions >>= maybe err action . M.lookup remoteHostId
|
|
|
|
where
|
|
|
|
err = throwError $ ChatErrorRemoteHost remoteHostId RHMissing
|
|
|
|
|
2023-10-04 18:36:10 +03:00
|
|
|
withRemoteHost :: (ChatMonad m) => RemoteHostId -> (RemoteHost -> m a) -> m a
|
|
|
|
withRemoteHost remoteHostId action =
|
|
|
|
withStore' (`getRemoteHost` remoteHostId) >>= \case
|
|
|
|
Nothing -> throwError $ ChatErrorRemoteHost remoteHostId RHMissing
|
|
|
|
Just rh -> action rh
|
|
|
|
|
2023-09-29 14:56:56 +03:00
|
|
|
startRemoteHost :: (ChatMonad m) => RemoteHostId -> m ChatResponse
|
|
|
|
startRemoteHost remoteHostId = do
|
2023-10-07 16:23:24 +03:00
|
|
|
asks remoteHostSessions >>= atomically . TM.lookup remoteHostId >>= \case
|
2023-10-04 18:36:10 +03:00
|
|
|
Just _ -> throwError $ ChatErrorRemoteHost remoteHostId RHBusy
|
2023-10-07 16:23:24 +03:00
|
|
|
Nothing -> withRemoteHost remoteHostId $ \rh -> do
|
|
|
|
announcer <- async $ run rh
|
2023-10-04 18:36:10 +03:00
|
|
|
chatModifyVar remoteHostSessions $ M.insert remoteHostId RemoteHostSessionStarting {announcer}
|
|
|
|
pure CRRemoteHostStarted {remoteHostId}
|
2023-10-07 16:23:24 +03:00
|
|
|
where
|
|
|
|
cleanup finished = do
|
|
|
|
logInfo "Remote host http2 client fininshed"
|
|
|
|
atomically $ writeTVar finished True
|
|
|
|
closeRemoteHostSession remoteHostId >>= toView
|
|
|
|
run RemoteHost {storePath, caKey, caCert} = do
|
|
|
|
finished <- newTVarIO False
|
|
|
|
let parent = (C.signatureKeyPair caKey, caCert)
|
|
|
|
sessionCreds <- liftIO $ genCredentials (Just parent) (0, 24) "Session"
|
|
|
|
let (fingerprint, credentials) = tlsCredentials $ sessionCreds :| [parent]
|
|
|
|
Discovery.announceRevHTTP2 (cleanup finished) fingerprint credentials >>= \case
|
|
|
|
Left h2ce -> do
|
|
|
|
logError $ "Failed to set up remote host connection: " <> tshow h2ce
|
|
|
|
cleanup finished
|
|
|
|
Right ctrlClient -> do
|
|
|
|
chatModifyVar remoteHostSessions $ M.insert remoteHostId RemoteHostSessionStarted {storePath, ctrlClient}
|
|
|
|
chatWriteVar currentRemoteHost $ Just remoteHostId
|
|
|
|
sendHello ctrlClient >>= \case
|
|
|
|
Left h2ce -> do
|
|
|
|
logError $ "Failed to send initial remote host request: " <> tshow h2ce
|
|
|
|
cleanup finished
|
|
|
|
Right HTTP2.HTTP2Response {respBody = HTTP2Body {bodyHead}} -> do
|
|
|
|
logDebug $ "Got initial from remote host: " <> tshow bodyHead
|
|
|
|
_ <- asks outputQ >>= async . pollRemote finished ctrlClient "/recv" (Nothing, Just remoteHostId,)
|
|
|
|
toView CRRemoteHostConnected {remoteHostId}
|
|
|
|
|
|
|
|
sendHello :: (ChatMonad m) => HTTP2Client -> m (Either HTTP2.HTTP2ClientError HTTP2.HTTP2Response)
|
|
|
|
sendHello http = liftIO (HTTP2.sendRequestDirect http req Nothing)
|
|
|
|
where
|
|
|
|
req = HTTP2Client.requestNoBody "GET" "/" mempty
|
|
|
|
|
|
|
|
pollRemote :: (ChatMonad m, J.FromJSON a) => TVar Bool -> HTTP2Client -> ByteString -> (a -> b) -> TBQueue b -> m ()
|
|
|
|
pollRemote finished http path f queue = loop
|
|
|
|
where
|
|
|
|
loop = do
|
|
|
|
liftIO (HTTP2.sendRequestDirect http req Nothing) >>= \case
|
|
|
|
Left e -> logError $ "pollRemote: " <> tshow (path, e)
|
|
|
|
Right HTTP2.HTTP2Response {respBody = HTTP2Body {bodyHead}} ->
|
|
|
|
case J.eitherDecodeStrict' bodyHead of
|
|
|
|
Left e -> logError $ "pollRemote/decode: " <> tshow (path, e)
|
|
|
|
Right o -> atomically $ writeTBQueue queue (f o)
|
|
|
|
readTVarIO finished >>= (`unless` loop)
|
|
|
|
req = HTTP2Client.requestNoBody "GET" path mempty
|
2023-10-04 18:36:10 +03:00
|
|
|
|
|
|
|
closeRemoteHostSession :: (ChatMonad m) => RemoteHostId -> m ChatResponse
|
|
|
|
closeRemoteHostSession remoteHostId = withRemoteHostSession remoteHostId $ \session -> do
|
2023-10-07 16:23:24 +03:00
|
|
|
liftIO $ cancelRemoteHostSession session
|
|
|
|
chatWriteVar currentRemoteHost Nothing
|
2023-10-04 18:36:10 +03:00
|
|
|
chatModifyVar remoteHostSessions $ M.delete remoteHostId
|
2023-10-07 16:23:24 +03:00
|
|
|
pure CRRemoteHostStopped {remoteHostId}
|
|
|
|
|
|
|
|
cancelRemoteHostSession :: MonadUnliftIO m => RemoteHostSession -> m ()
|
|
|
|
cancelRemoteHostSession = \case
|
|
|
|
RemoteHostSessionStarting {announcer} -> cancel announcer
|
|
|
|
RemoteHostSessionStarted {ctrlClient} -> liftIO $ HTTP2.closeHTTP2Client ctrlClient
|
2023-10-04 18:36:10 +03:00
|
|
|
|
|
|
|
createRemoteHost :: (ChatMonad m) => m ChatResponse
|
|
|
|
createRemoteHost = do
|
|
|
|
let displayName = "TODO" -- you don't have remote host name here, it will be passed from remote host
|
|
|
|
((_, caKey), caCert) <- liftIO $ genCredentials Nothing (-25, 24 * 365) displayName
|
|
|
|
storePath <- liftIO randomStorePath
|
|
|
|
remoteHostId <- withStore' $ \db -> insertRemoteHost db storePath displayName caKey caCert
|
2023-10-07 16:23:24 +03:00
|
|
|
let oobData = RemoteCtrlOOB {caFingerprint = C.certificateFingerprint caCert}
|
2023-10-04 18:36:10 +03:00
|
|
|
pure CRRemoteHostCreated {remoteHostId, oobData}
|
|
|
|
|
|
|
|
-- | Generate a random 16-char filepath without / in it by using base64url encoding.
|
|
|
|
randomStorePath :: IO FilePath
|
|
|
|
randomStorePath = B.unpack . B64U.encode <$> getRandomBytes 12
|
|
|
|
|
|
|
|
listRemoteHosts :: (ChatMonad m) => m ChatResponse
|
|
|
|
listRemoteHosts = do
|
|
|
|
stored <- withStore' getRemoteHosts
|
|
|
|
active <- chatReadVar remoteHostSessions
|
|
|
|
pure $ CRRemoteHostList $ do
|
|
|
|
RemoteHost {remoteHostId, storePath, displayName} <- stored
|
|
|
|
let sessionActive = M.member remoteHostId active
|
|
|
|
pure RemoteHostInfo {remoteHostId, storePath, displayName, sessionActive}
|
|
|
|
|
|
|
|
deleteRemoteHost :: (ChatMonad m) => RemoteHostId -> m ChatResponse
|
|
|
|
deleteRemoteHost remoteHostId = withRemoteHost remoteHostId $ \rh -> do
|
|
|
|
-- TODO: delete files
|
|
|
|
withStore' $ \db -> deleteRemoteHostRecord db remoteHostId
|
|
|
|
pure CRRemoteHostDeleted {remoteHostId}
|
2023-09-27 13:40:19 +03:00
|
|
|
|
2023-09-27 18:24:38 +03:00
|
|
|
processRemoteCommand :: (ChatMonad m) => RemoteHostSession -> (ByteString, ChatCommand) -> m ChatResponse
|
2023-10-07 16:23:24 +03:00
|
|
|
processRemoteCommand RemoteHostSessionStarting {} _ = pure . CRChatError Nothing . ChatError $ CEInternalError "sending remote commands before session started"
|
|
|
|
processRemoteCommand RemoteHostSessionStarted {ctrlClient} (s, cmd) = do
|
|
|
|
logDebug $ "processRemoteCommand: " <> T.pack (show s)
|
2023-09-27 11:41:02 +03:00
|
|
|
-- XXX: intercept and filter some commands
|
|
|
|
-- TODO: store missing files on remote host
|
2023-10-04 18:36:10 +03:00
|
|
|
relayCommand ctrlClient s
|
2023-09-27 11:41:02 +03:00
|
|
|
|
2023-10-04 18:36:10 +03:00
|
|
|
relayCommand :: (ChatMonad m) => HTTP2Client -> ByteString -> m ChatResponse
|
|
|
|
relayCommand http s =
|
2023-10-07 16:23:24 +03:00
|
|
|
postBytestring Nothing http "/send" mempty s >>= \case
|
|
|
|
Left e -> err $ "relayCommand/post: " <> show e
|
2023-09-27 18:24:38 +03:00
|
|
|
Right HTTP2.HTTP2Response {respBody = HTTP2Body {bodyHead}} -> do
|
2023-10-07 16:23:24 +03:00
|
|
|
logDebug $ "Got /send response: " <> T.pack (show bodyHead)
|
|
|
|
remoteChatResponse <- case J.eitherDecodeStrict bodyHead of -- XXX: large JSONs can overflow into buffered chunks
|
|
|
|
Left e -> err $ "relayCommand/decodeValue: " <> show e
|
|
|
|
Right json -> case J.fromJSON $ toTaggedJSON json of
|
|
|
|
J.Error e -> err $ "relayCommand/fromJSON: " <> show e
|
|
|
|
J.Success cr -> pure cr
|
2023-09-27 18:24:38 +03:00
|
|
|
case remoteChatResponse of
|
|
|
|
-- TODO: intercept file responses and fetch files when needed
|
|
|
|
-- XXX: is that even possible, to have a file response to a command?
|
|
|
|
_ -> pure remoteChatResponse
|
2023-09-27 11:41:02 +03:00
|
|
|
where
|
2023-10-07 16:23:24 +03:00
|
|
|
err = pure . CRChatError Nothing . ChatError . CEInternalError
|
|
|
|
toTaggedJSON :: J.Value -> J.Value
|
|
|
|
toTaggedJSON = id -- owsf2tagged TODO: get from RemoteHost
|
2023-09-27 11:41:02 +03:00
|
|
|
-- XXX: extract to http2 transport
|
2023-10-07 16:23:24 +03:00
|
|
|
postBytestring timeout' c path hs body = liftIO $ HTTP2.sendRequestDirect c req timeout'
|
2023-09-27 11:41:02 +03:00
|
|
|
where
|
|
|
|
req = HTTP2Client.requestBuilder "POST" path hs (Binary.fromByteString body)
|
|
|
|
|
2023-10-04 18:36:10 +03:00
|
|
|
-- | Convert swift single-field sum encoding into tagged/discriminator-field
|
2023-10-07 16:23:24 +03:00
|
|
|
owsf2tagged :: J.Value -> J.Value
|
|
|
|
owsf2tagged = \case
|
2023-10-04 18:36:10 +03:00
|
|
|
J.Object todo'convert -> J.Object todo'convert
|
|
|
|
skip -> skip
|
|
|
|
|
|
|
|
storeRemoteFile :: (ChatMonad m) => HTTP2Client -> FilePath -> m ChatResponse
|
|
|
|
storeRemoteFile http localFile = do
|
|
|
|
postFile Nothing http "/store" mempty localFile >>= \case
|
2023-09-29 14:56:56 +03:00
|
|
|
Left todo'err -> error "TODO: http2chatError"
|
2023-09-27 18:24:38 +03:00
|
|
|
Right HTTP2.HTTP2Response {response} -> case HTTP.statusCode <$> HTTP2Client.responseStatus response of
|
2023-09-27 11:41:02 +03:00
|
|
|
Just 200 -> pure $ CRCmdOk Nothing
|
2023-09-29 14:56:56 +03:00
|
|
|
todo'notOk -> error "TODO: http2chatError"
|
2023-09-27 11:41:02 +03:00
|
|
|
where
|
|
|
|
postFile timeout c path hs file = liftIO $ do
|
|
|
|
fileSize <- fromIntegral <$> getFileSize file
|
2023-10-07 16:23:24 +03:00
|
|
|
HTTP2.sendRequestDirect c (req fileSize) timeout
|
2023-09-27 11:41:02 +03:00
|
|
|
where
|
2023-10-07 16:23:24 +03:00
|
|
|
req size = HTTP2Client.requestFile "PUT" path hs (HTTP2Client.FileSpec file 0 size)
|
2023-09-27 11:41:02 +03:00
|
|
|
|
2023-10-04 18:36:10 +03:00
|
|
|
fetchRemoteFile :: (ChatMonad m) => HTTP2Client -> FilePath -> FileTransferId -> m ChatResponse
|
|
|
|
fetchRemoteFile http storePath remoteFileId = do
|
2023-10-07 16:23:24 +03:00
|
|
|
liftIO (HTTP2.sendRequestDirect http req Nothing) >>= \case
|
2023-09-27 11:41:02 +03:00
|
|
|
Left e -> error "TODO: http2chatError"
|
|
|
|
Right HTTP2.HTTP2Response {respBody} -> do
|
|
|
|
error "TODO: stream body into a local file" -- XXX: consult headers for a file name?
|
|
|
|
where
|
|
|
|
req = HTTP2Client.requestNoBody "GET" path mempty
|
|
|
|
path = "/fetch/" <> bshow remoteFileId
|
|
|
|
|
2023-10-07 16:23:24 +03:00
|
|
|
processControllerRequest :: forall m . (ChatMonad m) => (ByteString -> m ChatResponse) -> HTTP2.HTTP2Request -> m ()
|
|
|
|
processControllerRequest execChatCommand HTTP2.HTTP2Request {request, reqBody = HTTP2Body {bodyHead}, sendResponse} = do
|
|
|
|
logDebug $ "Remote controller request: " <> T.pack (show $ method <> " " <> path)
|
|
|
|
res <- tryChatError $ case (method, path) of
|
|
|
|
("GET", "/") -> getHello
|
|
|
|
("POST", "/send") -> sendCommand
|
|
|
|
("GET", "/recv") -> recvMessage
|
|
|
|
("PUT", "/store") -> storeFile
|
|
|
|
("GET", "/fetch") -> fetchFile
|
|
|
|
unexpected -> respondWith Status.badRequest400 $ "unexpected method/path: " <> Binary.putStringUtf8 (show unexpected)
|
|
|
|
case res of
|
|
|
|
Left e -> logError $ "Error handling remote controller request: (" <> tshow (method <> " " <> path) <> "): " <> tshow e
|
|
|
|
Right () -> logDebug $ "Remote controller request: " <> tshow (method <> " " <> path) <> " OK"
|
|
|
|
where
|
|
|
|
method = fromMaybe "" $ HTTP2Server.requestMethod request
|
|
|
|
path = fromMaybe "" $ HTTP2Server.requestPath request
|
|
|
|
getHello = respond "OK"
|
|
|
|
sendCommand = execChatCommand bodyHead >>= respondJSON
|
|
|
|
recvMessage = chatReadVar remoteCtrlSession >>= \case
|
|
|
|
Nothing -> respondWith Status.internalServerError500 "session not active"
|
|
|
|
Just rcs -> atomically (readTBQueue $ remoteOutputQ rcs) >>= respondJSON
|
|
|
|
storeFile = respondWith Status.notImplemented501 "TODO: storeFile"
|
|
|
|
fetchFile = respondWith Status.notImplemented501 "TODO: fetchFile"
|
|
|
|
|
|
|
|
respondJSON :: J.ToJSON a => a -> m ()
|
|
|
|
respondJSON = respond . Binary.fromLazyByteString . J.encode
|
|
|
|
|
|
|
|
respond = respondWith Status.ok200
|
|
|
|
respondWith status = liftIO . sendResponse . HTTP2Server.responseBuilder status []
|
2023-09-29 14:56:56 +03:00
|
|
|
|
|
|
|
-- * ChatRequest handlers
|
|
|
|
|
2023-10-07 16:23:24 +03:00
|
|
|
startRemoteCtrl :: (ChatMonad m) => (ByteString -> m ChatResponse) -> m ChatResponse
|
|
|
|
startRemoteCtrl execChatCommand =
|
2023-09-29 14:56:56 +03:00
|
|
|
chatReadVar remoteCtrlSession >>= \case
|
|
|
|
Just _busy -> throwError $ ChatErrorRemoteCtrl RCEBusy
|
|
|
|
Nothing -> do
|
2023-10-07 16:23:24 +03:00
|
|
|
size <- asks $ tbqSize . config
|
|
|
|
remoteOutputQ <- newTBQueueIO size
|
|
|
|
remoteNotifyQ <- newTBQueueIO size
|
2023-09-29 14:56:56 +03:00
|
|
|
discovered <- newTVarIO mempty
|
2023-10-04 18:36:10 +03:00
|
|
|
discoverer <- async $ discoverRemoteCtrls discovered
|
2023-10-07 16:23:24 +03:00
|
|
|
accepted <- newEmptyTMVarIO
|
2023-10-04 18:36:10 +03:00
|
|
|
supervisor <- async $ do
|
|
|
|
remoteCtrlId <- atomically (readTMVar accepted)
|
|
|
|
withRemoteCtrl remoteCtrlId $ \RemoteCtrl {displayName, fingerprint} -> do
|
|
|
|
source <- atomically $ TM.lookup fingerprint discovered >>= maybe retry pure
|
|
|
|
toView $ CRRemoteCtrlConnecting {remoteCtrlId, displayName}
|
|
|
|
atomically $ writeTVar discovered mempty -- flush unused sources
|
2023-10-07 16:23:24 +03:00
|
|
|
server <- async $ Discovery.connectRevHTTP2 source fingerprint (processControllerRequest execChatCommand)
|
2023-10-04 18:36:10 +03:00
|
|
|
chatModifyVar remoteCtrlSession $ fmap $ \s -> s {hostServer = Just server}
|
|
|
|
toView $ CRRemoteCtrlConnected {remoteCtrlId, displayName}
|
|
|
|
_ <- waitCatch server
|
|
|
|
chatWriteVar remoteCtrlSession Nothing
|
2023-10-07 16:23:24 +03:00
|
|
|
toView $ CRRemoteCtrlStopped Nothing
|
|
|
|
chatWriteVar remoteCtrlSession $ Just RemoteCtrlSession {discoverer, supervisor, hostServer = Nothing, discovered, accepted, remoteOutputQ, remoteNotifyQ}
|
2023-10-05 21:49:20 +03:00
|
|
|
pure $ CRRemoteCtrlStarted Nothing
|
2023-09-29 14:56:56 +03:00
|
|
|
|
2023-10-04 18:36:10 +03:00
|
|
|
discoverRemoteCtrls :: (ChatMonad m) => TM.TMap C.KeyHash TransportHost -> m ()
|
2023-10-07 16:23:24 +03:00
|
|
|
discoverRemoteCtrls discovered = Discovery.withListener go
|
2023-09-29 14:56:56 +03:00
|
|
|
where
|
|
|
|
go sock =
|
|
|
|
Discovery.recvAnnounce sock >>= \case
|
2023-10-07 16:23:24 +03:00
|
|
|
(SockAddrInet _sockPort sockAddr, invite) -> case strDecode invite of
|
2023-09-29 14:56:56 +03:00
|
|
|
Left _ -> go sock -- ignore malformed datagrams
|
|
|
|
Right fingerprint -> do
|
2023-10-07 16:23:24 +03:00
|
|
|
let addr = THIPv4 (hostAddressToTuple sockAddr)
|
|
|
|
ifM
|
|
|
|
(atomically $ TM.member fingerprint discovered)
|
|
|
|
(logDebug $ "Fingerprint announce already knwon: " <> T.pack (show (addr, invite)))
|
|
|
|
(do
|
|
|
|
logInfo $ "New fingerprint announce: " <> T.pack (show (addr, invite))
|
|
|
|
atomically $ TM.insert fingerprint addr discovered
|
|
|
|
)
|
2023-10-04 18:36:10 +03:00
|
|
|
withStore' (`getRemoteCtrlByFingerprint` fingerprint) >>= \case
|
2023-10-07 16:23:24 +03:00
|
|
|
Nothing -> toView $ CRRemoteCtrlAnnounce fingerprint -- unknown controller, ui "register" action required
|
2023-10-04 18:36:10 +03:00
|
|
|
Just found@RemoteCtrl {remoteCtrlId, accepted=storedChoice} -> case storedChoice of
|
2023-10-07 16:23:24 +03:00
|
|
|
Nothing -> toView $ CRRemoteCtrlFound found -- first-time controller, ui "accept" action required
|
2023-10-04 18:36:10 +03:00
|
|
|
Just False -> pure () -- skipping a rejected item
|
|
|
|
Just True -> chatReadVar remoteCtrlSession >>= \case
|
|
|
|
Nothing -> toView . CRChatError Nothing . ChatError $ CEInternalError "Remote host found without running a session"
|
|
|
|
Just RemoteCtrlSession {accepted} -> atomically $ void $ tryPutTMVar accepted remoteCtrlId -- previously accepted controller, connect automatically
|
2023-09-29 14:56:56 +03:00
|
|
|
_nonV4 -> go sock
|
|
|
|
|
2023-10-04 18:36:10 +03:00
|
|
|
registerRemoteCtrl :: (ChatMonad m) => RemoteCtrlOOB -> m ChatResponse
|
|
|
|
registerRemoteCtrl RemoteCtrlOOB {caFingerprint} = do
|
|
|
|
let displayName = "TODO" -- maybe include into OOB data
|
|
|
|
remoteCtrlId <- withStore' $ \db -> insertRemoteCtrl db displayName caFingerprint
|
|
|
|
pure $ CRRemoteCtrlRegistered {remoteCtrlId}
|
2023-09-29 14:56:56 +03:00
|
|
|
|
2023-10-04 18:36:10 +03:00
|
|
|
listRemoteCtrls :: (ChatMonad m) => m ChatResponse
|
|
|
|
listRemoteCtrls = do
|
|
|
|
stored <- withStore' getRemoteCtrls
|
|
|
|
active <-
|
|
|
|
chatReadVar remoteCtrlSession >>= \case
|
|
|
|
Nothing -> pure Nothing
|
|
|
|
Just RemoteCtrlSession {accepted} -> atomically (tryReadTMVar accepted)
|
|
|
|
pure $ CRRemoteCtrlList $ do
|
|
|
|
RemoteCtrl {remoteCtrlId, displayName} <- stored
|
|
|
|
let sessionActive = active == Just remoteCtrlId
|
|
|
|
pure RemoteCtrlInfo {remoteCtrlId, displayName, sessionActive}
|
|
|
|
|
|
|
|
acceptRemoteCtrl :: (ChatMonad m) => RemoteCtrlId -> m ChatResponse
|
|
|
|
acceptRemoteCtrl remoteCtrlId = do
|
|
|
|
withStore' $ \db -> markRemoteCtrlResolution db remoteCtrlId True
|
2023-09-29 14:56:56 +03:00
|
|
|
chatReadVar remoteCtrlSession >>= \case
|
|
|
|
Nothing -> throwError $ ChatErrorRemoteCtrl RCEInactive
|
2023-10-04 18:36:10 +03:00
|
|
|
Just RemoteCtrlSession {accepted} -> atomically . void $ tryPutTMVar accepted remoteCtrlId -- the remote host can now proceed with connection
|
|
|
|
pure $ CRRemoteCtrlAccepted {remoteCtrlId}
|
2023-09-29 14:56:56 +03:00
|
|
|
|
|
|
|
rejectRemoteCtrl :: (ChatMonad m) => RemoteCtrlId -> m ChatResponse
|
2023-10-04 18:36:10 +03:00
|
|
|
rejectRemoteCtrl remoteCtrlId = do
|
|
|
|
withStore' $ \db -> markRemoteCtrlResolution db remoteCtrlId False
|
2023-09-29 14:56:56 +03:00
|
|
|
chatReadVar remoteCtrlSession >>= \case
|
|
|
|
Nothing -> throwError $ ChatErrorRemoteCtrl RCEInactive
|
2023-10-04 18:36:10 +03:00
|
|
|
Just RemoteCtrlSession {discoverer, supervisor} -> do
|
|
|
|
cancel discoverer
|
|
|
|
cancel supervisor
|
|
|
|
pure $ CRRemoteCtrlRejected {remoteCtrlId}
|
2023-09-29 14:56:56 +03:00
|
|
|
|
2023-10-07 16:23:24 +03:00
|
|
|
stopRemoteCtrl :: (ChatMonad m) => m ChatResponse
|
|
|
|
stopRemoteCtrl =
|
2023-09-29 14:56:56 +03:00
|
|
|
chatReadVar remoteCtrlSession >>= \case
|
|
|
|
Nothing -> throwError $ ChatErrorRemoteCtrl RCEInactive
|
2023-10-07 16:23:24 +03:00
|
|
|
Just rcs -> do
|
|
|
|
cancelRemoteCtrlSession rcs $ do
|
|
|
|
chatWriteVar remoteCtrlSession Nothing
|
|
|
|
toView $ CRRemoteCtrlStopped Nothing
|
2023-10-04 18:36:10 +03:00
|
|
|
pure $ CRCmdOk Nothing
|
2023-09-29 14:56:56 +03:00
|
|
|
|
2023-10-07 16:23:24 +03:00
|
|
|
cancelRemoteCtrlSession_ :: MonadUnliftIO m => RemoteCtrlSession -> m ()
|
|
|
|
cancelRemoteCtrlSession_ rcs = cancelRemoteCtrlSession rcs $ pure ()
|
|
|
|
|
|
|
|
cancelRemoteCtrlSession :: MonadUnliftIO m => RemoteCtrlSession -> m () -> m ()
|
|
|
|
cancelRemoteCtrlSession RemoteCtrlSession {discoverer, supervisor, hostServer} cleanup = do
|
|
|
|
cancel discoverer -- may be gone by now
|
|
|
|
case hostServer of
|
|
|
|
Just host -> cancel host -- supervisor will clean up
|
|
|
|
Nothing -> do
|
|
|
|
cancel supervisor -- supervisor is blocked until session progresses
|
|
|
|
cleanup
|
|
|
|
|
2023-10-04 18:36:10 +03:00
|
|
|
deleteRemoteCtrl :: (ChatMonad m) => RemoteCtrlId -> m ChatResponse
|
|
|
|
deleteRemoteCtrl remoteCtrlId =
|
2023-09-29 14:56:56 +03:00
|
|
|
chatReadVar remoteCtrlSession >>= \case
|
|
|
|
Nothing -> do
|
2023-10-04 18:36:10 +03:00
|
|
|
withStore' $ \db -> deleteRemoteCtrlRecord db remoteCtrlId
|
|
|
|
pure $ CRRemoteCtrlDeleted {remoteCtrlId}
|
2023-09-29 14:56:56 +03:00
|
|
|
Just _ -> throwError $ ChatErrorRemoteCtrl RCEBusy
|
2023-10-04 18:36:10 +03:00
|
|
|
|
|
|
|
withRemoteCtrl :: (ChatMonad m) => RemoteCtrlId -> (RemoteCtrl -> m a) -> m a
|
|
|
|
withRemoteCtrl remoteCtrlId action =
|
|
|
|
withStore' (`getRemoteCtrl` remoteCtrlId) >>= \case
|
|
|
|
Nothing -> throwError $ ChatErrorRemoteCtrl RCEMissing {remoteCtrlId}
|
|
|
|
Just rc -> action rc
|