Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in / Register
Toggle navigation
C
capnproto
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Packages
Packages
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
submodule
capnproto
Commits
502baf92
Commit
502baf92
authored
May 30, 2018
by
Kenton Varda
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix regressions in HTTP drain.
parent
badade33
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
48 additions
and
23 deletions
+48
-23
http.c++
c++/src/kj/compat/http.c++
+48
-23
No files found.
c++/src/kj/compat/http.c++
View file @
502baf92
...
@@ -1051,19 +1051,7 @@ public:
...
@@ -1051,19 +1051,7 @@ public:
});
});
}
}
// Slightly-crappy code to snarf the expected line break. This will actually eat the leading
snarfBufferedLineBreak
();
// regex /\r*\n?/.
while
(
lineBreakBeforeNextHeader
&&
leftover
.
size
()
>
0
)
{
if
(
leftover
[
0
]
==
'\r'
)
{
leftover
=
leftover
.
slice
(
1
,
leftover
.
size
());
}
else
if
(
leftover
[
0
]
==
'\n'
)
{
leftover
=
leftover
.
slice
(
1
,
leftover
.
size
());
lineBreakBeforeNextHeader
=
false
;
}
else
{
// Err, missing line break, whatever.
lineBreakBeforeNextHeader
=
false
;
}
}
if
(
!
lineBreakBeforeNextHeader
&&
leftover
!=
nullptr
)
{
if
(
!
lineBreakBeforeNextHeader
&&
leftover
!=
nullptr
)
{
return
true
;
return
true
;
...
@@ -1080,6 +1068,13 @@ public:
...
@@ -1080,6 +1068,13 @@ public:
});
});
}
}
bool
isCleanDrain
()
{
// Returns whether we can cleanly drain the stream at this point.
if
(
onMessageDone
!=
nullptr
)
return
false
;
snarfBufferedLineBreak
();
return
!
lineBreakBeforeNextHeader
&&
leftover
!=
nullptr
;
}
kj
::
Promise
<
kj
::
ArrayPtr
<
char
>>
readMessageHeaders
()
{
kj
::
Promise
<
kj
::
ArrayPtr
<
char
>>
readMessageHeaders
()
{
++
pendingMessageCount
;
++
pendingMessageCount
;
auto
paf
=
kj
::
newPromiseAndFulfiller
<
void
>
();
auto
paf
=
kj
::
newPromiseAndFulfiller
<
void
>
();
...
@@ -1362,6 +1357,22 @@ private:
...
@@ -1362,6 +1357,22 @@ private:
}
}
});
});
}
}
void
snarfBufferedLineBreak
()
{
// Slightly-crappy code to snarf the expected line break. This will actually eat the leading
// regex /\r*\n?/.
while
(
lineBreakBeforeNextHeader
&&
leftover
.
size
()
>
0
)
{
if
(
leftover
[
0
]
==
'\r'
)
{
leftover
=
leftover
.
slice
(
1
,
leftover
.
size
());
}
else
if
(
leftover
[
0
]
==
'\n'
)
{
leftover
=
leftover
.
slice
(
1
,
leftover
.
size
());
lineBreakBeforeNextHeader
=
false
;
}
else
{
// Err, missing line break, whatever.
lineBreakBeforeNextHeader
=
false
;
}
}
}
};
};
// -----------------------------------------------------------------------------
// -----------------------------------------------------------------------------
...
@@ -4064,18 +4075,29 @@ public:
...
@@ -4064,18 +4075,29 @@ public:
}
}
kj
::
Promise
<
bool
>
loop
(
bool
firstRequest
)
{
kj
::
Promise
<
bool
>
loop
(
bool
firstRequest
)
{
if
(
!
firstRequest
&&
server
.
draining
&&
httpInput
.
isCleanDrain
())
{
// Don't call awaitNextMessage() in this case because that will initiate a read() which will
// immediately be canceled, losing data.
return
true
;
}
auto
firstByte
=
httpInput
.
awaitNextMessage
();
auto
firstByte
=
httpInput
.
awaitNextMessage
();
if
(
!
firstRequest
)
{
if
(
!
firstRequest
)
{
// For requests after the first, require that the first byte arrive before the pipeline
// For requests after the first, require that the first byte arrive before the pipeline
// timeout, otherwise treat it like the connection was simply closed.
// timeout, otherwise treat it like the connection was simply closed.
auto
timeoutPromise
=
server
.
timer
.
afterDelay
(
server
.
settings
.
pipelineTimeout
)
auto
timeoutPromise
=
server
.
timer
.
afterDelay
(
server
.
settings
.
pipelineTimeout
);
.
exclusiveJoin
(
server
.
onDrain
.
addBranch
())
.
then
([
this
]()
->
bool
{
if
(
httpInput
.
isCleanDrain
())
{
// If we haven't buffered any data, then we can safely drain here, so allow the wait to
// be canceled by the onDrain promise.
timeoutPromise
=
timeoutPromise
.
exclusiveJoin
(
server
.
onDrain
.
addBranch
());
}
firstByte
=
firstByte
.
exclusiveJoin
(
timeoutPromise
.
then
([
this
]()
->
bool
{
timedOut
=
true
;
timedOut
=
true
;
return
false
;
return
false
;
});
}));
firstByte
=
firstByte
.
exclusiveJoin
(
kj
::
mv
(
timeoutPromise
));
}
}
auto
receivedHeaders
=
firstByte
auto
receivedHeaders
=
firstByte
...
@@ -4114,10 +4136,6 @@ public:
...
@@ -4114,10 +4136,6 @@ public:
return
receivedHeaders
return
receivedHeaders
.
then
([
this
](
kj
::
Maybe
<
HttpHeaders
::
Request
>&&
request
)
->
kj
::
Promise
<
bool
>
{
.
then
([
this
](
kj
::
Maybe
<
HttpHeaders
::
Request
>&&
request
)
->
kj
::
Promise
<
bool
>
{
if
(
closed
)
{
// Client closed connection. Close our end too.
return
httpOutput
.
flush
().
then
([]()
{
return
false
;
});
}
if
(
timedOut
)
{
if
(
timedOut
)
{
// Client took too long to send anything, so we're going to close the connection. In
// Client took too long to send anything, so we're going to close the connection. In
// theory, we should send back an HTTP 408 error -- it is designed exactly for this
// theory, we should send back an HTTP 408 error -- it is designed exactly for this
...
@@ -4133,7 +4151,14 @@ public:
...
@@ -4133,7 +4151,14 @@ public:
// error in the case that the server is draining, which also sets timedOut = true; see
// error in the case that the server is draining, which also sets timedOut = true; see
// above.
// above.
return
httpOutput
.
flush
().
then
([
this
]()
{
return
server
.
draining
;
});
return
httpOutput
.
flush
().
then
([
this
]()
{
return
server
.
draining
&&
httpInput
.
isCleanDrain
();
});
}
if
(
closed
)
{
// Client closed connection. Close our end too.
return
httpOutput
.
flush
().
then
([]()
{
return
false
;
});
}
}
KJ_IF_MAYBE
(
req
,
request
)
{
KJ_IF_MAYBE
(
req
,
request
)
{
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment