{-# LANGUAGE OverloadedStrings, ScopedTypeVariables #-} module Web.Analyze.Client ( wrap, wrap' ) where import Prelude hiding (catch) import qualified Snap.Core as S (Request) import Snap.Core (rqContextPath, rqPathInfo, rqMethod, getRequest, urlEncode, Method(..)) import Snap.Snaplet (Handler) import Control.Monad (void) import Control.Monad.Trans (liftIO) import Control.Concurrent (forkIO) import Data.Time.Clock (UTCTime, getCurrentTime, diffUTCTime) import Network.HTTP.Conduit (Manager, parseUrl, Request(..), httpLbs) import Data.ByteString (ByteString) import qualified Data.ByteString as B (concat, append) import qualified Data.ByteString.Char8 as B8 (pack) import Control.Monad.CatchIO (catch) import Control.Exception.Base (SomeException) -- | wraps a request in code to log the requests and catch and report all errors. wrap :: Handler b v a -- ^ a handler to call in case of error, after it has been reported (presumably to show a 500 page) -> Manager -- ^ a conduit manager, used for http logging requests -> ByteString -- ^ the token for the analyze service -> Handler b v a -- ^ the handler to be wrapped -> Handler b v a wrap = wrap' (return Nothing) -- | like wrap, but also takes a handler that produces a user id if available wrap' :: Handler b v (Maybe ByteString) -- ^ a way to get a user id, if available -> Handler b v a -- ^ a handler to call in case of error, after it has been reported (presumably to show a 500 page) -> Manager -- ^ a conduit manager, used for http logging requests -> ByteString -- ^ the token for the analyze service -> Handler b v a -- ^ the handler to be wrapped -> Handler b v a wrap' userh errh man token h = handleErrors userh errh man token $ do start <- liftIO getCurrentTime res <- h end <- liftIO getCurrentTime req <- getRequest liftIO $ forkIO (sendResult man token req start end) return res handleErrors :: Handler b v (Maybe ByteString) -> Handler b v a -> Manager -> ByteString -> Handler b v a -> Handler b v a handleErrors userh errh man token h = catch h $ \(e::SomeException) -> do req <- getRequest uid <- userh liftIO $ forkIO (sendError man token req (B8.pack (show e)) uid) errh sendResult :: Manager -> ByteString -> S.Request -> UTCTime -> UTCTime -> IO () sendResult man token req start end = do let time = milliseconds (diffUTCTime end start) :: Int initreq <- parseUrl "http://analyze.positionstudios.com/submit/visit" let url = B.append (rqContextPath req) (rqPathInfo req) let meth = methodtobs (rqMethod req) let httpreq = initreq { method = "POST" , queryString = B.concat ["url=" , url , "&render=" , B8.pack (show time) , "&method=" , meth , "&token=" , token]} void (httpLbs httpreq man) where milliseconds = floor . fromRational . (1000 *) . toRational methodtobs GET = "get" methodtobs POST = "post" methodtobs PUT = "put" methodtobs DELETE = "delete" sendError :: Manager -> ByteString -> S.Request -> ByteString -> Maybe ByteString -> IO () sendError man token req message muid = do initreq <- parseUrl "http://analyze.positionstudios.com/submit/error" let url = B.append (rqContextPath req) (rqPathInfo req) let user = maybe "" (B.append "&uid=") muid let httpreq = initreq { method = "POST" , queryString = B.concat ["url=" , url , "&message=" , urlEncode message , user , "&token=" , token]} void (httpLbs httpreq man)