1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
|
{-# LANGUAGE OverloadedStrings #-}
-- | Simple functions to run TCP clients and servers.
module Network.Run.TCP.Timeout (
runTCPServer,
TimeoutServer,
-- * Generalized API
runTCPServerWithSocket,
openServerSocket,
openServerSocketWithOptions,
openServerSocketWithOpts,
) where
import Control.Concurrent (forkFinally)
import qualified Control.Exception as E
import Control.Monad (forever, void)
import qualified Data.List.NonEmpty as NE
import Network.Socket
import qualified System.TimeManager as T
import Network.Run.Core
-- | A server type
type TimeoutServer a =
T.Manager
-- ^ A global timeout manager
-> T.Handle
-- ^ A thread-local timeout handler
-> Socket
-- ^ A connected socket
-> IO a
-- | Running a TCP server with a connected socket.
runTCPServer
:: Int
-- ^ Timeout in second.
-> Maybe HostName
-> ServiceName
-> TimeoutServer a
-> IO a
runTCPServer tm mhost port server = do
addr <- resolve Stream mhost port [AI_PASSIVE] NE.head
E.bracket (openTCPServerSocket addr) close $ \sock ->
runTCPServerWithSocket tm sock server
-- | Running a TCP client with a connected socket for a given listen
-- socket.
runTCPServerWithSocket
:: Int
-- ^ Timeout in second.
-> Socket
-> TimeoutServer a
-> IO a
runTCPServerWithSocket tm sock server = do
T.withManager (tm * 1000000) $ \mgr -> forever $
E.bracketOnError (accept sock) (close . fst) $ \(conn, _peer) ->
void $ forkFinally (server' mgr conn) (const $ gclose conn)
where
server' mgr conn = do
labelMe "TCP timeout server"
T.withHandle mgr (return ()) $ \th -> server mgr th conn
|