# network-wait ![MIT](https://img.shields.io/github/license/mbg/network-wait) ![CI](https://github.com/mbg/network-wait/workflows/build/badge.svg?branch=main) [![Hackage](https://img.shields.io/hackage/v/network-wait)](https://hackage.haskell.org/package/network-wait) A lightweight Haskell library for waiting on networked services to become available. This is useful if you are e.g. building a web application which relies on a database server to be available, but which may not be immediately available on application startup. [**Full Haddock documentation for all modules**](https://mbg.github.io/network-wait/) ## Example All functions provided by this library work with retry policies from [`Control.Retry`](https://hackage.haskell.org/package/retry) which provide good control over the retry behaviour. To wait for a PostgreSQL server to become available on the same machine: ```haskell import Control.Retry import Network.Wait main :: IO () main = do waitTcp retryPolicyDefault "localhost" "5432" putStrLn "Yay, the server is available!" ``` The haddock documentation for this library contains more examples. ## Example: PostgreSQL The `network-wait` package can be compiled with the `network-wait:postgres` flag (e.g. `stack build --flag network-wait:postgres`) which enables support for checking the readiness of PostgreSQL servers specifically. Unlike the functions in the `Network.Wait` module, which only check that connections can be established, the functions in `Network.Wait.PostgreSQL` also check that a PostgreSQL server is ready to accept commands. To wait for a PostgreSQL server to become available and accept commands: ```haskell import Control.Retry (retryPolicyDefault) import Database.PostgreSQL.Simple (defaultConnectInfo) import Network.Wait.PostgreSQL (waitPostgreSQL) main :: IO () main = do waitPostgreSQL retryPolicyDefault defaultConnectInfo putStrLn "Yay, the PostgreSQL server is ready to accept commands!" ``` Internally, this uses `postgresql-simple` to connect to the specified server (`defaultConnectInfo` in the example above) and send a `SELECT 1;` query. If the query is answered correctly, we consider the server to be in a state ready to accept commands. Alternatively, a connection string may be used instead of a `ConnectInfo` value: ```haskell import Data.ByteString (ByteString) import Control.Retry (retryPolicyDefault) import Database.PostgreSQL.Simple (defaultConnectInfo) import Network.Wait.PostgreSQL (waitPostgreSQL) connStr :: ByteString connStr = "host=localhost port=5432" main :: IO () main = do waitPostgreSQL retryPolicyDefault connStr putStrLn "Yay, the PostgreSQL server is ready to accept commands!" ``` The `Network.Wait.PostgreSQL` module is gated behind the `network-wait:postgres` flag so that the PostgreSQL-specific dependencies are only required when PostgresSQL support is required by users of this library. ## Example: Redis The `network-wait` package can be compiled with the `network-wait:redis` flag (e.g. `stack build --flag network-wait:redis`) which enables support for checking the readiness of Redis servers specifically. Unlike the functions in the `Network.Wait` module, which only check that connections can be established, the functions in `Network.Wait.Redis` also check that a Redis server is ready to accept commands. To wait for a Redis server to become available and accept commands: ```haskell import Control.Retry (retryPolicyDefault) import Database.Redis (defaultConnectInfo) import Network.Wait.Redis (waitRedis) main :: IO () main = do waitRedis retryPolicyDefault defaultConnectInfo putStrLn "Yay, the Redis server is ready to accept commands!" ``` Internally, this uses `hedis` to connect to the specified server (`defaultConnectInfo` in the example above) and send a ping. If the ping is answered, we consider the server to be in a state ready to accept commands. The `Network.Wait.Redis` module is gated behind the `network-wait:redis` flag so that the Redis-specific dependencies are only required when Redis support is required by users of this library. ## See also - [wait-for](https://github.com/eficode/wait-for) is a popular shell script with the same objectives as this library. - The [port-utils](https://hackage.haskell.org/package/port-utils) package has some functions for waiting on TCP servers to become available, with a fixed timeout.