{-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE MultiParamTypeClasses #-} {-# LANGUAGE ScopedTypeVariables #-} {-# LANGUAGE TypeFamilies #-} {-# LANGUAGE UndecidableInstances #-} module Metro.TCPServer ( TCPServer , tcpServer ) where import Control.Monad (void) import Metro.Class (Servable (..)) import Metro.Socket (listen) import Metro.TP.TCPSocket (TCPSocket, tcpSocket_) import Network.Socket (Socket, SocketOption (KeepAlive), accept, setSocketOption) import qualified Network.Socket as Socket (close) import UnliftIO (async, liftIO) newtype TCPServer = TCPServer Socket instance Servable TCPServer where data ServerConfig TCPServer = TCPConfig String type SID TCPServer = Socket type STP TCPServer = TCPSocket newServer :: ServerConfig TCPServer -> m TCPServer newServer (TCPConfig hostPort) = IO TCPServer -> m TCPServer forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO TCPServer -> m TCPServer) -> IO TCPServer -> m TCPServer forall a b. (a -> b) -> a -> b $ Socket -> TCPServer TCPServer (Socket -> TCPServer) -> IO Socket -> IO TCPServer forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b <$> String -> IO Socket listen String hostPort servOnce :: TCPServer -> (Maybe (SID TCPServer, TransportConfig (STP TCPServer)) -> m ()) -> m () servOnce (TCPServer serv :: Socket serv) done :: Maybe (SID TCPServer, TransportConfig (STP TCPServer)) -> m () done = do (sock :: Socket sock, _) <- IO (Socket, SockAddr) -> m (Socket, SockAddr) forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO (Socket, SockAddr) -> m (Socket, SockAddr)) -> IO (Socket, SockAddr) -> m (Socket, SockAddr) forall a b. (a -> b) -> a -> b $ Socket -> IO (Socket, SockAddr) accept Socket serv IO () -> m () forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO () -> m ()) -> IO () -> m () forall a b. (a -> b) -> a -> b $ Socket -> SocketOption -> Int -> IO () setSocketOption Socket sock SocketOption KeepAlive 1 m (Async ()) -> m () forall (f :: * -> *) a. Functor f => f a -> f () void (m (Async ()) -> m ()) -> m (Async ()) -> m () forall a b. (a -> b) -> a -> b $ m () -> m (Async ()) forall (m :: * -> *) a. MonadUnliftIO m => m a -> m (Async a) async (m () -> m (Async ())) -> m () -> m (Async ()) forall a b. (a -> b) -> a -> b $ do Maybe (SID TCPServer, TransportConfig (STP TCPServer)) -> m () done (Maybe (SID TCPServer, TransportConfig (STP TCPServer)) -> m ()) -> Maybe (SID TCPServer, TransportConfig (STP TCPServer)) -> m () forall a b. (a -> b) -> a -> b $ (Socket, TransportConfig TCPSocket) -> Maybe (Socket, TransportConfig TCPSocket) forall a. a -> Maybe a Just (Socket sock, Socket -> TransportConfig TCPSocket tcpSocket_ Socket sock) IO () -> m () forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO () -> m ()) -> IO () -> m () forall a b. (a -> b) -> a -> b $ Socket -> IO () Socket.close Socket sock onConnEnter :: TCPServer -> SID TCPServer -> m () onConnEnter _ _ = () -> m () forall (m :: * -> *) a. Monad m => a -> m a return () onConnLeave :: TCPServer -> SID TCPServer -> m () onConnLeave _ _ = () -> m () forall (m :: * -> *) a. Monad m => a -> m a return () servClose :: TCPServer -> m () servClose (TCPServer serv :: Socket serv) = IO () -> m () forall (m :: * -> *) a. MonadIO m => IO a -> m a liftIO (IO () -> m ()) -> IO () -> m () forall a b. (a -> b) -> a -> b $ Socket -> IO () Socket.close Socket serv tcpServer :: String -> ServerConfig TCPServer tcpServer :: String -> ServerConfig TCPServer tcpServer = String -> ServerConfig TCPServer TCPConfig