mirror of
https://github.com/simplex-chat/simplexmq.git
synced 2026-06-09 17:13:06 +00:00
850d2fa423
* ntf server: PostgreSQL database storage * ntf server: import/export stubs * ntf server postgres db functions * some notifications tests pass * notifications tests pass * import/export notification store logs * fix ntf server CLI * log in parralel * update subscription statuses using executeMany * fix import/export * refactor * fix queries * prohibit token_id and subscription_id updates, dont update tokens and subscriptions on conflict, improve server insertion, remove duplicate tokens for import, remove subscriptions without tokens for import * comment * remame * increase test delay
456 lines
18 KiB
Haskell
456 lines
18 KiB
Haskell
{-# LANGUAGE CPP #-}
|
|
{-# LANGUAGE AllowAmbiguousTypes #-}
|
|
{-# LANGUAGE DataKinds #-}
|
|
{-# LANGUAGE DuplicateRecordFields #-}
|
|
{-# LANGUAGE FlexibleContexts #-}
|
|
{-# LANGUAGE GADTs #-}
|
|
{-# LANGUAGE KindSignatures #-}
|
|
{-# LANGUAGE LambdaCase #-}
|
|
{-# LANGUAGE NamedFieldPuns #-}
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
{-# LANGUAGE ScopedTypeVariables #-}
|
|
{-# LANGUAGE StrictData #-}
|
|
{-# LANGUAGE TypeFamilies #-}
|
|
{-# LANGUAGE TypeApplications #-}
|
|
{-# LANGUAGE TypeOperators #-}
|
|
#if __GLASGOW_HASKELL__ == 810
|
|
{-# LANGUAGE UndecidableInstances #-}
|
|
#endif
|
|
{-# OPTIONS_GHC -fno-warn-ambiguous-fields #-}
|
|
|
|
module Simplex.Messaging.Server.Env.STM where
|
|
|
|
import Control.Concurrent (ThreadId)
|
|
import Control.Logger.Simple
|
|
import Control.Monad
|
|
import qualified Crypto.PubKey.RSA as RSA
|
|
import Crypto.Random
|
|
import Data.ByteString.Char8 (ByteString)
|
|
import Data.Int (Int64)
|
|
import Data.IntMap.Strict (IntMap)
|
|
import qualified Data.IntMap.Strict as IM
|
|
import Data.Kind (Constraint)
|
|
import Data.List (intercalate)
|
|
import Data.List.NonEmpty (NonEmpty)
|
|
import Data.Maybe (isJust)
|
|
import qualified Data.Text as T
|
|
import Data.Time.Clock (getCurrentTime, nominalDay)
|
|
import Data.Time.Clock.System (SystemTime)
|
|
import qualified Data.X509 as X
|
|
import Data.X509.Validation (Fingerprint (..))
|
|
import GHC.TypeLits (TypeError)
|
|
import qualified GHC.TypeLits as TE
|
|
import Network.Socket (ServiceName)
|
|
import qualified Network.TLS as T
|
|
import Numeric.Natural
|
|
import Simplex.Messaging.Agent.Lock
|
|
import Simplex.Messaging.Agent.Store.Shared (MigrationConfirmation (..))
|
|
import Simplex.Messaging.Client.Agent (SMPClientAgent, SMPClientAgentConfig, newSMPClientAgent)
|
|
import Simplex.Messaging.Crypto (KeyHash (..))
|
|
import qualified Simplex.Messaging.Crypto as C
|
|
import Simplex.Messaging.Protocol
|
|
import Simplex.Messaging.Server.Expiration
|
|
import Simplex.Messaging.Server.Information
|
|
import Simplex.Messaging.Server.MsgStore.Journal
|
|
import Simplex.Messaging.Server.MsgStore.STM
|
|
import Simplex.Messaging.Server.MsgStore.Types
|
|
import Simplex.Messaging.Server.NtfStore
|
|
import Simplex.Messaging.Server.QueueStore
|
|
import Simplex.Messaging.Server.QueueStore.Postgres.Config
|
|
import Simplex.Messaging.Server.QueueStore.STM (STMQueueStore, setStoreLog)
|
|
import Simplex.Messaging.Server.QueueStore.Types
|
|
import Simplex.Messaging.Server.Stats
|
|
import Simplex.Messaging.Server.StoreLog
|
|
import Simplex.Messaging.Server.StoreLog.ReadWrite
|
|
import Simplex.Messaging.TMap (TMap)
|
|
import qualified Simplex.Messaging.TMap as TM
|
|
import Simplex.Messaging.Transport (ATransport, VersionRangeSMP, VersionSMP)
|
|
import Simplex.Messaging.Transport.Server
|
|
import System.Directory (doesFileExist)
|
|
import System.Exit (exitFailure)
|
|
import System.IO (IOMode (..))
|
|
import System.Mem.Weak (Weak)
|
|
import UnliftIO.STM
|
|
|
|
data ServerConfig = ServerConfig
|
|
{ transports :: [(ServiceName, ATransport, AddHTTP)],
|
|
smpHandshakeTimeout :: Int,
|
|
tbqSize :: Natural,
|
|
msgQueueQuota :: Int,
|
|
maxJournalMsgCount :: Int,
|
|
maxJournalStateLines :: Int,
|
|
queueIdBytes :: Int,
|
|
msgIdBytes :: Int,
|
|
serverStoreCfg :: AServerStoreCfg,
|
|
storeNtfsFile :: Maybe FilePath,
|
|
-- | set to False to prohibit creating new queues
|
|
allowNewQueues :: Bool,
|
|
-- | simple password that the clients need to pass in handshake to be able to create new queues
|
|
newQueueBasicAuth :: Maybe BasicAuth,
|
|
-- | control port passwords,
|
|
controlPortUserAuth :: Maybe BasicAuth,
|
|
controlPortAdminAuth :: Maybe BasicAuth,
|
|
-- | time after which the messages can be removed from the queues and check interval, seconds
|
|
messageExpiration :: Maybe ExpirationConfig,
|
|
expireMessagesOnStart :: Bool,
|
|
-- | interval of inactivity after which journal queue is closed
|
|
idleQueueInterval :: Int64,
|
|
-- | notification expiration interval (seconds)
|
|
notificationExpiration :: ExpirationConfig,
|
|
-- | time after which the socket with inactive client can be disconnected (without any messages or commands, incl. PING),
|
|
-- and check interval, seconds
|
|
inactiveClientExpiration :: Maybe ExpirationConfig,
|
|
-- | log SMP server usage statistics, only aggregates are logged, seconds
|
|
logStatsInterval :: Maybe Int64,
|
|
-- | time of the day when the stats are logged first, to log at consistent times,
|
|
-- irrespective of when the server is started (seconds from 00:00 UTC)
|
|
logStatsStartTime :: Int64,
|
|
-- | file to log stats
|
|
serverStatsLogFile :: FilePath,
|
|
-- | file to save and restore stats
|
|
serverStatsBackupFile :: Maybe FilePath,
|
|
-- | interval and file to save prometheus metrics
|
|
prometheusInterval :: Maybe Int,
|
|
prometheusMetricsFile :: FilePath,
|
|
-- | notification delivery interval
|
|
ntfDeliveryInterval :: Int,
|
|
-- | interval between sending pending END events to unsubscribed clients, seconds
|
|
pendingENDInterval :: Int,
|
|
smpCredentials :: ServerCredentials,
|
|
httpCredentials :: Maybe ServerCredentials,
|
|
-- | SMP client-server protocol version range
|
|
smpServerVRange :: VersionRangeSMP,
|
|
-- | TCP transport config
|
|
transportConfig :: TransportServerConfig,
|
|
-- | run listener on control port
|
|
controlPort :: Maybe ServiceName,
|
|
-- | SMP proxy config
|
|
smpAgentCfg :: SMPClientAgentConfig,
|
|
allowSMPProxy :: Bool, -- auth is the same with `newQueueBasicAuth`
|
|
serverClientConcurrency :: Int,
|
|
-- | server public information
|
|
information :: Maybe ServerPublicInfo,
|
|
startOptions :: StartOptions
|
|
}
|
|
|
|
data StartOptions = StartOptions
|
|
{ maintenance :: Bool,
|
|
compactLog :: Bool,
|
|
skipWarnings :: Bool,
|
|
confirmMigrations :: MigrationConfirmation
|
|
}
|
|
|
|
defMsgExpirationDays :: Int64
|
|
defMsgExpirationDays = 21
|
|
|
|
defaultMessageExpiration :: ExpirationConfig
|
|
defaultMessageExpiration =
|
|
ExpirationConfig
|
|
{ ttl = defMsgExpirationDays * 86400, -- seconds
|
|
checkInterval = 7200 -- seconds, 2 hours
|
|
}
|
|
|
|
defaultIdleQueueInterval :: Int64
|
|
defaultIdleQueueInterval = 14400 -- seconds, 4 hours
|
|
|
|
defNtfExpirationHours :: Int64
|
|
defNtfExpirationHours = 24
|
|
|
|
defaultNtfExpiration :: ExpirationConfig
|
|
defaultNtfExpiration =
|
|
ExpirationConfig
|
|
{ ttl = defNtfExpirationHours * 3600, -- seconds
|
|
checkInterval = 3600 -- seconds, 1 hour
|
|
}
|
|
|
|
defaultInactiveClientExpiration :: ExpirationConfig
|
|
defaultInactiveClientExpiration =
|
|
ExpirationConfig
|
|
{ ttl = 21600, -- seconds, 6 hours
|
|
checkInterval = 3600 -- seconds, 1 hours
|
|
}
|
|
|
|
defaultProxyClientConcurrency :: Int
|
|
defaultProxyClientConcurrency = 32
|
|
|
|
journalMsgStoreDepth :: Int
|
|
journalMsgStoreDepth = 5
|
|
|
|
defaultMaxJournalStateLines :: Int
|
|
defaultMaxJournalStateLines = 16
|
|
|
|
defaultMaxJournalMsgCount :: Int
|
|
defaultMaxJournalMsgCount = 256
|
|
|
|
defaultMsgQueueQuota :: Int
|
|
defaultMsgQueueQuota = 128
|
|
|
|
defaultStateTailSize :: Int
|
|
defaultStateTailSize = 512
|
|
|
|
data Env = Env
|
|
{ config :: ServerConfig,
|
|
serverActive :: TVar Bool,
|
|
serverInfo :: ServerInformation,
|
|
server :: Server,
|
|
serverIdentity :: KeyHash,
|
|
msgStore :: AMsgStore,
|
|
ntfStore :: NtfStore,
|
|
random :: TVar ChaChaDRG,
|
|
tlsServerCreds :: T.Credential,
|
|
httpServerCreds :: Maybe T.Credential,
|
|
serverStats :: ServerStats,
|
|
sockets :: TVar [(ServiceName, SocketState)],
|
|
clientSeq :: TVar ClientId,
|
|
clients :: TVar (IntMap (Maybe AClient)),
|
|
proxyAgent :: ProxyAgent -- senders served on this proxy
|
|
}
|
|
|
|
type family SupportedStore (qs :: QSType) (ms :: MSType) :: Constraint where
|
|
SupportedStore 'QSMemory 'MSMemory = ()
|
|
SupportedStore 'QSMemory 'MSJournal = ()
|
|
SupportedStore 'QSPostgres 'MSJournal = ()
|
|
SupportedStore 'QSPostgres 'MSMemory =
|
|
(Int ~ Bool, TypeError ('TE.Text "Storing messages in memory with Postgres DB is not supported"))
|
|
|
|
data AStoreType = forall qs ms. SupportedStore qs ms => ASType (SQSType qs) (SMSType ms)
|
|
|
|
data ServerStoreCfg qs ms where
|
|
SSCMemory :: Maybe StorePaths -> ServerStoreCfg 'QSMemory 'MSMemory
|
|
SSCMemoryJournal :: {storeLogFile :: FilePath, storeMsgsPath :: FilePath} -> ServerStoreCfg 'QSMemory 'MSJournal
|
|
SSCDatabaseJournal :: {storeCfg :: PostgresStoreCfg, storeMsgsPath' :: FilePath} -> ServerStoreCfg 'QSPostgres 'MSJournal
|
|
|
|
data StorePaths = StorePaths {storeLogFile :: FilePath, storeMsgsFile :: Maybe FilePath}
|
|
|
|
data AServerStoreCfg = forall qs ms. SupportedStore qs ms => ASSCfg (SQSType qs) (SMSType ms) (ServerStoreCfg qs ms)
|
|
|
|
type family MsgStore (qs :: QSType) (ms :: MSType) where
|
|
MsgStore 'QSMemory 'MSMemory = STMMsgStore
|
|
MsgStore qs 'MSJournal = JournalMsgStore qs
|
|
|
|
data AMsgStore =
|
|
forall qs ms. (SupportedStore qs ms, MsgStoreClass (MsgStore qs ms)) =>
|
|
AMS (SQSType qs) (SMSType ms) (MsgStore qs ms)
|
|
|
|
type Subscribed = Bool
|
|
|
|
data Server = Server
|
|
{ subscribedQ :: TQueue (RecipientId, ClientId, Subscribed),
|
|
subscribers :: TMap RecipientId (TVar AClient),
|
|
ntfSubscribedQ :: TQueue (NotifierId, ClientId, Subscribed),
|
|
notifiers :: TMap NotifierId (TVar AClient),
|
|
subClients :: TVar (IntMap AClient), -- clients with SMP subscriptions
|
|
ntfSubClients :: TVar (IntMap AClient), -- clients with Ntf subscriptions
|
|
pendingSubEvents :: TVar (IntMap (NonEmpty (RecipientId, Subscribed))),
|
|
pendingNtfSubEvents :: TVar (IntMap (NonEmpty (NotifierId, Subscribed))),
|
|
savingLock :: Lock
|
|
}
|
|
|
|
newtype ProxyAgent = ProxyAgent
|
|
{ smpAgent :: SMPClientAgent
|
|
}
|
|
|
|
type ClientId = Int
|
|
|
|
data AClient = forall qs ms. MsgStoreClass (MsgStore qs ms) => AClient (SQSType qs) (SMSType ms) (Client (MsgStore qs ms))
|
|
|
|
clientId' :: AClient -> ClientId
|
|
clientId' (AClient _ _ Client {clientId}) = clientId
|
|
{-# INLINE clientId' #-}
|
|
|
|
data Client s = Client
|
|
{ clientId :: ClientId,
|
|
subscriptions :: TMap RecipientId Sub,
|
|
ntfSubscriptions :: TMap NotifierId (),
|
|
rcvQ :: TBQueue (NonEmpty (Maybe (StoreQueue s, QueueRec), Transmission Cmd)),
|
|
sndQ :: TBQueue (NonEmpty (Transmission BrokerMsg)),
|
|
msgQ :: TBQueue (NonEmpty (Transmission BrokerMsg)),
|
|
procThreads :: TVar Int,
|
|
endThreads :: TVar (IntMap (Weak ThreadId)),
|
|
endThreadSeq :: TVar Int,
|
|
thVersion :: VersionSMP,
|
|
sessionId :: ByteString,
|
|
connected :: TVar Bool,
|
|
createdAt :: SystemTime,
|
|
rcvActiveAt :: TVar SystemTime,
|
|
sndActiveAt :: TVar SystemTime
|
|
}
|
|
|
|
data ServerSub = ServerSub (TVar SubscriptionThread) | ProhibitSub
|
|
|
|
data SubscriptionThread = NoSub | SubPending | SubThread (Weak ThreadId)
|
|
|
|
data Sub = Sub
|
|
{ subThread :: ServerSub, -- Nothing value indicates that sub
|
|
delivered :: TMVar MsgId
|
|
}
|
|
|
|
newServer :: IO Server
|
|
newServer = do
|
|
subscribedQ <- newTQueueIO
|
|
subscribers <- TM.emptyIO
|
|
ntfSubscribedQ <- newTQueueIO
|
|
notifiers <- TM.emptyIO
|
|
subClients <- newTVarIO IM.empty
|
|
ntfSubClients <- newTVarIO IM.empty
|
|
pendingSubEvents <- newTVarIO IM.empty
|
|
pendingNtfSubEvents <- newTVarIO IM.empty
|
|
savingLock <- createLockIO
|
|
return Server {subscribedQ, subscribers, ntfSubscribedQ, notifiers, subClients, ntfSubClients, pendingSubEvents, pendingNtfSubEvents, savingLock}
|
|
|
|
newClient :: SQSType qs -> SMSType ms -> ClientId -> Natural -> VersionSMP -> ByteString -> SystemTime -> IO (Client (MsgStore qs ms))
|
|
newClient _ _ clientId qSize thVersion sessionId createdAt = do
|
|
subscriptions <- TM.emptyIO
|
|
ntfSubscriptions <- TM.emptyIO
|
|
rcvQ <- newTBQueueIO qSize
|
|
sndQ <- newTBQueueIO qSize
|
|
msgQ <- newTBQueueIO qSize
|
|
procThreads <- newTVarIO 0
|
|
endThreads <- newTVarIO IM.empty
|
|
endThreadSeq <- newTVarIO 0
|
|
connected <- newTVarIO True
|
|
rcvActiveAt <- newTVarIO createdAt
|
|
sndActiveAt <- newTVarIO createdAt
|
|
return Client {clientId, subscriptions, ntfSubscriptions, rcvQ, sndQ, msgQ, procThreads, endThreads, endThreadSeq, thVersion, sessionId, connected, createdAt, rcvActiveAt, sndActiveAt}
|
|
|
|
newSubscription :: SubscriptionThread -> STM Sub
|
|
newSubscription st = do
|
|
delivered <- newEmptyTMVar
|
|
subThread <- ServerSub <$> newTVar st
|
|
return Sub {subThread, delivered}
|
|
|
|
newProhibitedSub :: STM Sub
|
|
newProhibitedSub = do
|
|
delivered <- newEmptyTMVar
|
|
return Sub {subThread = ProhibitSub, delivered}
|
|
|
|
newEnv :: ServerConfig -> IO Env
|
|
newEnv config@ServerConfig {smpCredentials, httpCredentials, serverStoreCfg, smpAgentCfg, information, messageExpiration, idleQueueInterval, msgQueueQuota, maxJournalMsgCount, maxJournalStateLines} = do
|
|
serverActive <- newTVarIO True
|
|
server <- newServer
|
|
msgStore <- case serverStoreCfg of
|
|
ASSCfg qt mt (SSCMemory storePaths_) -> do
|
|
let storePath = storeMsgsFile =<< storePaths_
|
|
ms <- newMsgStore STMStoreConfig {storePath, quota = msgQueueQuota}
|
|
forM_ storePaths_ $ \StorePaths {storeLogFile = f} -> loadStoreLog (mkQueue ms True) f $ queueStore ms
|
|
pure $ AMS qt mt ms
|
|
ASSCfg qt mt SSCMemoryJournal {storeLogFile, storeMsgsPath} -> do
|
|
let qsCfg = MQStoreCfg
|
|
cfg = mkJournalStoreConfig qsCfg storeMsgsPath msgQueueQuota maxJournalMsgCount maxJournalStateLines idleQueueInterval
|
|
ms <- newMsgStore cfg
|
|
loadStoreLog (mkQueue ms True) storeLogFile $ stmQueueStore ms
|
|
pure $ AMS qt mt ms
|
|
#if defined(dbServerPostgres)
|
|
ASSCfg qt mt SSCDatabaseJournal {storeCfg, storeMsgsPath'} -> do
|
|
let StartOptions {compactLog, confirmMigrations} = startOptions config
|
|
qsCfg = PQStoreCfg (storeCfg {confirmMigrations} :: PostgresStoreCfg)
|
|
cfg = mkJournalStoreConfig qsCfg storeMsgsPath' msgQueueQuota maxJournalMsgCount maxJournalStateLines idleQueueInterval
|
|
when compactLog $ compactDbStoreLog $ dbStoreLogPath storeCfg
|
|
ms <- newMsgStore cfg
|
|
pure $ AMS qt mt ms
|
|
#else
|
|
ASSCfg _ _ SSCDatabaseJournal {} -> noPostgresExit
|
|
#endif
|
|
ntfStore <- NtfStore <$> TM.emptyIO
|
|
random <- C.newRandom
|
|
tlsServerCreds <- getCredentials "SMP" smpCredentials
|
|
httpServerCreds <- mapM (getCredentials "HTTPS") httpCredentials
|
|
mapM_ checkHTTPSCredentials httpServerCreds
|
|
Fingerprint fp <- loadFingerprint smpCredentials
|
|
let serverIdentity = KeyHash fp
|
|
serverStats <- newServerStats =<< getCurrentTime
|
|
sockets <- newTVarIO []
|
|
clientSeq <- newTVarIO 0
|
|
clients <- newTVarIO mempty
|
|
proxyAgent <- newSMPProxyAgent smpAgentCfg random
|
|
pure Env {serverActive, config, serverInfo, server, serverIdentity, msgStore, ntfStore, random, tlsServerCreds, httpServerCreds, serverStats, sockets, clientSeq, clients, proxyAgent}
|
|
where
|
|
loadStoreLog :: StoreQueueClass q => (RecipientId -> QueueRec -> IO q) -> FilePath -> STMQueueStore q -> IO ()
|
|
loadStoreLog mkQ f st = do
|
|
logInfo $ "restoring queues from file " <> T.pack f
|
|
sl <- readWriteQueueStore False mkQ f st
|
|
setStoreLog st sl
|
|
#if defined(dbServerPostgres)
|
|
compactDbStoreLog = \case
|
|
Just f -> do
|
|
logInfo $ "compacting queues in file " <> T.pack f
|
|
st <- newMsgStore STMStoreConfig {storePath = Nothing, quota = msgQueueQuota}
|
|
-- we don't need to have locks in the map
|
|
sl <- readWriteQueueStore False (mkQueue st False) f (queueStore st)
|
|
setStoreLog (queueStore st) sl
|
|
closeMsgStore st
|
|
Nothing -> do
|
|
logError "Error: `--compact-log` used without `db_store_log` INI option"
|
|
exitFailure
|
|
#endif
|
|
getCredentials protocol creds = do
|
|
files <- missingCreds
|
|
unless (null files) $ do
|
|
putStrLn $ "----------\nError: no " <> protocol <> " credentials: " <> intercalate ", " files
|
|
when (protocol == "HTTPS") $ do
|
|
putStrLn "Server should serve static pages to show connection links in the browser."
|
|
putStrLn letsEncrypt
|
|
exitFailure
|
|
loadServerCredential creds
|
|
where
|
|
missingfile f = (\y -> [f | not y]) <$> doesFileExist f
|
|
missingCreds = do
|
|
let files = maybe id (:) (caCertificateFile creds) [certificateFile creds, privateKeyFile creds]
|
|
in concat <$> mapM missingfile files
|
|
checkHTTPSCredentials (X.CertificateChain cc, _k) =
|
|
-- LetsEncrypt provides ECDSA with insecure curve p256 (https://safecurves.cr.yp.to)
|
|
case map (X.signedObject . X.getSigned) cc of
|
|
X.Certificate {X.certPubKey = X.PubKeyRSA rsa} : _ca | RSA.public_size rsa >= 512 -> pure ()
|
|
_ -> do
|
|
putStrLn $ "Error: unsupported HTTPS credentials, required 4096-bit RSA\n" <> letsEncrypt
|
|
exitFailure
|
|
letsEncrypt = "Use Let's Encrypt to generate: certbot certonly --standalone -d yourdomainname --key-type rsa --rsa-key-size 4096\n----------"
|
|
serverInfo =
|
|
ServerInformation
|
|
{ information,
|
|
config =
|
|
ServerPublicConfig
|
|
{ persistence,
|
|
messageExpiration = ttl <$> messageExpiration,
|
|
statsEnabled = isJust $ logStatsInterval config,
|
|
newQueuesAllowed = allowNewQueues config,
|
|
basicAuthEnabled = isJust $ newQueueBasicAuth config
|
|
}
|
|
}
|
|
where
|
|
persistence = case serverStoreCfg of
|
|
ASSCfg _ _ (SSCMemory sp_) -> case sp_ of
|
|
Nothing -> SPMMemoryOnly
|
|
Just StorePaths {storeMsgsFile = Just _} -> SPMMessages
|
|
_ -> SPMQueues
|
|
_ -> SPMMessages
|
|
|
|
noPostgresExit :: IO a
|
|
noPostgresExit = do
|
|
putStrLn "Error: server binary is compiled without support for PostgreSQL database."
|
|
putStrLn "Please download `smp-server-postgres` or re-compile with `cabal build -fserver_postgres`."
|
|
exitFailure
|
|
|
|
mkJournalStoreConfig :: QStoreCfg s -> FilePath -> Int -> Int -> Int -> Int64 -> JournalStoreConfig s
|
|
mkJournalStoreConfig queueStoreCfg storePath msgQueueQuota maxJournalMsgCount maxJournalStateLines idleQueueInterval =
|
|
JournalStoreConfig
|
|
{ storePath,
|
|
quota = msgQueueQuota,
|
|
pathParts = journalMsgStoreDepth,
|
|
queueStoreCfg,
|
|
maxMsgCount = maxJournalMsgCount,
|
|
maxStateLines = maxJournalStateLines,
|
|
stateTailSize = defaultStateTailSize,
|
|
idleInterval = idleQueueInterval,
|
|
expireBackupsAfter = 14 * nominalDay,
|
|
keepMinBackups = 2
|
|
}
|
|
|
|
newSMPProxyAgent :: SMPClientAgentConfig -> TVar ChaChaDRG -> IO ProxyAgent
|
|
newSMPProxyAgent smpAgentCfg random = do
|
|
smpAgent <- newSMPClientAgent smpAgentCfg random
|
|
pure ProxyAgent {smpAgent}
|
|
|
|
readWriteQueueStore :: forall q s. QueueStoreClass q s => Bool -> (RecipientId -> QueueRec -> IO q) -> FilePath -> s -> IO (StoreLog 'WriteMode)
|
|
readWriteQueueStore tty mkQ = readWriteStoreLog (readQueueStore tty mkQ) (writeQueueStore @q)
|