module RFC.Servant
( ApiCtx
, apiCtxToHandler
, ResourceDefinition(..)
, ServerAPI
, ServerImpl
, module Servant
, module Servant.Docs
, module Servant.HTML.Blaze
, module Text.Blaze.Html
, module Data.Swagger
, module RFC.Data.IdAnd
, module RFC.API
) where
import Control.Natural (type (~>))
import Data.Aeson as JSON
import qualified Data.Aeson.Diff as JSON
import Data.Swagger (Swagger, ToSchema)
import Network.Wreq.Session as Wreq
import RFC.API
import RFC.Data.IdAnd
import RFC.HTTP.Client
import RFC.JSON ()
import RFC.Prelude hiding (Handler)
import qualified RFC.Psql as Psql
import qualified RFC.Redis as Redis
import Servant
import Servant.Docs hiding (API)
import Servant.HTML.Blaze (HTML)
import Servant.Server (Handler, runHandler)
import Text.Blaze.Html
type ApiCtx =
ReaderT Wreq.Session
( ReaderT Psql.ConnectionPool
( ReaderT Redis.ConnectionPool
Handler
)
)
instance MonadUnliftIO Handler where
askUnliftIO = return $ UnliftIO $ \handler -> do
either <- runHandler handler
case either of
Left err -> throwIO err
Right v -> return v
instance HasAPIClient ApiCtx where
getAPIClient = ask
instance Psql.HasPsql ApiCtx where
getPsqlPool = lift ask
instance Redis.HasRedis ApiCtx where
getRedisPool = lift $ lift ask
apiCtxToHandler :: Wreq.Session -> Redis.ConnectionPool -> Psql.ConnectionPool -> ApiCtx ~> Handler
apiCtxToHandler apiClient redisPool psqlPool = toHandler
where
toHandler :: ApiCtx ~> Handler
toHandler = withRedis . withPsql . withAPIClient
where
withAPIClient m = runReaderT m apiClient
withRedis m = runReaderT m redisPool
withPsql m = runReaderT m psqlPool
type FetchAllImpl a = ApiCtx (RefMap a)
type FetchAllAPI a = JGet (RefMap a)
type FetchImpl a = UUID -> ApiCtx (IdAnd a)
type FetchAPI a = Capture "id" UUID :> JGet (IdAnd a)
type CreateImpl a = a -> ApiCtx (IdAnd a)
type CreateAPI a = JReqBody a :> JPost (IdAnd a)
type PatchImpl a = UUID -> JSON.Patch -> ApiCtx (IdAnd a)
type PatchAPI a = Capture "id" UUID :> ReqBody '[JSON] JSON.Patch :> Patch '[JSON] (IdAnd a)
type ReplaceImpl a = UUID -> a -> ApiCtx (IdAnd a)
type ReplaceAPI a = Capture "id" UUID :> JReqBody a :> JPost (IdAnd a)
type ServerImpl a =
(FetchAllImpl a)
:<|> (FetchImpl a)
:<|> (CreateImpl a)
:<|> (PatchImpl a)
:<|> (ReplaceImpl a)
type ServerAPI a =
(FetchAllAPI a)
:<|> (FetchAPI a)
:<|> (CreateAPI a)
:<|> (PatchAPI a)
:<|> (ReplaceAPI a)
class (FromJSON a, ToJSON a, Show a) => ResourceDefinition a where
restFetchAll :: FetchAllImpl a
restFetchAll = do
resources <- fetchAllResources
return $ idAndsToMap resources
restFetch :: FetchImpl a
restFetch uuid = do
maybeResource <- fetchResource uuid
case maybeResource of
Nothing -> throwError $ err404
{ errReasonPhrase = "No resource found for id"
, errBody = asUTF8 $ "Could not find a resource with UUID: " ++ show uuid
}
Just value -> return $ IdAnd (uuid, value)
restCreate :: CreateImpl a
restCreate a = do
maybeId <- createResource a
case maybeId of
(Just id) -> restFetch id
Nothing -> throwIO $ err400
{ errReasonPhrase = "Could not create resource"
, errBody = asUTF8 $ show a
}
restPatch :: PatchImpl a
restPatch id patch = do
(IdAnd (_,original::a)) <- restFetch id
case JSON.patch patch $ toJSON original of
Error str -> throwError $ err400
{ errReasonPhrase = "Error applying patch"
, errBody = asUTF8 str
}
Success jsonValue ->
case JSON.eitherDecode' $ JSON.encode jsonValue of
Left err -> throwError $ err400
{ errReasonPhrase = "Error rebuilding object after patch"
, errBody = asUTF8 err
}
Right value -> restReplace id value
restReplace :: ReplaceImpl a
restReplace id value = do
replaceResource newValue
restFetch id
where
newValue = IdAnd (id,value)
restServer :: ServerImpl a
restServer =
restFetchAll
:<|> restFetch
:<|> restCreate
:<|> restPatch
:<|> restReplace
fetchResource :: UUID -> ApiCtx (Maybe a)
fetchAllResources :: ApiCtx [IdAnd a]
createResource :: a -> ApiCtx (Maybe UUID)
replaceResource :: (IdAnd a) -> ApiCtx ()