{% extends "global/layout.html" %} {% block title %}SSU Protocol Specification{% endblock %} {% block lastupdated %}February 2015{% endblock %} {% block accuratefor %}0.9.18{% endblock %} {% block content %}
See the SSU page for an overview of the SSU transport.
The initial 2048-bit DH key exchange is described on the SSU page. This exchange uses the same shared prime as that used for I2P's ElGamal encryption.
All UDP datagrams begin with a 16 byte MAC (Message Authentication Code) and a 16 byte IV (Initialization Vector) followed by a variable-size payload encrypted with the appropriate key. The MAC used is HMAC-MD5, truncated to 16 bytes, while the key is a full 32 byte AES256 key. The specific construct of the MAC is the first 16 bytes from:
HMAC-MD5(encryptedPayload + IV + (payloadLength ^ protocolVersion), macKey)where '+' means append and '^' means exclusive-or.
The IV is generated randomly for each packet. The encryptedPayload is the encrypted version of the message starting with the flag byte (encrypt-then-MAC). The payloadLength used in the MAC is a 2 byte unsigned integer. Note that protocolVersion is 0, so the exclusive-or is a no-op. The macKey is either the introduction key or is constructed from the exchanged DH key (see details below), as specified for each message below. WARNING - the HMAC-MD5-128 used here is non-standard, see the cryptography page for details.
The payload itself (that is, the message starting with the flag byte) is AES256/CBC encrypted with the IV and the sessionKey, with replay prevention addressed within its body, explained below.
The protocolVersion is a 2 byte unsigned integer and is currently set to 0. Peers using a different protocol version will not be able to communicate with this peer, though earlier versions not using this flag are.
Within the AES encrypted payload, there is a minimal common structure to the various messages - a one byte flag and a four byte sending timestamp (seconds since the unix epoch). The flag byte contains the following bitfields:
{% highlight %} Bit order: 76543210 (bit 7 is MSB) bits 7-4: payload type bit 3: rekey, always 0, unimplemented bit 2: extended options included, always 0, unimplemented bits 1-0: reserved, set to 0 for compatibility with future uses {% endhighlight %}The header format is:
{% highlight lang='dataspec' %} Header: 37+ bytes Encryption starts with the flag byte. +----+----+----+----+----+----+----+----+ | MAC | + + | | +----+----+----+----+----+----+----+----+ | IV | + + | | +----+----+----+----+----+----+----+----+ |flag| time | | +----+----+----+----+----+ + | keying material (optional) | + + | | ~ ~ | | + +----+----+----+ | |#opt| | +----+----+----+----+----+----+ + | #opt extended option bytes (optional) | ~ ~ ~ ~ +----+----+----+----+----+----+----+----+ {% endhighlight %}Note that rekeying and extended options are unimplemented, so the current header size is exactly 37 bytes.
If the rekey flag is set, 64 bytes of keying material follow the timestamp.
When rekeying, the first 32 bytes of the keying material is fed into a SHA256 to produce the new MAC key, and the next 32 bytes are fed into a SHA256 to produce the new session key, though the keys are not immediately used. The other side should also reply with the rekey flag set and that same keying material. Once both sides have sent and received those values, the new keys should be used and the previous keys discarded. It may be useful to keep the old keys around briefly, to address packet loss and reordering.
NOTE: Rekeying is currently unimplemented.
If the extended options flag is set, a one byte option size value is appended, followed by that many extended option bytes.
NOTE: Extended options is currently unimplemented.
All messages contain 0 or more bytes of padding. Each message must be padded to a 16 byte boundary, as required by the AES256 encryption layer. Through release 0.9.7, messages were only padded to the next 16 byte boundary, and messages not a multiple of 16 bytes could possibly be invalid. As of release 0.9.7, messages may be padded to any length as long as the current MTU is honored. Any extra 1-15 padding bytes beyond the last block of 16 bytes cannot be encrypted or decrypted and will be ignored. However, the full length and all padding is included in the MAC calculation. As of release 0.9.8, transmitted messages are not necessarily a multiple of 16 bytes. The SessionConfirmed message is an exception, see below.
Signatures in the SessionCreated and SessionConfirmed messages are generated using the Signing Public Key from the Router Identity which is distributed out-of-band by publishing in the network database, and the associated Signing Private Key. Through release 0.9.15, the signature algorithm was always DSA, with a 40 byte signature. As of release 0.9.16, the signature algorithm may be specified by a a Key Certificate in Bob's Router Identity.
Both introduction keys and session keys are 32 bytes, and are defined by the Common structures specification. The key used for the MAC and encryption is specified for each message below.
Introduction keys are delivered through an external channel (the network database, where they are identical to the router Hash for now).
There are 10 messages (payload types) defined:
Type | Message | Notes |
---|---|---|
0 | SessionRequest | |
1 | SessionCreated | |
2 | SessionConfirmed | |
3 | RelayRequest | |
4 | RelayResponse | |
5 | RelayIntro | |
6 | Data | |
7 | PeerTest | |
8 | SessionDestroyed | Implemented as of 0.8.9 |
n/a | HolePunch |
This is the first message sent to establish a session.
Peer: | Alice to Bob |
Data: |
|
Crypto Key used: | Bob's introKey, as retrieved from the network database |
MAC Key used: | Bob's introKey, as retrieved from the network database |
Message format:
{% highlight lang='dataspec' %} +----+----+----+----+----+----+----+----+ | X, as calculated from DH | ~ . . . ~ | | +----+----+----+----+----+----+----+----+ |size| that many byte IP address (4-16) | +----+----+----+----+----+----+----+----+ | arbitrary amount of uninterpreted data| ~ . . . ~ {% endhighlight %}Typical size including header, in current implementation: 304 (IPv4) or 320 (IPv6) bytes (before non-mod-16 padding)
This is the response to a Session Request.
Peer: | Bob to Alice |
Data: |
|
Crypto Key used: | Bob's introKey, with an additional layer of encryption over the 40 byte signature and the following 8 bytes padding. |
MAC Key used: | Bob's introKey |
Message format:
{% highlight lang='dataspec' %} +----+----+----+----+----+----+----+----+ | Y, as calculated from DH | ~ . . . ~ | | +----+----+----+----+----+----+----+----+ |size| that many byte IP address (4-16) | +----+----+----+----+----+----+----+----+ | Port (A)| public relay tag | signed +----+----+----+----+----+----+----+----+ on time | | +----+----+ + | | + + | signature | + + | | + + | | + +----+----+----+----+----+----+ | | (0-15 bytes of padding) +----+----+----+----+----+----+----+----+ | | +----+----+ + | arbitrary amount | ~ of uninterpreted data ~ ~ . . . ~ {% endhighlight %}Typical size including header, in current implementation: 368 bytes (IPv4 or IPv6) (before non-mod-16 padding)
This is the response to a Session Created message and the last step in establishing a session. There may be multiple Session Confirmed messages required if the Router Identity must be fragmented.
Peer: | Alice to Bob |
Data: |
|
Crypto Key used: | Alice/Bob sessionKey, as generated from the DH exchange |
MAC Key used: | Alice/Bob MAC Key, as generated from the DH exchange |
Fragment 0 through F-2 (only if F > 1; currently unused, see notes below) :
{% highlight lang='dataspec' %} +----+----+----+----+----+----+----+----+ |info| cursize | | +----+----+----+ + | fragment of Alice's full | ~ Router Identity ~ ~ . . . ~ | | +----+----+----+----+----+----+----+----+ | arbitrary amount of uninterpreted data| ~ . . . ~ {% endhighlight %}Fragment F-1 (last or only fragment):
{% highlight lang='dataspec' %} +----+----+----+----+----+----+----+----+ |info| cursize | | +----+----+----+ + | last fragment of Alice's full | ~ Router Identity ~ ~ . . . ~ | | +----+----+----+----+----+----+----+----+ | signed on time | | +----+----+----+----+ + | arbitrary amount of uninterpreted | ~ data, until the signature at ~ ~ end of the current packet ~ | Packet length must be mult. of 16 | +----+----+----+----+----+----+----+----+ + + | | + + | signature | + + | | + + | | +----+----+----+----+----+----+----+----+ {% endhighlight %}Typical size including header, in current implementation: 480 bytes (before non-mod-16 padding)
The Session Destroyed message was implemented (reception only) in release 0.8.1, and is sent as of release 0.8.9.
Peer: | Alice to Bob or Bob to Alice |
Data: | none |
Crypto Key used: | Alice/Bob sessionKey |
MAC Key used: | Alice/Bob MAC Key |
This message does not contain any data. Typical size including header, in current implementation: 48 bytes (before non-mod-16 padding)
This is the first message sent from Alice to Bob to request an introduction to Charlie.
Peer: | Alice to Bob |
Data: |
|
Crypto Key used: | Bob's introKey, as retrieved from the network database (or Alice/Bob sessionKey, if established) |
MAC Key used: | Bob's introKey, as retrieved from the network database (or Alice/Bob MAC Key, if established) |
Message format:
{% highlight lang='dataspec' %} +----+----+----+----+----+----+----+----+ | relay tag |size| Alice IP addr +----+----+----+----+----+----+----+----+ | Port (A)|size| challenge bytes | +----+----+----+----+ + | to be delivered to Charlie | +----+----+----+----+----+----+----+----+ | Alice's intro key | + + | | + + | | + + | | +----+----+----+----+----+----+----+----+ | nonce | | +----+----+----+----+ + | arbitrary amount of uninterpreted data| ~ . . . ~ {% endhighlight %}Typical size including header, in current implementation: 96 bytes (no Alice IP included) or 112 bytes (4-byte Alice IP included) (before non-mod-16 padding)
This is the response to a Relay Request and is sent from Bob to Alice.
Peer: | Bob to Alice |
Data: |
|
Crypto Key used: | Alice's introKey, as received in the Relay Request (or Alice/Bob sessionKey, if established) |
MAC Key used: | Alice's introKey, as received in the Relay Request (or Alice/Bob MAC Key, if established) |
Message format:
{% highlight lang='dataspec' %} +----+----+----+----+----+----+----+----+ |size| Charlie IP | Port (C)|size| +----+----+----+----+----+----+----+----+ | Alice IP | Port (A)| nonce +----+----+----+----+----+----+----+----+ | arbitrary amount of | +----+----+ + | uninterpreted data | ~ . . . ~ {% endhighlight %}Typical size including header, in current implementation: 64 (Alice IPv4) or 80 (Alice IPv6) bytes (before non-mod-16 padding)
This is the introduction for Alice, which is sent from Bob to Charlie.
Peer: | Bob to Charlie |
Data: |
|
Crypto Key used: | Bob/Charlie sessionKey |
MAC Key used: | Bob/Charlie MAC Key |
Message format:
{% highlight lang='dataspec' %} +----+----+----+----+----+----+----+----+ |size| Alice IP | Port (A)|size| +----+----+----+----+----+----+----+----+ | that many bytes of challenge | + + | data relayed from Alice | +----+----+----+----+----+----+----+----+ | arbitrary amount of uninterpreted data| ~ . . . ~ {% endhighlight %}Typical size including header, in current implementation: 48 bytes (before non-mod-16 padding)
This message is used for data transport and acknowledgment.
Peer: | Any |
Data: |
|
Crypto Key used: | Alice/Bob sessionKey |
MAC Key used: | Alice/Bob MAC Key |
Message format:
{% highlight lang='dataspec' %} +----+----+----+----+----+----+----+----+ |flag| (additional headers, determined | +----+ + ~ by the flags, such as ACKs or ~ | bitfields | +----+----+----+----+----+----+----+----+ |#frg| messageId | frag info | +----+----+----+----+----+----+----+----+ | that many bytes of fragment data | ~ . . . ~ | | +----+----+----+----+----+----+----+----+ | messageId | frag info | | +----+----+----+----+----+----+----+ + | that many bytes of fragment data | ~ . . . ~ | | +----+----+----+----+----+----+----+----+ | messageId | frag info | | +----+----+----+----+----+----+----+ + | that many bytes of fragment data | ~ . . . ~ | | +----+----+----+----+----+----+----+----+ | arbitrary amount of uninterpreted data| ~ . . . ~ {% endhighlight %}See the SSU overview page for details.
Peer: | Any |
Data: |
|
Crypto Key used: |
Listed in order of occurrence:
|
MAC Key used: |
Listed in order of occurrence:
|
Message format:
{% highlight lang='dataspec' %} +----+----+----+----+----+----+----+----+ | test nonce |size| Alice IP addr +----+----+----+----+----+----+----+----+ | Port (A)| | +----+----+----+ + | Alice or Charlie's | + introduction key (Alice's is sent to + | Bob and Charlie, while Charlie's is | + sent to Alice) + | | + +----+----+----+----+----+ | | arbitrary amount of | +----+----+----+ | | uninterpreted data | ~ . . . ~ {% endhighlight %}Typical size including header, in current implementation: 80 bytes (before non-mod-16 padding)
A HolePunch is simply a UDP packet with no data. It is unauthenticated and unencrypted. It does not contain a SSU header, so it does not have a message type number. It is sent from Charlie to Alice as a part of the Introduction sequence.