ADR-49: Signed Fetch V2

More details about this document
Latest published version:
https://adr.decentraland.org/adr/ADR-49
Authors:
2fd
cazala
nachomazzara
Feedback:
GitHub decentraland/adr (pull requests, new issue, open issues)
Edit this documentation:
GitHub View commits View commits on githistory.xyz

This document describe an improved mechanism to communicate Users with Decentraland Services over HTTP messages. This mechanism support all previously use case and includes new security features and follow common and well proved standards:

Summary

Preserved features include:

New features include:

Although this new mechanism is not backward compatible it doesn't include any incompatibility which means that a Service could accept request signed with either of the two mechanisms

Sign a Request

In this mechanism the Authorization header is used to transport the signature of the request:

Authorization

The Authorization request headers contain the credentials to authenticate a User with a Service. Here, the Type is used to differentiate the format on which the Credentials is encode/encrypted.

Authorization = Type + " " + Credentials

In mechanism Type is compose of 3 component:

Type = SignAlgorithm + "+" + HashAlgorithm[+"+" + SignEncoding]

Note: Other methods can be added into this list, like DCL2 for future versions of decentraland-crypto or the algorithm change and TYPED4 if signTypedData_v4 is use to generate the signature

Note: Other Hash methods can be added into this list, like SHA512 and SHA3-256

Example:

Authorization = "DCL+SHA256" + " " + Credentials
Authorization = "DCL+SHA256+BASE64" + " " + Credentials
Authorization = "SIGN+SHA256" + " " + Credentials

Meanwhile Credentials is the user signature of the request (called CanonicalRequest) using decentraland-crypto or eth_sign as appropriate

// Hash the request
Payload = SHA256(CanonicalRequest)

// Generate User AuthLink (Signature)
AuthLink = Authenticator.signPayload(Identity, Payload)

// Stringify AuthLink to get the credentials
Credentials = JSON.stringify(AuthLink)
// Hash the request
Payload = SHA256(CanonicalRequest)

// Generate User Signature
Credentials = Eth.signMessage(Payload)

Example:

Authorization: 'DCL+SHA256 [{"type":"SIGNER","payload":"0x978561a2fcf322d668906a30e561ec3e70756208","signature":""},{"type":"ECDSA_EPHEMERAL","payload":"Decentraland Login\\nEphemeral address: 0x0F7254618741D2FbBAaa2187195B241be2B06BB7\\nExpiration: 2022-01-07T19:38:17.741Z","signature":"0x29b5f488411f059b45b22eff66debb716b0617408e5d648f21d8ded12a15089e7232a591cad5a82f41b6020c779ed4427c8f6d84e4cd4b8be5e26c82eec374b71b"},{"type":"ECDSA_SIGNED_ENTITY","payload":"e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855","signature":"0x5b3cf13b6e21b41df56bbd5b8fb4ef6241306c666bb4136205a15ff74b698d5b10f2c1eab94306ae83d8b61350e19856cc6a610da135dd1b8601beac855e3d321b"}]'
Authorization: "DCL+SHA256+BASE64 W3sidHlwZSI6IlNJR05FUiIsInBheWxvYWQiOiIweDk3ODU2MWEyZmNmMzIyZDY2ODkwNmEzMGU1NjFlYzNlNzA3NTYyMDgiLCJzaWduYXR1cmUiOiIifSx7InR5cGUiOiJFQ0RTQV9FUEhFTUVSQUwiLCJwYXlsb2FkIjoiRGVjZW50cmFsYW5kIExvZ2luXFxuRXBoZW1lcmFsIGFkZHJlc3M6IDB4MEY3MjU0NjE4NzQxRDJGYkJBYWEyMTg3MTk1QjI0MWJlMkIwNkJCN1xcbkV4cGlyYXRpb246IDIwMjItMDEtMDdUMTk6Mzg6MTcuNzQxWiIsInNpZ25hdHVyZSI6IjB4MjliNWY0ODg0MTFmMDU5YjQ1YjIyZWZmNjZkZWJiNzE2YjA2MTc0MDhlNWQ2NDhmMjFkOGRlZDEyYTE1MDg5ZTcyMzJhNTkxY2FkNWE4MmY0MWI2MDIwYzc3OWVkNDQyN2M4ZjZkODRlNGNkNGI4YmU1ZTI2YzgyZWVjMzc0YjcxYiJ9LHsidHlwZSI6IkVDRFNBX1NJR05FRF9FTlRJVFkiLCJwYXlsb2FkIjoiZTNiMGM0NDI5OGZjMWMxNDlhZmJmNGM4OTk2ZmI5MjQyN2FlNDFlNDY0OWI5MzRjYTQ5NTk5MWI3ODUyYjg1NSIsInNpZ25hdHVyZSI6IjB4NWIzY2YxM2I2ZTIxYjQxZGY1NmJiZDViOGZiNGVmNjI0MTMwNmM2NjZiYjQxMzYyMDVhMTVmZjc0YjY5OGQ1YjEwZjJjMWVhYjk0MzA2YWU4M2Q4YjYxMzUwZTE5ODU2Y2M2YTYxMGRhMTM1ZGQxYjg2MDFiZWFjODU1ZTNkMzIxYiJ9XQ=="
Authorization: "SIGN+SHA256 0x5b3cf13b6e21b41df56bbd5b8fb4ef6241306c666bb4136205a15ff74b698d5b10f2c1eab94306ae83d8b61350e19856cc6a610da135dd1b8601beac855e3d321b"

