{-# LANGUAGE ConstraintKinds #-}
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE DuplicateRecordFields #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE NumericUnderscores #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TupleSections #-}

-- |
-- Module      : Simplex.Messaging.Agent
-- Copyright   : (c) simplex.chat
-- License     : AGPL-3
--
-- Maintainer  : chat@simplex.chat
-- Stability   : experimental
-- Portability : non-portable
--
-- This module defines SMP protocol agent with SQLite persistence.
--
-- See https://github.com/simplex-chat/simplexmq/blob/master/protocol/agent-protocol.md
module Simplex.Messaging.Agent
  ( -- * SMP agent over TCP
    runSMPAgent,
    runSMPAgentBlocking,

    -- * queue-based SMP agent
    getAgentClient,
    runAgentClient,

    -- * SMP agent functional API
    AgentClient (..),
    AgentMonad,
    AgentErrorMonad,
    getSMPAgentClient,
    disconnectAgentClient, -- used in tests
    withAgentLock,
    createConnection,
    joinConnection,
    acceptConnection,
    subscribeConnection,
    sendMessage,
    ackMessage,
    suspendConnection,
    deleteConnection,
  )
where

import Control.Concurrent.STM (stateTVar)
import Control.Logger.Simple (logInfo, showText)
import Control.Monad.Except
import Control.Monad.IO.Unlift (MonadUnliftIO)
import Control.Monad.Reader
import Crypto.Random (MonadRandom)
import Data.Bifunctor (second)
import Data.ByteString.Char8 (ByteString)
import qualified Data.ByteString.Char8 as B
import Data.Composition ((.:), (.:.))
import Data.Functor (($>))
import Data.List.NonEmpty (NonEmpty (..))
import qualified Data.List.NonEmpty as L
import Data.Map.Strict (Map)
import qualified Data.Map.Strict as M
import Data.Maybe (isJust)
import qualified Data.Text as T
import Data.Text.Encoding (decodeUtf8)
import Data.Time.Clock
import Database.SQLite.Simple (SQLError)
import Simplex.Messaging.Agent.Client
import Simplex.Messaging.Agent.Env.SQLite
import Simplex.Messaging.Agent.Protocol
import Simplex.Messaging.Agent.RetryInterval
import Simplex.Messaging.Agent.Store
import Simplex.Messaging.Agent.Store.SQLite (SQLiteStore)
import Simplex.Messaging.Client (SMPServerTransmission)
import qualified Simplex.Messaging.Crypto as C
import Simplex.Messaging.Protocol (MsgBody, SenderPublicKey)
import qualified Simplex.Messaging.Protocol as SMP
import Simplex.Messaging.Transport (ATransport (..), TProxy, Transport (..), runTransportServer)
import Simplex.Messaging.Util (bshow, tryError)
import System.Random (randomR)
import UnliftIO.Async (Async, async, race_)
import qualified UnliftIO.Exception as E
import UnliftIO.STM

-- | Runs an SMP agent as a TCP service using passed configuration.
--
-- See a full agent executable here: https://github.com/simplex-chat/simplexmq/blob/master/apps/smp-agent/Main.hs
runSMPAgent :: (MonadRandom m, MonadUnliftIO m) => ATransport -> AgentConfig -> m ()
runSMPAgent :: ATransport -> AgentConfig -> m ()
runSMPAgent ATransport
t AgentConfig
cfg = do
  TMVar Bool
started <- m (TMVar Bool)
forall (m :: * -> *) a. MonadIO m => m (TMVar a)
newEmptyTMVarIO
  ATransport -> TMVar Bool -> AgentConfig -> m ()
forall (m :: * -> *).
(MonadRandom m, MonadUnliftIO m) =>
ATransport -> TMVar Bool -> AgentConfig -> m ()
runSMPAgentBlocking ATransport
t TMVar Bool
started AgentConfig
cfg

-- | Runs an SMP agent as a TCP service using passed configuration with signalling.
--
-- This function uses passed TMVar to signal when the server is ready to accept TCP requests (True)
-- and when it is disconnected from the TCP socket once the server thread is killed (False).
runSMPAgentBlocking :: (MonadRandom m, MonadUnliftIO m) => ATransport -> TMVar Bool -> AgentConfig -> m ()
runSMPAgentBlocking :: ATransport -> TMVar Bool -> AgentConfig -> m ()
runSMPAgentBlocking (ATransport TProxy c
t) TMVar Bool
started cfg :: AgentConfig
cfg@AgentConfig {ServiceName
$sel:tcpPort:AgentConfig :: AgentConfig -> ServiceName
tcpPort :: ServiceName
tcpPort} = ReaderT Env m () -> Env -> m ()
forall r (m :: * -> *) a. ReaderT r m a -> r -> m a
runReaderT (TProxy c -> ReaderT Env m ()
forall c (m' :: * -> *).
(Transport c, MonadUnliftIO m', MonadReader Env m') =>
TProxy c -> m' ()
smpAgent TProxy c
t) (Env -> m ()) -> m Env -> m ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< AgentConfig -> m Env
forall (m :: * -> *).
(MonadUnliftIO m, MonadRandom m) =>
AgentConfig -> m Env
newSMPAgentEnv AgentConfig
cfg
  where
    smpAgent :: forall c m'. (Transport c, MonadUnliftIO m', MonadReader Env m') => TProxy c -> m' ()
    smpAgent :: TProxy c -> m' ()
smpAgent TProxy c
_ = TMVar Bool -> ServiceName -> (c -> m' ()) -> m' ()
forall c (m :: * -> *).
(Transport c, MonadUnliftIO m) =>
TMVar Bool -> ServiceName -> (c -> m ()) -> m ()
runTransportServer TMVar Bool
started ServiceName
tcpPort ((c -> m' ()) -> m' ()) -> (c -> m' ()) -> m' ()
forall a b. (a -> b) -> a -> b
$ \(c
h :: c) -> do
      IO () -> m' ()
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m' ()) -> IO () -> m' ()
forall a b. (a -> b) -> a -> b
$ c -> ByteString -> IO ()
forall c. Transport c => c -> ByteString -> IO ()
putLn c
h ByteString
"Welcome to SMP v0.4.1 agent"
      AgentClient
c <- m' AgentClient
forall (m :: * -> *).
(MonadUnliftIO m, MonadReader Env m) =>
m AgentClient
getAgentClient
      AgentClient -> Bool -> m' ()
forall (m :: * -> *).
MonadUnliftIO m =>
AgentClient -> Bool -> m ()
logConnection AgentClient
c Bool
True
      m' () -> m' () -> m' ()
forall (m :: * -> *) a b. MonadUnliftIO m => m a -> m b -> m ()
race_ (c -> AgentClient -> m' ()
forall c (m :: * -> *).
(Transport c, MonadUnliftIO m) =>
c -> AgentClient -> m ()
connectClient c
h AgentClient
c) (AgentClient -> m' ()
forall (m :: * -> *).
(MonadUnliftIO m, MonadReader Env m) =>
AgentClient -> m ()
runAgentClient AgentClient
c)
        m' () -> m' () -> m' ()
forall (m :: * -> *) a b. MonadUnliftIO m => m a -> m b -> m a
`E.finally` AgentClient -> m' ()
forall (m :: * -> *). MonadUnliftIO m => AgentClient -> m ()
disconnectAgentClient AgentClient
c

-- | Creates an SMP agent client instance
getSMPAgentClient :: (MonadRandom m, MonadUnliftIO m) => AgentConfig -> m AgentClient
getSMPAgentClient :: AgentConfig -> m AgentClient
getSMPAgentClient AgentConfig
cfg = AgentConfig -> m Env
forall (m :: * -> *).
(MonadUnliftIO m, MonadRandom m) =>
AgentConfig -> m Env
newSMPAgentEnv AgentConfig
cfg m Env -> (Env -> m AgentClient) -> m AgentClient
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= ReaderT Env m AgentClient -> Env -> m AgentClient
forall r (m :: * -> *) a. ReaderT r m a -> r -> m a
runReaderT ReaderT Env m AgentClient
runAgent
  where
    runAgent :: ReaderT Env m AgentClient
runAgent = do
      AgentClient
c <- ReaderT Env m AgentClient
forall (m :: * -> *).
(MonadUnliftIO m, MonadReader Env m) =>
m AgentClient
getAgentClient
      Async ()
action <- ReaderT Env m () -> ReaderT Env m (Async ())
forall (m :: * -> *) a. MonadUnliftIO m => m a -> m (Async a)
async (ReaderT Env m () -> ReaderT Env m (Async ()))
-> ReaderT Env m () -> ReaderT Env m (Async ())
forall a b. (a -> b) -> a -> b
$ AgentClient -> ReaderT Env m ()
forall (m :: * -> *).
(MonadUnliftIO m, MonadReader Env m) =>
AgentClient -> m ()
subscriber AgentClient
c ReaderT Env m () -> ReaderT Env m () -> ReaderT Env m ()
forall (m :: * -> *) a b. MonadUnliftIO m => m a -> m b -> m a
`E.finally` AgentClient -> ReaderT Env m ()
forall (m :: * -> *). MonadUnliftIO m => AgentClient -> m ()
disconnectAgentClient AgentClient
c
      AgentClient -> ReaderT Env m AgentClient
forall (f :: * -> *) a. Applicative f => a -> f a
pure AgentClient
c {$sel:smpSubscriber:AgentClient :: Async ()
smpSubscriber = Async ()
action}

disconnectAgentClient :: MonadUnliftIO m => AgentClient -> m ()
disconnectAgentClient :: AgentClient -> m ()
disconnectAgentClient AgentClient
c = AgentClient -> m ()
forall (m :: * -> *). MonadUnliftIO m => AgentClient -> m ()
closeAgentClient AgentClient
c m () -> m () -> m ()
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> AgentClient -> Bool -> m ()
forall (m :: * -> *).
MonadUnliftIO m =>
AgentClient -> Bool -> m ()
logConnection AgentClient
c Bool
False

-- |
type AgentErrorMonad m = (MonadUnliftIO m, MonadError AgentErrorType m)

-- | Create SMP agent connection (NEW command)
createConnection :: AgentErrorMonad m => AgentClient -> m (ConnId, SMPQueueInfo)
createConnection :: AgentClient -> m (ByteString, SMPQueueInfo)
createConnection AgentClient
c = AgentClient
-> ReaderT Env m (ByteString, SMPQueueInfo)
-> m (ByteString, SMPQueueInfo)
forall (m :: * -> *) a. AgentClient -> ReaderT Env m a -> m a
withAgentEnv AgentClient
c (ReaderT Env m (ByteString, SMPQueueInfo)
 -> m (ByteString, SMPQueueInfo))
-> ReaderT Env m (ByteString, SMPQueueInfo)
-> m (ByteString, SMPQueueInfo)
forall a b. (a -> b) -> a -> b
$ AgentClient
-> ByteString -> ReaderT Env m (ByteString, SMPQueueInfo)
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m (ByteString, SMPQueueInfo)
newConn AgentClient
c ByteString
""

-- | Join SMP agent connection (JOIN command)
joinConnection :: AgentErrorMonad m => AgentClient -> SMPQueueInfo -> ConnInfo -> m ConnId
joinConnection :: AgentClient -> SMPQueueInfo -> ByteString -> m ByteString
joinConnection AgentClient
c = AgentClient -> ReaderT Env m ByteString -> m ByteString
forall (m :: * -> *) a. AgentClient -> ReaderT Env m a -> m a
withAgentEnv AgentClient
c (ReaderT Env m ByteString -> m ByteString)
-> (SMPQueueInfo -> ByteString -> ReaderT Env m ByteString)
-> SMPQueueInfo
-> ByteString
-> m ByteString
forall c d a b. (c -> d) -> (a -> b -> c) -> a -> b -> d
.: AgentClient
-> ByteString
-> SMPQueueInfo
-> ByteString
-> ReaderT Env m ByteString
forall (m :: * -> *).
AgentMonad m =>
AgentClient
-> ByteString -> SMPQueueInfo -> ByteString -> m ByteString
joinConn AgentClient
c ByteString
""

-- | Approve confirmation (LET command)
acceptConnection :: AgentErrorMonad m => AgentClient -> ConnId -> ConfirmationId -> ConnInfo -> m ()
acceptConnection :: AgentClient -> ByteString -> ByteString -> ByteString -> m ()
acceptConnection AgentClient
c = AgentClient -> ReaderT Env m () -> m ()
forall (m :: * -> *) a. AgentClient -> ReaderT Env m a -> m a
withAgentEnv AgentClient
c (ReaderT Env m () -> m ())
-> (ByteString -> ByteString -> ByteString -> ReaderT Env m ())
-> ByteString
-> ByteString
-> ByteString
-> m ()
forall d e a b c.
(d -> e) -> (a -> b -> c -> d) -> a -> b -> c -> e
.:. AgentClient
-> ByteString -> ByteString -> ByteString -> ReaderT Env m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> ByteString -> ByteString -> m ()
acceptConnection' AgentClient
c

-- | Subscribe to receive connection messages (SUB command)
subscribeConnection :: AgentErrorMonad m => AgentClient -> ConnId -> m ()
subscribeConnection :: AgentClient -> ByteString -> m ()
subscribeConnection AgentClient
c = AgentClient -> ReaderT Env m () -> m ()
forall (m :: * -> *) a. AgentClient -> ReaderT Env m a -> m a
withAgentEnv AgentClient
c (ReaderT Env m () -> m ())
-> (ByteString -> ReaderT Env m ()) -> ByteString -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. AgentClient -> ByteString -> ReaderT Env m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
subscribeConnection' AgentClient
c

-- | Send message to the connection (SEND command)
sendMessage :: AgentErrorMonad m => AgentClient -> ConnId -> MsgBody -> m AgentMsgId
sendMessage :: AgentClient -> ByteString -> ByteString -> m AgentMsgId
sendMessage AgentClient
c = AgentClient -> ReaderT Env m AgentMsgId -> m AgentMsgId
forall (m :: * -> *) a. AgentClient -> ReaderT Env m a -> m a
withAgentEnv AgentClient
c (ReaderT Env m AgentMsgId -> m AgentMsgId)
-> (ByteString -> ByteString -> ReaderT Env m AgentMsgId)
-> ByteString
-> ByteString
-> m AgentMsgId
forall c d a b. (c -> d) -> (a -> b -> c) -> a -> b -> d
.: AgentClient -> ByteString -> ByteString -> ReaderT Env m AgentMsgId
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> ByteString -> m AgentMsgId
sendMessage' AgentClient
c

ackMessage :: AgentErrorMonad m => AgentClient -> ConnId -> AgentMsgId -> m ()
ackMessage :: AgentClient -> ByteString -> AgentMsgId -> m ()
ackMessage AgentClient
c = AgentClient -> ReaderT Env m () -> m ()
forall (m :: * -> *) a. AgentClient -> ReaderT Env m a -> m a
withAgentEnv AgentClient
c (ReaderT Env m () -> m ())
-> (ByteString -> AgentMsgId -> ReaderT Env m ())
-> ByteString
-> AgentMsgId
-> m ()
forall c d a b. (c -> d) -> (a -> b -> c) -> a -> b -> d
.: AgentClient -> ByteString -> AgentMsgId -> ReaderT Env m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> AgentMsgId -> m ()
ackMessage' AgentClient
c

-- | Suspend SMP agent connection (OFF command)
suspendConnection :: AgentErrorMonad m => AgentClient -> ConnId -> m ()
suspendConnection :: AgentClient -> ByteString -> m ()
suspendConnection AgentClient
c = AgentClient -> ReaderT Env m () -> m ()
forall (m :: * -> *) a. AgentClient -> ReaderT Env m a -> m a
withAgentEnv AgentClient
c (ReaderT Env m () -> m ())
-> (ByteString -> ReaderT Env m ()) -> ByteString -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. AgentClient -> ByteString -> ReaderT Env m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
suspendConnection' AgentClient
c

-- | Delete SMP agent connection (DEL command)
deleteConnection :: AgentErrorMonad m => AgentClient -> ConnId -> m ()
deleteConnection :: AgentClient -> ByteString -> m ()
deleteConnection AgentClient
c = AgentClient -> ReaderT Env m () -> m ()
forall (m :: * -> *) a. AgentClient -> ReaderT Env m a -> m a
withAgentEnv AgentClient
c (ReaderT Env m () -> m ())
-> (ByteString -> ReaderT Env m ()) -> ByteString -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. AgentClient -> ByteString -> ReaderT Env m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
deleteConnection' AgentClient
c

withAgentEnv :: AgentClient -> ReaderT Env m a -> m a
withAgentEnv :: AgentClient -> ReaderT Env m a -> m a
withAgentEnv AgentClient
c = (ReaderT Env m a -> Env -> m a
forall r (m :: * -> *) a. ReaderT r m a -> r -> m a
`runReaderT` AgentClient -> Env
agentEnv AgentClient
c)

-- withAgentClient :: AgentErrorMonad m => AgentClient -> ReaderT Env m a -> m a
-- withAgentClient c = withAgentLock c . withAgentEnv c

-- | Creates an SMP agent client instance that receives commands and sends responses via 'TBQueue's.
getAgentClient :: (MonadUnliftIO m, MonadReader Env m) => m AgentClient
getAgentClient :: m AgentClient
getAgentClient = m Env
forall r (m :: * -> *). MonadReader r m => m r
ask m Env -> (Env -> m AgentClient) -> m AgentClient
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= STM AgentClient -> m AgentClient
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM AgentClient -> m AgentClient)
-> (Env -> STM AgentClient) -> Env -> m AgentClient
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Env -> STM AgentClient
newAgentClient

connectClient :: Transport c => MonadUnliftIO m => c -> AgentClient -> m ()
connectClient :: c -> AgentClient -> m ()
connectClient c
h AgentClient
c = m () -> m () -> m ()
forall (m :: * -> *) a b. MonadUnliftIO m => m a -> m b -> m ()
race_ (c -> AgentClient -> m ()
forall c (m :: * -> *).
(Transport c, MonadUnliftIO m) =>
c -> AgentClient -> m ()
send c
h AgentClient
c) (c -> AgentClient -> m ()
forall c (m :: * -> *).
(Transport c, MonadUnliftIO m) =>
c -> AgentClient -> m ()
receive c
h AgentClient
c)

logConnection :: MonadUnliftIO m => AgentClient -> Bool -> m ()
logConnection :: AgentClient -> Bool -> m ()
logConnection AgentClient
c Bool
connected =
  let event :: Text
event = if Bool
connected then Text
"connected to" else Text
"disconnected from"
   in Text -> m ()
forall (m :: * -> *).
(?callStack::CallStack, MonadIO m) =>
Text -> m ()
logInfo (Text -> m ()) -> Text -> m ()
forall a b. (a -> b) -> a -> b
$ [Text] -> Text
T.unwords [Text
"client", Int -> Text
forall a. Show a => a -> Text
showText (AgentClient -> Int
clientId AgentClient
c), Text
event, Text
"Agent"]

-- | Runs an SMP agent instance that receives commands and sends responses via 'TBQueue's.
runAgentClient :: (MonadUnliftIO m, MonadReader Env m) => AgentClient -> m ()
runAgentClient :: AgentClient -> m ()
runAgentClient AgentClient
c = m () -> m () -> m ()
forall (m :: * -> *) a b. MonadUnliftIO m => m a -> m b -> m ()
race_ (AgentClient -> m ()
forall (m :: * -> *).
(MonadUnliftIO m, MonadReader Env m) =>
AgentClient -> m ()
subscriber AgentClient
c) (AgentClient -> m ()
forall (m :: * -> *).
(MonadUnliftIO m, MonadReader Env m) =>
AgentClient -> m ()
client AgentClient
c)

receive :: forall c m. (Transport c, MonadUnliftIO m) => c -> AgentClient -> m ()
receive :: c -> AgentClient -> m ()
receive c
h c :: AgentClient
c@AgentClient {TBQueue (ATransmission 'Client)
$sel:rcvQ:AgentClient :: AgentClient -> TBQueue (ATransmission 'Client)
rcvQ :: TBQueue (ATransmission 'Client)
rcvQ, TBQueue (ATransmission 'Agent)
$sel:subQ:AgentClient :: AgentClient -> TBQueue (ATransmission 'Agent)
subQ :: TBQueue (ATransmission 'Agent)
subQ} = m () -> m ()
forall (f :: * -> *) a b. Applicative f => f a -> f b
forever (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
  (ByteString
corrId, ByteString
connId, Either AgentErrorType (ACommand 'Client)
cmdOrErr) <- SAParty 'Client
-> c
-> m (ByteString, ByteString,
      Either AgentErrorType (ACommand 'Client))
forall c (m :: * -> *) (p :: AParty).
(Transport c, MonadIO m) =>
SAParty p -> c -> m (ATransmissionOrError p)
tGet SAParty 'Client
SClient c
h
  case Either AgentErrorType (ACommand 'Client)
cmdOrErr of
    Right ACommand 'Client
cmd -> TBQueue (ATransmission 'Client) -> ATransmission 'Client -> m ()
forall (p :: AParty).
TBQueue (ATransmission p) -> ATransmission p -> m ()
write TBQueue (ATransmission 'Client)
rcvQ (ByteString
corrId, ByteString
connId, ACommand 'Client
cmd)
    Left AgentErrorType
e -> TBQueue (ATransmission 'Agent) -> ATransmission 'Agent -> m ()
forall (p :: AParty).
TBQueue (ATransmission p) -> ATransmission p -> m ()
write TBQueue (ATransmission 'Agent)
subQ (ByteString
corrId, ByteString
connId, AgentErrorType -> ACommand 'Agent
ERR AgentErrorType
e)
  where
    write :: TBQueue (ATransmission p) -> ATransmission p -> m ()
    write :: TBQueue (ATransmission p) -> ATransmission p -> m ()
write TBQueue (ATransmission p)
q ATransmission p
t = do
      AgentClient -> ByteString -> ATransmission p -> m ()
forall (m :: * -> *) (a :: AParty).
MonadUnliftIO m =>
AgentClient -> ByteString -> ATransmission a -> m ()
logClient AgentClient
c ByteString
"-->" ATransmission p
t
      STM () -> m ()
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM () -> m ()) -> STM () -> m ()
forall a b. (a -> b) -> a -> b
$ TBQueue (ATransmission p) -> ATransmission p -> STM ()
forall a. TBQueue a -> a -> STM ()
writeTBQueue TBQueue (ATransmission p)
q ATransmission p
t

send :: (Transport c, MonadUnliftIO m) => c -> AgentClient -> m ()
send :: c -> AgentClient -> m ()
send c
h c :: AgentClient
c@AgentClient {TBQueue (ATransmission 'Agent)
subQ :: TBQueue (ATransmission 'Agent)
$sel:subQ:AgentClient :: AgentClient -> TBQueue (ATransmission 'Agent)
subQ} = m () -> m ()
forall (f :: * -> *) a b. Applicative f => f a -> f b
forever (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
  ATransmission 'Agent
t <- STM (ATransmission 'Agent) -> m (ATransmission 'Agent)
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM (ATransmission 'Agent) -> m (ATransmission 'Agent))
-> STM (ATransmission 'Agent) -> m (ATransmission 'Agent)
forall a b. (a -> b) -> a -> b
$ TBQueue (ATransmission 'Agent) -> STM (ATransmission 'Agent)
forall a. TBQueue a -> STM a
readTBQueue TBQueue (ATransmission 'Agent)
subQ
  c -> ATransmission 'Agent -> m ()
forall c (m :: * -> *) (p :: AParty).
(Transport c, MonadIO m) =>
c -> ATransmission p -> m ()
tPut c
h ATransmission 'Agent
t
  AgentClient -> ByteString -> ATransmission 'Agent -> m ()
forall (m :: * -> *) (a :: AParty).
MonadUnliftIO m =>
AgentClient -> ByteString -> ATransmission a -> m ()
logClient AgentClient
c ByteString
"<--" ATransmission 'Agent
t

logClient :: MonadUnliftIO m => AgentClient -> ByteString -> ATransmission a -> m ()
logClient :: AgentClient -> ByteString -> ATransmission a -> m ()
logClient AgentClient {Int
clientId :: Int
$sel:clientId:AgentClient :: AgentClient -> Int
clientId} ByteString
dir (ByteString
corrId, ByteString
connId, ACommand a
cmd) = do
  Text -> m ()
forall (m :: * -> *).
(?callStack::CallStack, MonadIO m) =>
Text -> m ()
logInfo (Text -> m ()) -> (ByteString -> Text) -> ByteString -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> Text
decodeUtf8 (ByteString -> m ()) -> ByteString -> m ()
forall a b. (a -> b) -> a -> b
$ [ByteString] -> ByteString
B.unwords [Int -> ByteString
forall a. Show a => a -> ByteString
bshow Int
clientId, ByteString
dir, ByteString
"A :", ByteString
corrId, ByteString
connId, (Char -> Bool) -> ByteString -> ByteString
B.takeWhile (Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/= Char
' ') (ByteString -> ByteString) -> ByteString -> ByteString
forall a b. (a -> b) -> a -> b
$ ACommand a -> ByteString
forall (p :: AParty). ACommand p -> ByteString
serializeCommand ACommand a
cmd]

client :: forall m. (MonadUnliftIO m, MonadReader Env m) => AgentClient -> m ()
client :: AgentClient -> m ()
client c :: AgentClient
c@AgentClient {TBQueue (ATransmission 'Client)
rcvQ :: TBQueue (ATransmission 'Client)
$sel:rcvQ:AgentClient :: AgentClient -> TBQueue (ATransmission 'Client)
rcvQ, TBQueue (ATransmission 'Agent)
subQ :: TBQueue (ATransmission 'Agent)
$sel:subQ:AgentClient :: AgentClient -> TBQueue (ATransmission 'Agent)
subQ} = m () -> m ()
forall (f :: * -> *) a b. Applicative f => f a -> f b
forever (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
  (ByteString
corrId, ByteString
connId, ACommand 'Client
cmd) <- STM (ATransmission 'Client) -> m (ATransmission 'Client)
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM (ATransmission 'Client) -> m (ATransmission 'Client))
-> STM (ATransmission 'Client) -> m (ATransmission 'Client)
forall a b. (a -> b) -> a -> b
$ TBQueue (ATransmission 'Client) -> STM (ATransmission 'Client)
forall a. TBQueue a -> STM a
readTBQueue TBQueue (ATransmission 'Client)
rcvQ
  AgentClient
-> m (Either AgentErrorType (ByteString, ACommand 'Agent))
-> m (Either AgentErrorType (ByteString, ACommand 'Agent))
forall (m :: * -> *) a.
MonadUnliftIO m =>
AgentClient -> m a -> m a
withAgentLock AgentClient
c (ExceptT AgentErrorType m (ByteString, ACommand 'Agent)
-> m (Either AgentErrorType (ByteString, ACommand 'Agent))
forall e (m :: * -> *) a. ExceptT e m a -> m (Either e a)
runExceptT (ExceptT AgentErrorType m (ByteString, ACommand 'Agent)
 -> m (Either AgentErrorType (ByteString, ACommand 'Agent)))
-> ExceptT AgentErrorType m (ByteString, ACommand 'Agent)
-> m (Either AgentErrorType (ByteString, ACommand 'Agent))
forall a b. (a -> b) -> a -> b
$ AgentClient
-> (ByteString, ACommand 'Client)
-> ExceptT AgentErrorType m (ByteString, ACommand 'Agent)
forall (m :: * -> *).
AgentMonad m =>
AgentClient
-> (ByteString, ACommand 'Client)
-> m (ByteString, ACommand 'Agent)
processCommand AgentClient
c (ByteString
connId, ACommand 'Client
cmd))
    m (Either AgentErrorType (ByteString, ACommand 'Agent))
-> (Either AgentErrorType (ByteString, ACommand 'Agent) -> m ())
-> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= STM () -> m ()
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM () -> m ())
-> (Either AgentErrorType (ByteString, ACommand 'Agent) -> STM ())
-> Either AgentErrorType (ByteString, ACommand 'Agent)
-> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TBQueue (ATransmission 'Agent) -> ATransmission 'Agent -> STM ()
forall a. TBQueue a -> a -> STM ()
writeTBQueue TBQueue (ATransmission 'Agent)
subQ (ATransmission 'Agent -> STM ())
-> (Either AgentErrorType (ByteString, ACommand 'Agent)
    -> ATransmission 'Agent)
-> Either AgentErrorType (ByteString, ACommand 'Agent)
-> STM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. \case
      Left AgentErrorType
e -> (ByteString
corrId, ByteString
connId, AgentErrorType -> ACommand 'Agent
ERR AgentErrorType
e)
      Right (ByteString
connId', ACommand 'Agent
resp) -> (ByteString
corrId, ByteString
connId', ACommand 'Agent
resp)

withStore ::
  AgentMonad m =>
  (forall m'. (MonadUnliftIO m', MonadError StoreError m') => SQLiteStore -> m' a) ->
  m a
withStore :: (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore forall (m' :: * -> *).
(MonadUnliftIO m', MonadError StoreError m') =>
SQLiteStore -> m' a
action = do
  SQLiteStore
st <- (Env -> SQLiteStore) -> m SQLiteStore
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks Env -> SQLiteStore
store
  ExceptT StoreError m a -> m (Either StoreError a)
forall e (m :: * -> *) a. ExceptT e m a -> m (Either e a)
runExceptT (SQLiteStore -> ExceptT StoreError m a
forall (m' :: * -> *).
(MonadUnliftIO m', MonadError StoreError m') =>
SQLiteStore -> m' a
action SQLiteStore
st ExceptT StoreError m a
-> (SQLError -> ExceptT StoreError m a) -> ExceptT StoreError m a
forall (m :: * -> *) e a.
(MonadUnliftIO m, Exception e) =>
m a -> (e -> m a) -> m a
`E.catch` SQLError -> ExceptT StoreError m a
forall (m' :: * -> *) a.
MonadError StoreError m' =>
SQLError -> m' a
handleInternal) m (Either StoreError a) -> (Either StoreError a -> m a) -> m a
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    Right a
c -> a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return a
c
    Left StoreError
e -> AgentErrorType -> m a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (AgentErrorType -> m a) -> AgentErrorType -> m a
forall a b. (a -> b) -> a -> b
$ StoreError -> AgentErrorType
storeError StoreError
e
  where
    -- TODO when parsing exception happens in store, the agent hangs;
    -- changing SQLError to SomeException does not help
    handleInternal :: (MonadError StoreError m') => SQLError -> m' a
    handleInternal :: SQLError -> m' a
handleInternal SQLError
e = StoreError -> m' a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (StoreError -> m' a)
-> (ByteString -> StoreError) -> ByteString -> m' a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString -> StoreError
SEInternal (ByteString -> m' a) -> ByteString -> m' a
forall a b. (a -> b) -> a -> b
$ SQLError -> ByteString
forall a. Show a => a -> ByteString
bshow SQLError
e
    storeError :: StoreError -> AgentErrorType
    storeError :: StoreError -> AgentErrorType
storeError = \case
      StoreError
SEConnNotFound -> ConnectionErrorType -> AgentErrorType
CONN ConnectionErrorType
NOT_FOUND
      StoreError
SEConnDuplicate -> ConnectionErrorType -> AgentErrorType
CONN ConnectionErrorType
DUPLICATE
      SEBadConnType ConnType
CRcv -> ConnectionErrorType -> AgentErrorType
CONN ConnectionErrorType
SIMPLEX
      SEBadConnType ConnType
CSnd -> ConnectionErrorType -> AgentErrorType
CONN ConnectionErrorType
SIMPLEX
      StoreError
e -> ServiceName -> AgentErrorType
INTERNAL (ServiceName -> AgentErrorType) -> ServiceName -> AgentErrorType
forall a b. (a -> b) -> a -> b
$ StoreError -> ServiceName
forall a. Show a => a -> ServiceName
show StoreError
e

-- | execute any SMP agent command
processCommand :: forall m. AgentMonad m => AgentClient -> (ConnId, ACommand 'Client) -> m (ConnId, ACommand 'Agent)
processCommand :: AgentClient
-> (ByteString, ACommand 'Client)
-> m (ByteString, ACommand 'Agent)
processCommand AgentClient
c (ByteString
connId, ACommand 'Client
cmd) = case ACommand 'Client
cmd of
  ACommand 'Client
NEW -> (SMPQueueInfo -> ACommand 'Agent)
-> (ByteString, SMPQueueInfo) -> (ByteString, ACommand 'Agent)
forall (p :: * -> * -> *) b c a.
Bifunctor p =>
(b -> c) -> p a b -> p a c
second SMPQueueInfo -> ACommand 'Agent
INV ((ByteString, SMPQueueInfo) -> (ByteString, ACommand 'Agent))
-> m (ByteString, SMPQueueInfo) -> m (ByteString, ACommand 'Agent)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> AgentClient -> ByteString -> m (ByteString, SMPQueueInfo)
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m (ByteString, SMPQueueInfo)
newConn AgentClient
c ByteString
connId
  JOIN SMPQueueInfo
smpQueueInfo ByteString
connInfo -> (,ACommand 'Agent
OK) (ByteString -> (ByteString, ACommand 'Agent))
-> m ByteString -> m (ByteString, ACommand 'Agent)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> AgentClient
-> ByteString -> SMPQueueInfo -> ByteString -> m ByteString
forall (m :: * -> *).
AgentMonad m =>
AgentClient
-> ByteString -> SMPQueueInfo -> ByteString -> m ByteString
joinConn AgentClient
c ByteString
connId SMPQueueInfo
smpQueueInfo ByteString
connInfo
  ACPT ByteString
confId ByteString
ownConnInfo -> AgentClient -> ByteString -> ByteString -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> ByteString -> ByteString -> m ()
acceptConnection' AgentClient
c ByteString
connId ByteString
confId ByteString
ownConnInfo m ()
-> (ByteString, ACommand 'Agent) -> m (ByteString, ACommand 'Agent)
forall (f :: * -> *) a b. Functor f => f a -> b -> f b
$> (ByteString
connId, ACommand 'Agent
OK)
  ACommand 'Client
SUB -> AgentClient -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
subscribeConnection' AgentClient
c ByteString
connId m ()
-> (ByteString, ACommand 'Agent) -> m (ByteString, ACommand 'Agent)
forall (f :: * -> *) a b. Functor f => f a -> b -> f b
$> (ByteString
connId, ACommand 'Agent
OK)
  SEND ByteString
msgBody -> (ByteString
connId,) (ACommand 'Agent -> (ByteString, ACommand 'Agent))
-> (AgentMsgId -> ACommand 'Agent)
-> AgentMsgId
-> (ByteString, ACommand 'Agent)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. AgentMsgId -> ACommand 'Agent
MID (AgentMsgId -> (ByteString, ACommand 'Agent))
-> m AgentMsgId -> m (ByteString, ACommand 'Agent)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> AgentClient -> ByteString -> ByteString -> m AgentMsgId
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> ByteString -> m AgentMsgId
sendMessage' AgentClient
c ByteString
connId ByteString
msgBody
  ACK AgentMsgId
msgId -> AgentClient -> ByteString -> AgentMsgId -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> AgentMsgId -> m ()
ackMessage' AgentClient
c ByteString
connId AgentMsgId
msgId m ()
-> (ByteString, ACommand 'Agent) -> m (ByteString, ACommand 'Agent)
forall (f :: * -> *) a b. Functor f => f a -> b -> f b
$> (ByteString
connId, ACommand 'Agent
OK)
  ACommand 'Client
OFF -> AgentClient -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
suspendConnection' AgentClient
c ByteString
connId m ()
-> (ByteString, ACommand 'Agent) -> m (ByteString, ACommand 'Agent)
forall (f :: * -> *) a b. Functor f => f a -> b -> f b
$> (ByteString
connId, ACommand 'Agent
OK)
  ACommand 'Client
DEL -> AgentClient -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
deleteConnection' AgentClient
c ByteString
connId m ()
-> (ByteString, ACommand 'Agent) -> m (ByteString, ACommand 'Agent)
forall (f :: * -> *) a b. Functor f => f a -> b -> f b
$> (ByteString
connId, ACommand 'Agent
OK)

newConn :: AgentMonad m => AgentClient -> ConnId -> m (ConnId, SMPQueueInfo)
newConn :: AgentClient -> ByteString -> m (ByteString, SMPQueueInfo)
newConn AgentClient
c ByteString
connId = do
  SMPServer
srv <- m SMPServer
forall (m :: * -> *). AgentMonad m => m SMPServer
getSMPServer
  (RcvQueue
rq, SMPQueueInfo
qInfo) <- AgentClient -> SMPServer -> m (RcvQueue, SMPQueueInfo)
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SMPServer -> m (RcvQueue, SMPQueueInfo)
newRcvQueue AgentClient
c SMPServer
srv
  TVar ChaChaDRG
g <- (Env -> TVar ChaChaDRG) -> m (TVar ChaChaDRG)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks Env -> TVar ChaChaDRG
idsDrg
  let cData :: ConnData
cData = ConnData :: ByteString -> ConnData
ConnData {ByteString
$sel:connId:ConnData :: ByteString
connId :: ByteString
connId}
  ByteString
connId' <- (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ByteString)
-> m ByteString
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ByteString)
 -> m ByteString)
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ByteString)
-> m ByteString
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore
-> TVar ChaChaDRG -> ConnData -> RcvQueue -> m' ByteString
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> TVar ChaChaDRG -> ConnData -> RcvQueue -> m ByteString
createRcvConn SQLiteStore
st TVar ChaChaDRG
g ConnData
cData RcvQueue
rq
  AgentClient -> RcvQueue -> ByteString -> m ()
forall (m :: * -> *).
MonadUnliftIO m =>
AgentClient -> RcvQueue -> ByteString -> m ()
addSubscription AgentClient
c RcvQueue
rq ByteString
connId'
  (ByteString, SMPQueueInfo) -> m (ByteString, SMPQueueInfo)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (ByteString
connId', SMPQueueInfo
qInfo)

joinConn :: AgentMonad m => AgentClient -> ConnId -> SMPQueueInfo -> ConnInfo -> m ConnId
joinConn :: AgentClient
-> ByteString -> SMPQueueInfo -> ByteString -> m ByteString
joinConn AgentClient
c ByteString
connId SMPQueueInfo
qInfo ByteString
cInfo = do
  (SndQueue
sq, SenderPublicKey
senderKey, SenderPublicKey
verifyKey) <- SMPQueueInfo -> m (SndQueue, SenderPublicKey, SenderPublicKey)
forall (m :: * -> *).
(MonadUnliftIO m, MonadReader Env m) =>
SMPQueueInfo -> m (SndQueue, SenderPublicKey, SenderPublicKey)
newSndQueue SMPQueueInfo
qInfo
  TVar ChaChaDRG
g <- (Env -> TVar ChaChaDRG) -> m (TVar ChaChaDRG)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks Env -> TVar ChaChaDRG
idsDrg
  AgentConfig
cfg <- (Env -> AgentConfig) -> m AgentConfig
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks Env -> AgentConfig
config
  let cData :: ConnData
cData = ConnData :: ByteString -> ConnData
ConnData {ByteString
connId :: ByteString
$sel:connId:ConnData :: ByteString
connId}
  ByteString
connId' <- (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ByteString)
-> m ByteString
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ByteString)
 -> m ByteString)
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ByteString)
-> m ByteString
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore
-> TVar ChaChaDRG -> ConnData -> SndQueue -> m' ByteString
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> TVar ChaChaDRG -> ConnData -> SndQueue -> m ByteString
createSndConn SQLiteStore
st TVar ChaChaDRG
g ConnData
cData SndQueue
sq
  AgentClient -> SndQueue -> SenderPublicKey -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SndQueue -> SenderPublicKey -> ByteString -> m ()
confirmQueue AgentClient
c SndQueue
sq SenderPublicKey
senderKey ByteString
cInfo
  AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
activateQueueJoining AgentClient
c ByteString
connId' SndQueue
sq SenderPublicKey
verifyKey (RetryInterval -> m ()) -> RetryInterval -> m ()
forall a b. (a -> b) -> a -> b
$ AgentConfig -> RetryInterval
retryInterval AgentConfig
cfg
  ByteString -> m ByteString
forall (f :: * -> *) a. Applicative f => a -> f a
pure ByteString
connId'

activateQueueJoining :: forall m. AgentMonad m => AgentClient -> ConnId -> SndQueue -> VerificationKey -> RetryInterval -> m ()
activateQueueJoining :: AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
activateQueueJoining AgentClient
c ByteString
connId SndQueue
sq SenderPublicKey
verifyKey RetryInterval
retryInterval =
  AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
-> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
-> m ()
activateQueue AgentClient
c ByteString
connId SndQueue
sq SenderPublicKey
verifyKey RetryInterval
retryInterval m ()
createReplyQueue
  where
    createReplyQueue :: m ()
    createReplyQueue :: m ()
createReplyQueue = do
      SMPServer
srv <- m SMPServer
forall (m :: * -> *). AgentMonad m => m SMPServer
getSMPServer
      (RcvQueue
rq, SMPQueueInfo
qInfo') <- AgentClient -> SMPServer -> m (RcvQueue, SMPQueueInfo)
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SMPServer -> m (RcvQueue, SMPQueueInfo)
newRcvQueue AgentClient
c SMPServer
srv
      AgentClient -> RcvQueue -> ByteString -> m ()
forall (m :: * -> *).
MonadUnliftIO m =>
AgentClient -> RcvQueue -> ByteString -> m ()
addSubscription AgentClient
c RcvQueue
rq ByteString
connId
      (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> ByteString -> RcvQueue -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> RcvQueue -> m ()
upgradeSndConnToDuplex SQLiteStore
st ByteString
connId RcvQueue
rq
      AgentClient -> SndQueue -> AMessage -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SndQueue -> AMessage -> m ()
sendControlMessage AgentClient
c SndQueue
sq (AMessage -> m ()) -> AMessage -> m ()
forall a b. (a -> b) -> a -> b
$ SMPQueueInfo -> AMessage
REPLY SMPQueueInfo
qInfo'

-- | Approve confirmation (LET command) in Reader monad
acceptConnection' :: AgentMonad m => AgentClient -> ConnId -> ConfirmationId -> ConnInfo -> m ()
acceptConnection' :: AgentClient -> ByteString -> ByteString -> ByteString -> m ()
acceptConnection' AgentClient
c ByteString
connId ByteString
confId ByteString
ownConnInfo =
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' SomeConn)
-> m SomeConn
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' SomeConn
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m SomeConn
`getConn` ByteString
connId) m SomeConn -> (SomeConn -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    SomeConn SConnType d
SCRcv (RcvConnection ConnData
_ RcvQueue
rq) -> do
      AcceptedConfirmation {SenderPublicKey
$sel:senderKey:AcceptedConfirmation :: AcceptedConfirmation -> SenderPublicKey
senderKey :: SenderPublicKey
senderKey} <- (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' AcceptedConfirmation)
-> m AcceptedConfirmation
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' AcceptedConfirmation)
 -> m AcceptedConfirmation)
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' AcceptedConfirmation)
-> m AcceptedConfirmation
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> ByteString -> ByteString -> m' AcceptedConfirmation
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> ByteString -> m AcceptedConfirmation
acceptConfirmation SQLiteStore
st ByteString
confId ByteString
ownConnInfo
      AgentClient -> RcvQueue -> SenderPublicKey -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> SenderPublicKey -> m ()
processConfirmation AgentClient
c RcvQueue
rq SenderPublicKey
senderKey
    SomeConn
_ -> AgentErrorType -> m ()
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (AgentErrorType -> m ()) -> AgentErrorType -> m ()
forall a b. (a -> b) -> a -> b
$ CommandErrorType -> AgentErrorType
CMD CommandErrorType
PROHIBITED

processConfirmation :: AgentMonad m => AgentClient -> RcvQueue -> SenderPublicKey -> m ()
processConfirmation :: AgentClient -> RcvQueue -> SenderPublicKey -> m ()
processConfirmation AgentClient
c RcvQueue
rq SenderPublicKey
sndKey = do
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> RcvQueue -> QueueStatus -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> RcvQueue -> QueueStatus -> m ()
setRcvQueueStatus SQLiteStore
st RcvQueue
rq QueueStatus
Confirmed
  AgentClient -> RcvQueue -> SenderPublicKey -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> SenderPublicKey -> m ()
secureQueue AgentClient
c RcvQueue
rq SenderPublicKey
sndKey
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> RcvQueue -> QueueStatus -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> RcvQueue -> QueueStatus -> m ()
setRcvQueueStatus SQLiteStore
st RcvQueue
rq QueueStatus
Secured

-- | Subscribe to receive connection messages (SUB command) in Reader monad
subscribeConnection' :: forall m. AgentMonad m => AgentClient -> ConnId -> m ()
subscribeConnection' :: AgentClient -> ByteString -> m ()
subscribeConnection' AgentClient
c ByteString
connId =
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' SomeConn)
-> m SomeConn
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' SomeConn
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m SomeConn
`getConn` ByteString
connId) m SomeConn -> (SomeConn -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    SomeConn SConnType d
_ (DuplexConnection ConnData
_ RcvQueue
rq SndQueue
sq) -> do
      SndQueue -> m ()
resumeDelivery SndQueue
sq
      case SndQueue -> QueueStatus
status (SndQueue
sq :: SndQueue) of
        QueueStatus
Confirmed -> SndQueue -> (SenderPublicKey -> m ()) -> m ()
withVerifyKey SndQueue
sq ((SenderPublicKey -> m ()) -> m ())
-> (SenderPublicKey -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \SenderPublicKey
verifyKey -> do
          AcceptedConfirmation
conf <- (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' AcceptedConfirmation)
-> m AcceptedConfirmation
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' AcceptedConfirmation
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m AcceptedConfirmation
`getAcceptedConfirmation` ByteString
connId)
          AgentClient -> RcvQueue -> SenderPublicKey -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> SenderPublicKey -> m ()
secureQueue AgentClient
c RcvQueue
rq (SenderPublicKey -> m ()) -> SenderPublicKey -> m ()
forall a b. (a -> b) -> a -> b
$ AcceptedConfirmation -> SenderPublicKey
senderKey (AcceptedConfirmation
conf :: AcceptedConfirmation)
          (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> RcvQueue -> QueueStatus -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> RcvQueue -> QueueStatus -> m ()
setRcvQueueStatus SQLiteStore
st RcvQueue
rq QueueStatus
Secured
          RcvQueue -> SndQueue -> SenderPublicKey -> m ()
activateSecuredQueue RcvQueue
rq SndQueue
sq SenderPublicKey
verifyKey
        QueueStatus
Secured -> SndQueue -> (SenderPublicKey -> m ()) -> m ()
withVerifyKey SndQueue
sq ((SenderPublicKey -> m ()) -> m ())
-> (SenderPublicKey -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ RcvQueue -> SndQueue -> SenderPublicKey -> m ()
activateSecuredQueue RcvQueue
rq SndQueue
sq
        QueueStatus
Active -> AgentClient -> RcvQueue -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> ByteString -> m ()
subscribeQueue AgentClient
c RcvQueue
rq ByteString
connId
        QueueStatus
_ -> AgentErrorType -> m ()
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (AgentErrorType -> m ()) -> AgentErrorType -> m ()
forall a b. (a -> b) -> a -> b
$ ServiceName -> AgentErrorType
INTERNAL ServiceName
"unexpected queue status"
    SomeConn SConnType d
_ (SndConnection ConnData
_ SndQueue
sq) -> do
      SndQueue -> m ()
resumeDelivery SndQueue
sq
      case SndQueue -> QueueStatus
status (SndQueue
sq :: SndQueue) of
        QueueStatus
Confirmed -> SndQueue -> (SenderPublicKey -> m ()) -> m ()
withVerifyKey SndQueue
sq ((SenderPublicKey -> m ()) -> m ())
-> (SenderPublicKey -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \SenderPublicKey
verifyKey ->
          AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
activateQueueJoining AgentClient
c ByteString
connId SndQueue
sq SenderPublicKey
verifyKey (RetryInterval -> m ()) -> m RetryInterval -> m ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< m RetryInterval
resumeInterval
        QueueStatus
Active -> AgentErrorType -> m ()
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (AgentErrorType -> m ()) -> AgentErrorType -> m ()
forall a b. (a -> b) -> a -> b
$ ConnectionErrorType -> AgentErrorType
CONN ConnectionErrorType
SIMPLEX
        QueueStatus
_ -> AgentErrorType -> m ()
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (AgentErrorType -> m ()) -> AgentErrorType -> m ()
forall a b. (a -> b) -> a -> b
$ ServiceName -> AgentErrorType
INTERNAL ServiceName
"unexpected queue status"
    SomeConn SConnType d
_ (RcvConnection ConnData
_ RcvQueue
rq) -> AgentClient -> RcvQueue -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> ByteString -> m ()
subscribeQueue AgentClient
c RcvQueue
rq ByteString
connId
  where
    resumeDelivery :: SndQueue -> m ()
    resumeDelivery :: SndQueue -> m ()
resumeDelivery SndQueue {SMPServer
$sel:server:SndQueue :: SndQueue -> SMPServer
server :: SMPServer
server} = do
      Bool
wasDelivering <- AgentClient -> ByteString -> SMPServer -> m Bool
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> SMPServer -> m Bool
resumeMsgDelivery AgentClient
c ByteString
connId SMPServer
server
      Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless Bool
wasDelivering (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
        [PendingMsg]
pending <- (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' [PendingMsg])
-> m [PendingMsg]
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' [PendingMsg]
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m [PendingMsg]
`getPendingMsgs` ByteString
connId)
        AgentClient -> ByteString -> [PendingMsg] -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> [PendingMsg] -> m ()
queuePendingMsgs AgentClient
c ByteString
connId [PendingMsg]
pending
    withVerifyKey :: SndQueue -> (C.PublicKey -> m ()) -> m ()
    withVerifyKey :: SndQueue -> (SenderPublicKey -> m ()) -> m ()
withVerifyKey SndQueue
sq SenderPublicKey -> m ()
action =
      let err :: m a
err = AgentErrorType -> m a
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (AgentErrorType -> m a) -> AgentErrorType -> m a
forall a b. (a -> b) -> a -> b
$ ServiceName -> AgentErrorType
INTERNAL ServiceName
"missing signing key public counterpart"
       in m () -> (SenderPublicKey -> m ()) -> Maybe SenderPublicKey -> m ()
forall b a. b -> (a -> b) -> Maybe a -> b
maybe m ()
forall a. m a
err SenderPublicKey -> m ()
action (Maybe SenderPublicKey -> m ())
-> (SignatureKey -> Maybe SenderPublicKey) -> SignatureKey -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. SignatureKey -> Maybe SenderPublicKey
forall k. PrivateKey k => k -> Maybe SenderPublicKey
C.publicKey (SignatureKey -> m ()) -> SignatureKey -> m ()
forall a b. (a -> b) -> a -> b
$ SndQueue -> SignatureKey
signKey SndQueue
sq
    activateSecuredQueue :: RcvQueue -> SndQueue -> C.PublicKey -> m ()
    activateSecuredQueue :: RcvQueue -> SndQueue -> SenderPublicKey -> m ()
activateSecuredQueue RcvQueue
rq SndQueue
sq SenderPublicKey
verifyKey = do
      AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
activateQueueInitiating AgentClient
c ByteString
connId SndQueue
sq SenderPublicKey
verifyKey (RetryInterval -> m ()) -> m RetryInterval -> m ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< m RetryInterval
resumeInterval
      AgentClient -> RcvQueue -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> ByteString -> m ()
subscribeQueue AgentClient
c RcvQueue
rq ByteString
connId
    resumeInterval :: m RetryInterval
    resumeInterval :: m RetryInterval
resumeInterval = do
      RetryInterval
r <- (Env -> RetryInterval) -> m RetryInterval
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks ((Env -> RetryInterval) -> m RetryInterval)
-> (Env -> RetryInterval) -> m RetryInterval
forall a b. (a -> b) -> a -> b
$ AgentConfig -> RetryInterval
retryInterval (AgentConfig -> RetryInterval)
-> (Env -> AgentConfig) -> Env -> RetryInterval
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Env -> AgentConfig
config
      RetryInterval -> m RetryInterval
forall (f :: * -> *) a. Applicative f => a -> f a
pure RetryInterval
r {initialInterval :: Int
initialInterval = Int
5_000_000}

-- | Send message to the connection (SEND command) in Reader monad
sendMessage' :: forall m. AgentMonad m => AgentClient -> ConnId -> MsgBody -> m AgentMsgId
sendMessage' :: AgentClient -> ByteString -> ByteString -> m AgentMsgId
sendMessage' AgentClient
c ByteString
connId ByteString
msg =
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' SomeConn)
-> m SomeConn
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' SomeConn
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m SomeConn
`getConn` ByteString
connId) m SomeConn -> (SomeConn -> m AgentMsgId) -> m AgentMsgId
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    SomeConn SConnType d
_ (DuplexConnection ConnData
_ RcvQueue
_ SndQueue
sq) -> SndQueue -> m AgentMsgId
enqueueMessage SndQueue
sq
    SomeConn SConnType d
_ (SndConnection ConnData
_ SndQueue
sq) -> SndQueue -> m AgentMsgId
enqueueMessage SndQueue
sq
    SomeConn
_ -> AgentErrorType -> m AgentMsgId
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (AgentErrorType -> m AgentMsgId) -> AgentErrorType -> m AgentMsgId
forall a b. (a -> b) -> a -> b
$ ConnectionErrorType -> AgentErrorType
CONN ConnectionErrorType
SIMPLEX
  where
    enqueueMessage :: SndQueue -> m AgentMsgId
    enqueueMessage :: SndQueue -> m AgentMsgId
enqueueMessage SndQueue {SMPServer
server :: SMPServer
$sel:server:SndQueue :: SndQueue -> SMPServer
server} = do
      InternalId
msgId <- m InternalId
storeSentMsg
      Bool
wasDelivering <- AgentClient -> ByteString -> SMPServer -> m Bool
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> SMPServer -> m Bool
resumeMsgDelivery AgentClient
c ByteString
connId SMPServer
server
      [PendingMsg]
pending <-
        if Bool
wasDelivering
          then [PendingMsg] -> m [PendingMsg]
forall (f :: * -> *) a. Applicative f => a -> f a
pure [PendingMsg :: ByteString -> InternalId -> PendingMsg
PendingMsg {ByteString
$sel:connId:PendingMsg :: ByteString
connId :: ByteString
connId, InternalId
$sel:msgId:PendingMsg :: InternalId
msgId :: InternalId
msgId}]
          else (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' [PendingMsg])
-> m [PendingMsg]
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' [PendingMsg]
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m [PendingMsg]
`getPendingMsgs` ByteString
connId)
      AgentClient -> ByteString -> [PendingMsg] -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> [PendingMsg] -> m ()
queuePendingMsgs AgentClient
c ByteString
connId [PendingMsg]
pending
      AgentMsgId -> m AgentMsgId
forall (f :: * -> *) a. Applicative f => a -> f a
pure (AgentMsgId -> m AgentMsgId) -> AgentMsgId -> m AgentMsgId
forall a b. (a -> b) -> a -> b
$ InternalId -> AgentMsgId
unId InternalId
msgId
      where
        storeSentMsg :: m InternalId
        storeSentMsg :: m InternalId
storeSentMsg = do
          UTCTime
internalTs <- IO UTCTime -> m UTCTime
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO IO UTCTime
getCurrentTime
          (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' InternalId)
-> m InternalId
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' InternalId)
 -> m InternalId)
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' InternalId)
-> m InternalId
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> do
            (InternalId
internalId, InternalSndId
internalSndId, ByteString
previousMsgHash) <- SQLiteStore
-> ByteString -> m' (InternalId, InternalSndId, ByteString)
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m (InternalId, InternalSndId, ByteString)
updateSndIds SQLiteStore
st ByteString
connId
            let msgBody :: ByteString
msgBody =
                  SMPMessage -> ByteString
serializeSMPMessage
                    SMPMessage :: AgentMsgId -> UTCTime -> ByteString -> AMessage -> SMPMessage
SMPMessage
                      { senderMsgId :: AgentMsgId
senderMsgId = InternalSndId -> AgentMsgId
unSndId InternalSndId
internalSndId,
                        senderTimestamp :: UTCTime
senderTimestamp = UTCTime
internalTs,
                        ByteString
previousMsgHash :: ByteString
previousMsgHash :: ByteString
previousMsgHash,
                        agentMessage :: AMessage
agentMessage = ByteString -> AMessage
A_MSG ByteString
msg
                      }
                internalHash :: ByteString
internalHash = ByteString -> ByteString
C.sha256Hash ByteString
msgBody
                msgData :: SndMsgData
msgData = SndMsgData :: InternalId
-> InternalSndId
-> UTCTime
-> ByteString
-> ByteString
-> ByteString
-> SndMsgData
SndMsgData {ByteString
UTCTime
InternalId
InternalSndId
$sel:previousMsgHash:SndMsgData :: ByteString
$sel:internalHash:SndMsgData :: ByteString
$sel:msgBody:SndMsgData :: ByteString
$sel:internalTs:SndMsgData :: UTCTime
$sel:internalSndId:SndMsgData :: InternalSndId
$sel:internalId:SndMsgData :: InternalId
internalHash :: ByteString
msgBody :: ByteString
previousMsgHash :: ByteString
internalSndId :: InternalSndId
internalId :: InternalId
internalTs :: UTCTime
..}
            SQLiteStore -> ByteString -> SndMsgData -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> SndMsgData -> m ()
createSndMsg SQLiteStore
st ByteString
connId SndMsgData
msgData
            InternalId -> m' InternalId
forall (f :: * -> *) a. Applicative f => a -> f a
pure InternalId
internalId

resumeMsgDelivery :: forall m. AgentMonad m => AgentClient -> ConnId -> SMPServer -> m Bool
resumeMsgDelivery :: AgentClient -> ByteString -> SMPServer -> m Bool
resumeMsgDelivery AgentClient
c ByteString
connId SMPServer
srv = do
  m Bool -> m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (m Bool -> m ()) -> m Bool -> m ()
forall a b. (a -> b) -> a -> b
$ SMPServer -> TVar (Map SMPServer (Async ())) -> m () -> m Bool
forall a. Ord a => a -> TVar (Map a (Async ())) -> m () -> m Bool
resume SMPServer
srv (AgentClient -> TVar (Map SMPServer (Async ()))
srvMsgDeliveries AgentClient
c) (m () -> m Bool) -> m () -> m Bool
forall a b. (a -> b) -> a -> b
$ AgentClient -> SMPServer -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SMPServer -> m ()
runSrvMsgDelivery AgentClient
c SMPServer
srv
  ByteString -> TVar (Map ByteString (Async ())) -> m () -> m Bool
forall a. Ord a => a -> TVar (Map a (Async ())) -> m () -> m Bool
resume ByteString
connId (AgentClient -> TVar (Map ByteString (Async ()))
connMsgDeliveries AgentClient
c) (m () -> m Bool) -> m () -> m Bool
forall a b. (a -> b) -> a -> b
$ AgentClient -> ByteString -> SMPServer -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> SMPServer -> m ()
runMsgDelivery AgentClient
c ByteString
connId SMPServer
srv
  where
    resume :: Ord a => a -> TVar (Map a (Async ())) -> m () -> m Bool
    resume :: a -> TVar (Map a (Async ())) -> m () -> m Bool
resume a
key TVar (Map a (Async ()))
actionMap m ()
actionProcess = do
      Bool
isDelivering <- Maybe (Async ()) -> Bool
forall a. Maybe a -> Bool
isJust (Maybe (Async ()) -> Bool)
-> (Map a (Async ()) -> Maybe (Async ()))
-> Map a (Async ())
-> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> Map a (Async ()) -> Maybe (Async ())
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup a
key (Map a (Async ()) -> Bool) -> m (Map a (Async ())) -> m Bool
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> TVar (Map a (Async ())) -> m (Map a (Async ()))
forall (m :: * -> *) a. MonadIO m => TVar a -> m a
readTVarIO TVar (Map a (Async ()))
actionMap
      Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless Bool
isDelivering (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$
        m () -> m (Async ())
forall (m :: * -> *) a. MonadUnliftIO m => m a -> m (Async a)
async m ()
actionProcess
          m (Async ()) -> (Async () -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= STM () -> m ()
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM () -> m ()) -> (Async () -> STM ()) -> Async () -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TVar (Map a (Async ()))
-> (Map a (Async ()) -> Map a (Async ())) -> STM ()
forall a. TVar a -> (a -> a) -> STM ()
modifyTVar TVar (Map a (Async ()))
actionMap ((Map a (Async ()) -> Map a (Async ())) -> STM ())
-> (Async () -> Map a (Async ()) -> Map a (Async ()))
-> Async ()
-> STM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> Async () -> Map a (Async ()) -> Map a (Async ())
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert a
key
      Bool -> m Bool
forall (f :: * -> *) a. Applicative f => a -> f a
pure Bool
isDelivering

queuePendingMsgs :: AgentMonad m => AgentClient -> ConnId -> [PendingMsg] -> m ()
queuePendingMsgs :: AgentClient -> ByteString -> [PendingMsg] -> m ()
queuePendingMsgs AgentClient
c ByteString
connId [PendingMsg]
pending =
  STM () -> m ()
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM () -> m ()) -> STM () -> m ()
forall a b. (a -> b) -> a -> b
$ ByteString
-> TVar (Map ByteString (TQueue PendingMsg))
-> STM (TQueue PendingMsg)
forall a.
Ord a =>
a -> TVar (Map a (TQueue PendingMsg)) -> STM (TQueue PendingMsg)
getPendingMsgQ ByteString
connId (AgentClient -> TVar (Map ByteString (TQueue PendingMsg))
connMsgQueues AgentClient
c) STM (TQueue PendingMsg) -> (TQueue PendingMsg -> STM ()) -> STM ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= [PendingMsg] -> (PendingMsg -> STM ()) -> STM ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
t a -> (a -> m b) -> m ()
forM_ [PendingMsg]
pending ((PendingMsg -> STM ()) -> STM ())
-> (TQueue PendingMsg -> PendingMsg -> STM ())
-> TQueue PendingMsg
-> STM ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TQueue PendingMsg -> PendingMsg -> STM ()
forall a. TQueue a -> a -> STM ()
writeTQueue

getPendingMsgQ :: Ord a => a -> TVar (Map a (TQueue PendingMsg)) -> STM (TQueue PendingMsg)
getPendingMsgQ :: a -> TVar (Map a (TQueue PendingMsg)) -> STM (TQueue PendingMsg)
getPendingMsgQ a
key TVar (Map a (TQueue PendingMsg))
queueMap = do
  STM (TQueue PendingMsg)
-> (TQueue PendingMsg -> STM (TQueue PendingMsg))
-> Maybe (TQueue PendingMsg)
-> STM (TQueue PendingMsg)
forall b a. b -> (a -> b) -> Maybe a -> b
maybe STM (TQueue PendingMsg)
newMsgQueue TQueue PendingMsg -> STM (TQueue PendingMsg)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Maybe (TQueue PendingMsg) -> STM (TQueue PendingMsg))
-> (Map a (TQueue PendingMsg) -> Maybe (TQueue PendingMsg))
-> Map a (TQueue PendingMsg)
-> STM (TQueue PendingMsg)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> Map a (TQueue PendingMsg) -> Maybe (TQueue PendingMsg)
forall k a. Ord k => k -> Map k a -> Maybe a
M.lookup a
key (Map a (TQueue PendingMsg) -> STM (TQueue PendingMsg))
-> STM (Map a (TQueue PendingMsg)) -> STM (TQueue PendingMsg)
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< TVar (Map a (TQueue PendingMsg)) -> STM (Map a (TQueue PendingMsg))
forall a. TVar a -> STM a
readTVar TVar (Map a (TQueue PendingMsg))
queueMap
  where
    newMsgQueue :: STM (TQueue PendingMsg)
    newMsgQueue :: STM (TQueue PendingMsg)
newMsgQueue = do
      TQueue PendingMsg
mq <- STM (TQueue PendingMsg)
forall a. STM (TQueue a)
newTQueue
      TVar (Map a (TQueue PendingMsg))
-> (Map a (TQueue PendingMsg) -> Map a (TQueue PendingMsg))
-> STM ()
forall a. TVar a -> (a -> a) -> STM ()
modifyTVar TVar (Map a (TQueue PendingMsg))
queueMap ((Map a (TQueue PendingMsg) -> Map a (TQueue PendingMsg))
 -> STM ())
-> (Map a (TQueue PendingMsg) -> Map a (TQueue PendingMsg))
-> STM ()
forall a b. (a -> b) -> a -> b
$ a
-> TQueue PendingMsg
-> Map a (TQueue PendingMsg)
-> Map a (TQueue PendingMsg)
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert a
key TQueue PendingMsg
mq
      TQueue PendingMsg -> STM (TQueue PendingMsg)
forall (f :: * -> *) a. Applicative f => a -> f a
pure TQueue PendingMsg
mq

runMsgDelivery :: AgentMonad m => AgentClient -> ConnId -> SMPServer -> m ()
runMsgDelivery :: AgentClient -> ByteString -> SMPServer -> m ()
runMsgDelivery AgentClient
c ByteString
connId SMPServer
srv = do
  TQueue PendingMsg
mq <- STM (TQueue PendingMsg) -> m (TQueue PendingMsg)
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM (TQueue PendingMsg) -> m (TQueue PendingMsg))
-> (TVar (Map ByteString (TQueue PendingMsg))
    -> STM (TQueue PendingMsg))
-> TVar (Map ByteString (TQueue PendingMsg))
-> m (TQueue PendingMsg)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. ByteString
-> TVar (Map ByteString (TQueue PendingMsg))
-> STM (TQueue PendingMsg)
forall a.
Ord a =>
a -> TVar (Map a (TQueue PendingMsg)) -> STM (TQueue PendingMsg)
getPendingMsgQ ByteString
connId (TVar (Map ByteString (TQueue PendingMsg))
 -> m (TQueue PendingMsg))
-> TVar (Map ByteString (TQueue PendingMsg))
-> m (TQueue PendingMsg)
forall a b. (a -> b) -> a -> b
$ AgentClient -> TVar (Map ByteString (TQueue PendingMsg))
connMsgQueues AgentClient
c
  TQueue PendingMsg
smq <- STM (TQueue PendingMsg) -> m (TQueue PendingMsg)
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM (TQueue PendingMsg) -> m (TQueue PendingMsg))
-> (TVar (Map SMPServer (TQueue PendingMsg))
    -> STM (TQueue PendingMsg))
-> TVar (Map SMPServer (TQueue PendingMsg))
-> m (TQueue PendingMsg)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. SMPServer
-> TVar (Map SMPServer (TQueue PendingMsg))
-> STM (TQueue PendingMsg)
forall a.
Ord a =>
a -> TVar (Map a (TQueue PendingMsg)) -> STM (TQueue PendingMsg)
getPendingMsgQ SMPServer
srv (TVar (Map SMPServer (TQueue PendingMsg)) -> m (TQueue PendingMsg))
-> TVar (Map SMPServer (TQueue PendingMsg))
-> m (TQueue PendingMsg)
forall a b. (a -> b) -> a -> b
$ AgentClient -> TVar (Map SMPServer (TQueue PendingMsg))
srvMsgQueues AgentClient
c
  m () -> m ()
forall (f :: * -> *) a b. Applicative f => f a -> f b
forever (m () -> m ()) -> (STM () -> m ()) -> STM () -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. STM () -> m ()
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM () -> m ()) -> STM () -> m ()
forall a b. (a -> b) -> a -> b
$ TQueue PendingMsg -> STM PendingMsg
forall a. TQueue a -> STM a
readTQueue TQueue PendingMsg
mq STM PendingMsg -> (PendingMsg -> STM ()) -> STM ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= TQueue PendingMsg -> PendingMsg -> STM ()
forall a. TQueue a -> a -> STM ()
writeTQueue TQueue PendingMsg
smq

runSrvMsgDelivery :: forall m. AgentMonad m => AgentClient -> SMPServer -> m ()
runSrvMsgDelivery :: AgentClient -> SMPServer -> m ()
runSrvMsgDelivery c :: AgentClient
c@AgentClient {TBQueue (ATransmission 'Agent)
subQ :: TBQueue (ATransmission 'Agent)
$sel:subQ:AgentClient :: AgentClient -> TBQueue (ATransmission 'Agent)
subQ} SMPServer
srv = do
  TQueue PendingMsg
mq <- STM (TQueue PendingMsg) -> m (TQueue PendingMsg)
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM (TQueue PendingMsg) -> m (TQueue PendingMsg))
-> (TVar (Map SMPServer (TQueue PendingMsg))
    -> STM (TQueue PendingMsg))
-> TVar (Map SMPServer (TQueue PendingMsg))
-> m (TQueue PendingMsg)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. SMPServer
-> TVar (Map SMPServer (TQueue PendingMsg))
-> STM (TQueue PendingMsg)
forall a.
Ord a =>
a -> TVar (Map a (TQueue PendingMsg)) -> STM (TQueue PendingMsg)
getPendingMsgQ SMPServer
srv (TVar (Map SMPServer (TQueue PendingMsg)) -> m (TQueue PendingMsg))
-> TVar (Map SMPServer (TQueue PendingMsg))
-> m (TQueue PendingMsg)
forall a b. (a -> b) -> a -> b
$ AgentClient -> TVar (Map SMPServer (TQueue PendingMsg))
srvMsgQueues AgentClient
c
  RetryInterval
ri <- (Env -> RetryInterval) -> m RetryInterval
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks ((Env -> RetryInterval) -> m RetryInterval)
-> (Env -> RetryInterval) -> m RetryInterval
forall a b. (a -> b) -> a -> b
$ AgentConfig -> RetryInterval
reconnectInterval (AgentConfig -> RetryInterval)
-> (Env -> AgentConfig) -> Env -> RetryInterval
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Env -> AgentConfig
config
  m () -> m ()
forall (f :: * -> *) a b. Applicative f => f a -> f b
forever (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
    PendingMsg {ByteString
connId :: ByteString
$sel:connId:PendingMsg :: PendingMsg -> ByteString
connId, InternalId
msgId :: InternalId
$sel:msgId:PendingMsg :: PendingMsg -> InternalId
msgId} <- STM PendingMsg -> m PendingMsg
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM PendingMsg -> m PendingMsg) -> STM PendingMsg -> m PendingMsg
forall a b. (a -> b) -> a -> b
$ TQueue PendingMsg -> STM PendingMsg
forall a. TQueue a -> STM a
readTQueue TQueue PendingMsg
mq
    let mId :: AgentMsgId
mId = InternalId -> AgentMsgId
unId InternalId
msgId
    (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' (Either SomeException (SndQueue, ByteString)))
-> m (Either SomeException (SndQueue, ByteString))
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (\SQLiteStore
st -> m' (SndQueue, ByteString)
-> m' (Either SomeException (SndQueue, ByteString))
forall (m :: * -> *) e a.
(MonadUnliftIO m, Exception e) =>
m a -> m (Either e a)
E.try (m' (SndQueue, ByteString)
 -> m' (Either SomeException (SndQueue, ByteString)))
-> m' (SndQueue, ByteString)
-> m' (Either SomeException (SndQueue, ByteString))
forall a b. (a -> b) -> a -> b
$ SQLiteStore
-> ByteString -> InternalId -> m' (SndQueue, ByteString)
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> InternalId -> m (SndQueue, ByteString)
getPendingMsgData SQLiteStore
st ByteString
connId InternalId
msgId) m (Either SomeException (SndQueue, ByteString))
-> (Either SomeException (SndQueue, ByteString) -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
      Left (SomeException
e :: E.SomeException) ->
        ByteString -> ACommand 'Agent -> m ()
notify ByteString
connId (ACommand 'Agent -> m ()) -> ACommand 'Agent -> m ()
forall a b. (a -> b) -> a -> b
$ AgentMsgId -> AgentErrorType -> ACommand 'Agent
MERR AgentMsgId
mId (ServiceName -> AgentErrorType
INTERNAL (ServiceName -> AgentErrorType) -> ServiceName -> AgentErrorType
forall a b. (a -> b) -> a -> b
$ SomeException -> ServiceName
forall a. Show a => a -> ServiceName
show SomeException
e)
      Right (SndQueue
sq, ByteString
msgBody) -> do
        RetryInterval -> (m () -> m ()) -> m ()
forall (m :: * -> *).
MonadIO m =>
RetryInterval -> (m () -> m ()) -> m ()
withRetryInterval RetryInterval
ri ((m () -> m ()) -> m ()) -> (m () -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \m ()
loop -> do
          m () -> m (Either AgentErrorType ())
forall e (m :: * -> *) a. MonadError e m => m a -> m (Either e a)
tryError (AgentClient -> SndQueue -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SndQueue -> ByteString -> m ()
sendAgentMessage AgentClient
c SndQueue
sq ByteString
msgBody) m (Either AgentErrorType ())
-> (Either AgentErrorType () -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
            Left AgentErrorType
e -> case AgentErrorType
e of
              SMP ErrorType
SMP.QUOTA -> m ()
loop
              SMP {} -> ByteString -> ACommand 'Agent -> m ()
notify ByteString
connId (ACommand 'Agent -> m ()) -> ACommand 'Agent -> m ()
forall a b. (a -> b) -> a -> b
$ AgentMsgId -> AgentErrorType -> ACommand 'Agent
MERR AgentMsgId
mId AgentErrorType
e
              CMD {} -> ByteString -> ACommand 'Agent -> m ()
notify ByteString
connId (ACommand 'Agent -> m ()) -> ACommand 'Agent -> m ()
forall a b. (a -> b) -> a -> b
$ AgentMsgId -> AgentErrorType -> ACommand 'Agent
MERR AgentMsgId
mId AgentErrorType
e
              AgentErrorType
_ -> m ()
loop
            Right () -> do
              ByteString -> ACommand 'Agent -> m ()
notify ByteString
connId (ACommand 'Agent -> m ()) -> ACommand 'Agent -> m ()
forall a b. (a -> b) -> a -> b
$ AgentMsgId -> ACommand 'Agent
SENT AgentMsgId
mId
              (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> ByteString -> InternalId -> SndMsgStatus -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> InternalId -> SndMsgStatus -> m ()
updateSndMsgStatus SQLiteStore
st ByteString
connId InternalId
msgId SndMsgStatus
SndMsgSent
  where
    notify :: ConnId -> ACommand 'Agent -> m ()
    notify :: ByteString -> ACommand 'Agent -> m ()
notify ByteString
connId ACommand 'Agent
cmd = STM () -> m ()
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM () -> m ()) -> STM () -> m ()
forall a b. (a -> b) -> a -> b
$ TBQueue (ATransmission 'Agent) -> ATransmission 'Agent -> STM ()
forall a. TBQueue a -> a -> STM ()
writeTBQueue TBQueue (ATransmission 'Agent)
subQ (ByteString
"", ByteString
connId, ACommand 'Agent
cmd)

ackMessage' :: forall m. AgentMonad m => AgentClient -> ConnId -> AgentMsgId -> m ()
ackMessage' :: AgentClient -> ByteString -> AgentMsgId -> m ()
ackMessage' AgentClient
c ByteString
connId AgentMsgId
msgId = do
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' SomeConn)
-> m SomeConn
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' SomeConn
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m SomeConn
`getConn` ByteString
connId) m SomeConn -> (SomeConn -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    SomeConn SConnType d
_ (DuplexConnection ConnData
_ RcvQueue
rq SndQueue
_) -> RcvQueue -> m ()
ack RcvQueue
rq
    SomeConn SConnType d
_ (RcvConnection ConnData
_ RcvQueue
rq) -> RcvQueue -> m ()
ack RcvQueue
rq
    SomeConn
_ -> AgentErrorType -> m ()
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (AgentErrorType -> m ()) -> AgentErrorType -> m ()
forall a b. (a -> b) -> a -> b
$ ConnectionErrorType -> AgentErrorType
CONN ConnectionErrorType
SIMPLEX
  where
    ack :: RcvQueue -> m ()
    ack :: RcvQueue -> m ()
ack RcvQueue
rq = do
      let mId :: InternalId
mId = AgentMsgId -> InternalId
InternalId AgentMsgId
msgId
      (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> ByteString -> InternalId -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> InternalId -> m ()
checkRcvMsg SQLiteStore
st ByteString
connId InternalId
mId
      AgentClient -> RcvQueue -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> m ()
sendAck AgentClient
c RcvQueue
rq
      (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> ByteString -> InternalId -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> InternalId -> m ()
updateRcvMsgAck SQLiteStore
st ByteString
connId InternalId
mId

-- | Suspend SMP agent connection (OFF command) in Reader monad
suspendConnection' :: AgentMonad m => AgentClient -> ConnId -> m ()
suspendConnection' :: AgentClient -> ByteString -> m ()
suspendConnection' AgentClient
c ByteString
connId =
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' SomeConn)
-> m SomeConn
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' SomeConn
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m SomeConn
`getConn` ByteString
connId) m SomeConn -> (SomeConn -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    SomeConn SConnType d
_ (DuplexConnection ConnData
_ RcvQueue
rq SndQueue
_) -> AgentClient -> RcvQueue -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> m ()
suspendQueue AgentClient
c RcvQueue
rq
    SomeConn SConnType d
_ (RcvConnection ConnData
_ RcvQueue
rq) -> AgentClient -> RcvQueue -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> m ()
suspendQueue AgentClient
c RcvQueue
rq
    SomeConn
_ -> AgentErrorType -> m ()
forall e (m :: * -> *) a. MonadError e m => e -> m a
throwError (AgentErrorType -> m ()) -> AgentErrorType -> m ()
forall a b. (a -> b) -> a -> b
$ ConnectionErrorType -> AgentErrorType
CONN ConnectionErrorType
SIMPLEX

-- | Delete SMP agent connection (DEL command) in Reader monad
deleteConnection' :: forall m. AgentMonad m => AgentClient -> ConnId -> m ()
deleteConnection' :: AgentClient -> ByteString -> m ()
deleteConnection' AgentClient
c ByteString
connId =
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' SomeConn)
-> m SomeConn
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' SomeConn
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m SomeConn
`getConn` ByteString
connId) m SomeConn -> (SomeConn -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    SomeConn SConnType d
_ (DuplexConnection ConnData
_ RcvQueue
rq SndQueue
_) -> RcvQueue -> m ()
delete RcvQueue
rq
    SomeConn SConnType d
_ (RcvConnection ConnData
_ RcvQueue
rq) -> RcvQueue -> m ()
delete RcvQueue
rq
    SomeConn
_ -> (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m ()
`deleteConn` ByteString
connId)
  where
    delete :: RcvQueue -> m ()
    delete :: RcvQueue -> m ()
delete RcvQueue
rq = do
      AgentClient -> RcvQueue -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> m ()
deleteQueue AgentClient
c RcvQueue
rq
      AgentClient -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
removeSubscription AgentClient
c ByteString
connId
      (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m ()
`deleteConn` ByteString
connId)

getSMPServer :: AgentMonad m => m SMPServer
getSMPServer :: m SMPServer
getSMPServer =
  (Env -> NonEmpty SMPServer) -> m (NonEmpty SMPServer)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks (AgentConfig -> NonEmpty SMPServer
smpServers (AgentConfig -> NonEmpty SMPServer)
-> (Env -> AgentConfig) -> Env -> NonEmpty SMPServer
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Env -> AgentConfig
config) m (NonEmpty SMPServer)
-> (NonEmpty SMPServer -> m SMPServer) -> m SMPServer
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    SMPServer
srv :| [] -> SMPServer -> m SMPServer
forall (f :: * -> *) a. Applicative f => a -> f a
pure SMPServer
srv
    NonEmpty SMPServer
servers -> do
      TVar StdGen
gen <- (Env -> TVar StdGen) -> m (TVar StdGen)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks Env -> TVar StdGen
randomServer
      Int
i <- STM Int -> m Int
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM Int -> m Int)
-> ((StdGen -> (Int, StdGen)) -> STM Int)
-> (StdGen -> (Int, StdGen))
-> m Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TVar StdGen -> (StdGen -> (Int, StdGen)) -> STM Int
forall s a. TVar s -> (s -> (a, s)) -> STM a
stateTVar TVar StdGen
gen ((StdGen -> (Int, StdGen)) -> m Int)
-> (StdGen -> (Int, StdGen)) -> m Int
forall a b. (a -> b) -> a -> b
$ (Int, Int) -> StdGen -> (Int, StdGen)
forall a g. (Random a, RandomGen g) => (a, a) -> g -> (a, g)
randomR (Int
0, NonEmpty SMPServer -> Int
forall a. NonEmpty a -> Int
L.length NonEmpty SMPServer
servers Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1)
      SMPServer -> m SMPServer
forall (f :: * -> *) a. Applicative f => a -> f a
pure (SMPServer -> m SMPServer) -> SMPServer -> m SMPServer
forall a b. (a -> b) -> a -> b
$ NonEmpty SMPServer
servers NonEmpty SMPServer -> Int -> SMPServer
forall a. NonEmpty a -> Int -> a
L.!! Int
i

sendControlMessage :: AgentMonad m => AgentClient -> SndQueue -> AMessage -> m ()
sendControlMessage :: AgentClient -> SndQueue -> AMessage -> m ()
sendControlMessage AgentClient
c SndQueue
sq AMessage
agentMessage = do
  UTCTime
senderTimestamp <- IO UTCTime -> m UTCTime
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO IO UTCTime
getCurrentTime
  AgentClient -> SndQueue -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SndQueue -> ByteString -> m ()
sendAgentMessage AgentClient
c SndQueue
sq (ByteString -> m ())
-> (SMPMessage -> ByteString) -> SMPMessage -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. SMPMessage -> ByteString
serializeSMPMessage (SMPMessage -> m ()) -> SMPMessage -> m ()
forall a b. (a -> b) -> a -> b
$
    SMPMessage :: AgentMsgId -> UTCTime -> ByteString -> AMessage -> SMPMessage
SMPMessage
      { senderMsgId :: AgentMsgId
senderMsgId = AgentMsgId
0,
        UTCTime
senderTimestamp :: UTCTime
senderTimestamp :: UTCTime
senderTimestamp,
        previousMsgHash :: ByteString
previousMsgHash = ByteString
"",
        AMessage
agentMessage :: AMessage
agentMessage :: AMessage
agentMessage
      }

subscriber :: (MonadUnliftIO m, MonadReader Env m) => AgentClient -> m ()
subscriber :: AgentClient -> m ()
subscriber c :: AgentClient
c@AgentClient {TBQueue SMPServerTransmission
$sel:msgQ:AgentClient :: AgentClient -> TBQueue SMPServerTransmission
msgQ :: TBQueue SMPServerTransmission
msgQ} = m () -> m ()
forall (f :: * -> *) a b. Applicative f => f a -> f b
forever (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
  SMPServerTransmission
t <- STM SMPServerTransmission -> m SMPServerTransmission
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM SMPServerTransmission -> m SMPServerTransmission)
-> STM SMPServerTransmission -> m SMPServerTransmission
forall a b. (a -> b) -> a -> b
$ TBQueue SMPServerTransmission -> STM SMPServerTransmission
forall a. TBQueue a -> STM a
readTBQueue TBQueue SMPServerTransmission
msgQ
  AgentClient
-> m (Either AgentErrorType ()) -> m (Either AgentErrorType ())
forall (m :: * -> *) a.
MonadUnliftIO m =>
AgentClient -> m a -> m a
withAgentLock AgentClient
c (ExceptT AgentErrorType m () -> m (Either AgentErrorType ())
forall e (m :: * -> *) a. ExceptT e m a -> m (Either e a)
runExceptT (ExceptT AgentErrorType m () -> m (Either AgentErrorType ()))
-> ExceptT AgentErrorType m () -> m (Either AgentErrorType ())
forall a b. (a -> b) -> a -> b
$ AgentClient -> SMPServerTransmission -> ExceptT AgentErrorType m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SMPServerTransmission -> m ()
processSMPTransmission AgentClient
c SMPServerTransmission
t) m (Either AgentErrorType ())
-> (Either AgentErrorType () -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    Left AgentErrorType
e -> IO () -> m ()
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO () -> m ()) -> IO () -> m ()
forall a b. (a -> b) -> a -> b
$ AgentErrorType -> IO ()
forall a. Show a => a -> IO ()
print AgentErrorType
e
    Right ()
_ -> () -> m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()

processSMPTransmission :: forall m. AgentMonad m => AgentClient -> SMPServerTransmission -> m ()
processSMPTransmission :: AgentClient -> SMPServerTransmission -> m ()
processSMPTransmission c :: AgentClient
c@AgentClient {TBQueue (ATransmission 'Agent)
subQ :: TBQueue (ATransmission 'Agent)
$sel:subQ:AgentClient :: AgentClient -> TBQueue (ATransmission 'Agent)
subQ} (SMPServer
srv, ByteString
rId, Command 'Broker
cmd) = do
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' SomeConn)
-> m SomeConn
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (\SQLiteStore
st -> SQLiteStore -> SMPServer -> ByteString -> m' SomeConn
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> SMPServer -> ByteString -> m SomeConn
getRcvConn SQLiteStore
st SMPServer
srv ByteString
rId) m SomeConn -> (SomeConn -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    SomeConn SConnType d
SCDuplex (DuplexConnection ConnData
cData RcvQueue
rq SndQueue
_) -> SConnType 'CDuplex -> ConnData -> RcvQueue -> m ()
forall (c :: ConnType). SConnType c -> ConnData -> RcvQueue -> m ()
processSMP SConnType 'CDuplex
SCDuplex ConnData
cData RcvQueue
rq
    SomeConn SConnType d
SCRcv (RcvConnection ConnData
cData RcvQueue
rq) -> SConnType 'CRcv -> ConnData -> RcvQueue -> m ()
forall (c :: ConnType). SConnType c -> ConnData -> RcvQueue -> m ()
processSMP SConnType 'CRcv
SCRcv ConnData
cData RcvQueue
rq
    SomeConn
_ -> STM () -> m ()
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM () -> m ()) -> STM () -> m ()
forall a b. (a -> b) -> a -> b
$ TBQueue (ATransmission 'Agent) -> ATransmission 'Agent -> STM ()
forall a. TBQueue a -> a -> STM ()
writeTBQueue TBQueue (ATransmission 'Agent)
subQ (ByteString
"", ByteString
"", AgentErrorType -> ACommand 'Agent
ERR (AgentErrorType -> ACommand 'Agent)
-> AgentErrorType -> ACommand 'Agent
forall a b. (a -> b) -> a -> b
$ ConnectionErrorType -> AgentErrorType
CONN ConnectionErrorType
NOT_FOUND)
  where
    processSMP :: SConnType c -> ConnData -> RcvQueue -> m ()
    processSMP :: SConnType c -> ConnData -> RcvQueue -> m ()
processSMP SConnType c
cType ConnData {ByteString
connId :: ByteString
$sel:connId:ConnData :: ConnData -> ByteString
connId} rq :: RcvQueue
rq@RcvQueue {QueueStatus
$sel:status:RcvQueue :: RcvQueue -> QueueStatus
status :: QueueStatus
status} =
      case Command 'Broker
cmd of
        SMP.MSG ByteString
srvMsgId UTCTime
srvTs ByteString
msgBody -> do
          -- TODO deduplicate with previously received
          ByteString
msg <- RcvQueue -> ByteString -> m ByteString
forall (m :: * -> *).
AgentMonad m =>
RcvQueue -> ByteString -> m ByteString
decryptAndVerify RcvQueue
rq ByteString
msgBody
          let msgHash :: ByteString
msgHash = ByteString -> ByteString
C.sha256Hash ByteString
msg
          case ByteString -> Either AgentErrorType SMPMessage
parseSMPMessage ByteString
msg of
            Left AgentErrorType
e -> ACommand 'Agent -> m ()
notify (ACommand 'Agent -> m ()) -> ACommand 'Agent -> m ()
forall a b. (a -> b) -> a -> b
$ AgentErrorType -> ACommand 'Agent
ERR AgentErrorType
e
            Right (SMPConfirmation SenderPublicKey
senderKey ByteString
cInfo) -> SenderPublicKey -> ByteString -> m ()
smpConfirmation SenderPublicKey
senderKey ByteString
cInfo m () -> m () -> m ()
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> AgentClient -> RcvQueue -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> m ()
sendAck AgentClient
c RcvQueue
rq
            Right SMPMessage {AMessage
agentMessage :: AMessage
agentMessage :: SMPMessage -> AMessage
agentMessage, AgentMsgId
senderMsgId :: AgentMsgId
senderMsgId :: SMPMessage -> AgentMsgId
senderMsgId, UTCTime
senderTimestamp :: UTCTime
senderTimestamp :: SMPMessage -> UTCTime
senderTimestamp, ByteString
previousMsgHash :: ByteString
previousMsgHash :: SMPMessage -> ByteString
previousMsgHash} ->
              case AMessage
agentMessage of
                HELLO SenderPublicKey
verifyKey AckMode
_ -> SenderPublicKey -> ByteString -> m ()
helloMsg SenderPublicKey
verifyKey ByteString
msgBody m () -> m () -> m ()
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> AgentClient -> RcvQueue -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> m ()
sendAck AgentClient
c RcvQueue
rq
                REPLY SMPQueueInfo
qInfo -> SMPQueueInfo -> m ()
replyMsg SMPQueueInfo
qInfo m () -> m () -> m ()
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> AgentClient -> RcvQueue -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> m ()
sendAck AgentClient
c RcvQueue
rq
                A_MSG ByteString
body -> ByteString
-> (AgentMsgId, UTCTime)
-> (ByteString, UTCTime)
-> ByteString
-> ByteString
-> m ()
agentClientMsg ByteString
previousMsgHash (AgentMsgId
senderMsgId, UTCTime
senderTimestamp) (ByteString
srvMsgId, UTCTime
srvTs) ByteString
body ByteString
msgHash
        Command 'Broker
SMP.END -> do
          AgentClient -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
removeSubscription AgentClient
c ByteString
connId
          ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
logServer ByteString
"<--" AgentClient
c SMPServer
srv ByteString
rId ByteString
"END"
          ACommand 'Agent -> m ()
notify ACommand 'Agent
END
        Command 'Broker
_ -> do
          ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
logServer ByteString
"<--" AgentClient
c SMPServer
srv ByteString
rId (ByteString -> m ()) -> ByteString -> m ()
forall a b. (a -> b) -> a -> b
$ ByteString
"unexpected: " ByteString -> ByteString -> ByteString
forall a. Semigroup a => a -> a -> a
<> Command 'Broker -> ByteString
forall a. Show a => a -> ByteString
bshow Command 'Broker
cmd
          ACommand 'Agent -> m ()
notify (ACommand 'Agent -> m ())
-> (AgentErrorType -> ACommand 'Agent) -> AgentErrorType -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. AgentErrorType -> ACommand 'Agent
ERR (AgentErrorType -> m ()) -> AgentErrorType -> m ()
forall a b. (a -> b) -> a -> b
$ BrokerErrorType -> AgentErrorType
BROKER BrokerErrorType
UNEXPECTED
      where
        notify :: ACommand 'Agent -> m ()
        notify :: ACommand 'Agent -> m ()
notify ACommand 'Agent
msg = STM () -> m ()
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM () -> m ()) -> STM () -> m ()
forall a b. (a -> b) -> a -> b
$ TBQueue (ATransmission 'Agent) -> ATransmission 'Agent -> STM ()
forall a. TBQueue a -> a -> STM ()
writeTBQueue TBQueue (ATransmission 'Agent)
subQ (ByteString
"", ByteString
connId, ACommand 'Agent
msg)

        prohibited :: m ()
        prohibited :: m ()
prohibited = ACommand 'Agent -> m ()
notify (ACommand 'Agent -> m ())
-> (AgentErrorType -> ACommand 'Agent) -> AgentErrorType -> m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. AgentErrorType -> ACommand 'Agent
ERR (AgentErrorType -> m ()) -> AgentErrorType -> m ()
forall a b. (a -> b) -> a -> b
$ SMPAgentError -> AgentErrorType
AGENT SMPAgentError
A_PROHIBITED

        smpConfirmation :: SenderPublicKey -> ConnInfo -> m ()
        smpConfirmation :: SenderPublicKey -> ByteString -> m ()
smpConfirmation SenderPublicKey
senderKey ByteString
cInfo = do
          ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
logServer ByteString
"<--" AgentClient
c SMPServer
srv ByteString
rId ByteString
"MSG <KEY>"
          case QueueStatus
status of
            QueueStatus
New -> case SConnType c
cType of
              SConnType c
SCRcv -> do
                TVar ChaChaDRG
g <- (Env -> TVar ChaChaDRG) -> m (TVar ChaChaDRG)
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks Env -> TVar ChaChaDRG
idsDrg
                let newConfirmation :: NewConfirmation
newConfirmation = NewConfirmation :: ByteString -> SenderPublicKey -> ByteString -> NewConfirmation
NewConfirmation {ByteString
$sel:connId:NewConfirmation :: ByteString
connId :: ByteString
connId, SenderPublicKey
$sel:senderKey:NewConfirmation :: SenderPublicKey
senderKey :: SenderPublicKey
senderKey, $sel:senderConnInfo:NewConfirmation :: ByteString
senderConnInfo = ByteString
cInfo}
                ByteString
confId <- (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ByteString)
-> m ByteString
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ByteString)
 -> m ByteString)
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ByteString)
-> m ByteString
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> TVar ChaChaDRG -> NewConfirmation -> m' ByteString
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> TVar ChaChaDRG -> NewConfirmation -> m ByteString
createConfirmation SQLiteStore
st TVar ChaChaDRG
g NewConfirmation
newConfirmation
                ACommand 'Agent -> m ()
notify (ACommand 'Agent -> m ()) -> ACommand 'Agent -> m ()
forall a b. (a -> b) -> a -> b
$ ByteString -> ByteString -> ACommand 'Agent
REQ ByteString
confId ByteString
cInfo
              SConnType c
SCDuplex -> do
                ACommand 'Agent -> m ()
notify (ACommand 'Agent -> m ()) -> ACommand 'Agent -> m ()
forall a b. (a -> b) -> a -> b
$ ByteString -> ACommand 'Agent
INFO ByteString
cInfo
                AgentClient -> RcvQueue -> SenderPublicKey -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> RcvQueue -> SenderPublicKey -> m ()
processConfirmation AgentClient
c RcvQueue
rq SenderPublicKey
senderKey
              SConnType c
_ -> m ()
prohibited
            QueueStatus
_ -> m ()
prohibited

        helloMsg :: SenderPublicKey -> ByteString -> m ()
        helloMsg :: SenderPublicKey -> ByteString -> m ()
helloMsg SenderPublicKey
verifyKey ByteString
msgBody = do
          ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
logServer ByteString
"<--" AgentClient
c SMPServer
srv ByteString
rId ByteString
"MSG <HELLO>"
          case QueueStatus
status of
            QueueStatus
Active -> m ()
prohibited
            QueueStatus
_ -> do
              m ByteString -> m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (m ByteString -> m ()) -> m ByteString -> m ()
forall a b. (a -> b) -> a -> b
$ Maybe SenderPublicKey -> ByteString -> m ByteString
forall (m :: * -> *).
AgentMonad m =>
Maybe SenderPublicKey -> ByteString -> m ByteString
verifyMessage (SenderPublicKey -> Maybe SenderPublicKey
forall a. a -> Maybe a
Just SenderPublicKey
verifyKey) ByteString
msgBody
              (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> RcvQueue -> SenderPublicKey -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> RcvQueue -> SenderPublicKey -> m ()
setRcvQueueActive SQLiteStore
st RcvQueue
rq SenderPublicKey
verifyKey
              case SConnType c
cType of
                SConnType c
SCDuplex -> AgentClient -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
notifyConnected AgentClient
c ByteString
connId
                SConnType c
_ -> () -> m ()
forall (f :: * -> *) a. Applicative f => a -> f a
pure ()

        replyMsg :: SMPQueueInfo -> m ()
        replyMsg :: SMPQueueInfo -> m ()
replyMsg SMPQueueInfo
qInfo = do
          ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
logServer ByteString
"<--" AgentClient
c SMPServer
srv ByteString
rId ByteString
"MSG <REPLY>"
          case SConnType c
cType of
            SConnType c
SCRcv -> do
              AcceptedConfirmation {ByteString
$sel:ownConnInfo:AcceptedConfirmation :: AcceptedConfirmation -> ByteString
ownConnInfo :: ByteString
ownConnInfo} <- (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' AcceptedConfirmation)
-> m AcceptedConfirmation
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' AcceptedConfirmation
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m AcceptedConfirmation
`getAcceptedConfirmation` ByteString
connId)
              (SndQueue
sq, SenderPublicKey
senderKey, SenderPublicKey
verifyKey) <- SMPQueueInfo -> m (SndQueue, SenderPublicKey, SenderPublicKey)
forall (m :: * -> *).
(MonadUnliftIO m, MonadReader Env m) =>
SMPQueueInfo -> m (SndQueue, SenderPublicKey, SenderPublicKey)
newSndQueue SMPQueueInfo
qInfo
              (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> ByteString -> SndQueue -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> SndQueue -> m ()
upgradeRcvConnToDuplex SQLiteStore
st ByteString
connId SndQueue
sq
              AgentClient -> SndQueue -> SenderPublicKey -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SndQueue -> SenderPublicKey -> ByteString -> m ()
confirmQueue AgentClient
c SndQueue
sq SenderPublicKey
senderKey ByteString
ownConnInfo
              (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore -> ByteString -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> m ()
`removeConfirmations` ByteString
connId)
              AgentConfig
cfg <- (Env -> AgentConfig) -> m AgentConfig
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks Env -> AgentConfig
config
              AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
activateQueueInitiating AgentClient
c ByteString
connId SndQueue
sq SenderPublicKey
verifyKey (RetryInterval -> m ()) -> RetryInterval -> m ()
forall a b. (a -> b) -> a -> b
$ AgentConfig -> RetryInterval
retryInterval AgentConfig
cfg
            SConnType c
_ -> m ()
prohibited

        agentClientMsg :: PrevRcvMsgHash -> (ExternalSndId, ExternalSndTs) -> (BrokerId, BrokerTs) -> MsgBody -> MsgHash -> m ()
        agentClientMsg :: ByteString
-> (AgentMsgId, UTCTime)
-> (ByteString, UTCTime)
-> ByteString
-> ByteString
-> m ()
agentClientMsg ByteString
externalPrevSndHash (AgentMsgId, UTCTime)
sender (ByteString, UTCTime)
broker ByteString
msgBody ByteString
internalHash = do
          ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
ByteString
-> AgentClient -> SMPServer -> ByteString -> ByteString -> m ()
logServer ByteString
"<--" AgentClient
c SMPServer
srv ByteString
rId ByteString
"MSG <MSG>"
          UTCTime
internalTs <- IO UTCTime -> m UTCTime
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO IO UTCTime
getCurrentTime
          (InternalId
internalId, InternalRcvId
internalRcvId, AgentMsgId
prevExtSndId, ByteString
prevRcvMsgHash) <- (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore
 -> m' (InternalId, InternalRcvId, AgentMsgId, ByteString))
-> m (InternalId, InternalRcvId, AgentMsgId, ByteString)
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore (SQLiteStore
-> ByteString
-> m' (InternalId, InternalRcvId, AgentMsgId, ByteString)
forall s (m :: * -> *).
MonadAgentStore s m =>
s
-> ByteString
-> m (InternalId, InternalRcvId, AgentMsgId, ByteString)
`updateRcvIds` ByteString
connId)
          let integrity :: MsgIntegrity
integrity = AgentMsgId
-> AgentMsgId -> ByteString -> ByteString -> MsgIntegrity
checkMsgIntegrity AgentMsgId
prevExtSndId ((AgentMsgId, UTCTime) -> AgentMsgId
forall a b. (a, b) -> a
fst (AgentMsgId, UTCTime)
sender) ByteString
prevRcvMsgHash ByteString
externalPrevSndHash
              recipient :: (AgentMsgId, UTCTime)
recipient = (InternalId -> AgentMsgId
unId InternalId
internalId, UTCTime
internalTs)
              msgMeta :: MsgMeta
msgMeta = MsgMeta :: MsgIntegrity
-> (AgentMsgId, UTCTime)
-> (ByteString, UTCTime)
-> (AgentMsgId, UTCTime)
-> MsgMeta
MsgMeta {MsgIntegrity
integrity :: MsgIntegrity
integrity :: MsgIntegrity
integrity, (AgentMsgId, UTCTime)
recipient :: (AgentMsgId, UTCTime)
recipient :: (AgentMsgId, UTCTime)
recipient, (AgentMsgId, UTCTime)
sender :: (AgentMsgId, UTCTime)
sender :: (AgentMsgId, UTCTime)
sender, (ByteString, UTCTime)
broker :: (ByteString, UTCTime)
broker :: (ByteString, UTCTime)
broker}
              rcvMsg :: RcvMsgData
rcvMsg = RcvMsgData :: MsgMeta
-> ByteString
-> InternalRcvId
-> ByteString
-> ByteString
-> RcvMsgData
RcvMsgData {ByteString
MsgMeta
InternalRcvId
$sel:externalPrevSndHash:RcvMsgData :: ByteString
$sel:internalHash:RcvMsgData :: ByteString
$sel:internalRcvId:RcvMsgData :: InternalRcvId
$sel:msgBody:RcvMsgData :: ByteString
$sel:msgMeta:RcvMsgData :: MsgMeta
msgMeta :: MsgMeta
internalRcvId :: InternalRcvId
internalHash :: ByteString
msgBody :: ByteString
externalPrevSndHash :: ByteString
..}
          (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> ByteString -> RcvMsgData -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> ByteString -> RcvMsgData -> m ()
createRcvMsg SQLiteStore
st ByteString
connId RcvMsgData
rcvMsg
          ACommand 'Agent -> m ()
notify (ACommand 'Agent -> m ()) -> ACommand 'Agent -> m ()
forall a b. (a -> b) -> a -> b
$ MsgMeta -> ByteString -> ACommand 'Agent
MSG MsgMeta
msgMeta ByteString
msgBody

        checkMsgIntegrity :: PrevExternalSndId -> ExternalSndId -> PrevRcvMsgHash -> ByteString -> MsgIntegrity
        checkMsgIntegrity :: AgentMsgId
-> AgentMsgId -> ByteString -> ByteString -> MsgIntegrity
checkMsgIntegrity AgentMsgId
prevExtSndId AgentMsgId
extSndId ByteString
internalPrevMsgHash ByteString
receivedPrevMsgHash
          | AgentMsgId
extSndId AgentMsgId -> AgentMsgId -> Bool
forall a. Eq a => a -> a -> Bool
== AgentMsgId
prevExtSndId AgentMsgId -> AgentMsgId -> AgentMsgId
forall a. Num a => a -> a -> a
+ AgentMsgId
1 Bool -> Bool -> Bool
&& ByteString
internalPrevMsgHash ByteString -> ByteString -> Bool
forall a. Eq a => a -> a -> Bool
== ByteString
receivedPrevMsgHash = MsgIntegrity
MsgOk
          | AgentMsgId
extSndId AgentMsgId -> AgentMsgId -> Bool
forall a. Ord a => a -> a -> Bool
< AgentMsgId
prevExtSndId = MsgErrorType -> MsgIntegrity
MsgError (MsgErrorType -> MsgIntegrity) -> MsgErrorType -> MsgIntegrity
forall a b. (a -> b) -> a -> b
$ AgentMsgId -> MsgErrorType
MsgBadId AgentMsgId
extSndId
          | AgentMsgId
extSndId AgentMsgId -> AgentMsgId -> Bool
forall a. Eq a => a -> a -> Bool
== AgentMsgId
prevExtSndId = MsgErrorType -> MsgIntegrity
MsgError MsgErrorType
MsgDuplicate -- ? deduplicate
          | AgentMsgId
extSndId AgentMsgId -> AgentMsgId -> Bool
forall a. Ord a => a -> a -> Bool
> AgentMsgId
prevExtSndId AgentMsgId -> AgentMsgId -> AgentMsgId
forall a. Num a => a -> a -> a
+ AgentMsgId
1 = MsgErrorType -> MsgIntegrity
MsgError (MsgErrorType -> MsgIntegrity) -> MsgErrorType -> MsgIntegrity
forall a b. (a -> b) -> a -> b
$ AgentMsgId -> AgentMsgId -> MsgErrorType
MsgSkipped (AgentMsgId
prevExtSndId AgentMsgId -> AgentMsgId -> AgentMsgId
forall a. Num a => a -> a -> a
+ AgentMsgId
1) (AgentMsgId
extSndId AgentMsgId -> AgentMsgId -> AgentMsgId
forall a. Num a => a -> a -> a
- AgentMsgId
1)
          | ByteString
internalPrevMsgHash ByteString -> ByteString -> Bool
forall a. Eq a => a -> a -> Bool
/= ByteString
receivedPrevMsgHash = MsgErrorType -> MsgIntegrity
MsgError MsgErrorType
MsgBadHash
          | Bool
otherwise = MsgErrorType -> MsgIntegrity
MsgError MsgErrorType
MsgDuplicate -- this case is not possible

confirmQueue :: AgentMonad m => AgentClient -> SndQueue -> SenderPublicKey -> ConnInfo -> m ()
confirmQueue :: AgentClient -> SndQueue -> SenderPublicKey -> ByteString -> m ()
confirmQueue AgentClient
c SndQueue
sq SenderPublicKey
senderKey ByteString
cInfo = do
  AgentClient -> SndQueue -> SenderPublicKey -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SndQueue -> SenderPublicKey -> ByteString -> m ()
sendConfirmation AgentClient
c SndQueue
sq SenderPublicKey
senderKey ByteString
cInfo
  (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> SndQueue -> QueueStatus -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> SndQueue -> QueueStatus -> m ()
setSndQueueStatus SQLiteStore
st SndQueue
sq QueueStatus
Confirmed

activateQueueInitiating :: AgentMonad m => AgentClient -> ConnId -> SndQueue -> VerificationKey -> RetryInterval -> m ()
activateQueueInitiating :: AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
activateQueueInitiating AgentClient
c ByteString
connId SndQueue
sq SenderPublicKey
verifyKey RetryInterval
retryInterval =
  AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
-> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
-> m ()
activateQueue AgentClient
c ByteString
connId SndQueue
sq SenderPublicKey
verifyKey RetryInterval
retryInterval (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ AgentClient -> ByteString -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> ByteString -> m ()
notifyConnected AgentClient
c ByteString
connId

activateQueue :: forall m. AgentMonad m => AgentClient -> ConnId -> SndQueue -> VerificationKey -> RetryInterval -> m () -> m ()
activateQueue :: AgentClient
-> ByteString
-> SndQueue
-> SenderPublicKey
-> RetryInterval
-> m ()
-> m ()
activateQueue AgentClient
c ByteString
connId SndQueue
sq SenderPublicKey
verifyKey RetryInterval
retryInterval m ()
afterActivation =
  AgentClient -> ByteString -> m (Maybe (Async ()))
forall (m :: * -> *).
MonadUnliftIO m =>
AgentClient -> ByteString -> m (Maybe (Async ()))
getActivation AgentClient
c ByteString
connId m (Maybe (Async ())) -> (Maybe (Async ()) -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
    Maybe (Async ())
Nothing -> m () -> m (Async ())
forall (m :: * -> *) a. MonadUnliftIO m => m a -> m (Async a)
async m ()
runActivation m (Async ()) -> (Async () -> m ()) -> m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= AgentClient -> ByteString -> Async () -> m ()
forall (m :: * -> *).
MonadUnliftIO m =>
AgentClient -> ByteString -> Async () -> m ()
addActivation AgentClient
c ByteString
connId
    Just Async ()
_ -> () -> m ()
forall (f :: * -> *) a. Applicative f => a -> f a
pure ()
  where
    runActivation :: m ()
    runActivation :: m ()
runActivation = do
      AgentClient -> SndQueue -> SenderPublicKey -> RetryInterval -> m ()
forall (m :: * -> *).
AgentMonad m =>
AgentClient -> SndQueue -> SenderPublicKey -> RetryInterval -> m ()
sendHello AgentClient
c SndQueue
sq SenderPublicKey
verifyKey RetryInterval
retryInterval
      (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> SndQueue -> QueueStatus -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> SndQueue -> QueueStatus -> m ()
setSndQueueStatus SQLiteStore
st SndQueue
sq QueueStatus
Active
      AgentClient -> ByteString -> m ()
forall (m :: * -> *).
MonadUnliftIO m =>
AgentClient -> ByteString -> m ()
removeActivation AgentClient
c ByteString
connId
      m ()
removeVerificationKey
      m ()
afterActivation
    removeVerificationKey :: m ()
    removeVerificationKey :: m ()
removeVerificationKey =
      let safeSignKey :: SignatureKey
safeSignKey = SignatureKey -> SignatureKey
C.removePublicKey (SignatureKey -> SignatureKey) -> SignatureKey -> SignatureKey
forall a b. (a -> b) -> a -> b
$ SndQueue -> SignatureKey
signKey SndQueue
sq
       in (forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' ())
-> m ()
forall (m :: * -> *) a.
AgentMonad m =>
(forall (m' :: * -> *).
 (MonadUnliftIO m', MonadError StoreError m') =>
 SQLiteStore -> m' a)
-> m a
withStore ((forall (m' :: * -> *).
  (MonadUnliftIO m', MonadError StoreError m') =>
  SQLiteStore -> m' ())
 -> m ())
-> (forall (m' :: * -> *).
    (MonadUnliftIO m', MonadError StoreError m') =>
    SQLiteStore -> m' ())
-> m ()
forall a b. (a -> b) -> a -> b
$ \SQLiteStore
st -> SQLiteStore -> SndQueue -> SignatureKey -> m' ()
forall s (m :: * -> *).
MonadAgentStore s m =>
s -> SndQueue -> SignatureKey -> m ()
updateSignKey SQLiteStore
st SndQueue
sq SignatureKey
safeSignKey

notifyConnected :: AgentMonad m => AgentClient -> ConnId -> m ()
notifyConnected :: AgentClient -> ByteString -> m ()
notifyConnected AgentClient
c ByteString
connId = STM () -> m ()
forall (m :: * -> *) a. MonadIO m => STM a -> m a
atomically (STM () -> m ()) -> STM () -> m ()
forall a b. (a -> b) -> a -> b
$ TBQueue (ATransmission 'Agent) -> ATransmission 'Agent -> STM ()
forall a. TBQueue a -> a -> STM ()
writeTBQueue (AgentClient -> TBQueue (ATransmission 'Agent)
subQ AgentClient
c) (ByteString
"", ByteString
connId, ACommand 'Agent
CON)

newSndQueue ::
  (MonadUnliftIO m, MonadReader Env m) => SMPQueueInfo -> m (SndQueue, SenderPublicKey, VerificationKey)
newSndQueue :: SMPQueueInfo -> m (SndQueue, SenderPublicKey, SenderPublicKey)
newSndQueue (SMPQueueInfo SMPServer
smpServer ByteString
senderId SenderPublicKey
encryptKey) = do
  Int
size <- (Env -> Int) -> m Int
forall r (m :: * -> *) a. MonadReader r m => (r -> a) -> m a
asks ((Env -> Int) -> m Int) -> (Env -> Int) -> m Int
forall a b. (a -> b) -> a -> b
$ AgentConfig -> Int
rsaKeySize (AgentConfig -> Int) -> (Env -> AgentConfig) -> Env -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Env -> AgentConfig
config
  (SenderPublicKey
senderKey, SenderPrivateKey
sndPrivateKey) <- IO (SenderPublicKey, SenderPrivateKey)
-> m (SenderPublicKey, SenderPrivateKey)
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO (SenderPublicKey, SenderPrivateKey)
 -> m (SenderPublicKey, SenderPrivateKey))
-> IO (SenderPublicKey, SenderPrivateKey)
-> m (SenderPublicKey, SenderPrivateKey)
forall a b. (a -> b) -> a -> b
$ Int -> IO (SenderPublicKey, SenderPrivateKey)
forall k. PrivateKey k => Int -> IO (KeyPair k)
C.generateKeyPair Int
size
  (SenderPublicKey
verifyKey, SignatureKey
signKey) <- IO (SenderPublicKey, SignatureKey)
-> m (SenderPublicKey, SignatureKey)
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO (SenderPublicKey, SignatureKey)
 -> m (SenderPublicKey, SignatureKey))
-> IO (SenderPublicKey, SignatureKey)
-> m (SenderPublicKey, SignatureKey)
forall a b. (a -> b) -> a -> b
$ Int -> IO (SenderPublicKey, SignatureKey)
forall k. PrivateKey k => Int -> IO (KeyPair k)
C.generateKeyPair Int
size
  let sndQueue :: SndQueue
sndQueue =
        SndQueue :: SMPServer
-> ByteString
-> SenderPrivateKey
-> SenderPublicKey
-> SignatureKey
-> QueueStatus
-> SndQueue
SndQueue
          { $sel:server:SndQueue :: SMPServer
server = SMPServer
smpServer,
            $sel:sndId:SndQueue :: ByteString
sndId = ByteString
senderId,
            SenderPrivateKey
$sel:sndPrivateKey:SndQueue :: SenderPrivateKey
sndPrivateKey :: SenderPrivateKey
sndPrivateKey,
            SenderPublicKey
$sel:encryptKey:SndQueue :: SenderPublicKey
encryptKey :: SenderPublicKey
encryptKey,
            SignatureKey
signKey :: SignatureKey
$sel:signKey:SndQueue :: SignatureKey
signKey,
            $sel:status:SndQueue :: QueueStatus
status = QueueStatus
New
          }
  (SndQueue, SenderPublicKey, SenderPublicKey)
-> m (SndQueue, SenderPublicKey, SenderPublicKey)
forall (m :: * -> *) a. Monad m => a -> m a
return (SndQueue
sndQueue, SenderPublicKey
senderKey, SenderPublicKey
verifyKey)