mirror of
https://github.com/simplex-chat/simplexmq.git
synced 2026-04-27 12:55:16 +00:00
9d12d76078
* add fourmolu * linebreak * simplify * comment
173 lines
6.2 KiB
Haskell
173 lines
6.2 KiB
Haskell
{-# LANGUAGE DataKinds #-}
|
|
{-# LANGUAGE DuplicateRecordFields #-}
|
|
{-# LANGUAGE GADTs #-}
|
|
{-# LANGUAGE KindSignatures #-}
|
|
{-# LANGUAGE NamedFieldPuns #-}
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
|
|
module Simplex.Messaging.Notifications.Server.Env where
|
|
|
|
import Control.Concurrent (ThreadId)
|
|
import Control.Concurrent.Async (Async)
|
|
import Control.Logger.Simple
|
|
import Control.Monad.IO.Unlift
|
|
import Crypto.Random
|
|
import Data.ByteString.Char8 (ByteString)
|
|
import Data.Int (Int64)
|
|
import Data.List.NonEmpty (NonEmpty)
|
|
import Data.Time.Clock (getCurrentTime)
|
|
import Data.Time.Clock.System (SystemTime)
|
|
import Data.Word (Word16)
|
|
import Data.X509.Validation (Fingerprint (..))
|
|
import Network.Socket
|
|
import qualified Network.TLS as T
|
|
import Numeric.Natural
|
|
import Simplex.Messaging.Client.Agent
|
|
import qualified Simplex.Messaging.Crypto as C
|
|
import Simplex.Messaging.Notifications.Protocol
|
|
import Simplex.Messaging.Notifications.Server.Push.APNS
|
|
import Simplex.Messaging.Notifications.Server.Stats
|
|
import Simplex.Messaging.Notifications.Server.Store
|
|
import Simplex.Messaging.Notifications.Server.StoreLog
|
|
import Simplex.Messaging.Protocol (CorrId, SMPServer, Transmission)
|
|
import Simplex.Messaging.Server.Expiration
|
|
import Simplex.Messaging.TMap (TMap)
|
|
import qualified Simplex.Messaging.TMap as TM
|
|
import Simplex.Messaging.Transport (ATransport)
|
|
import Simplex.Messaging.Transport.Server (TransportServerConfig, loadFingerprint, loadTLSServerParams)
|
|
import System.IO (IOMode (..))
|
|
import System.Mem.Weak (Weak)
|
|
import UnliftIO.STM
|
|
|
|
data NtfServerConfig = NtfServerConfig
|
|
{ transports :: [(ServiceName, ATransport)],
|
|
subIdBytes :: Int,
|
|
regCodeBytes :: Int,
|
|
clientQSize :: Natural,
|
|
subQSize :: Natural,
|
|
pushQSize :: Natural,
|
|
smpAgentCfg :: SMPClientAgentConfig,
|
|
apnsConfig :: APNSPushClientConfig,
|
|
subsBatchSize :: Int,
|
|
inactiveClientExpiration :: Maybe ExpirationConfig,
|
|
storeLogFile :: Maybe FilePath,
|
|
-- CA certificate private key is not needed for initialization
|
|
caCertificateFile :: FilePath,
|
|
privateKeyFile :: FilePath,
|
|
certificateFile :: FilePath,
|
|
-- stats config - see SMP server config
|
|
logStatsInterval :: Maybe Int64,
|
|
logStatsStartTime :: Int64,
|
|
serverStatsLogFile :: FilePath,
|
|
serverStatsBackupFile :: Maybe FilePath,
|
|
transportConfig :: TransportServerConfig
|
|
}
|
|
|
|
defaultInactiveClientExpiration :: ExpirationConfig
|
|
defaultInactiveClientExpiration =
|
|
ExpirationConfig
|
|
{ ttl = 7200, -- 2 hours
|
|
checkInterval = 3600 -- seconds, 1 hour
|
|
}
|
|
|
|
data NtfEnv = NtfEnv
|
|
{ config :: NtfServerConfig,
|
|
subscriber :: NtfSubscriber,
|
|
pushServer :: NtfPushServer,
|
|
store :: NtfStore,
|
|
storeLog :: Maybe (StoreLog 'WriteMode),
|
|
idsDrg :: TVar ChaChaDRG,
|
|
tlsServerParams :: T.ServerParams,
|
|
serverIdentity :: C.KeyHash,
|
|
serverStats :: NtfServerStats
|
|
}
|
|
|
|
newNtfServerEnv :: (MonadUnliftIO m, MonadRandom m) => NtfServerConfig -> m NtfEnv
|
|
newNtfServerEnv config@NtfServerConfig {subQSize, pushQSize, smpAgentCfg, apnsConfig, storeLogFile, caCertificateFile, certificateFile, privateKeyFile} = do
|
|
idsDrg <- newTVarIO =<< drgNew
|
|
store <- atomically newNtfStore
|
|
logInfo "restoring subscriptions..."
|
|
storeLog <- liftIO $ mapM (`readWriteNtfStore` store) storeLogFile
|
|
logInfo "restored subscriptions"
|
|
subscriber <- atomically $ newNtfSubscriber subQSize smpAgentCfg
|
|
pushServer <- atomically $ newNtfPushServer pushQSize apnsConfig
|
|
tlsServerParams <- liftIO $ loadTLSServerParams caCertificateFile certificateFile privateKeyFile
|
|
Fingerprint fp <- liftIO $ loadFingerprint caCertificateFile
|
|
serverStats <- atomically . newNtfServerStats =<< liftIO getCurrentTime
|
|
pure NtfEnv {config, subscriber, pushServer, store, storeLog, idsDrg, tlsServerParams, serverIdentity = C.KeyHash fp, serverStats}
|
|
|
|
data NtfSubscriber = NtfSubscriber
|
|
{ smpSubscribers :: TMap SMPServer SMPSubscriber,
|
|
newSubQ :: TBQueue [NtfEntityRec 'Subscription],
|
|
smpAgent :: SMPClientAgent
|
|
}
|
|
|
|
newNtfSubscriber :: Natural -> SMPClientAgentConfig -> STM NtfSubscriber
|
|
newNtfSubscriber qSize smpAgentCfg = do
|
|
smpSubscribers <- TM.empty
|
|
newSubQ <- newTBQueue qSize
|
|
smpAgent <- newSMPClientAgent smpAgentCfg
|
|
pure NtfSubscriber {smpSubscribers, newSubQ, smpAgent}
|
|
|
|
data SMPSubscriber = SMPSubscriber
|
|
{ newSubQ :: TQueue (NonEmpty (NtfEntityRec 'Subscription)),
|
|
subThreadId :: TVar (Maybe (Weak ThreadId))
|
|
}
|
|
|
|
newSMPSubscriber :: STM SMPSubscriber
|
|
newSMPSubscriber = do
|
|
newSubQ <- newTQueue
|
|
subThreadId <- newTVar Nothing
|
|
pure SMPSubscriber {newSubQ, subThreadId}
|
|
|
|
data NtfPushServer = NtfPushServer
|
|
{ pushQ :: TBQueue (NtfTknData, PushNotification),
|
|
pushClients :: TMap PushProvider PushProviderClient,
|
|
intervalNotifiers :: TMap NtfTokenId IntervalNotifier,
|
|
apnsConfig :: APNSPushClientConfig
|
|
}
|
|
|
|
data IntervalNotifier = IntervalNotifier
|
|
{ action :: Async (),
|
|
token :: NtfTknData,
|
|
interval :: Word16
|
|
}
|
|
|
|
newNtfPushServer :: Natural -> APNSPushClientConfig -> STM NtfPushServer
|
|
newNtfPushServer qSize apnsConfig = do
|
|
pushQ <- newTBQueue qSize
|
|
pushClients <- TM.empty
|
|
intervalNotifiers <- TM.empty
|
|
pure NtfPushServer {pushQ, pushClients, intervalNotifiers, apnsConfig}
|
|
|
|
newPushClient :: NtfPushServer -> PushProvider -> IO PushProviderClient
|
|
newPushClient NtfPushServer {apnsConfig, pushClients} pp = do
|
|
c <- apnsPushProviderClient <$> createAPNSPushClient (apnsProviderHost pp) apnsConfig
|
|
atomically $ TM.insert pp c pushClients
|
|
pure c
|
|
|
|
getPushClient :: NtfPushServer -> PushProvider -> IO PushProviderClient
|
|
getPushClient s@NtfPushServer {pushClients} pp =
|
|
atomically (TM.lookup pp pushClients) >>= maybe (newPushClient s pp) pure
|
|
|
|
data NtfRequest
|
|
= NtfReqNew CorrId ANewNtfEntity
|
|
| forall e. NtfEntityI e => NtfReqCmd (SNtfEntity e) (NtfEntityRec e) (Transmission (NtfCommand e))
|
|
| NtfReqPing CorrId NtfEntityId
|
|
|
|
data NtfServerClient = NtfServerClient
|
|
{ rcvQ :: TBQueue NtfRequest,
|
|
sndQ :: TBQueue (Transmission NtfResponse),
|
|
sessionId :: ByteString,
|
|
connected :: TVar Bool,
|
|
activeAt :: TVar SystemTime
|
|
}
|
|
|
|
newNtfServerClient :: Natural -> ByteString -> SystemTime -> STM NtfServerClient
|
|
newNtfServerClient qSize sessionId ts = do
|
|
rcvQ <- newTBQueue qSize
|
|
sndQ <- newTBQueue qSize
|
|
connected <- newTVar True
|
|
activeAt <- newTVar ts
|
|
return NtfServerClient {rcvQ, sndQ, sessionId, connected, activeAt}
|