CanonicalRequest

To create a signature that includes information from your request a standardized (canonical) format is required. This ensures that when a Service receives the request, it calculates the same signature that you calculated.

CanonicalRequest =
  HTTPRequestMethod +
  " " +
  CanonicalURI +
  CanonicalQueryString +
  "\n" +
  CanonicalHeaders +
  "\n" +
  BodyHashPayload

1. HTTPRequestMethod

Required: always MUST be included

The HTTP method that will be use to send the request.

HTTPRequestMethod =
  "GET" | "HEAD" | "POST" | "PUT" | "DELETE" | "CONNECT" | "OPTIONS" | "TRACE" | "PATCH"

2. CanonicalURI

Required: MUST be always included

Normalized URI pathname according with RFC 3986

CanonicalURI = "/" | "/path/to/resource" | "/wiki/%C3%91"

Note: In Javascript the URL API encodes the pathname using RFC 3986 as follow:

const url = new URL("https://en.wikipedia.org/wiki/Ñ")
url.pathname === "/wiki/%C3%91"

3. CanonicalQueryString

Required if: the request includes any query string

Normalized URI query string according with RFC 3986

CanonicalURI = "?order=asc" | "?q=%C3%B1"

Note: In Javascript URL API and URLS encodes the query string using RFC 3986 as follow:

const url = new URL("https://www.google.com/search?q=ñ")
url.search === "?q=%C3%B1"
const params = new URLSearchParams("?q=ñ")
params.toString() === "?q=%C3%B1"

4. CanonicalHeaders

The canonical headers consist of a list of all the HTTP headers that you are including with the signed request.

CanonicalHeaders =
  "host:" +
  Host +
  "\n" +
  ContentType +
  "\n" +
  "x-identity-expiration:" +
  Expiration +
  "\n" +
  "x-identity-metadata:" +
  Metadata +
  "\n" +
  ExtraHeaders

4.1. Host

Required: MUST be always included

Host encoded using RFC 3492 and port number, if a non standard http or https port is used, of the server to which the request will be send.

Host = "decentraland.org" | "xn--fiqs8s.asia" | "localhost:8000"

4.2 ContentType

Required if: the request includes some body content it MUST be present, otherwise it MUST be omitted

Indicate the original Media Type of the resource.

ContentType = "application/json"

If the Content-Type header includes the charset directive it MUST be also prensent in lowercase:

ContentType = "application/json; charset=utf-8"

If the Content-Type header is multipart/form-data the boundary MUST NOT be present :

ContentType = "multipart/form-data"

4.3. X-Identity-Expiration

Required: always MUST be included

The moment at which this signature is considered invalid (encoded with RFC 3986)

