Difference between revisions of "Reverse HTTP"

From Second Life Wiki
Jump to navigation Jump to search
 
(12 intermediate revisions by 5 users not shown)
Line 1: Line 1:
Reverse HTTP is an experimental protocol which takes advantage of the HTTP/1.1 Upgrade: header to turn one HTTP socket around. When a client makes a request to a server with the Upgrade: PTTH/0.9 header, the server may respond with an Upgrade: PTTH/1.0 header, after which point the server starts using the socket as a client, and the client starts using the socket as a server. There is also a COMET-style protocol which will work with HTTP/0.9 or 1.1 clients that do not know how to perform this upgrade.
Reverse HTTP is an experimental protocol which takes advantage of the HTTP/1.1 Upgrade: header to turn one HTTP socket around. When a client makes a request to a server with the Upgrade: PTTH/0.9 header, the server may respond with an Upgrade: PTTH/1.0 header, after which point the server starts using the socket as a client, and the client starts using the socket as a server. There is also a COMET-style protocol which will work with HTTP/0.9 or 1.1 clients that do not know how to perform this upgrade.


Below is an example transcript using Reverse HTTP. Lines on the left are traffic from the client to the server. Lines on the right are traffic from the server to client. The transcript was created by sniffing the traffic from running this script [[http://soundfarmer.com/paste/B4B75B7A-2E16-4CB1-B0BC-98ADAD2B9214.py]].
March 2009: Zero and Donovan submitted an [http://tools.ietf.org/html/draft-lentczner-rhttp-00 Internet-Draft] formally describing this.
----
Below is an example transcript using Reverse HTTP. Lines on the left are traffic from the client to the server. Lines on the right are traffic from the server to client. Lines with a red background are HTTP requests. Lines with a green background are HTTP responses.
 
The transcript was created by sniffing the traffic from running this script [http://soundfarmer.com/paste/B4B75B7A-2E16-4CB1-B0BC-98ADAD2B9214.py].
 
{|
|style="background-color: #FFA0A0"| Request
|-
|style="background-color: #A0FFA0"| Response
|}


{| width="100%"
{| width="100%"
|POST / HTTP/1.1
|- {{Hl2}}
! '''client -> server'''
! '''server -> client'''
|-
|style="background-color: #FFA0A0"|POST / HTTP/1.1
|
|
|-
|-
|Host: localhost:9999
|style="background-color: #FFA0A0"|Host: localhost:9999
|
|
|-
|-
|Accept-Encoding: identity
|style="background-color: #FFA0A0"|Accept-Encoding: identity
|
|
|-
|-
|Upgrade: PTTH/0.9
|style="background-color: #FFA0A0"|Upgrade: PTTH/0.9
|
|
|-
|-
Line 20: Line 34:
|-
|-
|
|
|HTTP/1.1 101 Switching Protocols
|style="background-color: #A0FFA0"|HTTP/1.1 101 Switching Protocols
|-
|-
|
|
|Content-type: text/plain
|style="background-color: #A0FFA0"|Content-type: text/plain
|-
|-
|
|
|Upgrade: PTTH/0.9
|style="background-color: #A0FFA0"|Upgrade: PTTH/0.9
|-
|-
|
|
|Date: Tue, 13 May 2008 20:14:45 GMT
|style="background-color: #A0FFA0"|Date: Tue, 13 May 2008 20:14:45 GMT
|-
|-
|
|
|Content-Length: 0
|style="background-color: #A0FFA0"|Content-Length: 0
|-
|-
|
|
Line 38: Line 52:
|-
|-
|
|
|GET / HTTP/1.1
|style="background-color: #FFA0A0"|GET / HTTP/1.1
|-
|-
|
|
|Host: 127.0.0.1:65331
|style="background-color: #FFA0A0"|Host: 127.0.0.1:65331
|-
|-
|
|
|Accept-Encoding: identity
|style="background-color: #FFA0A0"|Accept-Encoding: identity
|-
|-
|
|
|accept: text/plain;q=1,*/*;q=0
|style="background-color: #FFA0A0"|accept: text/plain;q=1,*/*;q=0
|-
|-
|
|
|
|
|-
|-
|HTTP/1.1 200 OK
|style="background-color: #A0FFA0"|HTTP/1.1 200 OK
|
|
|-
|-
|Content-type: text/plain
|style="background-color: #A0FFA0"|Content-type: text/plain
|
|
|-
|-
|Date: Tue, 13 May 2008 20:14:45 GMT
|style="background-color: #A0FFA0"|Date: Tue, 13 May 2008 20:14:45 GMT
|
|
|-
|-
|Content-Length: 15
|style="background-color: #A0FFA0"|Content-Length: 15
|
|
|-
|-
Line 67: Line 81:
|
|
|-
|-
|rdlrow ,olleH
|style="background-color: #A0FFA0"|!dlrow ,olleH
|
|
|-
|-
Line 80: Line 94:


{|
{|
|POST / HTTP/1.1
|style="background-color: #FFA0A0"|POST / HTTP/1.1
|
|-
|Host: localhost:9999
|
|
|-
|-
|Accept-Encoding: identity
|style="background-color: #FFA0A0"|Host: localhost:9999
|
|
|-
|-
|Upgrade: PTTH/0.9
|style="background-color: #FFA0A0"|Accept-Encoding: identity
|
|
|-
|-
Line 96: Line 107:
|-
|-
|
|
|HTTP/1.1 200 OK
|style="background-color: #A0FFA0"|HTTP/1.1 200 OK
|-
|-
|
|
|Content-type: application/json
|style="background-color: #A0FFA0"|Content-type: application/json
|-
|-
|
|
|Date: Tue, 13 May 2008 20:14:45 GMT
|style="background-color: #A0FFA0"|Date: Tue, 13 May 2008 20:14:45 GMT
|-
|-
|
|
|Content-Length: 210
|style="background-color: #A0FFA0"|Content-Length: 210
|-
|-
|
|
Line 111: Line 122:
|-
|-
|
|
|{'message-id': 'xxx', 'method': 'GET', 'request-uri': '/', 'http-version': 'PTTH/0.9', 'headers': [['Host', '127.0.0.1:65331'], ['Accept-Encoding', 'identity'], ['Accept','text/plain;q=1,*/*;q=0']], 'body': ''}
|style="background-color: #A0FFA0"|{'message-id': 'xxx', 'method': 'GET', 'request-uri': '/', 'http-version': 'PTTH/0.9', 'headers': [['Host', '127.0.0.1:65331'], ['Accept-Encoding', 'identity'], ['Accept','text/plain;q=1,*/*;q=0']], 'body': ''}
|POST / HTTP/1.1
|-
|style="background-color: #FFA0A0"|POST / HTTP/1.1
|
|
|-
|-
|Host: localhost:9999
|style="background-color: #FFA0A0"|Host: localhost:9999
|
|
|-
|-
|Accept-Encoding: identity
|style="background-color: #FFA0A0"|Accept-Encoding: identity
|
|
|-
|-
|Content-type: application/json
|style="background-color: #FFA0A0"|Content-type: application/json
|
|
|-
|-
|Content-length: 193
|style="background-color: #FFA0A0"|Content-length: 193
|
|
|-
|-
Line 130: Line 142:
|
|
|-
|-
|{'message-id': 'xxx', 'http-version': 'PTTH/0.9', 'status': 200, 'reason': 'OK', 'headers': [['Content-type', 'text/plain'], ['Date', 'Tue, 13 May 2008 20:14:45 GMT']], 'body': 'rdlrow ,olleH'}
|style="background-color: #FFA0A0"|{'message-id': 'xxx', 'http-version': 'PTTH/0.9', 'status': 200, 'reason': 'OK', 'headers': [['Content-type', 'text/plain'], ['Date', 'Tue, 13 May 2008 20:14:45 GMT']], 'body': '!dlrow ,olleH'}
|
|
|-
|-
|
|
|HTTP/1.1 200 OK
|style="background-color: #A0FFA0"|HTTP/1.1 200 OK
|-
|-
|
|
|Content-type: application/json
|style="background-color: #A0FFA0"|Content-type: application/json
|-
|-
|
|
|Date: Tue, 13 May 2008 20:14:45 GMT
|style="background-color: #A0FFA0"|Date: Tue, 13 May 2008 20:14:45 GMT
|-
|-
|
|
|Content-Length: 2
|style="background-color: #A0FFA0"|Content-Length: 2
|-
|-
|
|
Line 149: Line 161:
|-
|-
|
|
|{}
|style="background-color: #A0FFA0"|{}
|}
|}
</pre>


A response of {} from the server means no server-side event occurred within some timeout, and that the client should poll again if it wants to keep receiving events.
A response of {} from the server means no server-side event occurred within some timeout, and that the client should poll again if it wants to keep receiving events.


Another idea is to use text/plain encoding instead of JSON, and just have the actual text of the HTTP request in the response, and the text of the response in the next request. In this case, the message-id would be relayed in a header instead of in the body. Perhaps the message id should always be relayed in a header and never in the body?
Another idea is to use text/plain encoding instead of JSON, and just have the actual text of the HTTP request in the response, and the text of the response in the next request. In this case, the message-id would be relayed in a header instead of in the body. Perhaps the message id should always be relayed in a header and never in the body?

Latest revision as of 22:23, 4 March 2009

Reverse HTTP is an experimental protocol which takes advantage of the HTTP/1.1 Upgrade: header to turn one HTTP socket around. When a client makes a request to a server with the Upgrade: PTTH/0.9 header, the server may respond with an Upgrade: PTTH/1.0 header, after which point the server starts using the socket as a client, and the client starts using the socket as a server. There is also a COMET-style protocol which will work with HTTP/0.9 or 1.1 clients that do not know how to perform this upgrade.

March 2009: Zero and Donovan submitted an Internet-Draft formally describing this.


Below is an example transcript using Reverse HTTP. Lines on the left are traffic from the client to the server. Lines on the right are traffic from the server to client. Lines with a red background are HTTP requests. Lines with a green background are HTTP responses.

The transcript was created by sniffing the traffic from running this script [1].

Request
Response
client -> server server -> client
POST / HTTP/1.1
Host: localhost:9999
Accept-Encoding: identity
Upgrade: PTTH/0.9
HTTP/1.1 101 Switching Protocols
Content-type: text/plain
Upgrade: PTTH/0.9
Date: Tue, 13 May 2008 20:14:45 GMT
Content-Length: 0
GET / HTTP/1.1
Host: 127.0.0.1:65331
Accept-Encoding: identity
accept: text/plain;q=1,*/*;q=0
HTTP/1.1 200 OK
Content-type: text/plain
Date: Tue, 13 May 2008 20:14:45 GMT
Content-Length: 15
!dlrow ,olleH

COMET Fallback

If the client or server is incapable of upgrading from HTTP, a COMET-style protocol can be used instead. The server can maintain an event queue and require that the client repeatedly POST to the queue's resource to receive events. Below is a protocol trace using JSON as the protocol encoding.

POST / HTTP/1.1
Host: localhost:9999
Accept-Encoding: identity
HTTP/1.1 200 OK
Content-type: application/json
Date: Tue, 13 May 2008 20:14:45 GMT
Content-Length: 210
{'message-id': 'xxx', 'method': 'GET', 'request-uri': '/', 'http-version': 'PTTH/0.9', 'headers': [['Host', '127.0.0.1:65331'], ['Accept-Encoding', 'identity'], ['Accept','text/plain;q=1,*/*;q=0']], 'body': }
POST / HTTP/1.1
Host: localhost:9999
Accept-Encoding: identity
Content-type: application/json
Content-length: 193
{'message-id': 'xxx', 'http-version': 'PTTH/0.9', 'status': 200, 'reason': 'OK', 'headers': [['Content-type', 'text/plain'], ['Date', 'Tue, 13 May 2008 20:14:45 GMT']], 'body': '!dlrow ,olleH'}
HTTP/1.1 200 OK
Content-type: application/json
Date: Tue, 13 May 2008 20:14:45 GMT
Content-Length: 2
{}

A response of {} from the server means no server-side event occurred within some timeout, and that the client should poll again if it wants to keep receiving events.

Another idea is to use text/plain encoding instead of JSON, and just have the actual text of the HTTP request in the response, and the text of the response in the next request. In this case, the message-id would be relayed in a header instead of in the body. Perhaps the message id should always be relayed in a header and never in the body?