{-# LANGUAGE DeriveGeneric #-} {-# LANGUAGE FlexibleInstances #-} {-# LANGUAGE GeneralizedNewtypeDeriving #-} {-# LANGUAGE MultiParamTypeClasses #-} -- | -- Module : Network.Ethereum.Web3.Provider -- Copyright : Alexander Krupenkin 2016 -- License : BSD3 -- -- Maintainer : mail@akru.me -- Stability : experimental -- Portability : portable -- -- Web3 service provider. -- module Network.Ethereum.Web3.Provider where import Control.Concurrent.Async (Async, async) import Control.Exception (Exception, try) import Control.Monad.Catch (MonadThrow) import Control.Monad.IO.Class (MonadIO (..)) import Control.Monad.Reader (MonadReader (..)) import Control.Monad.Trans.Reader (ReaderT, mapReaderT, runReaderT) import Data.Aeson (FromJSON) import Data.Default (Default (..)) import GHC.Generics (Generic) import Network.HTTP.Client (Manager, newManager) import Network.HTTP.Client.TLS (tlsManagerSettings) import Network.JsonRpc.TinyClient (Remote, ServerUri) -- | Any communication with Ethereum node wrapped with 'Web3' monad newtype Web3 a = Web3 { unWeb3 :: ReaderT (ServerUri, Manager) IO a } deriving (Functor, Applicative, Monad, MonadIO, MonadThrow) instance MonadReader (ServerUri, Manager) Web3 where ask = Web3 ask local f = Web3 . local f . unWeb3 instance FromJSON a => Remote Web3 (Web3 a) -- | Some peace of error response data Web3Error = JsonRpcFail !String -- ^ JSON-RPC communication error | ParserFail !String -- ^ Error in parser state | UserFail !String -- ^ Common head for user errors deriving (Show, Eq, Generic) instance Exception Web3Error --TODO: Change to `HttpProvider ServerUri | IpcProvider FilePath` to support IPC -- | Web3 Provider data Provider = HttpProvider ServerUri deriving (Show, Eq, Generic) instance Default Provider where def = HttpProvider "http://localhost:8545" -- | 'Web3' monad runner, using the supplied Manager runWeb3With :: MonadIO m => Manager -> Provider -> Web3 a -> m (Either Web3Error a) runWeb3With manager (HttpProvider uri) f = liftIO . try . flip runReaderT (uri, manager) . unWeb3 $ f -- | 'Web3' monad runner runWeb3' :: MonadIO m => Provider -> Web3 a -> m (Either Web3Error a) runWeb3' provider f = do manager <- liftIO $ newManager tlsManagerSettings runWeb3With manager provider f -- | 'Web3' runner for default provider runWeb3 :: MonadIO m => Web3 a -> m (Either Web3Error a) {-# INLINE runWeb3 #-} runWeb3 = runWeb3' def -- | Fork 'Web3' with the same 'Provider' forkWeb3 :: Web3 a -> Web3 (Async a) {-# INLINE forkWeb3 #-} forkWeb3 = Web3 . mapReaderT async . unWeb3