-
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
Timeout for pure values #118
Comments
Sure. I think this is worth investigating. One thing I'm curious about is whether you can come up with another handler for which |
Here's an example that still fails to timeout with {-# LANGUAGE OverloadedStrings #-}
import Snap.Core
import Snap.Http.Server
import qualified Data.ByteString.Lazy as LBS
import qualified Data.List as L
main = quickHttpServe hmm
hmm :: Snap ()
hmm = do
setTimeout 5
let x = if L.find (==1) (repeat 0) == Just 1 then "A" else "B"
writeLBS $! 33 `LBS.cons` x (Again just |
Before we attempt any fixes it's important to understand why this is happening -- there might not be anything that can be done about it. The GHC runtime relies on calls into the allocator to do scheduling or to check for delivery of async exceptions. See the ghc commentary. Tight loops that don't allocate won't get exceptions and never yield, it's a known problem. See also https://ghc.haskell.org/trac/ghc/ticket/367. You could try building with The question is why it works in one position but not the other. You'd think that the scheduler would get stuck either way -- maybe there's some inlining happening in one case but not the other, or maybe our exception masking is different in one phase (monad computation) vs the other (response delivery). |
Building with I've also, in less-simple repro cases, observed the same behavior with code that's not a tight non-allocating loop. If it helps I can provide an example. |
In this case, yes, please. We may be gobbling the ThreadKilled exception |
Continuing the conversation from here: snapframework/snap-core#283
tl;dr Handlers seem not to correctly time out if computing a pure value causes them to overrun their time deadline.
Repro:
If you
curl 'localhost:8000'
, it won't time out in 5 seconds, or seemingly ever. If you changewriteBS $
towriteBS $!
, though, forcing the value to WHNF is enough to properly time out.The problem, I hypothesize, is that the
hmm
function is immediately returning a thunk, and thus not timing out.My (hand-wavy) proposal is to force to WHNF (or normal form) the ByteString response body, in order to correctly time out. This'll prevent malicious input (to vulnerable handlers) from creating "permanent" un-killed response threads.
Functions I might try and update are
processRequest
orrunServerHandler
inSnap.Internal.Http.Server.Session
.I can try to hack on this but it'd be good to know if it's the right way forward first.
The text was updated successfully, but these errors were encountered: