-
Notifications
You must be signed in to change notification settings - Fork 85
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Issue/62 custom access and error log handlers (against 0.9-stable) #71
base: 0.9-stable
Are you sure you want to change the base?
Changes from 4 commits
fb438cc
b3be5db
e28ec29
6b01854
1427db8
f379fb4
2f637e3
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -93,6 +93,15 @@ type ServerHandler = (ByteString -> IO ()) | |
------------------------------------------------------------------------------ | ||
type ServerMonad = StateT ServerState (Iteratee ByteString IO) | ||
|
||
------------------------------------------------------------------------------ | ||
-- | This handler may be used (in conjunction with setErrorLogHandler) to write out error logs in a | ||
-- custom manner. | ||
type ErrorLogHandler = ByteString -> IO ByteString | ||
|
||
------------------------------------------------------------------------------ | ||
-- | This handler may be used (in conjunction with setAccessLogHandler) to write out access logs in a | ||
-- custom manner. | ||
type AccessLogHandler = Request -> Response -> IO ByteString | ||
|
||
------------------------------------------------------------------------------ | ||
data ListenPort = | ||
|
@@ -155,11 +164,13 @@ httpServe :: Int -- ^ default timeout | |
-> [ListenPort] -- ^ ports to listen on | ||
-> ByteString -- ^ local hostname (server name) | ||
-> Maybe (ByteString -> IO ()) -- ^ access log action | ||
-> Maybe AccessLogHandler | ||
-> Maybe (ByteString -> IO ()) -- ^ error log action | ||
-> Maybe ErrorLogHandler | ||
-> ([Socket] -> IO ()) -- ^ initialisation | ||
-> ServerHandler -- ^ handler procedure | ||
-> IO () | ||
httpServe defaultTimeout ports localHostname alog' elog' initial handler = | ||
httpServe defaultTimeout ports localHostname alog' alh elog' elh initial handler = | ||
withSocketsDo $ spawnAll alog' elog' `catches` errorHandlers | ||
|
||
where | ||
|
@@ -170,7 +181,7 @@ httpServe defaultTimeout ports localHostname alog' elog' initial handler = | |
|
||
-------------------------------------------------------------------------- | ||
sslException (e@(TLS.TLSException msg)) = do | ||
logE elog' msg | ||
logE errorHandle elog' msg | ||
SC.hPutStrLn stderr msg | ||
throw e | ||
|
||
|
@@ -183,14 +194,14 @@ httpServe defaultTimeout ports localHostname alog' elog' initial handler = | |
"Error on startup: \n" | ||
, T.encodeUtf8 $ T.pack $ show e | ||
] | ||
logE elog' msg | ||
logE errorHandle elog' msg | ||
SC.hPutStrLn stderr msg | ||
throw e | ||
|
||
-------------------------------------------------------------------------- | ||
spawnAll alog elog = {-# SCC "httpServe/spawnAll" #-} do | ||
|
||
logE elog $ S.concat [ "Server.httpServe: START, binding to " | ||
logE errorHandle elog $ S.concat [ "Server.httpServe: START, binding to " | ||
, bshow ports ] | ||
|
||
let isHttps p = case p of { (HttpsPort _ _ _ _ _) -> True; _ -> False;} | ||
|
@@ -203,36 +214,58 @@ httpServe defaultTimeout ports localHostname alog' elog' initial handler = | |
nports <- mapM bindPort ports | ||
let socks = map (\x -> case x of ListenHttp s -> s; ListenHttps s _ -> s) nports | ||
|
||
(simpleEventLoop defaultTimeout nports numCapabilities (logE elog) (initial socks) | ||
$ runHTTP defaultTimeout alog elog handler localHostname) | ||
(simpleEventLoop defaultTimeout nports numCapabilities (logE errorHandle elog) (initial socks) | ||
$ runHTTP defaultTimeout alog alh elog elh handler localHostname) | ||
`finally` do | ||
logE elog "Server.httpServe: SHUTDOWN" | ||
logE errorHandle elog "Server.httpServe: SHUTDOWN" | ||
|
||
if initHttps | ||
then TLS.stopTLS | ||
else return () | ||
|
||
logE elog "Server.httpServe: BACKEND STOPPED" | ||
logE errorHandle elog "Server.httpServe: BACKEND STOPPED" | ||
|
||
-------------------------------------------------------------------------- | ||
bindPort (HttpPort baddr port ) = bindHttp baddr port | ||
bindPort (HttpsPort baddr port cert chainCert key) = | ||
TLS.bindHttps baddr port cert chainCert key | ||
|
||
errorHandle = fromMaybe defaultErrorLogHandler elh | ||
|
||
|
||
------------------------------------------------------------------------------ | ||
debugE :: (MonadIO m) => ByteString -> m () | ||
debugE s = debug $ "Server: " ++ (map w2c $ S.unpack s) | ||
|
||
------------------------------------------------------------------------------ | ||
defaultAccessLogHandler :: AccessLogHandler | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Export this one too, in case users want to mix and match? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't understand, if you simply don't set anything in the config instance then won't this be the default. Do we need to export it if it is the result of the customer doing nothing? I thought that was the beauty of Config being an instance of Monoid. That said I just wanted to be really sure that was what you wanted me to do and add it to the public api. I'm more than happy to do it if you say yes again. 😄 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes I meant that it should be exported from the module. |
||
defaultAccessLogHandler req rsp = do | ||
let hdrs = rqHeaders req | ||
let host = rqRemoteAddr req | ||
let user = Nothing -- TODO we don't do authentication yet | ||
let (v, v') = rqVersion req | ||
let ver = S.concat [ "HTTP/", bshow v, ".", bshow v' ] | ||
let method = toBS $ show (rqMethod req) | ||
let reql = S.intercalate " " [ method, rqURI req, ver ] | ||
let status = rspStatus rsp | ||
let cl = rspContentLength rsp | ||
let referer = maybe Nothing (Just . head) $ H.lookup "referer" hdrs | ||
let userAgent = maybe "-" head $ H.lookup "user-agent" hdrs | ||
|
||
combinedLogEntry host user reql status cl referer userAgent | ||
|
||
------------------------------------------------------------------------------ | ||
defaultErrorLogHandler :: ErrorLogHandler | ||
defaultErrorLogHandler = timestampedLogEntry | ||
|
||
------------------------------------------------------------------------------ | ||
logE :: Maybe (ByteString -> IO ()) -> ByteString -> IO () | ||
logE elog = maybe debugE (\l s -> debugE s >> logE' l s) elog | ||
logE :: ErrorLogHandler -> Maybe (ByteString -> IO ()) -> ByteString -> IO () | ||
logE elh elog = maybe debugE (\l s -> debugE s >> logE' elh l s) elog | ||
|
||
|
||
------------------------------------------------------------------------------ | ||
logE' :: (ByteString -> IO ()) -> ByteString -> IO () | ||
logE' logger s = (timestampedLogEntry s) >>= logger | ||
logE' :: ErrorLogHandler -> (ByteString -> IO ()) -> ByteString -> IO () | ||
logE' elh logger s = logger =<< elh s | ||
|
||
|
||
------------------------------------------------------------------------------ | ||
|
@@ -241,33 +274,21 @@ bshow = toBS . show | |
|
||
|
||
------------------------------------------------------------------------------ | ||
logA :: Maybe (ByteString -> IO ()) -> Request -> Response -> IO () | ||
logA alog = maybe (\_ _ -> return ()) logA' alog | ||
logA :: AccessLogHandler -> Maybe (ByteString -> IO ()) -> Request -> Response -> IO () | ||
logA alh alog = maybe (\_ _ -> return ()) (logA' alh) alog | ||
|
||
|
||
------------------------------------------------------------------------------ | ||
logA' :: (ByteString -> IO ()) -> Request -> Response -> IO () | ||
logA' logger req rsp = do | ||
let hdrs = rqHeaders req | ||
let host = rqRemoteAddr req | ||
let user = Nothing -- TODO we don't do authentication yet | ||
let (v, v') = rqVersion req | ||
let ver = S.concat [ "HTTP/", bshow v, ".", bshow v' ] | ||
let method = toBS $ show (rqMethod req) | ||
let reql = S.intercalate " " [ method, rqURI req, ver ] | ||
let status = rspStatus rsp | ||
let cl = rspContentLength rsp | ||
let referer = maybe Nothing (Just . head) $ H.lookup "referer" hdrs | ||
let userAgent = maybe "-" head $ H.lookup "user-agent" hdrs | ||
|
||
msg <- combinedLogEntry host user reql status cl referer userAgent | ||
logger msg | ||
logA' :: AccessLogHandler -> (ByteString -> IO ()) -> Request -> Response -> IO () | ||
logA' alh logger req rsp = logger =<< alh req rsp | ||
|
||
|
||
------------------------------------------------------------------------------ | ||
runHTTP :: Int -- ^ default timeout | ||
-> Maybe (ByteString -> IO ()) -- ^ access logger | ||
-> Maybe AccessLogHandler | ||
-> Maybe (ByteString -> IO ()) -- ^ error logger | ||
-> Maybe ErrorLogHandler | ||
-> ServerHandler -- ^ handler procedure | ||
-> ByteString -- ^ local host name | ||
-> SessionInfo -- ^ session port information | ||
|
@@ -277,7 +298,7 @@ runHTTP :: Int -- ^ default timeout | |
-- ^ sendfile end | ||
-> ((Int -> Int) -> IO ()) -- ^ timeout tickler | ||
-> IO () | ||
runHTTP defaultTimeout alog elog handler lh sinfo readEnd writeEnd onSendFile | ||
runHTTP defaultTimeout alog alh elog elh handler lh sinfo readEnd writeEnd onSendFile | ||
tickle = | ||
go `catches` [ Handler $ \(_ :: TerminatedBeforeHandlerException) -> do | ||
return () | ||
|
@@ -288,7 +309,7 @@ runHTTP defaultTimeout alog elog handler lh sinfo readEnd writeEnd onSendFile | |
, Handler $ \(e :: AsyncException) -> do | ||
throwIO e | ||
, Handler $ \(e :: SomeException) -> | ||
logE elog $ toByteString $ lmsg e | ||
logE errorHandle elog $ toByteString $ lmsg e | ||
] | ||
|
||
where | ||
|
@@ -301,7 +322,7 @@ runHTTP defaultTimeout alog elog handler lh sinfo readEnd writeEnd onSendFile | |
|
||
go = do | ||
buf <- allocBuffer 16384 | ||
let iter1 = runServerMonad lh sinfo (logA alog) (logE elog) $ | ||
let iter1 = runServerMonad lh sinfo (logA accessHandle alog) (logE errorHandle elog) $ | ||
httpSession defaultTimeout writeEnd buf | ||
onSendFile tickle handler | ||
let iter = iterateeDebugWrapper "httpSession iteratee" iter1 | ||
|
@@ -314,6 +335,9 @@ runHTTP defaultTimeout alog elog handler lh sinfo readEnd writeEnd onSendFile | |
run_ $ readEnd step | ||
debug "runHTTP/go: finished" | ||
|
||
accessHandle = fromMaybe defaultAccessLogHandler alh | ||
errorHandle = fromMaybe defaultErrorLogHandler elh | ||
|
||
|
||
------------------------------------------------------------------------------ | ||
requestErrorMessage :: Request -> SomeException -> Builder | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Unfortunately since we're changing this signature on an exported module (even if it does live in .Internal), this will force a version bump to 1.0.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oh, so you are saying that I can't backport this? I'm confused.
Looking at the exported method it is a completely different type (https://hackage.haskell.org/package/snap-server-0.9.5.1/docs/Snap-Http-Server.html#v:httpServe):
That is because httpServer is different to Int.httpServe in the public Server.hs file.
However, I have added records to the Config instance. Since the type of the Config instance has changed would that cause the same issue. I would hope not. 😞 And...thinking about it some more. I don't think that modifying the Config instance should cause issues in the common case. Config is an instance of Monoid and the docs instruct people to use it as such: https://github.com/snapframework/snap-server/blob/0.9-stable/src/Snap/Internal/Http/Server/Config.hs#L74
So I am hoping that we can backport it. 🙏
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@gregorycollins Sorry to bother you again but any news? :) Sorry. I hope that this reasoning is solid and I'm excited that it could be merged so soon.