Expiration = `2020-01-01T00:00:00Z`

4.4 X-Identity-Metadata

Required if: the request includes X-Identity-Metadata header

Extra metadata sent to the server (encoded as JSON)

Metadata = `{"catalyst": "peer.decentraland.org"}`

4.5. ExtraHeaders

Optional

To sign other headers not listed previously Users can include them using X-Identity-Headers, which is a list separated by semicolons. If this header is present, all extra headers listed MUST be included in the signature as well in the same order they were listed

ExtraHeaders =
  "x-identity-headers:" +
  LOWERCASE(SIGNED_HEADER_1) +
  ";" +
  LOWERCASE(SIGNED_HEADER_2) +
  ";" +
  LOWERCASE(SIGNED_HEADER_N) +
  "\n" +
  LOWERCASE(SIGNED_HEADER_1) +
  ":" +
  TRIM(Headers[SIGNED_HEADER_1]) +
  "\n" +
  LOWERCASE(SIGNED_HEADER_2) +
  ":" +
  TRIM(Headers[SIGNED_HEADER_2]) +
  "\n" +
  LOWERCASE(SIGNED_HEADER_N) +
  ":" +
  TRIM(Headers[SIGNED_HEADER_N])

Example: sign Accept and Cookie headers

ExtraHeaders = "x-identity-headers:accept;cookie\n" + "accept:application/json\n" + "cookie:lang=en"

5. BodyHashPayload

Required if: the request includes some body content it MUST be present, otherwise it MUST be omitted

For almost every case this is the result of hashing the content of the body, but for multipart/form-data request a different approach is needed in order to be used on web applications

When ContentType != 'multipart/form-data'

5.1 ContentType != multipart/form-data

Just hash the entire body

BodyHashPayload = SHA265(REQUEST_BODY)

5.2 ContentType == multipart/form-data

The ordered list of all fields in the request, each field MUST be normalized as well

BodyHashPayload = SORT(CanonicalField1, CanonicalField2 /*, ... */)

Each field MUST include name and size directives as prefix of the content hash

CanonicalFieldX = 'name="description";' + "size=50;" + SHA256(FieldContent)

Additionally if some fields are files they MUST also include filename and type directives

CanonicalFieldX =
  'name="description";' +
  'filename="image.png";' +
  'type="image/png";' +
  "size=99999999999;" +
  SHA256(FieldContent)

Canonical Request Examples

GET https://decentraland.org/api/status

GET /api/status
host:decentraland.org
x-identity-expiration:2020-01-01T00:00:00Z

GET https://decentraland.org/api/status with metadata

GET /api/status
host:decentraland.org
x-identity-expiration:2020-01-01T00:00:00Z
x-identity-metadata:{"service":"market.decentraland.org"}

POST https://decentraland.org/api/status?filter=asc with metadata

POST /api/status?filter=asc
host:decentraland.org
x-identity-expiration:2020-01-01T00:00:00Z
x-identity-metadata:{"service":"market.decentraland.org"}

POST https://decentraland.org/api/status with metadata and extra headers

POST /api/status
host:decentraland.org
x-identity-expiration:2020-01-01T00:00:00Z
x-identity-metadata:{"service":"market.decentraland.org"}
x-identity-headers:accept;cookie
accept:*/*
cookie:eu_cn=1;

POST https://decentraland.org/api/status with json data

POST /api/status
host:decentraland.org
content-type:application/json; charset=utf-8
x-identity-expiration:2020-01-01T00:00:00Z
0xe3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855

POST https://decentraland.org/api/status with multipart data

POST /api/status
host:decentraland.org
content-type:multipart/form-data
x-identity-expiration:2020-01-01T00:00:00Z
name="avatar";filename="avatar.png";type="application/png";0x585460e3d01c950dd755f4c369bbf2edb9e6025fa88db029c02bfe6a89e5ec7f
name="email";size=22;0xfefe75065b68e4fb6ef79e1e5f542b84cfe6b8050b01f4ba05a64060131d534b

References

License

Copyright and related rights waived via CC0-1.0. Draft