-
Notifications
You must be signed in to change notification settings - Fork 233
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
2 changed files
with
190 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,89 @@ | ||
# IPIP 0000: Delegated Routing HTTP API | ||
|
||
- Start Date: 2022-10-18 | ||
- Related Issues: | ||
- (add links here) | ||
|
||
## Summary | ||
|
||
This IPIP specifies an HTTP API for delegated routing. | ||
|
||
## Motivation | ||
|
||
Idiomatic and first-class HTTP support for delegated routing is an important requirement for large content routing providers, | ||
and supporting large content providers is a key strategy for driving down IPFS latency. | ||
These providers must handle high volumes of traffic and support many users, so leveraging industry-standard tools and services | ||
such as HTTP load balancers, CDNs, reverse proxies, etc. is a requirement. | ||
To maximize compatibility with standard tools, IPFS needs an HTTP API specification that uses standard HTTP idioms and payload encoding. | ||
The [Reframe spec](https://github.com/ipfs/specs/blob/main/reframe/REFRAME_PROTOCOL.md) for delegated content routing was an experimental attempt at this, | ||
but it has resulted in a very unidiomatic HTTP API which is difficult to implement and is incompatible with many existing tools. | ||
The cost of a proper redesign, implementation, and maintenance of Reframe and its implementation is too high relative to the urgency of having a delegated routing HTTP API. | ||
|
||
Note that this does not supplant nor deprecate Reframe. Ideally in the future, Reframe and its implementation would receive the resources needed to map the IDL to idiomatic HTTP, | ||
and this spec could then be rewritten in the IDL, maintaining backwards compatibility. | ||
|
||
## Detailed design | ||
|
||
See the [API design](../routing/DELEGATED_ROUTING_HTTP.md) included with this IPIP. | ||
|
||
## Design rationale | ||
To understand the design rationale, it is important to consider the concrete Reframe limitations that we know about: | ||
|
||
- Reframe methods are encoded inside messages | ||
- This prevents URL-based pattern matching on methods | ||
- Configuring different caching strategies for different methods | ||
- Configuring reverse proxies on a per-method basis | ||
- Routing methods to specific backends | ||
- Method-specific reverse proxy config such as timeouts | ||
- Developer UX is poor as a result, e.g. for CDN caching you must encode the entire request message and pass it as a query parameter | ||
- This was initially done by URL-escaping the raw bytes | ||
- Not possible to consume correctly using standard JavaScript | ||
- Shipped in Kubo 0.16 | ||
- Packing a CID into a struct, encoding it with DAG-CBOR, multibase-encoding that, percent-encoding that, and then passing it in a URL, rather than merely passing the CID in the URL, is needlessly complex from a user's perspective | ||
- Added complexity of "Cacheable" methods supporting both POSTs and GETs | ||
- The required streaming support adds a lot of implementation complexity but isn’t very useful | ||
- Ex for FindProviders, the response is buffered anyway for ETag calculation | ||
- There are no limits on response sizes nor ways to impose limits and paginate | ||
- Message groups (newline-delimited messages) also add unnecessary complexity | ||
- Due to this complexity, we currently have a bug that requires a newline at the end of the HTTP message body | ||
- Nothing is actually using this | ||
- The Identify method is not implemented because it is not currently useful | ||
- Client and server implementations are difficult to write correctly, because of the non-standard wire formats and conventions | ||
- The Go implementation is [complex](https://github.com/ipfs/go-delegated-routing/blob/main/gen/proto/proto_edelweiss.go) and [brittle](https://github.com/ipfs/go-delegated-routing/blame/main/client/provide.go#L51), and is currently maintained by IPFS Stewards who are already over-committed with other priorities | ||
- Only the HTTP transport has been designed and implemented, so it's unclear if the existing design will work for other transports, and what their use cases and requirements are | ||
|
||
So this API proposal makes the following changes: | ||
|
||
- The API is defined in HTTP directly | ||
- "Methods" and cache-relevant parameters are pushed into the URL path | ||
- Streaming support and "message groups" are removed, and pagination is added, which limits the response size and provides a scalable mechanism for iterating over arbitrarily-large collections | ||
- Bodies are encoded using standard JSON or CBOR, instead of using IPLD codecs | ||
- The "Identify" method is removed | ||
|
||
### User benefit | ||
|
||
The cost of building and operating content routing services will be much lower, as developers will be able to reuse existing industry-standard tooling. | ||
This will result in more content routing providers, each providing a better experience for users, driving down content routing latency across the IPFS netowrk | ||
and increasing data availability. | ||
|
||
### Compatibility | ||
|
||
#### Backwards Compatibility | ||
IPFS Stewards will implement this API in [go-delegated-routing](https://github.com/ipfs/go-delegated-routing), using breaking changes in a new minor version. | ||
Because the existing Reframe spec can't be safely used in JavaScript, the experimental support for Reframe in Kubo will be removed in the next release, | ||
and delegated routing will subsequently use this HTTP API. We may decide to re-add Reframe support in the future once these issues have been resolved. | ||
|
||
#### Forwards Compatibility | ||
Standard HTTP mechanisms for forward compatibility are used--the API is versioned using a version number in the path. The `Accept` and `Content-Type` headers are used for content type negotiation. new methods will result in new paths, and parameters can be added using either new query parameters or new fields in the request/response body. Certain parts of bodies are labeled as "opaque bytes", which are passed through by the implementation, with no schema enforcement. | ||
|
||
### Security | ||
|
||
None | ||
|
||
### Alternatives | ||
|
||
This *is* an alternative. | ||
|
||
### Copyright | ||
|
||
Copyright and related rights waived via [CC0](https://creativecommons.org/publicdomain/zero/1.0/). |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,101 @@ | ||
# ![](https://img.shields.io/badge/status-wip-orange.svg?style=flat-square) Delegated Routing HTTP API | ||
|
||
**Author(s)**: | ||
- Gus Eggert | ||
|
||
**Maintainer(s)**: | ||
|
||
* * * | ||
|
||
**Abstract** | ||
|
||
"Delegated routing" is a mechanism for IPFS implementations to use for offloading content routing to another process/server. This spec describes an HTTP API for delegated routing. | ||
|
||
# Organization of this document | ||
|
||
- [Introduction](#introduction) | ||
- [Spec](#spec) | ||
- [Interaction Pattern](#interaction-pattern) | ||
- [Cachability](#cachability) | ||
- [Transports](#transports) | ||
- [Protocol Message Overview](#protocol-message-overview) | ||
- [Known Methods](#known-methods) | ||
- [Method Upgrade Paths](#method-upgrade-paths) | ||
- [Implementations](#implementations) | ||
|
||
# API Specification | ||
By default, the Delegated Routing HTTP API uses the `application/json` content type. Clients and servers may optionally negotiate other content types such as `application/cbor`, `application/vnd.ipfs.rpc+dag-json`, etc. using the standard `Accept` and `Content-Type` headers. | ||
|
||
- `GET /v1/providers/{CID}` | ||
- Reframe equivalent: FindProviders | ||
- Response | ||
|
||
```json | ||
{ | ||
"Providers": [ | ||
{ | ||
"PeerID": "...", | ||
"Multiaddrs": ["...", "..."] | ||
"Protocols": [ | ||
{ | ||
"Codec": 2320, | ||
"Payload": <opaque data> | ||
} | ||
] | ||
} | ||
] | ||
"NextPageToken": "<token>" | ||
} | ||
``` | ||
|
||
- Default limit: 100 providers | ||
- Optional query parameters | ||
- `transfer` only return providers who support the passed transfer protocols, expressed as a comma-separated list of multicodec IDs such as `2304,2320`, | ||
- `transport` only return providers whose published multiaddrs explicitly support the passed transport protocols, such as `/quic` or `/tls/ws`. | ||
- `GET /v1/providers/hash/{multihash}` | ||
- This is the same as `GET /v1/providers/{CID}`, but takes a hashed CID encoded as a multihash | ||
- `GET /v1/ipns/{ID}` | ||
- Reframe equivalent: GetIPNS | ||
- Response | ||
- record bytes | ||
- `POST /v1/ipns/{ID}` | ||
- Reframe equivalent: PutIPNS | ||
- Body | ||
- record bytes | ||
- No need for idempotency | ||
- `PUT /v1/providers/{CID}` | ||
- Reframe equivalent: Provide | ||
- Body | ||
|
||
```json | ||
{ | ||
"Keys": ["cid1", "cid2"], | ||
"Timestamp": 1234, | ||
"AdvisoryTTL": 1234, | ||
"Signature": "multibase bytes", | ||
"Provider": { | ||
"Peer": { | ||
"ID": "peerID", | ||
"Addrs": ["multiaddr1", "multiaddr2"] | ||
}, | ||
"Protocols": [ | ||
{ | ||
"Codec": 1234, | ||
"Payload": <opaque data> | ||
} | ||
] | ||
} | ||
} | ||
``` | ||
|
||
- Idempotent | ||
- `GET /v1/ping` | ||
- This is absent from Reframe but is necessary for supporting e.g. the accelerated DHT client which can take many minutes to bootstrap | ||
- Returns 200 once the server is ready to accept requests | ||
- An alternate approach is w/ an orchestration dance in the server by not listening on the socket until the dependencies are ready, but this makes the “dance” easier to implement | ||
- Pagination | ||
- Responses with collections of results must have a default limit on the number of results that will be returned in a single response | ||
- Servers may optionally implement pagination by responding with an opaque page token which, when provided as a subsequent query parameter, will fetch the next page of results. | ||
- Clients may continue paginating until no `NextPageToken` is returned. | ||
- Clients making calls that return collections may limit the number of per-page results returned with the `limit` query parameter, i.e. `GET /v1/providers/{CID}?limit=10` | ||
- Additional filtering/sorting operations may be defined on a per-path basis, as needed |