om-socket
Overview
This package provides some utilities for Haskell programs to communicate raw
binary messages over the network. It includes:
-
Opening an "Ingress" service.
It provides a way for a program to open a socket and accept a
stream of messages without responding to any of them.
-
Open an "Egress" socket.
It provides a way to connect to an "Ingress" service and dump a stream of
messages to it.
-
Open a bidirectional "server".
It provides a way to open a "server", which provides your program with a
stream of requests paired with a way to respond to each request. Responses are
allowed to be supplied in an order different than that from which the
corresponding requests were received.
-
Open a client to a bidirectional "server".
It provides a way to connect to an open server and provides a convenient
(request -> IO response)
interface to talk to the server.
Examples
Open an Ingress service
{-# LANGUAGE DeriveAnyClass #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE OverloadedStrings #-}
module Main (main) where
import Conduit ((.|), awaitForever, runConduit)
import Control.Monad.IO.Class (MonadIO(liftIO))
import Data.Binary (Binary)
import GHC.Generics (Generic)
import OM.Socket (openIngress)
{- | The messages that arrive on the socket. -}
data Msg
= A
| B
deriving stock (Generic)
deriving anyclass (Binary)
main :: IO ()
main =
runConduit $
openIngress "localhost:9000"
.| awaitForever (\msg ->
case msg of
A -> liftIO $ putStrLn "Got A"
B -> liftIO $ putStrLn "Got B"
)
Open an Egress connection
{-# LANGUAGE DeriveAnyClass #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE OverloadedStrings #-}
module Main (main) where
import Conduit ((.|), runConduit, yield)
import Data.Binary (Binary)
import GHC.Generics (Generic)
import OM.Socket (openEgress)
{- | The messages that arrive on the socket. -}
data Msg
= A
| B
deriving stock (Generic)
deriving anyclass (Binary)
main :: IO ()
main =
runConduit $
mapM_ yield [A, B, B, A, A, A, B]
.| openEgress "localhost:9000"
Start a server process
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE OverloadedStrings #-}
module Main (main) where
import Conduit ((.|), awaitForever, runConduit)
import Control.Monad.Logger (runStdoutLoggingT)
import Control.Monad.Trans.Class (MonadTrans(lift))
import Data.Binary (Binary)
import OM.Socket (openServer)
{- | The requests accepted by the server. -}
newtype Request = EchoRequest String
deriving newtype (Binary, Show)
{- | The response sent back to the client. -}
newtype Responsee = EchoResponse String
deriving newtype (Binary, Show)
{- | Simple echo resposne server. -}
main :: IO ()
main =
runStdoutLoggingT . runConduit $
pure ()
.| openServer "localhost:9000" Nothing
.| awaitForever (\(EchoRequest str, respond) ->
{-
You don't necessarily have to respond right away if you don't
want to. You can cache the responder away in some state and
get back to it at some later time if you like.
-}
lift $ respond (EchoResponse str)
)
Connect a client to a server
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE OverloadedStrings #-}
module Main (main) where
import Control.Monad.Logger (runStdoutLoggingT)
import Data.Binary (Binary)
import OM.Socket (connectServer)
{- | The requests accepted by the server. -}
newtype Request = EchoRequest String
deriving newtype (Binary, Show)
{- | The response sent back to the client. -}
newtype Responsee = EchoResponse String
deriving newtype (Binary, Show)
{- | Simple echo resposne client. -}
main :: IO ()
main = do
client <-
runStdoutLoggingT $
connectServer "localhost:9000" Nothing
putStrLn =<< client (EchoRequest "hello")
putStrLn =<< client (EchoRequest "world")