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
b5425465
Unverified
Commit
b5425465
authored
Feb 16, 2018
by
Kenton Varda
Committed by
GitHub
Feb 16, 2018
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #637 from capnproto/rewrite-websocket
Redesign server-side WebSocket handling.
parents
9032d060
632888d6
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
31 additions
and
40 deletions
+31
-40
http-test.c++
c++/src/kj/compat/http-test.c++
+20
-24
http.c++
c++/src/kj/compat/http.c++
+0
-0
http.h
c++/src/kj/compat/http.h
+11
-16
No files found.
c++/src/kj/compat/http-test.c++
View file @
b5425465
...
@@ -1691,11 +1691,8 @@ public:
...
@@ -1691,11 +1691,8 @@ public:
kj
::
Promise
<
void
>
request
(
kj
::
Promise
<
void
>
request
(
HttpMethod
method
,
kj
::
StringPtr
url
,
const
HttpHeaders
&
headers
,
HttpMethod
method
,
kj
::
StringPtr
url
,
const
HttpHeaders
&
headers
,
kj
::
AsyncInputStream
&
requestBody
,
Response
&
response
)
override
{
kj
::
AsyncInputStream
&
requestBody
,
Response
&
response
)
override
{
KJ_FAIL_ASSERT
(
"can't get here"
);
KJ_ASSERT
(
headers
.
isWebSocket
());
}
kj
::
Promise
<
void
>
openWebSocket
(
kj
::
StringPtr
url
,
const
HttpHeaders
&
headers
,
WebSocketResponse
&
response
)
override
{
HttpHeaders
responseHeaders
(
headerTable
);
HttpHeaders
responseHeaders
(
headerTable
);
KJ_IF_MAYBE
(
h
,
headers
.
get
(
hMyHeader
))
{
KJ_IF_MAYBE
(
h
,
headers
.
get
(
hMyHeader
))
{
responseHeaders
.
set
(
hMyHeader
,
kj
::
str
(
"respond-"
,
*
h
));
responseHeaders
.
set
(
hMyHeader
,
kj
::
str
(
"respond-"
,
*
h
));
...
@@ -2491,26 +2488,25 @@ public:
...
@@ -2491,26 +2488,25 @@ public:
kj
::
Promise
<
void
>
request
(
kj
::
Promise
<
void
>
request
(
HttpMethod
method
,
kj
::
StringPtr
url
,
const
HttpHeaders
&
headers
,
HttpMethod
method
,
kj
::
StringPtr
url
,
const
HttpHeaders
&
headers
,
kj
::
AsyncInputStream
&
requestBody
,
Response
&
response
)
override
{
kj
::
AsyncInputStream
&
requestBody
,
Response
&
response
)
override
{
KJ_ASSERT
(
url
!=
"/throw"
);
if
(
!
headers
.
isWebSocket
())
{
KJ_ASSERT
(
url
!=
"/throw"
);
auto
body
=
kj
::
str
(
headers
.
get
(
HttpHeaderId
::
HOST
).
orDefault
(
"null"
),
":"
,
url
);
auto
stream
=
response
.
send
(
200
,
"OK"
,
HttpHeaders
(
headerTable
),
body
.
size
());
auto
body
=
kj
::
str
(
headers
.
get
(
HttpHeaderId
::
HOST
).
orDefault
(
"null"
),
":"
,
url
);
auto
promises
=
kj
::
heapArrayBuilder
<
kj
::
Promise
<
void
>>
(
2
);
auto
stream
=
response
.
send
(
200
,
"OK"
,
HttpHeaders
(
headerTable
),
body
.
size
());
promises
.
add
(
stream
->
write
(
body
.
begin
(),
body
.
size
()));
auto
promises
=
kj
::
heapArrayBuilder
<
kj
::
Promise
<
void
>>
(
2
);
promises
.
add
(
requestBody
.
readAllBytes
().
ignoreResult
());
promises
.
add
(
stream
->
write
(
body
.
begin
(),
body
.
size
()));
return
kj
::
joinPromises
(
promises
.
finish
()).
attach
(
kj
::
mv
(
stream
),
kj
::
mv
(
body
));
promises
.
add
(
requestBody
.
readAllBytes
().
ignoreResult
());
}
return
kj
::
joinPromises
(
promises
.
finish
()).
attach
(
kj
::
mv
(
stream
),
kj
::
mv
(
body
));
}
else
{
kj
::
Promise
<
void
>
openWebSocket
(
auto
ws
=
response
.
acceptWebSocket
(
HttpHeaders
(
headerTable
));
kj
::
StringPtr
url
,
const
HttpHeaders
&
headers
,
WebSocketResponse
&
response
)
override
{
auto
body
=
kj
::
str
(
headers
.
get
(
HttpHeaderId
::
HOST
).
orDefault
(
"null"
),
":"
,
url
);
auto
ws
=
response
.
acceptWebSocket
(
HttpHeaders
(
headerTable
));
auto
sendPromise
=
ws
->
send
(
body
);
auto
body
=
kj
::
str
(
headers
.
get
(
HttpHeaderId
::
HOST
).
orDefault
(
"null"
),
":"
,
url
);
auto
sendPromise
=
ws
->
send
(
body
);
auto
promises
=
kj
::
heapArrayBuilder
<
kj
::
Promise
<
void
>>
(
2
);
promises
.
add
(
sendPromise
.
attach
(
kj
::
mv
(
body
)));
auto
promises
=
kj
::
heapArrayBuilder
<
kj
::
Promise
<
void
>>
(
2
);
promises
.
add
(
ws
->
receive
().
ignoreResult
());
promises
.
add
(
sendPromise
.
attach
(
kj
::
mv
(
body
)));
return
kj
::
joinPromises
(
promises
.
finish
()).
attach
(
kj
::
mv
(
ws
));
promises
.
add
(
ws
->
receive
().
ignoreResult
());
}
return
kj
::
joinPromises
(
promises
.
finish
()).
attach
(
kj
::
mv
(
ws
));
}
}
private
:
private
:
...
...
c++/src/kj/compat/http.c++
View file @
b5425465
This diff is collapsed.
Click to expand it.
c++/src/kj/compat/http.h
View file @
b5425465
...
@@ -260,6 +260,13 @@ public:
...
@@ -260,6 +260,13 @@ public:
// Creates a shallow clone of the HttpHeaders. The returned object references the same strings
// Creates a shallow clone of the HttpHeaders. The returned object references the same strings
// as the original, owning none of them.
// as the original, owning none of them.
bool
isWebSocket
()
const
;
// Convenience method that checks for the presence of the header `Upgrade: websocket`.
//
// Note that this does not actually validate that the request is a complete WebSocket handshake
// with the correct version number -- such validation will occur if and when you call
// acceptWebSocket().
kj
::
Maybe
<
kj
::
StringPtr
>
get
(
HttpHeaderId
id
)
const
;
kj
::
Maybe
<
kj
::
StringPtr
>
get
(
HttpHeaderId
id
)
const
;
// Read a header.
// Read a header.
...
@@ -413,7 +420,7 @@ public:
...
@@ -413,7 +420,7 @@ public:
virtual
kj
::
Promise
<
Message
>
receive
()
=
0
;
virtual
kj
::
Promise
<
Message
>
receive
()
=
0
;
// 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
EndOfStream
is received.
// again after
Close
is received.
kj
::
Promise
<
void
>
pumpTo
(
WebSocket
&
other
);
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`.
...
@@ -513,6 +520,9 @@ public:
...
@@ -513,6 +520,9 @@ public:
// `statusText` and `headers` need only remain valid until send() returns (they can be
// `statusText` and `headers` need only remain valid until send() returns (they can be
// stack-allocated).
// stack-allocated).
virtual
kj
::
Own
<
WebSocket
>
acceptWebSocket
(
const
HttpHeaders
&
headers
)
=
0
;
// If headers.isWebSocket() is true then you can call acceptWebSocket() instead of send().
kj
::
Promise
<
void
>
sendError
(
uint
statusCode
,
kj
::
StringPtr
statusText
,
kj
::
Promise
<
void
>
sendError
(
uint
statusCode
,
kj
::
StringPtr
statusText
,
const
HttpHeaders
&
headers
);
const
HttpHeaders
&
headers
);
kj
::
Promise
<
void
>
sendError
(
uint
statusCode
,
kj
::
StringPtr
statusText
,
kj
::
Promise
<
void
>
sendError
(
uint
statusCode
,
kj
::
StringPtr
statusText
,
...
@@ -536,21 +546,6 @@ public:
...
@@ -536,21 +546,6 @@ public:
// `url` and `headers` are invalidated on the first read from `requestBody` or when the returned
// `url` and `headers` are invalidated on the first read from `requestBody` or when the returned
// promise resolves, whichever comes first.
// promise resolves, whichever comes first.
class
WebSocketResponse
:
public
Response
{
public
:
virtual
kj
::
Own
<
WebSocket
>
acceptWebSocket
(
const
HttpHeaders
&
headers
)
=
0
;
// Accept and open the WebSocket.
//
// `headers` need only remain valid until acceptWebSocket() returns (it can be stack-allocated).
};
virtual
kj
::
Promise
<
void
>
openWebSocket
(
kj
::
StringPtr
url
,
const
HttpHeaders
&
headers
,
WebSocketResponse
&
response
);
// Tries to open a WebSocket. Default implementation calls request() and never returns a
// WebSocket.
//
// `url` and `headers` are invalidated when the returned promise resolves.
virtual
kj
::
Promise
<
kj
::
Own
<
kj
::
AsyncIoStream
>>
connect
(
kj
::
StringPtr
host
);
virtual
kj
::
Promise
<
kj
::
Own
<
kj
::
AsyncIoStream
>>
connect
(
kj
::
StringPtr
host
);
// Handles CONNECT requests. Only relevant for proxy services. Default implementation throws
// Handles CONNECT requests. Only relevant for proxy services. Default implementation throws
// UNIMPLEMENTED.
// UNIMPLEMENTED.
...
...
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