-
Notifications
You must be signed in to change notification settings - Fork 372
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
net::ERR_CONNECTION_RESET in browser when sending large JSON body and not consuming #1479
Comments
Any chance you could try reproing with just Warp? I can take a stab at that
myself, but it likely won't be till next week.
…On Mon, Jan 22, 2018, 11:39 PM Maximilian Tagher ***@***.***> wrote:
Cross-posted from Stackoverflow
<https://stackoverflow.com/questions/48351199/yesod-neterr-connection-reset-in-browser-when-sending-large-json-body>,
since after some investigation I think this is a bug.
I'm sending a large JSON body (~3MB) in a production app (it's a base64
encoded file). When I send a JSON body and don't consume it (just return
()), Yesod seems to think this worked:
POST /send-json
Accept: */*
Status: 200 OK 0.000098s
But the browser doesn't get a response back and instead gets
net::ERR_CONNECTION_RESET. I've reproduced this in Chrome, Safari and
Firefox.
If I actually consume the body using requireCheckJsonBody, then the issue
goes away, which makes me think this is a bug.
If this is a bug, I imagine it could be somewhere lower level than Yesod,
e.g. warp/wai.
I reproduced this issue in this repo here:
https://github.com/MaxGabriel/large-files. It's the yesod-simple template
with just 1 additional commit: ***@***.***
<MaxGabriel/large-files@38addf7>.
Instead of sending a file, I just copy a string a bunch of times to create
a huge JSON body (a few megabytes).
I couldn't reproduce this issue sending a large binary file in a form, but
I think that may have been because the form was consumed by the Yesod app—I
haven't checked that bit.
I can't reproduce this issue from curl.
Yesod is configured with:
maximumContentLength _ _ = Just (10 * 1024 * 1024) -- 10 megabytes
so receiving a large request body shouldn't be a problem.
Version Info:
https://gist.github.com/MaxGabriel/10e8a999eabcc9cb566c087978f5c3ac
LTS: LTS-Haskell 9.14
—
You are receiving this because you are subscribed to this thread.
Reply to this email directly, view it on GitHub
<#1479>, or mute the thread
<https://github.com/notifications/unsubscribe-auth/AADBBxjNFmZSLx4kQyoxj4DJIVyhjLFYks5tNQAbgaJpZM4RorXe>
.
|
Yes I can reproduce with just Warp: #!/usr/bin/env stack
-- stack --resolver lts-9.14 script
{-# LANGUAGE OverloadedStrings #-}
module Main where
import Network.HTTP.Types (status200)
import Network.Wai
import Network.Wai.Handler.Warp
main = run 3000 $ \request respond -> do
putStrLn "returning response"
_body <- strictRequestBody request -- Comment out this line to trigger the error
respond $ responseLBS status200 [] "Hello World" I used the following HTML to generate the request: <head>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.2.1/jquery.min.js"></script>
</head>
<body>
<script>
$(function() {
$("#send-json").on('click', function(event) {
var string = "";
for (i = 0; i < 55000; i++) {
string += "a".repeat(64)
string += "\n"
}
$.ajax({
url: 'http://localhost:3000/large-json',
type: 'POST',
contentType: "application/json",
data: JSON.stringify({
bigString: string,
}),
success: function (data) {
console.log(data);
},
error: function (data) {
console.log("Error: " + data);
},
});
});
});
</script>
<button id="send-json">Send Large JSON</button>
</body> (Note: to workaround CORS issues, I initially served the HTML file using So far I can't reproduce with a
given a 6.2MB file gets the "Hello World" response delivered as a new web page. |
I'm not sure that this is a bug on the Warp side. I can explain what's happening. When you don't consume the request body yourself in your application, Warp tries to implement an optimization to avoid excessive bandwidth usage: it will send the response body immediately, then close the connection. You can see this with the following telnet session against the example Warp application you gave above:
Even though I indicated that I had a 3mb request body, Warp did not wait for it, gave the response, and then closed the connection. This seems to work fine with |
Ah, ok. This might be the relevant part of the spec? https://tools.ietf.org/html/rfc7230#section-6.6 Some discussion about this on the nodeJS side: nodejs/node#9085 nodejs/node#12339 This does seem somewhat edge-case, but I could definitely see situations where e.g. the server immediately rejects the request without sending a response because the client was missing certain headers. It doesn’t affect me so I’m in no hurry about this, either. In a car so haven’t looked too closely into this since your reply |
Yes, those links look correct. The only other thing I can think of here is that, perhaps, Warp is in violation by not sending a |
I don't think that I understand this perfectly. But what will happen if we use |
I'm not sure how that would change things.
…On Wed, Jan 31, 2018 at 1:52 PM, Kazu Yamamoto ***@***.***> wrote:
I don't think that I understand this perfectly. But what will happen if we
use shutdown instead of close in Warp when a request body remains?
—
You are receiving this because you commented.
Reply to this email directly, view it on GitHub
<#1479 (comment)>,
or mute the thread
<https://github.com/notifications/unsubscribe-auth/AADBB5uikWmpxiF3Your2gQdP0MERWfDks5tQFPsgaJpZM4RorXe>
.
|
If a browser gets an error of connection close while sending data, the browser has no chance to read a response from the server. If the server uses |
If my understanding is correct, this is what @kazu-yamamoto is referencing from the HTTP spec I linked to:
|
In the https://github.com/yesodweb/wai/blob/master/warp/Network/Wai/Handler/Warp/Run.hs#L455 I don't understand what the code is trying to do exactly but I guess that |
As Kazu linked correctly, I think the issue here is exactly what I bring up in yesodweb/wai#673. I also just commented that for nodejs in nodejs/node#12339 (comment).
So I think that
is not a legal optimisation to do. If you To terminate the connection, you must
Step (3) guarantees the other side has actually seen all data sent by the server. Why? Because when the other side sees So I think we need to remove this optimisation. @MaxGabriel Can you confirm whether the problem goes away if you set |
@nh2 Yes, I can confirm that #!/usr/bin/env stack
-- stack --resolver lts-9.14 script
{-# LANGUAGE OverloadedStrings #-}
module Main where
import Network.HTTP.Types (status200)
import Network.Wai
import Network.Wai.Handler.Warp
main = runSettings (setMaximumBodyFlush Nothing defaultSettings) $ \request respond -> do
putStrLn "returning response"
-- _body <- strictRequestBody request -- This line no longer necessary
respond $ responseLBS status200 [] "Hello World" |
Cross-posted from Stackoverflow, since after some investigation I think this is a bug.
I'm sending a large JSON body (~3MB) in a production app (it's a base64 encoded file). When I send a JSON body and don't consume it (just
return ()
), Yesod seems to think this worked:But the browser doesn't get a response back and instead gets
net::ERR_CONNECTION_RESET
. I've reproduced this in Chrome, Safari and Firefox.If I actually consume the body using
requireCheckJsonBody
, then the issue goes away, which makes me think this is a bug.If this is a bug, I imagine it could be somewhere lower level than Yesod, e.g. warp/wai.
I reproduced this issue in this repo here: https://github.com/MaxGabriel/large-files. It's the yesod-simple template with just 1 additional commit: MaxGabriel/large-files@38addf7. Instead of sending a file, I just copy a string a bunch of times to create a huge JSON body (a few megabytes).
I couldn't reproduce this issue sending a large binary file in a form, but I think that may have been because the form was consumed by the Yesod app—I haven't checked that bit.
I can't reproduce this issue from
curl
.Yesod is configured with:
so receiving a large request body shouldn't be a problem.
Version Info: https://gist.github.com/MaxGabriel/10e8a999eabcc9cb566c087978f5c3ac
LTS: LTS-Haskell 9.14
The text was updated successfully, but these errors were encountered: