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
fac09cf7
Unverified
Commit
fac09cf7
authored
Apr 27, 2018
by
Kenton Varda
Committed by
GitHub
Apr 27, 2018
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #662 from capnproto/http-client-adapter
Implement newHttpClient(HttpService&).
parents
8243bab5
8d2ed06b
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
152 additions
and
76 deletions
+152
-76
async-io-test.c++
c++/src/kj/async-io-test.c++
+20
-0
async-io.c++
c++/src/kj/async-io.c++
+21
-1
http-test.c++
c++/src/kj/compat/http-test.c++
+95
-74
http.c++
c++/src/kj/compat/http.c++
+0
-0
http.h
c++/src/kj/compat/http.h
+16
-1
No files found.
c++/src/kj/async-io-test.c++
View file @
fac09cf7
...
@@ -1289,5 +1289,25 @@ KJ_TEST("Userland pipe pumpTo less than write amount") {
...
@@ -1289,5 +1289,25 @@ KJ_TEST("Userland pipe pumpTo less than write amount") {
writePromise
.
wait
(
ws
);
writePromise
.
wait
(
ws
);
}
}
KJ_TEST
(
"Userland pipe pumpFrom EOF on abortRead()"
)
{
kj
::
EventLoop
loop
;
WaitScope
ws
(
loop
);
auto
pipe
=
newOneWayPipe
();
auto
pipe2
=
newOneWayPipe
();
auto
pumpPromise
=
KJ_ASSERT_NONNULL
(
pipe2
.
out
->
tryPumpFrom
(
*
pipe
.
in
));
auto
promise
=
pipe
.
out
->
write
(
"foobar"
,
6
);
KJ_EXPECT
(
!
promise
.
poll
(
ws
));
expectRead
(
*
pipe2
.
in
,
"foobar"
).
wait
(
ws
);
promise
.
wait
(
ws
);
KJ_EXPECT
(
!
pumpPromise
.
poll
(
ws
));
pipe
.
out
=
nullptr
;
pipe2
.
in
=
nullptr
;
// force pump to notice EOF
KJ_EXPECT
(
pumpPromise
.
wait
(
ws
)
==
6
);
pipe2
.
out
=
nullptr
;
}
}
// namespace
}
// namespace
}
// namespace kj
}
// namespace kj
c++/src/kj/async-io.c++
View file @
fac09cf7
...
@@ -522,7 +522,26 @@ private:
...
@@ -522,7 +522,26 @@ private:
void
abortRead
()
override
{
void
abortRead
()
override
{
canceler
.
cancel
(
"abortRead() was called"
);
canceler
.
cancel
(
"abortRead() was called"
);
fulfiller
.
reject
(
KJ_EXCEPTION
(
FAILED
,
"read end of pipe was aborted"
));
// The input might have reached EOF, but we haven't detected it yet because we haven't tried
// to read that far. If we had not optimized tryPumpFrom() and instead used the default
// pumpTo() implementation, then the input would not have called write() again once it
// reached EOF, and therefore the abortRead() on the other end would *not* propagate an
// exception! We need the same behavior here. To that end, we need to detect if we're at EOF
// by reading one last byte.
checkEofTask
=
kj
::
evalNow
([
&
]()
{
static
char
junk
;
return
input
.
tryRead
(
&
junk
,
1
,
1
).
then
([
this
](
uint64_t
n
)
{
if
(
n
==
0
)
{
fulfiller
.
fulfill
(
kj
::
cp
(
pumpedSoFar
));
}
else
{
fulfiller
.
reject
(
KJ_EXCEPTION
(
FAILED
,
"read end of pipe was aborted"
));
}
}).
eagerlyEvaluate
([
this
](
kj
::
Exception
&&
e
)
{
fulfiller
.
reject
(
kj
::
mv
(
e
));
});
});
pipe
.
endState
(
*
this
);
pipe
.
endState
(
*
this
);
pipe
.
abortRead
();
pipe
.
abortRead
();
}
}
...
@@ -547,6 +566,7 @@ private:
...
@@ -547,6 +566,7 @@ private:
uint64_t
amount
;
uint64_t
amount
;
uint64_t
pumpedSoFar
=
0
;
uint64_t
pumpedSoFar
=
0
;
Canceler
canceler
;
Canceler
canceler
;
kj
::
Promise
<
void
>
checkEofTask
=
nullptr
;
};
};
class
BlockedRead
final
:
public
AsyncIoStream
{
class
BlockedRead
final
:
public
AsyncIoStream
{
...
...
c++/src/kj/compat/http-test.c++
View file @
fac09cf7
...
@@ -512,6 +512,33 @@ void testHttpClientResponse(kj::WaitScope& waitScope, const HttpResponseTestCase
...
@@ -512,6 +512,33 @@ void testHttpClientResponse(kj::WaitScope& waitScope, const HttpResponseTestCase
KJ_EXPECT
(
pipe
.
ends
[
1
]
->
readAllText
().
wait
(
waitScope
)
==
""
);
KJ_EXPECT
(
pipe
.
ends
[
1
]
->
readAllText
().
wait
(
waitScope
)
==
""
);
}
}
void
testHttpClient
(
kj
::
WaitScope
&
waitScope
,
HttpHeaderTable
&
table
,
HttpClient
&
client
,
const
HttpTestCase
&
testCase
)
{
KJ_CONTEXT
(
testCase
.
request
.
raw
,
testCase
.
response
.
raw
);
HttpHeaders
headers
(
table
);
for
(
auto
&
header
:
testCase
.
request
.
requestHeaders
)
{
headers
.
set
(
header
.
id
,
header
.
value
);
}
auto
request
=
client
.
request
(
testCase
.
request
.
method
,
testCase
.
request
.
path
,
headers
,
testCase
.
request
.
requestBodySize
);
for
(
auto
&
part
:
testCase
.
request
.
requestBodyParts
)
{
request
.
body
->
write
(
part
.
begin
(),
part
.
size
()).
wait
(
waitScope
);
}
request
.
body
=
nullptr
;
auto
response
=
request
.
response
.
wait
(
waitScope
);
KJ_EXPECT
(
response
.
statusCode
==
testCase
.
response
.
statusCode
);
auto
body
=
response
.
body
->
readAllText
().
wait
(
waitScope
);
if
(
testCase
.
request
.
method
==
HttpMethod
::
HEAD
)
{
KJ_EXPECT
(
body
==
""
);
}
else
{
KJ_EXPECT
(
body
==
kj
::
strArray
(
testCase
.
response
.
responseBodyParts
,
""
),
body
);
}
}
class
TestHttpService
final
:
public
HttpService
{
class
TestHttpService
final
:
public
HttpService
{
public
:
public
:
TestHttpService
(
const
HttpRequestTestCase
&
expectedRequest
,
TestHttpService
(
const
HttpRequestTestCase
&
expectedRequest
,
...
@@ -1062,29 +1089,7 @@ KJ_TEST("HttpClient pipeline") {
...
@@ -1062,29 +1089,7 @@ KJ_TEST("HttpClient pipeline") {
auto
client
=
newHttpClient
(
table
,
*
pipe
.
ends
[
0
]);
auto
client
=
newHttpClient
(
table
,
*
pipe
.
ends
[
0
]);
for
(
auto
&
testCase
:
PIPELINE_TESTS
)
{
for
(
auto
&
testCase
:
PIPELINE_TESTS
)
{
KJ_CONTEXT
(
testCase
.
request
.
raw
,
testCase
.
response
.
raw
);
testHttpClient
(
waitScope
,
table
,
*
client
,
testCase
);
HttpHeaders
headers
(
table
);
for
(
auto
&
header
:
testCase
.
request
.
requestHeaders
)
{
headers
.
set
(
header
.
id
,
header
.
value
);
}
auto
request
=
client
->
request
(
testCase
.
request
.
method
,
testCase
.
request
.
path
,
headers
,
testCase
.
request
.
requestBodySize
);
for
(
auto
&
part
:
testCase
.
request
.
requestBodyParts
)
{
request
.
body
->
write
(
part
.
begin
(),
part
.
size
()).
wait
(
waitScope
);
}
request
.
body
=
nullptr
;
auto
response
=
request
.
response
.
wait
(
waitScope
);
KJ_EXPECT
(
response
.
statusCode
==
testCase
.
response
.
statusCode
);
auto
body
=
response
.
body
->
readAllText
().
wait
(
waitScope
);
if
(
testCase
.
request
.
method
==
HttpMethod
::
HEAD
)
{
KJ_EXPECT
(
body
==
""
);
}
else
{
KJ_EXPECT
(
body
==
kj
::
strArray
(
testCase
.
response
.
responseBodyParts
,
""
),
body
);
}
}
}
client
=
nullptr
;
client
=
nullptr
;
...
@@ -1232,29 +1237,7 @@ KJ_TEST("HttpClient <-> HttpServer") {
...
@@ -1232,29 +1237,7 @@ KJ_TEST("HttpClient <-> HttpServer") {
auto
client
=
newHttpClient
(
table
,
*
pipe
.
ends
[
0
]);
auto
client
=
newHttpClient
(
table
,
*
pipe
.
ends
[
0
]);
for
(
auto
&
testCase
:
PIPELINE_TESTS
)
{
for
(
auto
&
testCase
:
PIPELINE_TESTS
)
{
KJ_CONTEXT
(
testCase
.
request
.
raw
,
testCase
.
response
.
raw
);
testHttpClient
(
waitScope
,
table
,
*
client
,
testCase
);
HttpHeaders
headers
(
table
);
for
(
auto
&
header
:
testCase
.
request
.
requestHeaders
)
{
headers
.
set
(
header
.
id
,
header
.
value
);
}
auto
request
=
client
->
request
(
testCase
.
request
.
method
,
testCase
.
request
.
path
,
headers
,
testCase
.
request
.
requestBodySize
);
for
(
auto
&
part
:
testCase
.
request
.
requestBodyParts
)
{
request
.
body
->
write
(
part
.
begin
(),
part
.
size
()).
wait
(
waitScope
);
}
request
.
body
=
nullptr
;
auto
response
=
request
.
response
.
wait
(
waitScope
);
KJ_EXPECT
(
response
.
statusCode
==
testCase
.
response
.
statusCode
);
auto
body
=
response
.
body
->
readAllText
().
wait
(
waitScope
);
if
(
testCase
.
request
.
method
==
HttpMethod
::
HEAD
)
{
KJ_EXPECT
(
body
==
""
);
}
else
{
KJ_EXPECT
(
body
==
kj
::
strArray
(
testCase
.
response
.
responseBodyParts
,
""
),
body
);
}
}
}
client
=
nullptr
;
client
=
nullptr
;
...
@@ -1628,7 +1611,7 @@ public:
...
@@ -1628,7 +1611,7 @@ public:
if
(
url
==
"/return-error"
)
{
if
(
url
==
"/return-error"
)
{
response
.
send
(
404
,
"Not Found"
,
responseHeaders
,
uint64_t
(
0
));
response
.
send
(
404
,
"Not Found"
,
responseHeaders
,
uint64_t
(
0
));
return
kj
::
READY_NOW
;
return
kj
::
READY_NOW
;
}
else
if
(
url
==
"/w
s-inline
"
)
{
}
else
if
(
url
==
"/w
ebsocket
"
)
{
auto
ws
=
response
.
acceptWebSocket
(
responseHeaders
);
auto
ws
=
response
.
acceptWebSocket
(
responseHeaders
);
return
doWebSocket
(
*
ws
,
"start-inline"
).
attach
(
kj
::
mv
(
ws
));
return
doWebSocket
(
*
ws
,
"start-inline"
).
attach
(
kj
::
mv
(
ws
));
}
else
{
}
else
{
...
@@ -1704,33 +1687,11 @@ kj::ArrayPtr<const byte> asBytes(const char (&chars)[s]) {
...
@@ -1704,33 +1687,11 @@ kj::ArrayPtr<const byte> asBytes(const char (&chars)[s]) {
return
kj
::
ArrayPtr
<
const
char
>
(
chars
,
s
-
1
).
asBytes
();
return
kj
::
ArrayPtr
<
const
char
>
(
chars
,
s
-
1
).
asBytes
();
}
}
KJ_TEST
(
"HttpClient WebSocket handshake"
)
{
void
testWebSocketClient
(
kj
::
WaitScope
&
waitScope
,
HttpHeaderTable
&
headerTable
,
kj
::
EventLoop
eventLoop
;
kj
::
HttpHeaderId
hMyHeader
,
HttpClient
&
client
)
{
kj
::
WaitScope
waitScope
(
eventLoop
);
kj
::
HttpHeaders
headers
(
headerTable
);
auto
pipe
=
kj
::
newTwoWayPipe
();
auto
request
=
kj
::
str
(
"GET /websocket"
,
WEBSOCKET_REQUEST_HANDSHAKE
);
auto
serverTask
=
expectRead
(
*
pipe
.
ends
[
1
],
request
)
.
then
([
&
]()
{
return
pipe
.
ends
[
1
]
->
write
({
asBytes
(
WEBSOCKET_RESPONSE_HANDSHAKE
)});
})
.
then
([
&
]()
{
return
pipe
.
ends
[
1
]
->
write
({
WEBSOCKET_FIRST_MESSAGE_INLINE
});
})
.
then
([
&
]()
{
return
expectRead
(
*
pipe
.
ends
[
1
],
WEBSOCKET_SEND_MESSAGE
);
})
.
then
([
&
]()
{
return
pipe
.
ends
[
1
]
->
write
({
WEBSOCKET_REPLY_MESSAGE
});
})
.
then
([
&
]()
{
return
expectRead
(
*
pipe
.
ends
[
1
],
WEBSOCKET_SEND_CLOSE
);
})
.
then
([
&
]()
{
return
pipe
.
ends
[
1
]
->
write
({
WEBSOCKET_REPLY_CLOSE
});
})
.
eagerlyEvaluate
([](
kj
::
Exception
&&
e
)
{
KJ_LOG
(
ERROR
,
e
);
});
HttpHeaderTable
::
Builder
tableBuilder
;
HttpHeaderId
hMyHeader
=
tableBuilder
.
add
(
"My-Header"
);
auto
headerTable
=
tableBuilder
.
build
();
FakeEntropySource
entropySource
;
auto
client
=
newHttpClient
(
*
headerTable
,
*
pipe
.
ends
[
0
],
entropySource
);
kj
::
HttpHeaders
headers
(
*
headerTable
);
headers
.
set
(
hMyHeader
,
"foo"
);
headers
.
set
(
hMyHeader
,
"foo"
);
auto
response
=
client
->
openWebSocket
(
"/websocket"
,
headers
).
wait
(
waitScope
);
auto
response
=
client
.
openWebSocket
(
"/websocket"
,
headers
).
wait
(
waitScope
);
KJ_EXPECT
(
response
.
statusCode
==
101
);
KJ_EXPECT
(
response
.
statusCode
==
101
);
KJ_EXPECT
(
response
.
statusText
==
"Switching Protocols"
,
response
.
statusText
);
KJ_EXPECT
(
response
.
statusText
==
"Switching Protocols"
,
response
.
statusText
);
...
@@ -1758,6 +1719,33 @@ KJ_TEST("HttpClient WebSocket handshake") {
...
@@ -1758,6 +1719,33 @@ KJ_TEST("HttpClient WebSocket handshake") {
KJ_EXPECT
(
message
.
get
<
WebSocket
::
Close
>
().
code
==
0x1235
);
KJ_EXPECT
(
message
.
get
<
WebSocket
::
Close
>
().
code
==
0x1235
);
KJ_EXPECT
(
message
.
get
<
WebSocket
::
Close
>
().
reason
==
"close-reply:qux"
);
KJ_EXPECT
(
message
.
get
<
WebSocket
::
Close
>
().
reason
==
"close-reply:qux"
);
}
}
}
KJ_TEST
(
"HttpClient WebSocket handshake"
)
{
kj
::
EventLoop
eventLoop
;
kj
::
WaitScope
waitScope
(
eventLoop
);
auto
pipe
=
kj
::
newTwoWayPipe
();
auto
request
=
kj
::
str
(
"GET /websocket"
,
WEBSOCKET_REQUEST_HANDSHAKE
);
auto
serverTask
=
expectRead
(
*
pipe
.
ends
[
1
],
request
)
.
then
([
&
]()
{
return
pipe
.
ends
[
1
]
->
write
({
asBytes
(
WEBSOCKET_RESPONSE_HANDSHAKE
)});
})
.
then
([
&
]()
{
return
pipe
.
ends
[
1
]
->
write
({
WEBSOCKET_FIRST_MESSAGE_INLINE
});
})
.
then
([
&
]()
{
return
expectRead
(
*
pipe
.
ends
[
1
],
WEBSOCKET_SEND_MESSAGE
);
})
.
then
([
&
]()
{
return
pipe
.
ends
[
1
]
->
write
({
WEBSOCKET_REPLY_MESSAGE
});
})
.
then
([
&
]()
{
return
expectRead
(
*
pipe
.
ends
[
1
],
WEBSOCKET_SEND_CLOSE
);
})
.
then
([
&
]()
{
return
pipe
.
ends
[
1
]
->
write
({
WEBSOCKET_REPLY_CLOSE
});
})
.
eagerlyEvaluate
([](
kj
::
Exception
&&
e
)
{
KJ_LOG
(
ERROR
,
e
);
});
HttpHeaderTable
::
Builder
tableBuilder
;
HttpHeaderId
hMyHeader
=
tableBuilder
.
add
(
"My-Header"
);
auto
headerTable
=
tableBuilder
.
build
();
FakeEntropySource
entropySource
;
auto
client
=
newHttpClient
(
*
headerTable
,
*
pipe
.
ends
[
0
],
entropySource
);
testWebSocketClient
(
waitScope
,
*
headerTable
,
hMyHeader
,
*
client
);
serverTask
.
wait
(
waitScope
);
serverTask
.
wait
(
waitScope
);
}
}
...
@@ -1821,7 +1809,7 @@ KJ_TEST("HttpServer WebSocket handshake") {
...
@@ -1821,7 +1809,7 @@ KJ_TEST("HttpServer WebSocket handshake") {
auto
listenTask
=
server
.
listenHttp
(
kj
::
mv
(
pipe
.
ends
[
0
]));
auto
listenTask
=
server
.
listenHttp
(
kj
::
mv
(
pipe
.
ends
[
0
]));
auto
request
=
kj
::
str
(
"GET /w
s-inline
"
,
WEBSOCKET_REQUEST_HANDSHAKE
);
auto
request
=
kj
::
str
(
"GET /w
ebsocket
"
,
WEBSOCKET_REQUEST_HANDSHAKE
);
pipe
.
ends
[
1
]
->
write
({
request
.
asBytes
()}).
wait
(
waitScope
);
pipe
.
ends
[
1
]
->
write
({
request
.
asBytes
()}).
wait
(
waitScope
);
expectRead
(
*
pipe
.
ends
[
1
],
WEBSOCKET_RESPONSE_HANDSHAKE
).
wait
(
waitScope
);
expectRead
(
*
pipe
.
ends
[
1
],
WEBSOCKET_RESPONSE_HANDSHAKE
).
wait
(
waitScope
);
...
@@ -2314,6 +2302,39 @@ KJ_TEST("newHttpService from HttpClient WebSockets disconnect") {
...
@@ -2314,6 +2302,39 @@ KJ_TEST("newHttpService from HttpClient WebSockets disconnect") {
// -----------------------------------------------------------------------------
// -----------------------------------------------------------------------------
KJ_TEST
(
"newHttpClient from HttpService"
)
{
auto
PIPELINE_TESTS
=
pipelineTestCases
();
kj
::
EventLoop
eventLoop
;
kj
::
WaitScope
waitScope
(
eventLoop
);
kj
::
TimerImpl
timer
(
kj
::
origin
<
kj
::
TimePoint
>
());
HttpHeaderTable
table
;
TestHttpService
service
(
PIPELINE_TESTS
,
table
);
auto
client
=
newHttpClient
(
service
);
for
(
auto
&
testCase
:
PIPELINE_TESTS
)
{
testHttpClient
(
waitScope
,
table
,
*
client
,
testCase
);
}
}
KJ_TEST
(
"newHttpClient from HttpService WebSockets"
)
{
kj
::
EventLoop
eventLoop
;
kj
::
WaitScope
waitScope
(
eventLoop
);
kj
::
TimerImpl
timer
(
kj
::
origin
<
kj
::
TimePoint
>
());
auto
pipe
=
kj
::
newTwoWayPipe
();
HttpHeaderTable
::
Builder
tableBuilder
;
HttpHeaderId
hMyHeader
=
tableBuilder
.
add
(
"My-Header"
);
auto
headerTable
=
tableBuilder
.
build
();
TestWebSocketService
service
(
*
headerTable
,
hMyHeader
);
auto
client
=
newHttpClient
(
service
);
testWebSocketClient
(
waitScope
,
*
headerTable
,
hMyHeader
,
*
client
);
}
// -----------------------------------------------------------------------------
class
CountingIoStream
final
:
public
kj
::
AsyncIoStream
{
class
CountingIoStream
final
:
public
kj
::
AsyncIoStream
{
// An AsyncIoStream wrapper which decrements a counter when destroyed (allowing us to count how
// An AsyncIoStream wrapper which decrements a counter when destroyed (allowing us to count how
// many connections are open).
// many connections are open).
...
...
c++/src/kj/compat/http.c++
View file @
fac09cf7
This diff is collapsed.
Click to expand it.
c++/src/kj/compat/http.h
View file @
fac09cf7
...
@@ -424,7 +424,7 @@ public:
...
@@ -424,7 +424,7 @@ public:
// Read one message from the WebSocket and return it. Can only call once at a time. Do not call
// Read one message from the WebSocket and return it. Can only call once at a time. Do not call
// again after Close is received.
// again after Close is received.
kj
::
Promise
<
void
>
pumpTo
(
WebSocket
&
other
);
virtual
kj
::
Promise
<
void
>
pumpTo
(
WebSocket
&
other
);
// Continuously receives messages from this WebSocket and send them to `other`.
// Continuously receives messages from this WebSocket and send them to `other`.
//
//
// On EOF, calls other.disconnect(), then resolves.
// On EOF, calls other.disconnect(), then resolves.
...
@@ -432,6 +432,12 @@ public:
...
@@ -432,6 +432,12 @@ public:
// On other read errors, calls other.close() with the error, then resolves.
// On other read errors, calls other.close() with the error, then resolves.
//
//
// On write error, rejects with the error.
// On write error, rejects with the error.
virtual
kj
::
Maybe
<
kj
::
Promise
<
void
>>
tryPumpFrom
(
WebSocket
&
other
);
// Either returns null, or performs the equivalent of other.pumpTo(*this). Only returns non-null
// if this WebSocket implementation is able to perform the pump in an optimized way, better than
// the default implementation of pumpTo(). The default implementation of pumpTo() always tries
// calling this first, and the default implementation of tryPumpFrom() always returns null.
};
};
class
HttpClient
{
class
HttpClient
{
...
@@ -634,6 +640,15 @@ kj::Own<WebSocket> newWebSocket(kj::Own<kj::AsyncIoStream> stream,
...
@@ -634,6 +640,15 @@ kj::Own<WebSocket> newWebSocket(kj::Own<kj::AsyncIoStream> stream,
// like HTTP requests" in a message as being actual HTTP requests, which could result in cache
// like HTTP requests" in a message as being actual HTTP requests, which could result in cache
// poisoning. See RFC6455 section 10.3.
// poisoning. See RFC6455 section 10.3.
struct
WebSocketPipe
{
kj
::
Own
<
WebSocket
>
ends
[
2
];
};
WebSocketPipe
newWebSocketPipe
();
// Create a WebSocket pipe. Messages written to one end of the pipe will be readable from the other
// end. No buffering occurs -- a message send does not complete until a corresponding receive
// accepts the message.
struct
HttpServerSettings
{
struct
HttpServerSettings
{
kj
::
Duration
headerTimeout
=
15
*
kj
::
SECONDS
;
kj
::
Duration
headerTimeout
=
15
*
kj
::
SECONDS
;
// After initial connection open, or after receiving the first byte of a pipelined request,
// After initial connection open, or after receiving the first byte of a pipelined 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