module Metro.Socket
( Socket
, close
, listen
, connect
, getHost
, getService
, bindTo
, getDatagramAddr
) where
import Control.Exception (bracketOnError, throwIO)
import Control.Monad (when)
import Data.List (isPrefixOf)
import Data.Maybe (listToMaybe)
import Network.Socket hiding (bind, connect, listen)
import qualified Network.Socket as S (bind, connect, listen)
import System.Directory (doesFileExist, removeFile)
import System.Exit (exitFailure)
import UnliftIO (tryIO)
firstSuccessful :: [IO a] -> IO a
firstSuccessful :: [IO a] -> IO a
firstSuccessful = Maybe IOException -> [IO a] -> IO a
forall b. Maybe IOException -> [IO b] -> IO b
go Maybe IOException
forall a. Maybe a
Nothing
where
go :: Maybe IOException -> [IO b] -> IO b
go _ (p :: IO b
p:ps :: [IO b]
ps) =
do Either IOException b
r <- IO b -> IO (Either IOException b)
forall (m :: * -> *) a.
MonadUnliftIO m =>
m a -> m (Either IOException a)
tryIO IO b
p
case Either IOException b
r of
Right x :: b
x -> b -> IO b
forall (m :: * -> *) a. Monad m => a -> m a
return b
x
Left e :: IOException
e -> Maybe IOException -> [IO b] -> IO b
go (IOException -> Maybe IOException
forall a. a -> Maybe a
Just IOException
e) [IO b]
ps
go Nothing [] = [Char] -> IO b
forall a. HasCallStack => [Char] -> a
error "firstSuccessful: empty list"
go (Just e :: IOException
e) [] = IOException -> IO b
forall e a. Exception e => e -> IO a
throwIO IOException
e
connectTo :: Maybe HostName -> Maybe ServiceName -> IO Socket
connectTo :: Maybe [Char] -> Maybe [Char] -> IO Socket
connectTo host :: Maybe [Char]
host serv :: Maybe [Char]
serv = do
let hints :: AddrInfo
hints = AddrInfo
defaultHints { addrFlags :: [AddrInfoFlag]
addrFlags = [AddrInfoFlag
AI_ADDRCONFIG]
, addrSocketType :: SocketType
addrSocketType = SocketType
Stream }
[AddrInfo]
addrs <- Maybe AddrInfo -> Maybe [Char] -> Maybe [Char] -> IO [AddrInfo]
getAddrInfo (AddrInfo -> Maybe AddrInfo
forall a. a -> Maybe a
Just AddrInfo
hints) Maybe [Char]
host Maybe [Char]
serv
[IO Socket] -> IO Socket
forall a. [IO a] -> IO a
firstSuccessful ([IO Socket] -> IO Socket) -> [IO Socket] -> IO Socket
forall a b. (a -> b) -> a -> b
$ (AddrInfo -> IO Socket) -> [AddrInfo] -> [IO Socket]
forall a b. (a -> b) -> [a] -> [b]
map AddrInfo -> IO Socket
tryToConnect [AddrInfo]
addrs
where
tryToConnect :: AddrInfo -> IO Socket
tryToConnect addr :: AddrInfo
addr =
IO Socket
-> (Socket -> IO ()) -> (Socket -> IO Socket) -> IO Socket
forall a b c. IO a -> (a -> IO b) -> (a -> IO c) -> IO c
bracketOnError
(Family -> SocketType -> ProtocolNumber -> IO Socket
socket (AddrInfo -> Family
addrFamily AddrInfo
addr) (AddrInfo -> SocketType
addrSocketType AddrInfo
addr) (AddrInfo -> ProtocolNumber
addrProtocol AddrInfo
addr))
Socket -> IO ()
close
(\sock :: Socket
sock -> do
Socket -> SockAddr -> IO ()
S.connect Socket
sock (AddrInfo -> SockAddr
addrAddress AddrInfo
addr)
Socket -> IO Socket
forall (m :: * -> *) a. Monad m => a -> m a
return Socket
sock
)
connectToFile :: FilePath -> IO Socket
connectToFile :: [Char] -> IO Socket
connectToFile path :: [Char]
path =
IO Socket
-> (Socket -> IO ()) -> (Socket -> IO Socket) -> IO Socket
forall a b c. IO a -> (a -> IO b) -> (a -> IO c) -> IO c
bracketOnError
(Family -> SocketType -> ProtocolNumber -> IO Socket
socket Family
AF_UNIX SocketType
Stream 0)
Socket -> IO ()
close
(\sock :: Socket
sock -> do
Socket -> SockAddr -> IO ()
S.connect Socket
sock ([Char] -> SockAddr
SockAddrUnix [Char]
path)
Socket -> IO Socket
forall (m :: * -> *) a. Monad m => a -> m a
return Socket
sock
)
listenOnFile :: FilePath -> IO Socket
listenOnFile :: [Char] -> IO Socket
listenOnFile path :: [Char]
path =
IO Socket
-> (Socket -> IO ()) -> (Socket -> IO Socket) -> IO Socket
forall a b c. IO a -> (a -> IO b) -> (a -> IO c) -> IO c
bracketOnError
(Family -> SocketType -> ProtocolNumber -> IO Socket
socket Family
AF_UNIX SocketType
Stream 0)
Socket -> IO ()
close
(\sock :: Socket
sock -> do
Socket -> SocketOption -> Int -> IO ()
setSocketOption Socket
sock SocketOption
ReuseAddr 1
Socket -> SockAddr -> IO ()
S.bind Socket
sock ([Char] -> SockAddr
SockAddrUnix [Char]
path)
Socket -> Int -> IO ()
S.listen Socket
sock Int
maxListenQueue
Socket -> IO Socket
forall (m :: * -> *) a. Monad m => a -> m a
return Socket
sock
)
listenOn :: Maybe HostName -> Maybe ServiceName -> IO Socket
listenOn :: Maybe [Char] -> Maybe [Char] -> IO Socket
listenOn host :: Maybe [Char]
host serv :: Maybe [Char]
serv = do
let hints :: AddrInfo
hints = AddrInfo
defaultHints { addrFlags :: [AddrInfoFlag]
addrFlags = [AddrInfoFlag
AI_ADDRCONFIG, AddrInfoFlag
AI_PASSIVE]
, addrSocketType :: SocketType
addrSocketType = SocketType
Stream
}
[AddrInfo]
addrs <- Maybe AddrInfo -> Maybe [Char] -> Maybe [Char] -> IO [AddrInfo]
getAddrInfo (AddrInfo -> Maybe AddrInfo
forall a. a -> Maybe a
Just AddrInfo
hints) Maybe [Char]
host Maybe [Char]
serv
let addrs' :: [AddrInfo]
addrs' = (AddrInfo -> Bool) -> [AddrInfo] -> [AddrInfo]
forall a. (a -> Bool) -> [a] -> [a]
filter (\x :: AddrInfo
x -> AddrInfo -> Family
addrFamily AddrInfo
x Family -> Family -> Bool
forall a. Eq a => a -> a -> Bool
== Family
AF_INET6) [AddrInfo]
addrs
addr :: AddrInfo
addr = if [AddrInfo] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [AddrInfo]
addrs' then [AddrInfo] -> AddrInfo
forall a. [a] -> a
head [AddrInfo]
addrs else [AddrInfo] -> AddrInfo
forall a. [a] -> a
head [AddrInfo]
addrs'
IO Socket
-> (Socket -> IO ()) -> (Socket -> IO Socket) -> IO Socket
forall a b c. IO a -> (a -> IO b) -> (a -> IO c) -> IO c
bracketOnError
(Family -> SocketType -> ProtocolNumber -> IO Socket
socket (AddrInfo -> Family
addrFamily AddrInfo
addr) (AddrInfo -> SocketType
addrSocketType AddrInfo
addr) (AddrInfo -> ProtocolNumber
addrProtocol AddrInfo
addr))
Socket -> IO ()
close
(\sock :: Socket
sock -> do
Socket -> SocketOption -> Int -> IO ()
setSocketOption Socket
sock SocketOption
ReuseAddr 1
Socket -> SocketOption -> Int -> IO ()
setSocketOption Socket
sock SocketOption
NoDelay 1
Socket -> SockAddr -> IO ()
S.bind Socket
sock (AddrInfo -> SockAddr
addrAddress AddrInfo
addr)
Socket -> Int -> IO ()
S.listen Socket
sock Int
maxListenQueue
Socket -> IO Socket
forall (m :: * -> *) a. Monad m => a -> m a
return Socket
sock
)
listen :: String -> IO Socket
listen :: [Char] -> IO Socket
listen port :: [Char]
port =
if "tcp" [Char] -> [Char] -> Bool
forall a. Eq a => [a] -> [a] -> Bool
`isPrefixOf` [Char]
port then
Maybe [Char] -> Maybe [Char] -> IO Socket
listenOn ([Char] -> Maybe [Char]
getHost [Char]
port) ([Char] -> Maybe [Char]
getService [Char]
port)
else do
let sockFile :: [Char]
sockFile = [Char] -> [Char]
dropS [Char]
port
Bool
exists <- [Char] -> IO Bool
doesFileExist [Char]
sockFile
Bool -> IO () -> IO ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when Bool
exists (IO () -> IO ()) -> IO () -> IO ()
forall a b. (a -> b) -> a -> b
$ do
Either IOException Socket
e <- IO Socket -> IO (Either IOException Socket)
forall (m :: * -> *) a.
MonadUnliftIO m =>
m a -> m (Either IOException a)
tryIO (IO Socket -> IO (Either IOException Socket))
-> IO Socket -> IO (Either IOException Socket)
forall a b. (a -> b) -> a -> b
$ [Char] -> IO Socket
connectToFile [Char]
sockFile
case Either IOException Socket
e of
Left _ -> [Char] -> IO ()
removeFile [Char]
sockFile
Right _ -> do
[Char] -> IO ()
putStrLn "periodicd: bind: resource busy (Address already in use)"
IO ()
forall a. IO a
exitFailure
[Char] -> IO Socket
listenOnFile [Char]
sockFile
connect :: String -> IO Socket
connect :: [Char] -> IO Socket
connect h :: [Char]
h | "tcp" [Char] -> [Char] -> Bool
forall a. Eq a => [a] -> [a] -> Bool
`isPrefixOf` [Char]
h = Maybe [Char] -> Maybe [Char] -> IO Socket
connectTo ([Char] -> Maybe [Char]
getHost [Char]
h) ([Char] -> Maybe [Char]
getService [Char]
h)
| Bool
otherwise = [Char] -> IO Socket
connectToFile ([Char] -> [Char]
dropS [Char]
h)
getDatagramAddrList :: String -> IO [AddrInfo]
getDatagramAddrList :: [Char] -> IO [AddrInfo]
getDatagramAddrList hostPort :: [Char]
hostPort = Maybe AddrInfo -> Maybe [Char] -> Maybe [Char] -> IO [AddrInfo]
getAddrInfo (AddrInfo -> Maybe AddrInfo
forall a. a -> Maybe a
Just AddrInfo
hints) Maybe [Char]
host Maybe [Char]
port
where hints :: AddrInfo
hints = AddrInfo
defaultHints
{ addrFlags :: [AddrInfoFlag]
addrFlags = [AddrInfoFlag
AI_PASSIVE]
, addrSocketType :: SocketType
addrSocketType = SocketType
Datagram
}
host :: Maybe [Char]
host = [Char] -> Maybe [Char]
getHost [Char]
hostPort
port :: Maybe [Char]
port = [Char] -> Maybe [Char]
getService [Char]
hostPort
getDatagramAddr :: String -> IO (Maybe AddrInfo)
getDatagramAddr :: [Char] -> IO (Maybe AddrInfo)
getDatagramAddr hostPort :: [Char]
hostPort = [AddrInfo] -> Maybe AddrInfo
forall a. [a] -> Maybe a
listToMaybe ([AddrInfo] -> Maybe AddrInfo)
-> IO [AddrInfo] -> IO (Maybe AddrInfo)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> [Char] -> IO [AddrInfo]
getDatagramAddrList [Char]
hostPort
bindTo :: String -> IO Socket
bindTo :: [Char] -> IO Socket
bindTo hostPort :: [Char]
hostPort = do
[AddrInfo]
addrs <- [Char] -> IO [AddrInfo]
getDatagramAddrList [Char]
hostPort
[IO Socket] -> IO Socket
forall a. [IO a] -> IO a
firstSuccessful ([IO Socket] -> IO Socket) -> [IO Socket] -> IO Socket
forall a b. (a -> b) -> a -> b
$ (AddrInfo -> IO Socket) -> [AddrInfo] -> [IO Socket]
forall a b. (a -> b) -> [a] -> [b]
map AddrInfo -> IO Socket
tryToConnect [AddrInfo]
addrs
where
tryToConnect :: AddrInfo -> IO Socket
tryToConnect addr :: AddrInfo
addr =
IO Socket
-> (Socket -> IO ()) -> (Socket -> IO Socket) -> IO Socket
forall a b c. IO a -> (a -> IO b) -> (a -> IO c) -> IO c
bracketOnError
(Family -> SocketType -> ProtocolNumber -> IO Socket
socket (AddrInfo -> Family
addrFamily AddrInfo
addr) (AddrInfo -> SocketType
addrSocketType AddrInfo
addr) (AddrInfo -> ProtocolNumber
addrProtocol AddrInfo
addr))
Socket -> IO ()
close
(\sock :: Socket
sock -> do
Socket -> SocketOption -> Int -> IO ()
setSocketOption Socket
sock SocketOption
ReuseAddr 1
Socket -> SockAddr -> IO ()
S.bind Socket
sock (SockAddr -> IO ()) -> SockAddr -> IO ()
forall a b. (a -> b) -> a -> b
$ AddrInfo -> SockAddr
addrAddress AddrInfo
addr
Socket -> IO Socket
forall (m :: * -> *) a. Monad m => a -> m a
return Socket
sock
)
countColon :: String -> Int
countColon :: [Char] -> Int
countColon = [Char] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length ([Char] -> Int) -> ([Char] -> [Char]) -> [Char] -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Char -> Bool) -> [Char] -> [Char]
forall a. (a -> Bool) -> [a] -> [a]
filter (Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
==':')
splitHostPort :: String -> (Maybe String, Maybe String)
splitHostPort :: [Char] -> (Maybe [Char], Maybe [Char])
splitHostPort hostPort :: [Char]
hostPort
| Int
colon Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== 0 = ([Char] -> Maybe [Char]
forall a. a -> Maybe a
Just [Char]
hostPort, Maybe [Char]
forall a. Maybe a
Nothing)
| Int
colon Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== 1 = (([Char] -> [Char]) -> [Char] -> Maybe [Char]
takeFst [Char] -> [Char]
forall a. a -> a
id [Char]
hostPort, ([Char] -> [Char]) -> [Char] -> Maybe [Char]
takeSnd [Char] -> [Char]
forall a. a -> a
id [Char]
hostPort)
| Int
colon Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== 5 = ([Char] -> Maybe [Char]
forall a. a -> Maybe a
Just [Char]
hostPort, Maybe [Char]
forall a. Maybe a
Nothing)
| Int
colon Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== 6 = (([Char] -> [Char]) -> [Char] -> Maybe [Char]
takeSnd [Char] -> [Char]
forall a. [a] -> [a]
reverse [Char]
hostPort, ([Char] -> [Char]) -> [Char] -> Maybe [Char]
takeFst [Char] -> [Char]
forall a. [a] -> [a]
reverse [Char]
hostPort )
| Bool
otherwise = (Maybe [Char]
forall a. Maybe a
Nothing, Maybe [Char]
forall a. Maybe a
Nothing)
where colon :: Int
colon = [Char] -> Int
countColon [Char]
hostPort
takeFst :: ([Char] -> [Char]) -> [Char] -> Maybe [Char]
takeFst f :: [Char] -> [Char]
f = [Char] -> Maybe [Char]
toMaybe ([Char] -> Maybe [Char])
-> ([Char] -> [Char]) -> [Char] -> Maybe [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Char] -> [Char]
f ([Char] -> [Char]) -> ([Char] -> [Char]) -> [Char] -> [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Char -> Bool) -> [Char] -> [Char]
forall a. (a -> Bool) -> [a] -> [a]
takeWhile (Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/=':') ([Char] -> [Char]) -> ([Char] -> [Char]) -> [Char] -> [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Char] -> [Char]
f
takeSnd :: ([Char] -> [Char]) -> [Char] -> Maybe [Char]
takeSnd f :: [Char] -> [Char]
f = [Char] -> Maybe [Char]
toMaybe ([Char] -> Maybe [Char])
-> ([Char] -> [Char]) -> [Char] -> Maybe [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Char] -> [Char]
f ([Char] -> [Char]) -> ([Char] -> [Char]) -> [Char] -> [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> [Char] -> [Char]
forall a. Int -> [a] -> [a]
drop 1 ([Char] -> [Char]) -> ([Char] -> [Char]) -> [Char] -> [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Char -> Bool) -> [Char] -> [Char]
forall a. (a -> Bool) -> [a] -> [a]
dropWhile (Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/=':') ([Char] -> [Char]) -> ([Char] -> [Char]) -> [Char] -> [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Char] -> [Char]
f
dropS :: String -> String
dropS :: [Char] -> [Char]
dropS = Int -> [Char] -> [Char]
forall a. Int -> [a] -> [a]
drop 3 ([Char] -> [Char]) -> ([Char] -> [Char]) -> [Char] -> [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Char -> Bool) -> [Char] -> [Char]
forall a. (a -> Bool) -> [a] -> [a]
dropWhile (Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/= ':')
toMaybe :: String -> Maybe String
toMaybe :: [Char] -> Maybe [Char]
toMaybe [] = Maybe [Char]
forall a. Maybe a
Nothing
toMaybe xs :: [Char]
xs = [Char] -> Maybe [Char]
forall a. a -> Maybe a
Just [Char]
xs
getHost :: String -> Maybe String
getHost :: [Char] -> Maybe [Char]
getHost = (Maybe [Char], Maybe [Char]) -> Maybe [Char]
forall a b. (a, b) -> a
fst ((Maybe [Char], Maybe [Char]) -> Maybe [Char])
-> ([Char] -> (Maybe [Char], Maybe [Char]))
-> [Char]
-> Maybe [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Char] -> (Maybe [Char], Maybe [Char])
splitHostPort ([Char] -> (Maybe [Char], Maybe [Char]))
-> ([Char] -> [Char]) -> [Char] -> (Maybe [Char], Maybe [Char])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Char] -> [Char]
dropS
getService :: String -> Maybe String
getService :: [Char] -> Maybe [Char]
getService = (Maybe [Char], Maybe [Char]) -> Maybe [Char]
forall a b. (a, b) -> b
snd ((Maybe [Char], Maybe [Char]) -> Maybe [Char])
-> ([Char] -> (Maybe [Char], Maybe [Char]))
-> [Char]
-> Maybe [Char]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Char] -> (Maybe [Char], Maybe [Char])
splitHostPort ([Char] -> (Maybe [Char], Maybe [Char]))
-> ([Char] -> [Char]) -> [Char] -> (Maybe [Char], Maybe [Char])
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Char] -> [Char]
dropS