-
Notifications
You must be signed in to change notification settings - Fork 328
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
Create PROTOCOL.md #71
Changes from 4 commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,174 @@ | ||
# Twirp Wire Protocol | ||
|
||
This document defines the Twirp wire protocol over HTTP. The | ||
current protocol version is v5. | ||
|
||
## Overview | ||
|
||
The Twirp wire protocol is a simple RPC protocol based on HTTP and | ||
Protocol Buffers (proto). The protocol uses HTTP URLs to specify the | ||
RPC endpoints, and sends/receives proto messages as HTTP | ||
request/response bodies. | ||
|
||
To use Twirp, developers first define their APIs using proto files, | ||
then use Twirp tools to generate the client and the server libraries. | ||
The generated libraries implement the Twirp wire protocol, using the | ||
standard HTTP library provided by the programming language runtime or | ||
the operating system. Once the client and the server are implemented, | ||
the client can communicate with the server by making RPC calls. | ||
|
||
The Twirp wire protocol supports both binary and JSON encodings of | ||
proto messages, and works with any HTTP client and any HTTP version. | ||
|
||
### URLs | ||
|
||
In [ABNF syntax](https://tools.ietf.org/html/rfc5234), Twirp's URLs | ||
have the following format: | ||
|
||
**URL ::= Base-URL "/twirp/" [ Package "." ] Service "/" Method** | ||
|
||
The Twirp wire protocol uses HTTP URLs to specify the RPC | ||
endpoints on the server for sending the requests. Such direct mapping | ||
makes the request routing simple and efficient. The Twirp URLs have | ||
the following components. | ||
|
||
* **Base-URL** is the virtual location of a Twirp API server, which is | ||
typically published via API documentation or service discovery. | ||
Currently, it should only contain URL `scheme` and `authority`. For | ||
example, "https://example.com". | ||
|
||
* **Package** is the proto `package` name for an API, which is often | ||
considered as an API version. For example, | ||
`example.calendar.v1`. This component is omitted if the API | ||
definition doesn't have a package name. | ||
|
||
* **Service** is the proto `service` name for an API. For example, | ||
`CalendarService`. | ||
|
||
* **Method** is the proto `rpc` name for an API method. For example, | ||
`CreateEvent`. | ||
|
||
### Requests | ||
|
||
Twirp always uses HTTP POST method to send requests, because it | ||
closely matches the semantics of RPC methods. | ||
|
||
The **Request-Headers** are normal HTTP headers. The Twirp wire | ||
protocol uses the following headers. | ||
|
||
* **Content-Type** header indicates the proto message encoding, which | ||
should be one of "application/protobuf", "application/json". The | ||
server uses this value to decide how to parse the request body, | ||
and encode the response body. | ||
|
||
The **Request-Body** is the encoded request message, contained in the | ||
HTTP request body. The encoding is specified by the `Content-Type` | ||
header. | ||
|
||
### Responses | ||
|
||
The **Response-Headers** are just normal HTTP response headers. The | ||
Twirp wire protocol uses the following headers. | ||
|
||
* **Content-Type** The value should be either "application/protobuf" | ||
or "application/json" to indicate the encoding of the response | ||
message. It must match the "Content-Type" header in the request. | ||
|
||
The **Request-Body** is the encoded response message contained in the | ||
HTTP response body. The encoding is specified by the `Content-Type` | ||
header. | ||
|
||
## Example | ||
|
||
The following example shows a simple Echo API definition and its | ||
corresponding wire payloads. | ||
|
||
The example assumes the server base URL is "https://example.com". | ||
|
||
```proto | ||
syntax = "proto3"; | ||
|
||
package twirp; | ||
|
||
service Echo { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Let's change this to the classic Twirp example, There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Echo is the most classic example for network APIs used in many places. As a protocol spec, we should use the example most developers are already familiar with. For programming languages, everyone chooses hello-world. I think we should just follow the historical tradition. |
||
rpc Hello(HelloRequest) returns (HelloResponse); | ||
} | ||
|
||
message HelloRequest { | ||
string message; | ||
} | ||
|
||
message HelloResponse { | ||
string message; | ||
} | ||
``` | ||
|
||
**Proto Request** | ||
|
||
``` | ||
POST /twirp.Echo/Hello HTTP/1.1 | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The URL here should be The stuttering of 'twirp' here is not ideal. Could you change the package to something like |
||
Host: example.com | ||
Content-Type: application/protobuf | ||
Content-Length: 15 | ||
|
||
<encoded HelloRequest> | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Should we try to represent the encoded protobuf here in some way? I think that might be valuable. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is a public spec. I believe 99% developers don't know or care about proto encoding format. Let's wait and see. We can always add it if many people ask for it. |
||
``` | ||
|
||
**JSON Request** | ||
|
||
``` | ||
POST /twirp.Echo/Hello HTTP/1.1 | ||
Host: example.com | ||
Content-Type: application/json | ||
Content-Length: 27 | ||
|
||
{"message":"Hello, World!"} | ||
``` | ||
|
||
**Proto Response** | ||
|
||
``` | ||
HTTP/1.1 200 OK | ||
Content-Type: application/protobuf | ||
Content-Length: 15 | ||
|
||
<encoded HelloResponse> | ||
``` | ||
|
||
**JSON Response** | ||
|
||
``` | ||
HTTP/1.1 200 OK | ||
Content-Type: application/json | ||
Content-Length: 27 | ||
|
||
{"message":"Hello, World!"} | ||
``` | ||
|
||
## Errors | ||
|
||
Twirp error responses are always JSON-encoded, regardless of | ||
the request's Content-Type, with a corresponding | ||
`Content-Type: application/json` header. This ensures that | ||
the errors are human-readable in any setting. | ||
|
||
Twirp errors are a JSON object with three keys: | ||
|
||
* **code**: One of the Twirp error codes as a string. | ||
* **msg**: A human-readable message describing the error | ||
as a string. | ||
* **meta**: An object with string keys and values holding | ||
arbitrary additional metadata describing the error. | ||
|
||
Example: | ||
``` | ||
{ | ||
"code": "permission_denied", | ||
"msg": "thou shall not pass", | ||
"meta": { | ||
"target": "Balrog" | ||
} | ||
} | ||
``` | ||
|
||
For more information, see https://github.com/twitchtv/twirp/wiki/Errors. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Missing
syntax = "proto3";
at the top of this proto file.